ITSMConfigurationManagement
6.5.1
6.5.x
ITSMCore
Znuny
https://www.znuny.org/
GNU GENERAL PUBLIC LICENSE Version 3, 29 June 2007
The Znuny::ITSM Configuration Management package.
Das Znuny::ITSM Configuration Management Paket.
El paquete Znuny::ITSM Configuration Management.
Az Znuny::ITSM konfigurációmenedzsment csomag.
Initial release for Znuny 6.5.
Updated translations.
Initial release for Znuny 6.4.2.
Release for Znuny 6.3.3.
Ported to 6.3.
Znuny LTS
build.znuny.com
2023-02-20 14:46:19 UTC
<?xml version="1.0" encoding="utf-8" ?>
<otrs_config version="2.0" init="Config">
    <!-- nofilter(TidyAll::Plugin::Znuny::Config::ACLKeysLevel3Actions) -->
    <Setting Name="Frontend::Module###AdminITSMConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Navigation>Frontend::Admin::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="GroupRo">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="Group">
                        <Array>
                            <Item>admin</Item>
                        </Array>
                    </Item>
                    <Item Key="Description" Translatable="1">Admin.</Item>
                    <Item Key="Title" Translatable="1">Config Item</Item>
                    <Item Key="NavBarName">Admin</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Frontend::Navigation###AdminITSMConfigItem###003-ITSMConfigItem" Required="0" Valid="0">
        <Description Translatable="1">Main menu item registration.</Description>
        <Navigation>Frontend::Admin::ModuleRegistration::MainMenu</Navigation>
        <Value>
            <Array>
                <DefaultItem ValueType="FrontendNavigation">
                    <Hash>
                    </Hash>
                </DefaultItem>
            </Array>
        </Value>
    </Setting>
    <Setting Name="Frontend::NavigationModule###AdminITSMConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Admin area navigation for the agent interface.</Description>
        <Navigation>Frontend::Admin::ModuleRegistration::AdminOverview</Navigation>
        <Value>
            <Hash>
                <Item Key="Group">
                    <Array>
                        <Item>admin</Item>
                    </Array>
                </Item>
                <Item Key="GroupRo">
                    <Array>
                    </Array>
                </Item>
                <Item Key="Module">Kernel::Output::HTML::NavBar::ModuleAdmin</Item>
                <Item Key="Name" Translatable="1">Config Items</Item>
                <Item Key="Block">CMDBSettings</Item>
                <Item Key="Description" Translatable="1">Create and manage the definitions for Configuration Items.</Item>
                <Item Key="IconBig">fa-laptop</Item>
                <Item Key="IconSmall"></Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AgentITSMConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="Group">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="GroupRo">
                        <Array>
                            <Item>itsm-configitem</Item>
                        </Array>
                    </Item>
                    <Item Key="Description" Translatable="1">ITSM config item overview.</Item>
                    <Item Key="NavBarName">Config Item</Item>
                    <Item Key="Title" Translatable="1">Config Item</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Loader::Module::AgentITSMConfigItem###003-ITSMConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Loader module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration::Loader</Navigation>
        <Value>
            <Hash>
                <Item Key="CSS">
                    <Array>
                        <Item>ITSM.Agent.Default.css</Item>
                    </Array>
                </Item>
                <Item Key="JavaScript">
                    <Array>
                        <Item>ITSM.Agent.ConfigItem.Overview.js</Item>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Frontend::Navigation###AgentITSMConfigItem###003-ITSMConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Main menu item registration.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration::MainMenu</Navigation>
        <Value>
            <Array>
                <DefaultItem ValueType="FrontendNavigation">
                    <Hash>
                    </Hash>
                </DefaultItem>
                <Item>
                    <Hash>
                        <Item Key="Group">
                            <Array>
                            </Array>
                        </Item>
                        <Item Key="GroupRo">
                            <Array>
                                <Item>itsm-configitem</Item>
                            </Array>
                        </Item>
                        <Item Key="Description" Translatable="1">Configuration Management Database.</Item>
                        <Item Key="Name">CMDB</Item>
                        <Item Key="Link">Action=AgentITSMConfigItem</Item>
                        <Item Key="LinkOption"></Item>
                        <Item Key="NavBar">Config Item</Item>
                        <Item Key="Type">Menu</Item>
                        <Item Key="Block">ItemArea</Item>
                        <Item Key="AccessKey"></Item>
                        <Item Key="Prio">3200</Item>
                    </Hash>
                </Item>
                <Item>
                    <Hash>
                        <Item Key="Group">
                            <Array>
                            </Array>
                        </Item>
                        <Item Key="GroupRo">
                            <Array>
                                <Item>itsm-configitem</Item>
                            </Array>
                        </Item>
                        <Item Key="Description" Translatable="1">Overview.</Item>
                        <Item Key="Name">Overview</Item>
                        <Item Key="Link">Action=AgentITSMConfigItem</Item>
                        <Item Key="LinkOption"></Item>
                        <Item Key="NavBar">Config Item</Item>
                        <Item Key="Type"></Item>
                        <Item Key="Block"></Item>
                        <Item Key="AccessKey"></Item>
                        <Item Key="Prio">100</Item>
                    </Hash>
                </Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AgentITSMConfigItemZoom" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="Group">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="GroupRo">
                        <Array>
                            <Item>itsm-configitem</Item>
                        </Array>
                    </Item>
                    <Item Key="Description" Translatable="1">Config item zoom.</Item>
                    <Item Key="Title" Translatable="1">Zoom</Item>
                    <Item Key="NavBarName">Config Item</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Loader::Module::AgentITSMConfigItemZoom###003-ITSMConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Loader module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration::Loader</Navigation>
        <Value>
            <Hash>
                <Item Key="CSS">
                    <Array>
                        <Item>ITSM.Agent.Default.css</Item>
                        <Item>ITSM.Agent.Detail.css</Item>
                        <Item>Core.AllocationList.css</Item>
                        <Item>ITSM.Print.css</Item>
                    </Array>
                </Item>
                <Item Key="JavaScript">
                    <Array>
                        <Item>thirdparty/jquery-tablesorter-2.28.14/jquery.tablesorter.js</Item>
                        <Item>Core.UI.AllocationList.js</Item>
                        <Item>Core.UI.Table.Sort.js</Item>
                        <Item>Core.Agent.TableFilters.js</Item>
                        <Item>Core.Agent.LinkObject.js</Item>
                        <Item>ITSM.Agent.Zoom.js</Item>
                        <Item>ITSM.Agent.ConfirmDialog.js</Item>
                        <Item>ITSM.Agent.ConfigItem.Zoom.js</Item>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AgentITSMConfigItemEdit" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="GroupRo">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="Group">
                        <Array>
                            <Item>itsm-configitem</Item>
                        </Array>
                    </Item>
                    <Item Key="Description" Translatable="1">Config item edit.</Item>
                    <Item Key="Title" Translatable="1">Edit</Item>
                    <Item Key="NavBarName">Config Item</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Loader::Module::AgentITSMConfigItemEdit###003-ITSMConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Loader module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration::Loader</Navigation>
        <Value>
            <Hash>
                <Item Key="CSS">
                    <Array>
                        <Item>ITSM.Agent.ConfigItem.Edit.css</Item>
                    </Array>
                </Item>
                <Item Key="JavaScript">
                    <Array>
                        <Item>ITSM.Agent.ConfigItem.UserSearch.js</Item>
                        <Item>ITSM.Agent.ConfigItem.CustomerSearch.js</Item>
                        <Item>ITSM.Agent.ConfigItem.Edit.js</Item>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AgentITSMConfigItemPrint" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="Group">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="GroupRo">
                        <Array>
                            <Item>itsm-configitem</Item>
                        </Array>
                    </Item>
                    <Item Key="Description" Translatable="1">Config item print.</Item>
                    <Item Key="Title" Translatable="1">Print</Item>
                    <Item Key="NavBarName">Config Item</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AgentITSMConfigItemDelete" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="GroupRo">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="Group">
                        <Array>
                            <Item>itsm-configitem</Item>
                        </Array>
                    </Item>
                    <Item Key="Description">Delete a configuration item</Item>
                    <Item Key="Title">Delete</Item>
                    <Item Key="NavBarName">ITSM ConfigItem</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AgentITSMConfigItemAdd" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="GroupRo">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="Group">
                        <Array>
                            <Item>itsm-configitem</Item>
                        </Array>
                    </Item>
                    <Item Key="Description" Translatable="1">Config item add.</Item>
                    <Item Key="Title" Translatable="1">Add</Item>
                    <Item Key="NavBarName">Config Item</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Loader::Module::AgentITSMConfigItemAdd###003-ITSMConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Loader module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration::Loader</Navigation>
        <Value>
            <Hash>
                <Item Key="JavaScript">
                    <Array>
                        <Item>ITSM.Agent.ConfigItem.Add.js</Item>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Frontend::Navigation###AgentITSMConfigItemAdd###003-ITSMConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Main menu item registration.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration::MainMenu</Navigation>
        <Value>
            <Array>
                <DefaultItem ValueType="FrontendNavigation">
                    <Hash>
                    </Hash>
                </DefaultItem>
                <Item>
                    <Hash>
                        <Item Key="Group">
                            <Array>
                                <Item>itsm-configitem</Item>
                            </Array>
                        </Item>
                        <Item Key="GroupRo">
                            <Array>
                            </Array>
                        </Item>
                        <Item Key="Description" Translatable="1">New</Item>
                        <Item Key="Name">New</Item>
                        <Item Key="Link">Action=AgentITSMConfigItemAdd</Item>
                        <Item Key="LinkOption"></Item>
                        <Item Key="NavBar">Config Item</Item>
                        <Item Key="Type"></Item>
                        <Item Key="Block"></Item>
                        <Item Key="AccessKey"></Item>
                        <Item Key="Prio">200</Item>
                    </Hash>
                </Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AgentITSMConfigItemSearch" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="Group">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="GroupRo">
                        <Array>
                            <Item>itsm-configitem</Item>
                        </Array>
                    </Item>
                    <Item Key="Description" Translatable="1">Search config items.</Item>
                    <Item Key="Title" Translatable="1">Search</Item>
                    <Item Key="NavBarName">Config Item</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Loader::Module::AgentITSMConfigItemSearch###003-ITSMConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Loader module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration::Loader</Navigation>
        <Value>
            <Hash>
                <Item Key="CSS">
                    <Array>
                        <Item>ITSM.Agent.Detail.css</Item>
                        <Item>ITSM.Table.css</Item>
                        <Item>ITSM.Print.css</Item>
                    </Array>
                </Item>
                <Item Key="JavaScript">
                    <Array>
                        <Item>thirdparty/jquery-tablesorter-2.28.14/jquery.tablesorter.js</Item>
                        <Item>Core.UI.Table.Sort.js</Item>
                        <Item>ITSM.Agent.Zoom.js</Item>
                        <Item>ITSM.Agent.ConfigItem.Overview.js</Item>
                        <Item>ITSM.Agent.ConfigItem.Search.js</Item>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Frontend::Navigation###AgentITSMConfigItemSearch###003-ITSMConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Main menu item registration.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration::MainMenu</Navigation>
        <Value>
            <Array>
                <DefaultItem ValueType="FrontendNavigation">
                    <Hash>
                    </Hash>
                </DefaultItem>
                <Item>
                    <Hash>
                        <Item Key="Group">
                            <Array>
                            </Array>
                        </Item>
                        <Item Key="GroupRo">
                            <Array>
                                <Item>itsm-configitem</Item>
                            </Array>
                        </Item>
                        <Item Key="Description" Translatable="1">Search config items.</Item>
                        <Item Key="Name">Search</Item>
                        <Item Key="Link">Action=AgentITSMConfigItemSearch</Item>
                        <Item Key="LinkOption">onclick="ITSM.Agent.ConfigItem.Search.OpenSearchDialog('AgentITSMConfigItemSearch'); if (event.stopPropagation) { event.stopPropagation(); } else { window.event.cancelBubble = true; } return false;"</Item>
                        <Item Key="NavBar">Config Item</Item>
                        <Item Key="Type"></Item>
                        <Item Key="Block"></Item>
                        <Item Key="AccessKey"></Item>
                        <Item Key="Prio">300</Item>
                    </Hash>
                </Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::MenuModule###000-Back" Required="0" Valid="1">
        <Description Translatable="1">Shows a link in the menu to go back in the configuration item zoom view of the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::MenuModule</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::ITSMConfigItem::MenuGeneric</Item>
                <Item Key="Name" Translatable="1">Back</Item>
                <Item Key="Description" Translatable="1">Back</Item>
                <Item Key="Action"></Item>
                <Item Key="Link">[% Env("LastScreenOverview") %]</Item>
                <Item Key="Target"></Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::MenuModule###200-History" Required="0" Valid="1">
        <Description Translatable="1">Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::MenuModule</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::ITSMConfigItem::MenuGeneric</Item>
                <Item Key="Name" Translatable="1">History</Item>
                <Item Key="Description" Translatable="1">History</Item>
                <Item Key="Action">AgentITSMConfigItemHistory</Item>
                <Item Key="Link">Action=AgentITSMConfigItemHistory;ConfigItemID=[% Data.ConfigItemID | html %];VersionID=[% Data.VersionID | html %]</Item>
                <Item Key="Target">PopUp</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::MenuModule###300-Edit" Required="0" Valid="1">
        <Description Translatable="1">Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::MenuModule</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::ITSMConfigItem::MenuGeneric</Item>
                <Item Key="Name" Translatable="1">Edit</Item>
                <Item Key="Description" Translatable="1">Edit</Item>
                <Item Key="Action">AgentITSMConfigItemEdit</Item>
                <Item Key="Link">Action=AgentITSMConfigItemEdit;ConfigItemID=[% Data.ConfigItemID | html %]</Item>
                <Item Key="Target">PopUp</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::MenuModule###400-Print" Required="0" Valid="1">
        <Description Translatable="1">Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::MenuModule</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::ITSMConfigItem::MenuGeneric</Item>
                <Item Key="Name" Translatable="1">Print</Item>
                <Item Key="Description" Translatable="1">Print</Item>
                <Item Key="Action">AgentITSMConfigItemPrint</Item>
                <Item Key="Link">Action=AgentITSMConfigItemPrint;ConfigItemID=[% Data.ConfigItemID | html %];VersionID=[% Data.VersionID | html %]</Item>
                <Item Key="LinkParam">target="print"</Item>
                <Item Key="Target">PopUp</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::MenuModule###500-Link" Required="0" Valid="1">
        <Description Translatable="1">Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::MenuModule</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::ITSMConfigItem::MenuGeneric</Item>
                <Item Key="Name" Translatable="1">Link</Item>
                <Item Key="Description" Translatable="1">Link</Item>
                <Item Key="Action">AgentLinkObject</Item>
                <Item Key="Link">Action=AgentLinkObject;SourceObject=ITSMConfigItem;SourceKey=[% Data.ConfigItemID | html %]</Item>
                <Item Key="Target">PopUp</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::MenuModule###600-Duplicate" Required="0" Valid="1">
        <Description Translatable="1">Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::MenuModule</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::ITSMConfigItem::MenuGeneric</Item>
                <Item Key="Name" Translatable="1">Duplicate</Item>
                <Item Key="Description" Translatable="1">Duplicate</Item>
                <Item Key="Action">AgentITSMConfigItemEdit</Item>
                <Item Key="Link">Action=AgentITSMConfigItemEdit;DuplicateID=[% Data.ConfigItemID | html %];VersionID=[% Data.VersionID | html %]</Item>
                <Item Key="Target">PopUp</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::MenuModule###700-ConfigItemDelete" Required="1" Valid="1">
        <Description Translatable="1">Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::MenuModule</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::ITSMConfigItem::MenuGeneric</Item>
                <Item Key="Name" Translatable="1">Delete</Item>
                <Item Key="Description" Translatable="1">Delete Configuration Item</Item>
                <Item Key="Action">AgentITSMConfigItemDelete</Item>
                <Item Key="Link">Action=AgentITSMConfigItemDelete;ConfigItemID=[% Data.ConfigItemID | html %]</Item>
                <Item Key="Target">ConfirmDialog</Item>
                <Item Key="ElementSelector">#Menu[% Data.MenuID | html %]</Item>
                <Item Key="DialogContentQueryString">Action=AgentITSMConfigItemDelete;ConfigItemID=[% Data.ConfigItemID | html %]</Item>
                <Item Key="ConfirmedActionQueryString">Action=AgentITSMConfigItemDelete;Subaction=ConfigItemDelete;ConfigItemID=[% Data.ConfigItemID | html %]</Item>
                <Item Key="DialogTitle">[% Translate("Delete") | html %] [% Config("ITSMConfigItem::Hook") %] [% Data.Number | html %]</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="LinkObject::ComplexTable::SettingsVisibility###ITSMConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.</Description>
        <Navigation>Frontend::Agent::LinkObject</Navigation>
        <Value>
            <Array>
                <Item>AgentITSMConfigItemZoom</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::NumberGenerator" Required="1" Valid="1">
        <Description Translatable="1">Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.</Description>
        <Navigation>Core::ITSMConfigItem</Navigation>
        <Value>
            <Item ValueType="PerlModule" ValueFilter="Kernel/System/ITSMConfigItem/Number/*.pm">Kernel::System::ITSMConfigItem::Number::AutoIncrement</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::CINameRegex" Required="0" Valid="0">
        <Description Translatable="1">Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.</Description>
        <Navigation>Core::ITSMConfigItem</Navigation>
        <Value>
            <Hash>
                <Item Key="Computer::CINameRegex">^ABC.*</Item>
                <Item Key="Computer::CINameRegexErrorMessage">A computer name must start with ABC!</Item>
                <Item Key="Hardware::CINameRegex">.*\d\d$</Item>
                <Item Key="Hardware::CINameRegexErrorMessage">A hardware name must end with two digits!</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="LinkObject::DefaultSubObject###ITSMConfigItem" Required="1" Valid="1">
        <Description Translatable="1">Defines the default subobject of the class 'ITSMConfigItem'.</Description>
        <Navigation>Core::LinkObject</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">Computer</Item>
        </Value>
    </Setting>
    <Setting Name="LinkObject::ITSMConfigItem::ShowColumns" Required="0" Valid="1">
        <Description Translatable="1">Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.</Description>
        <Navigation>Core::LinkObject</Navigation>
        <Value>
            <Array>
                <Item>CurInciSignal</Item>
                <Item>CurDeplSignal</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="LinkObject::ITSMConfigItem::ShowColumnsByClass" Required="0" Valid="0">
        <Description Translatable="1">Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.</Description>
        <Navigation>Core::LinkObject</Navigation>
        <Value>
            <Array>
                <Item>Computer::Name</Item>
                <Item>Computer::CurDeplState</Item>
                <Item>Computer::HardDisk::1</Item>
                <Item>Computer::HardDisk::1::Capacity::1</Item>
                <Item>Computer::HardDisk::2</Item>
                <Item>Computer::HardDisk::2::Capacity::1</Item>
                <Item>Computer::HardDisk::3</Item>
                <Item>Computer::HardDisk::3::Capacity::1</Item>
                <Item>Computer::WarrantyExpirationDate::1</Item>
                <Item>Computer::CreateTime</Item>
                <Item>Location::Type::1</Item>
                <Item>Location::Address::1</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="Stats::DynamicObjectRegistration###ITSMConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Module to generate ITSM config item statistics.</Description>
        <Navigation>Core::Stats</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::Stats::Dynamic::ITSMConfigItem</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ImportExport::ObjectBackendRegistration###ITSMConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Object backend module registration for the import/export module.</Description>
        <Navigation>Core::ImportExport::ObjectBackend::ModuleRegistration</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::ImportExport::ObjectBackend::ITSMConfigItem</Item>
                <Item Key="Name">Config Item</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::EventModulePost###100-History" Required="0" Valid="1">
        <Description Translatable="1">Config item event module that enables logging to history in the agent interface.</Description>
        <Navigation>Core::Event::ITSMConfigItem</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::ITSMConfigItem::Event::DoHistory</Item>
                <Item Key="Event">(ConfigItemCreate|VersionCreate|DeploymentStateUpdate|IncidentStateUpdate|ConfigItemDelete|LinkAdd|LinkDelete|DefinitionUpdate|NameUpdate|ValueUpdate|DefinitionCreate|VersionDelete|AttachmentAddPost|AttachmentDeletePost)</Item>
                <Item Key="Transaction">0</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::HistoryOrder" Required="1" Valid="1">
        <Description Translatable="1">Shows the config item history (reverse ordered) in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::ITSMConfigItemHistory</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="normal">
                <Item ValueType="Option" Value="reverse" Translatable="1">reverse</Item>
                <Item ValueType="Option" Value="normal" Translatable="1">normal</Item>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AgentITSMConfigItemHistory" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="Group">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="GroupRo">
                        <Array>
                            <Item>itsm-configitem</Item>
                        </Array>
                    </Item>
                    <Item Key="Description" Translatable="1">Config item history.</Item>
                    <Item Key="Title" Translatable="1">History</Item>
                    <Item Key="NavBarName">Config Item</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Loader::Module::AgentITSMConfigItemHistory###003-ITSMConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Loader module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration::Loader</Navigation>
        <Value>
            <Hash>
                <Item Key="JavaScript">
                    <Array>
                        <Item>ITSM.Agent.ConfigItem.History.js</Item>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="GeneralCatalogPreferences###DeploymentStates" Required="0" Valid="1">
        <Description Translatable="1">Parameters for the deployment states in the preferences view of the agent interface.</Description>
        <Navigation>Core::GeneralCatalog</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::GeneralCatalogPreferences::Generic</Item>
                <Item Key="Class">ITSM::ConfigItem::DeploymentState</Item>
                <Item Key="Label" Translatable="1">Deployment State Type</Item>
                <Item Key="Desc" Translatable="1"></Item>
                <Item Key="Data">
                    <Hash>
                        <Item Key="postproductive" Translatable="1">postproductive</Item>
                        <Item Key="preproductive" Translatable="1">preproductive</Item>
                        <Item Key="productive" Translatable="1">productive</Item>
                    </Hash>
                </Item>
                <Item Key="PrefKey">Functionality</Item>
                <Item Key="Block">Option</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="GeneralCatalogPreferences###DeploymentStatesColors" Required="0" Valid="1">
        <Description Translatable="1">Parameters for the deployment states color in the preferences view of the agent interface.</Description>
        <Navigation>Core::GeneralCatalog</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::GeneralCatalogPreferences::Generic</Item>
                <Item Key="Class">ITSM::ConfigItem::DeploymentState</Item>
                <Item Key="Label" Translatable="1">Deployment State Color</Item>
                <Item Key="Desc" Translatable="1"></Item>
                <Item Key="Data"></Item>
                <Item Key="PrefKey">Color</Item>
                <Item Key="Block">ColorPicker</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Permission::Class###010-ClassGroupCheck" Required="0" Valid="1">
        <Description Translatable="1">Module to check the group responsible for a class.</Description>
        <Navigation>Core::ITSMConfigItem</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::ITSMConfigItem::Permission::ClassGroupCheck</Item>
                <Item Key="Required">0</Item>
                <Item Key="Granted">1</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Permission::Item###010-ItemClassGroupCheck" Required="0" Valid="1">
        <Description Translatable="1">Module to check the group responsible for a configuration item.</Description>
        <Navigation>Core::ITSMConfigItem</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::ITSMConfigItem::Permission::ItemClassGroupCheck</Item>
                <Item Key="Required">0</Item>
                <Item Key="Granted">1</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItem###Permission" Required="1" Valid="1">
        <Description Translatable="1">Required permissions to use the ITSM configuration item screen in the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::Permission</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">ro</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemEdit###Permission" Required="1" Valid="1">
        <Description Translatable="1">Required permissions to use the edit ITSM configuration item screen in the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::Permission</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">rw</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemAdd###Permission" Required="1" Valid="1">
        <Description Translatable="1">Required permissions to use the add ITSM configuration item screen in the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::Permission</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">rw</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemHistory###Permission" Required="1" Valid="1">
        <Description Translatable="1">Required permissions to use the history ITSM configuration item screen in the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::Permission</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">ro</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemPrint###Permission" Required="1" Valid="1">
        <Description Translatable="1">Required permissions to use the print ITSM configuration item screen in the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::Permission</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">ro</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemDelete###Permission" Required="1" Valid="1">
        <Description Translatable="1">Required privileges to delete config items.</Description>
        <Navigation>Frontend::Agent::View::ITSMConfigItemDelete</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">rw</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemZoom###Permission" Required="1" Valid="1">
        <Description Translatable="1">Required permissions to use the ITSM configuration item zoom screen in the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::Permission</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">ro</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemSearch###Permission" Required="1" Valid="1">
        <Description Translatable="1">Required permissions to use the ITSM configuration item search screen in the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::Permission</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">ro</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AdminITSMConfigItem###EditorRows" Required="1" Valid="1">
        <Description Translatable="1">Defines the number of rows for the CI definition editor in the admin interface.</Description>
        <Navigation>Frontend::Admin::View::ITSMConfigItemDefinition</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="^\d+$">30</Item>
        </Value>
    </Setting>
    <Setting Name="GeneralCatalogPreferences###Permissions" Required="0" Valid="1">
        <Description Translatable="1">Parameters for the example permission groups of the general catalog attributes.</Description>
        <Navigation>Core::GeneralCatalog</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::GeneralCatalogPreferences::Generic</Item>
                <Item Key="Label" Translatable="1">Permission Group</Item>
                <Item Key="Block">Permission</Item>
                <Item Key="Class">ITSM::ConfigItem::Class</Item>
                <Item Key="PrefKey">Permission</Item>
                <Item Key="Mandatory">1</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Frontend::Search###ConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Configuration item search backend router of the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration::MainMenu::Search</Navigation>
        <Value>
            <Hash>
                <Item Key="^(?:Agent|Admin)ITSMConfigItem">Action=AgentITSMConfigItemSearch;Subaction=AJAX</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Frontend::Search::JavaScript###ConfigItem" Required="0" Valid="1">
        <Description Translatable="1">JavaScript function for the search frontend.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration::MainMenu::Search</Navigation>
        <Value>
            <Hash>
                <Item Key="^(?:Agent|Admin)ITSMConfigItem">ITSM.Agent.ConfigItem.Search.OpenSearchDialog('AgentITSMConfigItemSearch')</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Loader::Agent::CommonJS###100-ConfigurationManagement" Required="1" Valid="1">
        <Description Translatable="1">List of JS files to always be loaded for the agent interface.</Description>
        <Navigation>Frontend::Base::Loader</Navigation>
        <Value>
            <Array>
                <Item>ITSM.Agent.ConfigItem.Search.js</Item>
                <Item>ITSM.UI.ConfigItemActionRow.js</Item>
                <Item>ITSM.Agent.ConfigItem.Dashboard.js</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="PreferencesGroups###ConfigItemOverviewSmallPageShown" Required="0" Valid="1">
        <Description Translatable="1">Parameters for the pages (in which the configuration items are shown).</Description>
        <Navigation>Frontend::Agent::View::Preferences</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::Preferences::Generic</Item>
                <Item Key="PreferenceGroup">Miscellaneous</Item>
                <Item Key="Label" Translatable="1">Configuration Item Limit</Item>
                <Item Key="Desc" Translatable="1">Configuration Item limit per page.</Item>
                <Item Key="Key" Translatable="1"></Item>
                <Item Key="Data">
                    <Hash>
                        <Item Key="10">10</Item>
                        <Item Key="15">15</Item>
                        <Item Key="20">20</Item>
                        <Item Key="25">25</Item>
                        <Item Key="30">30</Item>
                        <Item Key="35">35</Item>
                    </Hash>
                </Item>
                <Item Key="DataSelected">25</Item>
                <Item Key="PrefKey">UserConfigItemOverviewSmallPageShown</Item>
                <Item Key="Prio">8000</Item>
                <Item Key="Active">0</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::Overview###Small" Required="0" Valid="1">
        <Description Translatable="1">Defines an overview module to show the small view of a configuration item list.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItemOverview</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::ITSMConfigItem::OverviewSmall</Item>
                <Item Key="Name">Small</Item>
                <Item Key="NameShort">S</Item>
                <Item Key="PageShown">25</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Hook" Required="1" Valid="1">
        <Description Translatable="1">The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.</Description>
        <Navigation>Core::ITSMConfigItem</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">ConfigItem#</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemSearch###SearchLimit" Required="1" Valid="1">
        <Description Translatable="1">Defines the search limit for the AgentITSMConfigItemSearch screen.</Description>
        <Navigation>Frontend::Agent::View::ITSMConfigItemSearch</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="^\d+$">10000</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemSearch###ShowColumns" Required="1" Valid="1">
        <Description Translatable="1">Defines the shown columns in the config item search. This option has no effect on the position of the column.</Description>
        <Navigation>Frontend::Agent::View::ITSMConfigItemSearch</Navigation>
        <Value>
            <Hash>
                <DefaultItem ValueType="Select">
                    <Item ValueType="Option" Value="0" Translatable="1">0 - Disabled</Item>
                    <Item ValueType="Option" Value="1" Translatable="1">1 - Enabled</Item>
                </DefaultItem>
                <Item Key="CurInciSignal" SelectedID="1"></Item>
                <Item Key="CurDeplSignal" SelectedID="1"></Item>
                <Item Key="Number" SelectedID="1"></Item>
                <Item Key="Name" SelectedID="1"></Item>
                <Item Key="Class" SelectedID="0"></Item>
                <Item Key="CurDeplState" SelectedID="1"></Item>
                <Item Key="CurDeplStateType" SelectedID="0"></Item>
                <Item Key="CurInciState" SelectedID="1">1</Item>
                <Item Key="CurInciStateType" SelectedID="0"></Item>
                <Item Key="LastChanged" SelectedID="1"></Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemSearch###ShowColumnsByClass" Required="0" Valid="0">
        <Description Translatable="1">Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.</Description>
        <Navigation>Frontend::Agent::View::ITSMConfigItemSearch</Navigation>
        <Value>
            <Array>
                <Item>Computer::Name</Item>
                <Item>Computer::Number</Item>
                <Item>Computer::CurInciSignal</Item>
                <Item>Computer::CurDeplSignal</Item>
                <Item>Computer::CurDeplState</Item>
                <Item>Computer::HardDisk::1</Item>
                <Item>Computer::HardDisk::1::Capacity::1</Item>
                <Item>Computer::HardDisk::2</Item>
                <Item>Computer::HardDisk::2::Capacity::1</Item>
                <Item>Computer::HardDisk::3</Item>
                <Item>Computer::HardDisk::3::Capacity::1</Item>
                <Item>Computer::WarrantyExpirationDate::1</Item>
                <Item>Computer::CreateTime</Item>
                <Item>Location::CurInciSignal</Item>
                <Item>Location::Number</Item>
                <Item>Location::Type::1</Item>
                <Item>Location::Name</Item>
                <Item>Location::Address::1</Item>
                <Item>Location::LastChanged</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemSearch###SearchCSVData" Required="1" Valid="1">
        <Description Translatable="1">Data used to export the search result in CSV format.</Description>
        <Navigation>Frontend::Agent::View::ITSMConfigItemSearch</Navigation>
        <Value>
            <Array>
                <Item Translatable="1">Class</Item>
                <Item Translatable="1">Incident State</Item>
                <Item Translatable="1">Name</Item>
                <Item Translatable="1">ConfigItemNumber</Item>
                <Item Translatable="1">Deployment State</Item>
                <Item Translatable="1">Version</Item>
                <Item Translatable="1">Create Time</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItem###SearchLimit" Required="1" Valid="1">
        <Description Translatable="1">Defines the search limit for the AgentITSMConfigItem screen.</Description>
        <Navigation>Frontend::Agent::View::ITSMConfigItem</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="^\d+$">10000</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns" Required="1" Valid="1">
        <Description Translatable="1">Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter 'All' is selected.</Description>
        <Navigation>Frontend::Agent::View::ITSMConfigItem</Navigation>
        <Value>
            <Hash>
                <DefaultItem ValueType="Select">
                    <Item ValueType="Option" Value="0" Translatable="1">0 - Disabled</Item>
                    <Item ValueType="Option" Value="1" Translatable="1">1 - Enabled</Item>
                </DefaultItem>
                <Item Key="CurInciSignal" SelectedID="1"></Item>
                <Item Key="CurDeplSignal" SelectedID="1"></Item>
                <Item Key="Number" SelectedID="1"></Item>
                <Item Key="Name" SelectedID="1"></Item>
                <Item Key="Class" SelectedID="0"></Item>
                <Item Key="CurDeplState" SelectedID="1"></Item>
                <Item Key="CurDeplStateType" SelectedID="0"></Item>
                <Item Key="CurInciState" SelectedID="1"></Item>
                <Item Key="CurInciStateType" SelectedID="1"></Item>
                <Item Key="LastChanged" SelectedID="1"></Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumnsByClass" Required="0" Valid="0">
        <Description Translatable="1">Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.</Description>
        <Navigation>Frontend::Agent::View::ITSMConfigItem</Navigation>
        <Value>
            <Array>
                <Item>Computer::Name</Item>
                <Item>Computer::Number</Item>
                <Item>Computer::CurInciSignal</Item>
                <Item>Computer::CurDeplSignal</Item>
                <Item>Computer::CurDeplState</Item>
                <Item>Computer::HardDisk::1</Item>
                <Item>Computer::HardDisk::1::Capacity::1</Item>
                <Item>Computer::HardDisk::2</Item>
                <Item>Computer::HardDisk::2::Capacity::1</Item>
                <Item>Computer::HardDisk::3</Item>
                <Item>Computer::HardDisk::3::Capacity::1</Item>
                <Item>Computer::WarrantyExpirationDate::1</Item>
                <Item>Computer::CreateTime</Item>
                <Item>Location::CurInciSignal</Item>
                <Item>Location::Number</Item>
                <Item>Location::Type::1</Item>
                <Item>Location::Name</Item>
                <Item>Location::Address::1</Item>
                <Item>Location::LastChanged</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="UniqueCIName::EnableUniquenessCheck" Required="0" Valid="1">
        <Description Translatable="1">Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.</Description>
        <Navigation>Core::ITSMConfigItem</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="0">
                <Item ValueType="Option" Value="1" Translatable="1">Enabled</Item>
                <Item ValueType="Option" Value="0" Translatable="1">Disabled</Item>
            </Item>
        </Value>
    </Setting>
    <Setting Name="UniqueCIName::UniquenessCheckScope" Required="0" Valid="1">
        <Description Translatable="1">Check for a unique name only within the same ConfigItem class ('class') or globally ('global'), which means every existing ConfigItem is taken into account when looking for duplicates.</Description>
        <Navigation>Core::ITSMConfigItem</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="global">
                <Item ValueType="Option" Value="global" Translatable="1">global</Item>
                <Item ValueType="Option" Value="class" Translatable="1">class</Item>
            </Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::BulkFeature" Required="1" Valid="1">
        <Description Translatable="1">Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.</Description>
        <Navigation>Core::BulkAction</Navigation>
        <Value>
            <Item ValueType="Checkbox">1</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::BulkFeatureGroup" Required="0" Valid="0">
        <Description Translatable="1">Enables configuration item bulk action feature only for the listed groups.</Description>
        <Navigation>Core::BulkAction</Navigation>
        <Value>
            <Array>
                <Item>admin</Item>
                <Item>users</Item>
                <Item>itsm-configitem</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AgentITSMConfigItemBulk" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="Group">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="GroupRo">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="Description" Translatable="1">Configuration item bulk module.</Item>
                    <Item Key="Title" Translatable="1">Bulk Action</Item>
                    <Item Key="NavBarName">CMDB</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Frontend::Navigation###AgentITSMConfigItemBulk###003-ITSMConfigItem" Required="0" Valid="0">
        <Description Translatable="1">Main menu item registration.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration::MainMenu</Navigation>
        <Value>
            <Array>
                <DefaultItem ValueType="FrontendNavigation">
                    <Hash>
                    </Hash>
                </DefaultItem>
            </Array>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::PreMenuModule###100-Zoom" Required="0" Valid="1">
        <Description Translatable="1">Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::MenuModulePre</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::ITSMConfigItem::MenuGeneric</Item>
                <Item Key="Name" Translatable="1">Zoom</Item>
                <Item Key="Description" Translatable="1">Zoom</Item>
                <Item Key="Action">AgentITSMConfigItemZoom</Item>
                <Item Key="Link">Action=AgentITSMConfigItemZoom;ConfigItemID=[% Data.ConfigItemID | html %]</Item>
                <Item Key="Target"></Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::PreMenuModule###200-History" Required="0" Valid="1">
        <Description Translatable="1">Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::MenuModulePre</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::ITSMConfigItem::MenuGeneric</Item>
                <Item Key="Name" Translatable="1">History</Item>
                <Item Key="Description" Translatable="1">History</Item>
                <Item Key="Action">AgentITSMConfigItemHistory</Item>
                <Item Key="Link">Action=AgentITSMConfigItemHistory;ConfigItemID=[% Data.ConfigItemID | html %];VersionID=[% Data.VersionID | html %]</Item>
                <Item Key="Target">PopUp</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::PreMenuModule###300-Duplicate" Required="0" Valid="1">
        <Description Translatable="1">Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.</Description>
        <Navigation>Frontend::Agent::ITSMConfigItem::MenuModulePre</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::ITSMConfigItem::MenuGeneric</Item>
                <Item Key="Name" Translatable="1">Duplicate</Item>
                <Item Key="Description" Translatable="1">Duplicate</Item>
                <Item Key="Action">AgentITSMConfigItemEdit</Item>
                <Item Key="Link">Action=AgentITSMConfigItemEdit;DuplicateID=[% Data.ConfigItemID | html %];VersionID=[% Data.VersionID | html %];ReturnToLastScreen=1</Item>
                <Item Key="Target">PopUp</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemBulk###DeplState" Required="0" Valid="1">
        <Description Translatable="1">Sets the deployment state in the configuration item bulk screen of the agent interface.</Description>
        <Navigation>Frontend::Agent::View::ITSMConfigItemBulk</Navigation>
        <Value>
            <Item ValueType="Checkbox">1</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemBulk###InciState" Required="0" Valid="1">
        <Description Translatable="1">Sets the incident state in the configuration item bulk screen of the agent interface.</Description>
        <Navigation>Frontend::Agent::View::ITSMConfigItemBulk</Navigation>
        <Value>
            <Item ValueType="Checkbox">1</Item>
        </Value>
    </Setting>
    <Setting Name="Frontend::AdminModuleGroups###200-ITSMConfigurationManagement" Required="1" Valid="1">
        <Description Translatable="1">Defines available groups for the admin overview screen.</Description>
        <Navigation>Frontend::Base::NavBarModule</Navigation>
        <Value>
            <Hash>
                <Item Key="CMDBSettings">
                    <Hash>
                        <Item Key="Title" Translatable="1">CMDB Settings</Item>
                        <Item Key="Order">8000</Item>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>

    <Setting Name="AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany" Required="0" Valid="1">
        <Description Translatable="1">Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.</Description>
        <Navigation>Frontend::Agent::View::CustomerInformationCenter</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::Dashboard::ITSMConfigItemGeneric</Item>
                <Item Key="Title" Translatable="1">Assigned CIs</Item>
                <Item Key="Description" Translatable="1">CIs assigned to customer company</Item>
                <Item Key="Attributes">CustomerCompany</Item>
                <Item Key="Limit">10</Item>
                <Item Key="Permission">ro</Item>
                <Item Key="Block">ContentLarge</Item>
                <Item Key="Group"></Item>
                <Item Key="Default">1</Item>
                <Item Key="CacheTTLLocal">0.5</Item>
                <Item Key="ConfigItemKey">
                    <Hash>
                        <Item Key="Computer">CustomerID</Item>
                        <Item Key="Hardware">CustomerID</Item>
                        <Item Key="Location">CustomerID</Item>
                        <Item Key="Network">CustomerID</Item>
                        <Item Key="Software">CustomerID</Item>
                    </Hash>
                </Item>
                <Item Key="DefaultColumns">
                    <Hash>
                        <DefaultItem ValueType="Select">
                            <Item ValueType="Option" Value="0" Translatable="1">0 - Disabled</Item>
                            <Item ValueType="Option" Value="1" Translatable="1">1 - Enabled</Item>
                        </DefaultItem>
                        <Item Key="CurInciSignal" SelectedID="1"></Item>
                        <Item Key="CurDeplSignal" SelectedID="1"></Item>
                        <Item Key="Number" SelectedID="1"></Item>
                        <Item Key="Name" SelectedID="1"></Item>
                        <Item Key="Class" SelectedID="0"></Item>
                        <Item Key="CurDeplState" SelectedID="1"></Item>
                        <Item Key="CurDeplStateType" SelectedID="0"></Item>
                        <Item Key="CurInciState" SelectedID="1"></Item>
                        <Item Key="CurInciStateType" SelectedID="0"></Item>
                        <Item Key="LastChanged" SelectedID="1"></Item>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Frontend::Agent::CustomerInformationCenter::ITSMConfigItem###ShowColumnsByClass" Required="0" Valid="0">
        <Description Translatable="1">Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).</Description>
        <Navigation>Frontend::Agent::View::CustomerInformationCenter</Navigation>
        <Value>
            <Array>
                <Item>Computer::Name</Item>
                <Item>Computer::Number</Item>
                <Item>Computer::CurInciSignal</Item>
                <Item>Computer::CurDeplSignal</Item>
                <Item>Computer::CurDeplState</Item>
                <Item>Computer::HardDisk::1</Item>
                <Item>Computer::HardDisk::1::Capacity::1</Item>
                <Item>Computer::HardDisk::2</Item>
                <Item>Computer::HardDisk::2::Capacity::1</Item>
                <Item>Computer::HardDisk::3</Item>
                <Item>Computer::HardDisk::3::Capacity::1</Item>
                <Item>Computer::WarrantyExpirationDate::1</Item>
                <Item>Computer::CreateTime</Item>
                <Item>Location::CurInciSignal</Item>
                <Item>Location::Number</Item>
                <Item>Location::Type::1</Item>
                <Item>Location::Name</Item>
                <Item>Location::Address::1</Item>
                <Item>Location::LastChanged</Item>
            </Array>
        </Value>
    </Setting>

    <Setting Name="AgentCustomerUserInformationCenter::Backend###0060-CUIC-ITSMConfigItemCustomerUser" Required="0" Valid="1">
        <Description Translatable="1">Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.</Description>
        <Navigation>Frontend::Agent::View::CustomerUserInformationCenter</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::Dashboard::ITSMConfigItemGeneric</Item>
                <Item Key="Title" Translatable="1">Assigned CIs</Item>
                <Item Key="Description" Translatable="1">CIs assigned to customer user</Item>
                <Item Key="Attributes">Customer</Item>
                <Item Key="Limit">10</Item>
                <Item Key="Permission">ro</Item>
                <Item Key="Block">ContentLarge</Item>
                <Item Key="Group"></Item>
                <Item Key="Default">1</Item>
                <Item Key="CacheTTLLocal">0.5</Item>
                <Item Key="ConfigItemKey">
                    <Hash>
                        <Item Key="Computer">Owner</Item>
                        <Item Key="Hardware">Owner</Item>
                        <Item Key="Location">Owner</Item>
                        <Item Key="Network">Owner</Item>
                        <Item Key="Software">Owner</Item>
                    </Hash>
                </Item>
                <Item Key="DefaultColumns">
                    <Hash>
                        <DefaultItem ValueType="Select">
                            <Item ValueType="Option" Value="0" Translatable="1">0 - Disabled</Item>
                            <Item ValueType="Option" Value="1" Translatable="1">1 - Enabled</Item>
                        </DefaultItem>
                        <Item Key="CurInciSignal" SelectedID="1"></Item>
                        <Item Key="CurDeplSignal" SelectedID="1"></Item>
                        <Item Key="Number" SelectedID="1"></Item>
                        <Item Key="Name" SelectedID="1"></Item>
                        <Item Key="Class" SelectedID="0"></Item>
                        <Item Key="CurDeplState" SelectedID="1"></Item>
                        <Item Key="CurDeplStateType" SelectedID="0"></Item>
                        <Item Key="CurInciState" SelectedID="1"></Item>
                        <Item Key="CurInciStateType" SelectedID="0"></Item>
                        <Item Key="LastChanged" SelectedID="1"></Item>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Frontend::Agent::CustomerUserInformationCenter::ITSMConfigItem###ShowColumnsByClass" Required="0" Valid="0">
        <Description Translatable="1">Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).</Description>
        <Navigation>Frontend::Agent::View::CustomerUserInformationCenter</Navigation>
        <Value>
            <Array>
                <Item>Computer::Name</Item>
                <Item>Computer::Number</Item>
                <Item>Computer::CurInciSignal</Item>
                <Item>Computer::CurDeplSignal</Item>
                <Item>Computer::CurDeplState</Item>
                <Item>Computer::HardDisk::1</Item>
                <Item>Computer::HardDisk::1::Capacity::1</Item>
                <Item>Computer::HardDisk::2</Item>
                <Item>Computer::HardDisk::2::Capacity::1</Item>
                <Item>Computer::HardDisk::3</Item>
                <Item>Computer::HardDisk::3::Capacity::1</Item>
                <Item>Computer::WarrantyExpirationDate::1</Item>
                <Item>Computer::CreateTime</Item>
                <Item>Location::CurInciSignal</Item>
                <Item>Location::Number</Item>
                <Item>Location::Type::1</Item>
                <Item>Location::Name</Item>
                <Item>Location::Address::1</Item>
                <Item>Location::LastChanged</Item>
            </Array>
        </Value>
    </Setting>

    <!-- ITSMConfigItemTicketLink -->
    <Setting Name="ITSMConfigItem::SetIncidentStateOnLink" Required="1" Valid="1">
        <Description Translatable="1">Set the incident state of a CI automatically when a Ticket is Linked to a CI.</Description>
        <Navigation>Core::LinkStatus</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::EventModulePost###042-ITSMConfigItemTicketStatusLink" Required="1" Valid="1">
        <Description Translatable="1">Event module to set configitem-status on ticket-configitem-link.</Description>
        <Navigation>Core::Event::Ticket</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::Ticket::Event::TicketStatusLink</Item>
                <Item Key="Event">(LinkAdd|LinkDelete)</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Ticket::EventModulePost###042-ITSMConfigItemTicketStatusLink" Required="1" Valid="1">
        <Description Translatable="1">Event module to set configitem-status on ticket-configitem-link.</Description>
        <Navigation>Core::Event::Ticket</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::Ticket::Event::TicketStatusLink</Item>
                <Item Key="Event">(TicketStateUpdate|TicketTypeUpdate)</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::LinkStatus::TicketTypes" Required="0" Valid="0">
        <Description Translatable="1">Defines which type of ticket can affect the status of a linked CI.</Description>
        <Navigation>Core::LinkStatus</Navigation>
        <Value>
            <Array>
                <Item>Incident</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::LinkStatus::DeploymentStates" Required="0" Valid="0">
        <Description Translatable="1">Defines the relevant deployment states where linked tickets can affect the status of a CI.</Description>
        <Navigation>Core::LinkStatus</Navigation>
        <Value>
            <Array>
                <Item>Production</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::LinkStatus::IncidentStates" Required="1" Valid="1">
        <Description Translatable="1">Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).</Description>
        <Navigation>Core::LinkStatus</Navigation>
        <Value>
            <Array>
                <Item>Incident</Item>
                <Item>Warning</Item>
                <Item>Operational</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::LinkStatus::LinkTypes" Required="1" Valid="1">
        <Description Translatable="1">Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.</Description>
        <Navigation>Core::LinkStatus</Navigation>
        <Value>
            <Hash>
                <Item Key="RelevantTo">Incident</Item>
            </Hash>
        </Value>
    </Setting>

    <!-- ITSMConfigItemGenericInterface -->
    <Setting Name="GenericInterface::Operation::Module###ConfigItem::ConfigItemCreate" Required="0" Valid="1">
        <Description Translatable="1">GenericInterface module registration for the operation layer.</Description>
        <Navigation>GenericInterface::Operation::ModuleRegistration</Navigation>
        <Value>
            <Hash>
                <Item Key="Name">ConfigItemCreate</Item>
                <Item Key="Controller">ConfigItem</Item>
                <Item Key="ConfigDialog">AdminGenericInterfaceOperationDefault</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="GenericInterface::Operation::Module###ConfigItem::ConfigItemGet" Required="0" Valid="1">
        <Description Translatable="1">GenericInterface module registration for the operation layer.</Description>
        <Navigation>GenericInterface::Operation::ModuleRegistration</Navigation>
        <Value>
            <Hash>
                <Item Key="Name">ConfigItemGet</Item>
                <Item Key="Controller">ConfigItem</Item>
                <Item Key="ConfigDialog">AdminGenericInterfaceOperationDefault</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="GenericInterface::Operation::Module###ConfigItem::ConfigItemUpdate" Required="0" Valid="1">
        <Description Translatable="1">GenericInterface module registration for the operation layer.</Description>
        <Navigation>GenericInterface::Operation::ModuleRegistration</Navigation>
        <Value>
            <Hash>
                <Item Key="Name">ConfigItemUpdate</Item>
                <Item Key="Controller">ConfigItem</Item>
                <Item Key="ConfigDialog">AdminGenericInterfaceOperationDefault</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="GenericInterface::Operation::Module###ConfigItem::ConfigItemSearch" Required="0" Valid="1">
        <Description Translatable="1">GenericInterface module registration for the operation layer.</Description>
        <Navigation>GenericInterface::Operation::ModuleRegistration</Navigation>
        <Value>
            <Hash>
                <Item Key="Name">ConfigItemSearch</Item>
                <Item Key="Controller">ConfigItem</Item>
                <Item Key="ConfigDialog">AdminGenericInterfaceOperationDefault</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="GenericInterface::Operation::Module###ConfigItem::ConfigItemDelete" Required="0" Valid="1">
        <Description Translatable="1">GenericInterface module registration for the operation layer.</Description>
        <Navigation>GenericInterface::Operation::ModuleRegistration</Navigation>
        <Value>
            <Hash>
                <Item Key="Name">ConfigItemDelete</Item>
                <Item Key="Controller">ConfigItem</Item>
                <Item Key="ConfigDialog">AdminGenericInterfaceOperationDefault</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="GenericInterface::Operation::ConfigItemGet###Permission" Required="1" Valid="1">
        <Description Translatable="1">Defines Required permissions to get ITSM configuration items using the Generic Interface.</Description>
        <Navigation>GenericInterface::Operation::ConfigItemGet</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">ro</Item>
        </Value>
    </Setting>
    <Setting Name="GenericInterface::Operation::ConfigItemSearch###Permission" Required="1" Valid="1">
        <Description Translatable="1">Defines Required permissions to search ITSM configuration items using the Generic Interface.</Description>
        <Navigation>GenericInterface::Operation::ConfigItemSearch</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">ro</Item>
        </Value>
    </Setting>
    <Setting Name="GenericInterface::Operation::ConfigItemCreate###Permission" Required="1" Valid="1">
        <Description Translatable="1">Defines Required permissions to create ITSM configuration items using the Generic Interface.</Description>
        <Navigation>GenericInterface::Operation::ConfigItemCreate</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">rw</Item>
        </Value>
    </Setting>
    <Setting Name="GenericInterface::Operation::ConfigItemUpdate###Permission" Required="1" Valid="1">
        <Description Translatable="1">Defines Required permissions to update ITSM configuration items using the Generic Interface.</Description>
        <Navigation>GenericInterface::Operation::ConfigItemUpdate</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">rw</Item>
        </Value>
    </Setting>
    <Setting Name="GenericInterface::Operation::ConfigItemDelete###Permission" Required="1" Valid="1">
        <Description Translatable="1">Defines Required permissions to delete ITSM configuration items using the Generic Interface.</Description>
        <Navigation>GenericInterface::Operation::ConfigItemDelete</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">rw</Item>
        </Value>
    </Setting>

    <!-- Znuny(4OTRS)-ITSMConfigItemInvoker -->
    <Setting Name="Events###ITSMConfigItem" Required="1" Valid="1">
        <Description Translatable="1">List of events that are available for generic interface event object type ITSMConfigItem.</Description>
        <Navigation>Frontend::Admin</Navigation>
        <Value>
            <Array>
                <Item>ConfigItemCreate</Item>
                <Item>ConfigItemDelete</Item>
                <Item>DeploymentStateUpdate</Item>
                <Item>IncidentStateUpdate</Item>
                <Item>NameUpdate</Item>
                <Item>ValueUpdate</Item>
                <Item>VersionCreate</Item>

                <!--
                    Other possible events:
                <Item>LinkAdd</Item>
                <Item>LinkDelete</Item>
                <Item>DefinitionUpdate</Item>
                <Item>DefinitionCreate</Item>
                <Item>VersionDelete</Item>
                <Item>AttachmentAddPost</Item>
                <Item>AttachmentDeletePost</Item>
                -->
            </Array>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::EventModulePost###9900-GenericInterface" Required="0" Valid="1">
        <Description Translatable="1">Performs the configured action for each event (as an Invoker) for each configured web service.</Description>
        <Navigation>Core::Event::ITSMConfigItem</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::GenericInterface::Event::Handler</Item>
                <Item Key="Transaction">1</Item>
                <Item Key="Event"></Item><!-- run on all events -->
            </Hash>
        </Value>
    </Setting>
    <Setting Name="GenericInterface::Invoker::Module###ITSMConfigItem::Generic" Required="1" Valid="1">
        <Description Translatable="1">GenericInterface module registration for the invoker layer.</Description>
        <Navigation>GenericInterface::Invoker::ModuleRegistration</Navigation>
        <Value>
            <Hash>
                <Item Key="Name">Generic</Item>
                <Item Key="Controller">ITSMConfigItem</Item>
                <Item Key="ConfigDialog">AdminGenericInterfaceInvokerDefault</Item>
            </Hash>
        </Value>
    </Setting>

    <!-- Znuny(4OTRS)-ITSMConfigItemTypes -->
    <Setting Name="Frontend::Module###AgentITSMConfigItemAttachment" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="Group">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="GroupRo">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="Description">Returns CIAttachments</Item>
                    <Item Key="Title">CIAttachment</Item>
                    <Item Key="NavBarName"></Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::EventModulePost###100-CIAttachment" Required="0" Valid="1">
        <Description Translatable="1">Registers a ITSMConfigItem-EventModule that will save related attachments.</Description>
        <Navigation>Core::Event::Ticket</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::ITSMConfigItem::Event::CIAttachment</Item>
                <Item Key="Transaction">0</Item>
                <Item Key="Event">(VersionCreate|VersionDelete)</Item>
            </Hash>
        </Value>
    </Setting>

    <!-- Znuny(4OTRS)-CustomerCIs -->
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemCustomerCIs###SearchLimit" Required="1" Valid="1">
        <Description Translatable="1">Limit for config item search.</Description>
        <Navigation>Frontend::Agent::View::ConfigItem</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="^\d+$">100</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemCustomerCIs###CustomerUser" Required="1" Valid="1">
        <Description Translatable="1">Enable search for config items based on the customer user id.</Description>
        <Navigation>Frontend::Agent::View::ConfigItem</Navigation>
        <Value>
            <Item ValueType="Checkbox">1</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemCustomerCIs###CustomerCompany" Required="1" Valid="1">
        <Description Translatable="1">Enable search for config items based on the customer id.</Description>
        <Navigation>Frontend::Agent::View::ConfigItem</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemCustomerCIs###Mapping" Required="0" Valid="0">
        <Description Translatable="1">Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::ConfigItem</Navigation>
        <Value>
            <Hash>
                <Item Key="Computer">fa-laptop</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="ITSMConfigItem::Frontend::AgentITSMConfigItemCustomerCIs###Permission" Required="1" Valid="1">
        <Description Translatable="1">Required permissions to use the config item dialog in the agent interface.</Description>
        <Navigation>Frontend::Agent::View::ConfigItem</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">ro</Item>
        </Value>
    </Setting>
    <Setting Name="Frontend::Output::FilterElementPost###AgentITSMConfigItemCustomerCIs" Required="0" Valid="1">
        <Description Translatable="1">Registers an output filter that shows a config item widget.</Description>
        <Navigation>Frontend::Base::OutputFilter</Navigation>
            <Value>
                <Hash>
                    <Item Key="Module">Kernel::Output::HTML::FilterElementPost::AgentITSMConfigItemCustomerCIs</Item>
                    <Item Key="Templates">
                        <Hash>
                            <Item Key="AgentTicketPhone">1</Item>
                            <Item Key="AgentTicketEmail">1</Item>
                            <Item Key="AgentTicketZoom">1</Item>
                        </Hash>
                    </Item>
                </Hash>
            </Value>
    </Setting>
    <Setting Name="AgentITSMConfigItemCustomerCIsWidget###Group" Required="1" Valid="1">
        <Description Translatable="1">Required group permissions to use the customer config item widget in the agent interface.</Description>
        <Navigation>Core::Znuny::CustomerCIs</Navigation>
        <Value>
            <Array>
                <Item>users</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="AgentITSMConfigItemCustomerCIsWidget###LinkType" Required="1" Valid="1">
        <Description Translatable="1">Defines which type of link (named from the ticket perspective) is used to link tickets and config items.</Description>
        <Navigation>Core::Znuny::CustomerCIs</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">RelevantTo</Item>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AgentITSMConfigItemCustomerCIsWidget" Required="0" Valid="1">
        <Description Translatable="1">This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="Group">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="GroupRo">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="NavBarName">AgentITSMConfigItemCustomerCIsWidget</Item>
                    <Item Key="Title">CustomerCIsWidget AJAX interface</Item>
                    <Item Key="Description">AJAX interface for CustomerCIsWidget.</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Loader::Agent::CommonJS###999-AgentITSMConfigItemCustomerCIs" Required="0" Valid="1">
        <Description Translatable="1">List of JS files to always be loaded for the agent interface.</Description>
        <Navigation>Frontend::Base::Loader</Navigation>
        <Value>
            <Array>
                <Item>ITSM.Agent.ConfigItem.CustomerCIsWidget.js</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="Ticket::EventModulePost###999-ITSMConfigItemCustomerCIs" Required="0" Valid="1">
        <Description Translatable="1">Event Module that links a ConfigItem.</Description>
        <Navigation>Core::Event::Ticket</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::Ticket::Event::ITSMConfigItemCustomerCIs</Item>
                <Item Key="Transaction">1</Item>
                <Item Key="Event">TicketCreate</Item>
            </Hash>
        </Value>
    </Setting>

    <!-- Znuny(4OTRS)-DynamicFieldConfigItem -->
    <Setting Name="Frontend::Module###AdminDynamicFieldConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Navigation>Frontend::Admin::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="GroupRo">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="Group">
                        <Array>
                            <Item>admin</Item>
                        </Array>
                    </Item>
                    <Item Key="Description">Admin</Item>
                    <Item Key="Title" Translatable="1">Dynamic fields administration</Item>
                    <Item Key="NavBarName">Admin</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AJAXDynamicFieldConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the agent interface.</Description>
        <Navigation>Frontend::Agent::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="Group">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="GroupRo">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="Description" Translatable="1"></Item>
                    <Item Key="NavBarName">Ticket</Item>
                    <Item Key="Title" Translatable="1">DynamicFieldConfigItem</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="CustomerFrontend::Module###AJAXDynamicFieldConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Frontend module registration for the customer interface.</Description>
        <Navigation>Frontend::Customer::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="Group">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="GroupRo">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="Description" Translatable="1"></Item>
                    <Item Key="NavBarName">Ticket</Item>
                    <Item Key="Title" Translatable="1">DynamicFieldConfigItem</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="DynamicFields::Driver###ConfigItemDropdown" Required="0" Valid="1">
        <Description Translatable="1">Dynamic field backend registration.</Description>
        <Navigation>Core::DynamicFields::DriverRegistration</Navigation>
        <Value>
            <Hash>
                <Item Key="DisplayName" Translatable="1">Config item (dropdown)</Item>
                <Item Key="Module">Kernel::System::DynamicField::Driver::ConfigItemDropdown</Item>
                <Item Key="ConfigDialog">AdminDynamicFieldConfigItem</Item>
                <Item Key="Config">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="DynamicFields::Driver###ConfigItemMultiselect" Required="0" Valid="1">
        <Description Translatable="1">Dynamic field backend registration.</Description>
        <Navigation>Core::DynamicFields::DriverRegistration</Navigation>
        <Value>
            <Hash>
                <Item Key="DisplayName" Translatable="1">Config item (multiselect)</Item>
                <Item Key="Module">Kernel::System::DynamicField::Driver::ConfigItemMultiselect</Item>
                <Item Key="ConfigDialog">AdminDynamicFieldConfigItem</Item>
                <Item Key="Config">
                    <Hash>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Ticket::EventModulePost###999-DynamicFieldConfigItemTicketLink" Required="0" Valid="1">
        <Description Translatable="1">Ticket event module that creates and removes links between tickets and config items.</Description>
        <Navigation>Core::Event::Znuny::DynamicFieldConfigItem</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::Ticket::Event::DynamicFieldConfigItemTicketLink</Item>
                <Item Key="Transaction">0</Item>
                <Item Key="Event">TicketDynamicFieldUpdate</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Ticket::EventModulePost###999-DynamicFieldConfigItemAdditionalDFStorage" Required="0" Valid="1">
        <Description Translatable="1">Ticket event module that updates DynamicFields.</Description>
        <Navigation>Core::Event::Znuny::DynamicFieldConfigItem</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::Ticket::Event::DynamicFieldConfigItemAdditionalDFStorage</Item>
                <Item Key="Transaction">0</Item>
                <Item Key="Event">TicketDynamicFieldUpdate</Item>
            </Hash>
        </Value>
    </Setting>
    <!-- Loaders -->
    <Setting Name="Loader::Agent::CommonJS###999-DynamicFieldConfigItem" Required="0" Valid="1">
        <Description Translatable="1">List of JS files to always be loaded for the agent interface.</Description>
        <Navigation>Frontend::Base::Loader</Navigation>
        <Value>
            <Array>
                <Item>Znuny.DynamicField.ConfigItem.js</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="Loader::Customer::CommonJS###999-DynamicFieldConfigItem" Required="0" Valid="1">
        <Description Translatable="1">List of JS files to always be loaded for the customer interface.</Description>
        <Navigation>Frontend::Base::Loader</Navigation>
        <Value>
            <Array>
                <Item>Znuny.DynamicField.ConfigItem.js</Item>
            </Array>
        </Value>
    </Setting>
    <Setting Name="Loader::Module::AdminDynamicFieldConfigItem###003-DynamicFieldConfigItem" Required="0" Valid="1">
        <Description Translatable="1">Loader module registration for the agent interface.</Description>
        <Navigation>Frontend::Admin::ModuleRegistration::Loader</Navigation>
        <Value>
            <Hash>
                <Item Key="CSS">
                    <Array>
                    </Array>
                </Item>
                <Item Key="JavaScript">
                    <Array>
                        <Item>Core.Agent.Admin.DynamicFieldConfigItem.js</Item>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <!-- These settings are only here for backwards compatibility -->
    <Setting Name="ZnunyDynamicFieldConfigItem###TicketLink" Required="1" Valid="1" ReadOnly="1">
        <Description Translatable="1">No longer in use. Do not change. Required due to compatibility with previous package versions.</Description>
        <Navigation>Core::Znuny::DynamicFieldConfigItem</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="1">
                <Item ValueType="Option" Value="0" Translatable="1">No</Item>
                <Item ValueType="Option" Value="1" Translatable="1">Yes</Item>
            </Item>
        </Value>
    </Setting>
    <Setting Name="ZnunyDynamicFieldConfigItem###LinkType" Required="1" Valid="1" ReadOnly="1">
        <Description Translatable="1">No longer in use. Do not change. Required due to compatibility with previous package versions.</Description>
        <Navigation>Core::Znuny::DynamicFieldConfigItem</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="RelevantTo">
                <Item ValueType="Option" Value="AlternativeTo" Translatable="1">AlternativeTo</Item>
                <Item ValueType="Option" Value="DependsOn" Translatable="1">DependsOn</Item>
                <Item ValueType="Option" Value="RelevantTo" Translatable="1">RelevantTo</Item>
            </Item>
        </Value>
    </Setting>
</otrs_config>
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpHZW5lcmljSW50ZXJmYWNlOjpFdmVudDo6T2JqZWN0VHlwZTo6SVRTTUNvbmZpZ0l0ZW07Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgS2VybmVsOjpTeXN0ZW06OlZhcmlhYmxlQ2hlY2sgcXcoOmFsbCk7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6TG9nJywKICAgICdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW1JbnZva2VyJywKKTsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6OkdlbmVyaWNJbnRlcmZhY2U6OkV2ZW50OjpPYmplY3RUeXBlOjpJVFNNQ29uZmlnSXRlbQoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9CgpzdWIgRGF0YUdldCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRMb2dPYmplY3QgICAgICAgICAgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyk7CiAgICBteSAkSVRTTUNvbmZpZ0l0ZW1JbnZva2VyT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtSW52b2tlcicpOwoKICAgIE5FRURFRDoKICAgIGZvciBteSAkTmVlZGVkIChxdyhEYXRhKSkgewogICAgICAgIG5leHQgTkVFREVEIGlmIGRlZmluZWQgJFBhcmFteyROZWVkZWR9OwoKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJOZWVkICROZWVkZWQhIgogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIE5FRURFRDoKICAgIGZvciBteSAkTmVlZGVkIChxdyhDb25maWdJdGVtSUQpKSB7CiAgICAgICAgbmV4dCBORUVERUQgaWYgZGVmaW5lZCAkUGFyYW17RGF0YX0tPnskTmVlZGVkfTsKCiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkTmVlZGVkIGluIERhdGEhIgogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIG15ICRDb25maWdJdGVtSUQgPSAkUGFyYW17RGF0YX0tPntDb25maWdJdGVtSUR9OwogICAgcmV0dXJuIGlmICEkQ29uZmlnSXRlbUlEOwoKICAgIG15ICRDb25maWdJdGVtRGF0YSA9ICRJVFNNQ29uZmlnSXRlbUludm9rZXJPYmplY3QtPkdldENvbmZpZ0l0ZW1EYXRhKAogICAgICAgIENvbmZpZ0l0ZW1JRCA9PiAkQ29uZmlnSXRlbUlELAogICAgICAgIEV2ZW50ICAgICAgICA9PiAkUGFyYW17RGF0YX0tPntFdmVudH0sCiAgICApOwogICAgcmV0dXJuIGlmICFJc0hhc2hSZWZXaXRoRGF0YSgkQ29uZmlnSXRlbURhdGEpOwoKICAgIHJldHVybiAleyRDb25maWdJdGVtRGF0YX07Cn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpHZW5lcmljSW50ZXJmYWNlOjpJbnZva2VyOjpJVFNNQ29uZmlnSXRlbTo6R2VuZXJpYzsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6T2JqZWN0TWFuYWdlcjsKdXNlIEtlcm5lbDo6U3lzdGVtOjpWYXJpYWJsZUNoZWNrIHF3KDphbGwpOwoKb3VyICRPYmplY3RNYW5hZ2VyRGlzYWJsZWQgPSAxOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6R2VuZXJpY0ludGVyZmFjZTo6SW52b2tlcjo6SVRTTUNvbmZpZ0l0ZW06OkdlbmVyaWMKCj1oZWFkMSBQVUJMSUMgSU5URVJGQUNFCgo9aGVhZDIgbmV3KCkKCnVzdWFsbHksIHlvdSB3YW50IHRvIGNyZWF0ZSBhbiBpbnN0YW5jZSBvZiB0aGlzIGJ5IHVzaW5nCgogICAgS2VybmVsOjpHZW5lcmljSW50ZXJmYWNlOjpJbnZva2VyLT5uZXcoKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICBteSAkU2VsZiA9IHt9OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgIGlmICggISRQYXJhbXtEZWJ1Z2dlck9iamVjdH0gKSB7CiAgICAgICAgcmV0dXJuIHsKICAgICAgICAgICAgU3VjY2VzcyAgICAgID0+IDAsCiAgICAgICAgICAgIEVycm9yTWVzc2FnZSA9PiAiR290IG5vIERlYnVnZ2VyT2JqZWN0ISIKICAgICAgICB9OwogICAgfQoKICAgICRTZWxmLT57RGVidWdnZXJPYmplY3R9ID0gJFBhcmFte0RlYnVnZ2VyT2JqZWN0fTsKCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1oZWFkMiBQcmVwYXJlUmVxdWVzdCgpCgpwcmVwYXJlIHRoZSBpbnZvY2F0aW9uIG9mIHRoZSBjb25maWd1cmVkIHJlbW90ZSB3ZWIgc2VydmljZS4KCiAgICBteSAkUmVzdWx0ID0gJEludm9rZXJPYmplY3QtPlByZXBhcmVSZXF1ZXN0KAogICAgICAgIERhdGEgPT4geyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAjIGRhdGEgcGF5bG9hZAogICAgICAgICAgICAuLi4KICAgICAgICB9LAogICAgKTsKCiAgICAkUmVzdWx0ID0gewogICAgICAgIFN1Y2Nlc3MgICAgICAgICA9PiAxLCAgICAgICAgICAgICAgICAgICAjIDAgb3IgMQogICAgICAgIEVycm9yTWVzc2FnZSAgICA9PiAnJywgICAgICAgICAgICAgICAgICAjIGluIGNhc2Ugb2YgZXJyb3IKICAgICAgICBEYXRhICAgICAgICAgICAgPT4geyAgICAgICAgICAgICAgICAgICAgIyBkYXRhIHBheWxvYWQgYWZ0ZXIgSW52b2tlcgogICAgICAgICAgICAuLi4KICAgICAgICB9LAogICAgfTsKCj1jdXQKCnN1YiBQcmVwYXJlUmVxdWVzdCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRDb25maWdPYmplY3QgICAgICAgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OkNvbmZpZycpOwogICAgbXkgJElUU01Db25maWdJdGVtSW52b2tlck9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbUludm9rZXInKTsKCiAgICBteSAkQ29uZmlnSXRlbURhdGEgPSAkSVRTTUNvbmZpZ0l0ZW1JbnZva2VyT2JqZWN0LT5HZXRDb25maWdJdGVtRGF0YSgKICAgICAgICBDb25maWdJdGVtSUQgPT4gJFBhcmFte0RhdGF9LT57Q29uZmlnSXRlbUlEfSwKCiAgICAgICAgIyBBZHZhbmNlZEdJIHB1dHMgZXZlbnQgaW50byAkUGFyYW17RGF0YX0tPntFdmVudH0sCiAgICAgICAgIyBpZiBleGVjdXRlZCB2aWEgS2VybmVsOjpHZW5lcmljSW50ZXJmYWNlOjpFdmVudDo6SGFuZGxlcgogICAgICAgIEV2ZW50ID0+ICRQYXJhbXtEYXRhfS0+e0V2ZW50fSwKICAgICk7CgogICAgaWYgKCAhSXNIYXNoUmVmV2l0aERhdGEoJENvbmZpZ0l0ZW1EYXRhKSApIHsKICAgICAgICByZXR1cm4gewogICAgICAgICAgICBTdWNjZXNzICAgICAgPT4gMCwKICAgICAgICAgICAgRXJyb3JNZXNzYWdlID0+ICJEYXRhIG9mIGNvbmZpZyBpdGVtIHdpdGggSUQgJFBhcmFte0RhdGF9LT57Q29uZmlnSXRlbUlEfSBjb3VsZCBub3QgYmUgcmV0cmlldmVkLiIsCiAgICAgICAgfTsKICAgIH0KCiAgICBteSAlRGF0YSA9ICgKICAgICAgICBDb25maWdJdGVtID0+ICRDb25maWdJdGVtRGF0YSAvLyB7fSwKICAgICAgICBFdmVudCAgICAgID0+ICRQYXJhbXtEYXRhfS0+e0V2ZW50fSwKICAgICk7CgogICAgcmV0dXJuIHsKICAgICAgICBTdWNjZXNzID0+IDEsCiAgICAgICAgRGF0YSAgICA9PiBcJURhdGEsCiAgICB9Owp9Cgo9aGVhZDIgSGFuZGxlUmVzcG9uc2UoKQoKaGFuZGxlIHJlc3BvbnNlIGRhdGEgb2YgdGhlIGNvbmZpZ3VyZWQgcmVtb3RlIHdlYiBzZXJ2aWNlLgoKICAgIG15ICRSZXN1bHQgPSAkSW52b2tlck9iamVjdC0+SGFuZGxlUmVzcG9uc2UoCiAgICAgICAgUmVzcG9uc2VTdWNjZXNzICAgICAgPT4gMSwgICAgICAgICAgICAgICMgc3VjY2VzcyBzdGF0dXMgb2YgdGhlIHJlbW90ZSB3ZWIgc2VydmljZQogICAgICAgIFJlc3BvbnNlRXJyb3JNZXNzYWdlID0+ICcnLCAgICAgICAgICAgICAjIGluIGNhc2Ugb2Ygd2ViIHNlcnZpY2UgZXJyb3IKICAgICAgICBEYXRhID0+IHsgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIyBkYXRhIHBheWxvYWQKICAgICAgICAgICAgLi4uCiAgICAgICAgfSwKICAgICk7CgogICAgJFJlc3VsdCA9IHsKICAgICAgICBTdWNjZXNzICAgICAgICAgPT4gMSwgICAgICAgICAgICAgICAgICAgIyAwIG9yIDEKICAgICAgICBFcnJvck1lc3NhZ2UgICAgPT4gJycsICAgICAgICAgICAgICAgICAgIyBpbiBjYXNlIG9mIGVycm9yCiAgICAgICAgRGF0YSAgICAgICAgICAgID0+IHsgICAgICAgICAgICAgICAgICAgICMgZGF0YSBwYXlsb2FkIGFmdGVyIEludm9rZXIKICAgICAgICAgICAgLi4uCiAgICAgICAgfSwKICAgIH07Cgo9Y3V0CgpzdWIgSGFuZGxlUmVzcG9uc2UgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICBteSAkQmFja2VuZE9iamVjdCAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkR5bmFtaWNGaWVsZDo6QmFja2VuZCcpOwogICAgbXkgJER5bmFtaWNGaWVsZE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpEeW5hbWljRmllbGQnKTsKICAgIG15ICRUaWNrZXRPYmplY3QgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VGlja2V0Jyk7CiAgICBteSAkQXJ0aWNsZU9iamVjdCAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlRpY2tldDo6QXJ0aWNsZScpOwoKICAgICMgaWYgdGhlcmUgd2FzIGFuIGVycm9yIGluIHRoZSByZXNwb25zZSwgZm9yd2FyZCBpdAogICAgaWYgKCAhJFBhcmFte1Jlc3BvbnNlU3VjY2Vzc30gKSB7CiAgICAgICAgaWYgKCAhSXNTdHJpbmdXaXRoRGF0YSggJFBhcmFte1Jlc3BvbnNlRXJyb3JNZXNzYWdlfSApICkgewogICAgICAgICAgICByZXR1cm4gJFNlbGYtPntEZWJ1Z2dlck9iamVjdH0tPkVycm9yKAogICAgICAgICAgICAgICAgU3VtbWFyeSA9PiAnR290IHJlc3BvbnNlIGVycm9yLCBidXQgbm8gcmVzcG9uc2UgZXJyb3IgbWVzc2FnZSEnLAogICAgICAgICAgICApOwogICAgICAgIH0KICAgICAgICByZXR1cm4gewogICAgICAgICAgICBTdWNjZXNzICAgICAgPT4gMCwKICAgICAgICAgICAgRXJyb3JNZXNzYWdlID0+ICRQYXJhbXtSZXNwb25zZUVycm9yTWVzc2FnZX0sCiAgICAgICAgfTsKICAgIH0KCiAgICByZXR1cm4gewogICAgICAgIFN1Y2Nlc3MgPT4gMSwKICAgICAgICBEYXRhICAgID0+ICRQYXJhbXtEYXRhfSwKICAgIH07Cn0KCjE7Cg==
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::GenericInterface::Operation::ConfigItem::Common;

use strict;
use warnings;

use MIME::Base64();

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

our $ObjectManagerDisabled = 1;

=head1 NAME

Kernel::GenericInterface::Operation::ConfigItem::Common - Base class for all CI Operations

=head1 PUBLIC INTERFACE

=cut

=head2 Init()

initialize the operation by checking the web-service configuration

    my $Return = $CommonObject->Init(
        WebserviceID => 1,
    );

    $Return = {
        Success => 1,                       # or 0 in case of failure,
        ErrorMessage => 'Error Message',
    }

=cut

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

    # check needed
    if ( !$Param{WebserviceID} ) {
        return {
            Success      => 0,
            ErrorMessage => "Got no WebserviceID!",
        };
    }

    # get webservice configuration
    my $Webservice = $Kernel::OM->Get('Kernel::System::GenericInterface::Webservice')->WebserviceGet(
        ID => $Param{WebserviceID},
    );

    if ( !IsHashRefWithData($Webservice) ) {
        return {
            Success => 0,
            ErrorMessage =>
                'Could not determine Web service configuration'
                . ' in Kernel::GenericInterface::Operation::ConfigItem::Common::new()',
        };
    }

    return {
        Success => 1,
    };
}

=head2 ValidateClass()

checks if the given Class is valid.

    my $Sucess = $CommonObject->ValidateClass(
        Class => 'some class',
    );

    returns
    $Success = 1            # or 0

=cut

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

    # check needed stuff
    return if !$Param{Class};

    # check for Class sent
    my $ItemDataRef = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemGet(
        Class => 'ITSM::ConfigItem::Class',
        Name  => $Param{Class},
    );

    # return false if item data is empty
    return if !IsHashRefWithData($ItemDataRef);

    return 1;
}

=head2 ValidateDeplState()

checks if the given DeplState is valid.

    my $Sucess = $CommonObject->ValidateDeplState(
        DelpState => 'some DeplState',
    );

    returns
    $Success = 1            # or 0

=cut

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

    # check needed stuff
    return if !$Param{DeplState};

    # check for Class sent
    my $ItemDataRef = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemGet(
        Class => 'ITSM::ConfigItem::DeploymentState',
        Name  => $Param{DeplState},
    );

    # return false if item data is empty
    return if !IsHashRefWithData($ItemDataRef);

    return 1;
}

=head2 ValidateInciState()

checks if the given InciState is valid.

    my $Sucess = $CommonObject->ValidateInciState(
        InciState => 'some InciState',
    );

    returns
    $Success = 1            # or 0

=cut

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

    # check needed stuff
    return if !$Param{InciState};

    # check for Class sent
    my $ItemDataRef = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemGet(
        Class => 'ITSM::Core::IncidentState',
        Name  => $Param{InciState},
    );

    # return false if item data is empty
    return if !IsHashRefWithData($ItemDataRef);

    return 1;
}

=head2 ValidateInputText()

checks if the given value is valid.

    my $Sucess = $CommonObject->ValidateInputText(
        Value     => 'some value',
        MaxLength => 123,
    );

    returns
    $Success = 1            # or 0

=cut

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

    # check length
    if (
        defined $Param{Input}->{MaxLength}
        && $Param{Input}->{MaxLength}
        && length $Param{Value} > $Param{Input}->{MaxLength}
        )
    {
        return;
    }

    return 1;
}

=head2 ValidateInputDate()

checks if the given value is valid.

    my $Sucess = $CommonObject->ValidateInputDate(
        Value => '12/12/1977',
    );

    or

    my $Sucess = $CommonObject->ValidateInputDate(
        Value => '1977-12-12',
    );

    returns
    $Success = 1            # or 0

=cut

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

    my $Value = $Param{Value};

    # check and convert for date format like "05/18/2011"
    if ( $Value =~ m{\A (\d{2}) / (\d{2}) / (\d{4}) \z}xms ) {
        $Value = $3 . '-' . $1 . '-' . $2 . ' 00:00:00';
    }

    # check and convert for date format like "2011-05-18"
    if ( $Value =~ m{\A (\d{4} - \d{2} - \d{2} \z) }xms ) {
        $Value = $1 . ' 00:00:00';
    }

    # Convert the raw data to a system datetime object.
    my $DateTimeObject = $Kernel::OM->Create(
        'Kernel::System::DateTime',
        ObjectParams => {
            String => $Value,
        },
    );

    return if !$DateTimeObject;

    return 1;
}

=head2 ValidateInputDateTime()

checks if the given value is valid.

    my $Sucess = $CommonObject->ValidateInputDateTime(
        Value => '12/12/1977 12:00:00',
    );

    or

    my $Sucess = $CommonObject->ValidateInputDateTime(
        Value => '1977-12-12 12:00:00',
    );

    returns
    $Success = 1            # or 0

=cut

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

    my $Value = $Param{Value};

    # check and convert for date format like "05/18/2011"
    if ( $Value =~ m{\A (\d{2}) / (\d{2}) / (\d{4}) \z}xms ) {
        $Value = $3 . '-' . $1 . '-' . $2 . ' 00:00:00';
    }
    elsif ( $Value =~ m{\A (\d{2}) / (\d{2}) / (\d{4}) (\d{2} : \d{2} : \d{2}) \z}xms ) {
        $Value = $3 . '-' . $1 . '-' . $2 . $4;
    }

    # check and convert for date format like "2011-05-18"
    if ( $Value =~ m{\A (\d{4} - \d{2} - \d{2} \z) }xms ) {
        $Value = $1 . ' 00:00:00';
    }

    # Convert the raw data to a system datetime object.
    my $DateTimeObject = $Kernel::OM->Create(
        'Kernel::System::DateTime',
        ObjectParams => {
            String => $Value,
        },
    );

    return if !$DateTimeObject;

    return 1;
}

=head2 ValidateInputInteger()

checks if the given value is valid.

    my $Sucess = $CommonObject->ValidateInputInteger(
        Value => 123,
    );

    returns
    $Success = 1            # or 0

=cut

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

    my $Value = $Param{Value};

    return if !IsInteger($Value);

    return if defined $Param{ValueMin} && $Value < $Param{ValueMin};
    return if defined $Param{ValueMax} && $Value > $Param{ValueMax};

    return 1;
}

=head2 ValidateInputGeneralCatalog()

checks if the given value is valid.

    my $Sucess = $CommonObject->ValidateInputGeneralCatalog(
        Value => 123,
        Class => 'Some general catalog class'
    );

    returns
    $Success = 1            # or 0

=cut

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

    my $Value = $Param{Value};

    # get the values for the General catalog class
    my $ItemList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => $Param{Input}->{Class},
    );

    # create a lokup list
    my %ItemListLookup = reverse %{$ItemList};

    return if !$ItemListLookup{$Value};

    return 1;
}

=head2 ValidateInputCustomer()

checks if the given value is valid.

    my $Sucess = $CommonObject->ValidateInputCustomer(
        Value => 'some customer login',
    );

    returns
    $Success = 1            # or 0

=cut

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

    my $Value = $Param{Value};

    return if !$Value;

    my %CustomerData = $Kernel::OM->Get('Kernel::System::CustomerUser')->CustomerUserDataGet(
        User => $Param{Value},
    );

    # if customer is not registered in the database
    return if !IsHashRefWithData( \%CustomerData );

    # if ValidID is present, check if it is valid!
    if ( defined $CustomerData{ValidID} ) {

        # return false if customer is not valid
        return
            if $Kernel::OM->Get('Kernel::System::Valid')->ValidLookup( ValidID => $CustomerData{ValidID} ) ne 'valid';
    }

    return 1;
}

=head2 ValidateInputCustomerCompany()

checks if the given value is valid.

    my $Sucess = $CommonObject->ValidateInputCustomerCompany(
        Value => 'some customer company name',
    );

    returns
    $Success = 1            # or 0

=cut

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

    my $Value = $Param{Value};

    return if !$Value;

    my %CompanyList = $Kernel::OM->Get('Kernel::System::CustomerCompany')->CustomerCompanyList(
        Limit => 0,    # Display all Customer Companies
    );

    return if !$CompanyList{ $Param{Value} };

    return 1;
}

=head2 ValidateMimeType()

checks if the given MimeType is valid.

    my $Sucess = $CommonObject->ValidateMimeType(
        MimeTypeID => 'some MimeType',
    );

    returns
    $Success = 1            # or 0

=cut

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

    # check needed stuff
    return if !$Param{MimeType};

    return if $Param{MimeType} !~ m{\A\w+\/\w+\z};

    return 1;
}

=head2 ValidateCharset()

checks if the given Charset is valid.

    my $Sucess = $CommonObject->ValidateCharset(
        Charset => 'some charset',
    );

    returns
    $Success = 1            # or 0

=cut

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

    # check needed stuff
    return if !$Param{Charset};

    my $CharsetList = $Self->_CharsetList();

    return if !$CharsetList->{ $Param{Charset} };

    return 1;
}

=head2 ReplaceInputDate()

replaces the user value with a system valid value.

    my $NewValue = $CommonObject->ReplaceInputDate(
        Value => '12/12/1977',
    );

    or

    my $NewValue = $CommonObject->ReplaceInputDate(
        Value => '1977-12-12',
    );

    returns
    $NewValue = '1977-12-12',

=cut

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

    my $Value = $Param{Value};

    # check and convert for date format like "05/18/2011"
    if ( $Value =~ m{\A (\d{2}) / (\d{2}) / (\d{4}) \z}xms ) {
        $Value = $3 . '-' . $1 . '-' . $2 . ' 00:00:00';
    }

    # check and convert for date format like "2011-05-18"
    if ( $Value =~ m{\A (\d{4} - \d{2} - \d{2} \z) }xms ) {
        $Value = $1 . ' 00:00:00';
    }

    # Convert the raw data to a system datetime object.
    my $DateTimeObject = $Kernel::OM->Create(
        'Kernel::System::DateTime',
        ObjectParams => {
            String => $Value,
        },
    );

    return $DateTimeObject->Format( Format => '%F' );
}

=head2 ReplaceInputDateTime()

replaces the user value with a system valid value.

    my $NewValue = $CommonObject->ReplaceInputDateTime(
        Value => '12/12/1977 12:00:00',
    );

    or

    my $NewValue = $CommonObject->ReplaceInputDateTime(
        Value => '1977-12-12 12:00:00',
    );

    returns
    $NewValue = '1977-12-12 12:00:00';

=cut

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

    my $Value = $Param{Value};

    # check and convert for date format like "05/18/2011"
    if ( $Value =~ m{\A (\d{2}) / (\d{2}) / (\d{4}) \z}xms ) {
        $Value = $3 . '-' . $1 . '-' . $2 . ' 00:00:00';
    }
    elsif ( $Value =~ m{\A (\d{2}) / (\d{2}) / (\d{4}) (\d{2} : \d{2} : \d{2}) \z}xms ) {
        $Value = $3 . '-' . $1 . '-' . $2 . $4;
    }

    # check and convert for date format like "2011-05-18"
    if ( $Value =~ m{\A (\d{4} - \d{2} - \d{2} \z) }xms ) {
        $Value = $1 . ' 00:00:00';
    }

    # Convert the raw data to a system datetime object.
    my $DateTimeObject = $Kernel::OM->Create(
        'Kernel::System::DateTime',
        ObjectParams => {
            String => $Value,
        },
    );

    return $DateTimeObject->ToString();
}

=head2 ReplaceInputGeneralCatalog()

replaces the user value with a system valid value.

    my $NewValue = $CommonObject->ReplaceInputGeneralCatalog(
        Value => 'some value',
        Class => 'Some general catalog class'
    );

    returns
    $NewValue = 123

=cut

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

    my $Value = $Param{Value};

    # get the values for the General catalog class
    my $ItemList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => $Param{Input}->{Class},
    );

    # create a lokup list
    my %ItemListLookup = reverse %{$ItemList};

    return $ItemListLookup{$Value};
}

=head2 InvertReplaceInputDate()

replaces the system value with a user value.

    my $NewValue = $CommonObject->InvertReplaceInputDate(
        Value => '12-12-1977 00:00:00',
    );

    returns
    $NewValue = '1977-12-12',

=cut

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

    my $Value = $Param{Value};

    $Value =~ s{\A (\d{4} - \d{2} - \d{2}) [ ] 00:00:00 \z}{$1}xms;

    return $Value;
}

=head2 InvertReplaceInputGeneralCatalog()

replaces the system value with a user value.

    my $NewValue = $CommonObject->InvertReplaceInputGeneralCatalog(
        Value => 123,
        Class => 'Some general catalog class'
    );

    returns
    $NewValue = 'some value'

=cut

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

    my $Value = $Param{Value};

    # get the values for the General catalog class
    my $ItemList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => $Param{Input}->{Class},
    );

    # create a lokup list
    return $ItemList->{$Value};
}

=head2 CreateAttachment()

creates a new attachment for the given ConfigItem.

    my $Result = $CommonObject->CreateAttachment(
        Content      => $Data,                   # file content (Base64 encoded)
        ContentType  => 'some content type',
        Filename     => 'some filename',
        ConfigItemID => 456,
        UserID       => 123,
    );

    returns
    $Result = {
        Success => 1,                        # if everything is ok
    }

    $Result = {
        Success      => 0,
        ErrorMessage => 'Error description'
    }

=cut

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

    # check needed stuff
    for my $Needed (qw(Attachment ConfigItemID UserID)) {
        if ( !$Param{$Needed} ) {
            return {
                Success      => 0,
                ErrorMessage => "CreateAttachment() Got no $Needed!",
            };
        }
    }

    # write attachment
    my $Success = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemAttachmentAdd(
        %{ $Param{Attachment} },
        ConfigItemID => $Param{ConfigItemID},
        Content      => MIME::Base64::decode_base64( $Param{Attachment}->{Content} ),
        UserID       => $Param{UserID},
    );

    return {
        Success => $Success,
    };
}

=head2 CheckXMLData()

checks if the given XMLData value are valid.

    my $XMLDataCheck = $CommonObject->CheckXMLData(
        Definition => $DefinitionArrayRef,          # Config Item Definition ot just part of it
        XMLData    => $XMLDataHashRef,
        Parent     => 'some parent',
    );

    returns:

    $XMLDataCheck = {
        Success => 1,                               # if everything is OK
    }

    $XMLDataCheck = {
        ErrorCode    => 'Function.Error',           # if error
        ErrorMessage => 'Error description',
    }

=cut

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

    my $Definition = $Param{Definition};
    my $XMLData    = $Param{XMLData};
    my $Parent     = $Param{Parent} || '';

    my $CheckValueResult;
    for my $DefItem ( @{$Definition} ) {
        my $ItemKey = $DefItem->{Key};

        # check if at least one element should exist
        if (
            (
                defined $DefItem->{CountMin}
                && $DefItem->{CountMin} >= 1
                && defined $DefItem->{Input}->{Required}
                && $DefItem->{Input}->{Required}
            )
            && ( !defined $XMLData->{$ItemKey} || !$XMLData->{$ItemKey} )
            )
        {
            return {
                ErrorCode => "$Self->{OperationName}.MissingParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter"
                    . " is missing!",
            };
        }

        if ( ref $XMLData->{$ItemKey} eq 'ARRAY' ) {
            for my $ArrayItem ( @{ $XMLData->{$ItemKey} } ) {
                if ( ref $ArrayItem eq 'HASH' ) {
                    $CheckValueResult = $Self->_CheckValue(
                        Value   => $ArrayItem->{$ItemKey},
                        Input   => $DefItem->{Input},
                        ItemKey => $ItemKey,
                        Parent  => $Parent,
                    );
                    if ( !$CheckValueResult->{Success} ) {
                        return $CheckValueResult;
                    }
                }
                elsif ( ref $ArrayItem eq '' ) {
                    $CheckValueResult = $Self->_CheckValue(
                        Value   => $ArrayItem,
                        Input   => $DefItem->{Input},
                        ItemKey => $ItemKey,
                        Parent  => $Parent,
                    );
                    if ( !$CheckValueResult->{Success} ) {
                        return $CheckValueResult;
                    }
                }
                else {
                    return {
                        ErrorCode => "$Self->{OperationName}.InvalidParameter",
                        ErrorMessage =>
                            "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter"
                            . " is invalid!",
                    };
                }
            }
        }
        elsif ( ref $XMLData->{$ItemKey} eq 'HASH' ) {
            $CheckValueResult = $Self->_CheckValue(
                Value   => $XMLData->{$ItemKey}->{$ItemKey},
                Input   => $DefItem->{Input},
                ItemKey => $ItemKey,
                Parent  => $Parent,
            );
            if ( !$CheckValueResult->{Success} ) {
                return $CheckValueResult;
            }
        }
        else {

            # only perform checks if item really exits in the XMLData
            # CountNin checks was verified and passed before!, so it is safe to skip if needed
            if ( $XMLData->{$ItemKey} ) {
                $CheckValueResult = $Self->_CheckValue(
                    Value   => $XMLData->{$ItemKey},
                    Input   => $DefItem->{Input},
                    ItemKey => $ItemKey,
                    Parent  => $Parent,
                );
                if ( !$CheckValueResult->{Success} ) {
                    return $CheckValueResult;
                }
            }
        }

        # check if exists more elements than the ones they should
        if ( defined $DefItem->{CountMax} )
        {
            if (
                ref $XMLData->{$ItemKey} eq 'ARRAY'
                && scalar @{ $XMLData->{$ItemKey} } > $DefItem->{CountMax}
                )
            {
                return {
                    ErrorCode => "$Self->{OperationName}.InvalidParameter",
                    ErrorMessage =>
                        "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter"
                        . " repetitions is higher than the maxium value!",
                };
            }
        }

        # check if there is a sub and start recursion
        if ( defined $DefItem->{Sub} ) {

            if ( ref $XMLData->{$ItemKey} eq 'ARRAY' ) {
                my $Counter = 0;
                for my $ArrayItem ( @{ $XMLData->{$ItemKey} } ) {

                    # start recursion for each array item
                    my $XMLDataCheck = $Self->CheckXMLData(
                        Definition => $DefItem->{Sub},
                        XMLData    => $ArrayItem,
                        Parent     => $Parent . $ItemKey . "[$Counter]->",
                    );
                    if ( !$XMLDataCheck->{Success} ) {
                        return $XMLDataCheck;
                    }
                    $Counter++;
                }
            }
            elsif ( ref $XMLData->{$ItemKey} eq 'HASH' ) {

                # start recursion
                my $XMLDataCheck = $Self->CheckXMLData(
                    Definition => $DefItem->{Sub},
                    XMLData    => $XMLData->{$ItemKey},
                    Parent     => $Parent . $ItemKey . '->',
                );
                if ( !$XMLDataCheck->{Success} ) {
                    return $XMLDataCheck;
                }
            }
            else {

                # start recusrsion
                my $XMLDataCheck = $Self->CheckXMLData(
                    Definition => $DefItem->{Sub},
                    XMLData    => {},
                    Parent     => $Parent . $ItemKey . '->',
                );
                if ( !$XMLDataCheck->{Success} ) {
                    return $XMLDataCheck;
                }
            }
        }
    }

    return {
        Success => 1,
    };
}

=head2 ReplaceXMLData()

replace the XMLData to one that uses internal values.

    my $NewXMLData = $CommonObject->ReplaceXMLData(
        Definition => $DefinitionArrayRef,          # Config Item Definition ot just part of it
        XMLData    => $XMLDataHashRef,
        Parent     => 'some parent',
    );

    returns:

    $NewXMLData = $XMLDataHashRef,                  # with replaced values

=cut

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

    my $Definition = $Param{Definition};
    my $XMLData    = $Param{XMLData};
    my $Parent     = $Param{Parent} || '';

    my $NewXMLData = $XMLData;

    ITEM:
    for my $DefItem ( @{$Definition} ) {
        my $ItemKey = $DefItem->{Key};

        my $NewValue;

        if ( ref $XMLData->{$ItemKey} eq 'ARRAY' ) {
            for my $ArrayItem ( @{ $XMLData->{$ItemKey} } ) {
                if ( ref $ArrayItem eq 'HASH' ) {

                    # get the new value
                    $NewValue = $Self->_ReplaceValue(
                        Value   => $ArrayItem->{$ItemKey},
                        Input   => $DefItem->{Input},
                        ItemKey => $ItemKey,
                        Parent  => $Parent,
                    );

                    # replace the value in the array
                    $ArrayItem->{$ItemKey} = $NewValue;
                }

                elsif ( ref $ArrayItem eq '' ) {
                    $NewValue = $Self->_ReplaceValue(
                        Value   => $ArrayItem,
                        Input   => $DefItem->{Input},
                        ItemKey => $ItemKey,
                        Parent  => $Parent,
                    );

                    # replace the value in the array
                    $ArrayItem = $NewValue;
                }
            }
        }
        elsif ( ref $XMLData->{$ItemKey} eq 'HASH' ) {
            $NewValue = $Self->_ReplaceValue(
                Value   => $XMLData->{$ItemKey}->{$ItemKey},
                Input   => $DefItem->{Input},
                ItemKey => $ItemKey,
                Parent  => $Parent,
            );

            # replace the value in the hash
            $XMLData->{$ItemKey}->{$ItemKey} = $NewValue;
        }
        else {

            # only perform replace if item really exits in the XMLData
            if ( $XMLData->{$ItemKey} ) {
                $NewValue = $Self->_ReplaceValue(
                    Value   => $XMLData->{$ItemKey},
                    Input   => $DefItem->{Input},
                    ItemKey => $ItemKey,
                    Parent  => $Parent,
                );

                # replace the root value
                $XMLData->{$ItemKey} = $NewValue;
            }
        }

        # replace value in the resulting XMLData
        if ( $XMLData->{$ItemKey} ) {
            $NewXMLData->{$ItemKey} = $XMLData->{$ItemKey};
        }

        # check if there is a sub and start recursion
        if ( defined $DefItem->{Sub} ) {

            if ( ref $XMLData->{$ItemKey} eq 'ARRAY' ) {
                my $Counter = 0;
                for my $ArrayItem ( @{ $XMLData->{$ItemKey} } ) {

                    # start recursion for each array item
                    my $NewXMLDataPart = $Self->ReplaceXMLData(
                        Definition => $DefItem->{Sub},
                        XMLData    => $ArrayItem,
                        Parent     => $Parent . $ItemKey . "[$Counter]->",
                    );
                    $NewXMLData->{$ItemKey}->[$Counter] = $NewXMLDataPart;
                    $Counter++;
                }
            }
            elsif ( ref $XMLData->{$ItemKey} eq 'HASH' ) {

                # start recursion
                my $NewXMLDataPart = $Self->ReplaceXMLData(
                    Definition => $DefItem->{Sub},
                    XMLData    => $XMLData->{$ItemKey},
                    Parent     => $Parent . $ItemKey . '->',
                );
                $NewXMLData->{$ItemKey} = $NewXMLDataPart;
            }
            else {
                if ( $XMLData->{$ItemKey} ) {

                    # start recusrsion
                    my $NewXMLDataPart = $Self->ReplaceXMLData(
                        Definition => $DefItem->{Sub},
                        XMLData    => {},
                        Parent     => $Parent . $ItemKey . '->',
                    );
                    $NewXMLData->{$ItemKey} = $NewXMLDataPart;
                }
            }
        }
    }
    return $NewXMLData;
}

=head2 FormatXMLData()

Create a XMLData suitable for VersionAdd.

    my $NewXMLData = $CommonObject->FormatXMLData(
        XMLData    => $XMLDataHashRef,
        Child      => 1,                    # or 0, optional
    );

    returns:

    $NewXMLData = $XMLDataHashRef,                  # suitable for version add

=cut

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

    my $XMLData = $Param{XMLData};
    my $Child   = $Param{Child};

    my $NewXMLData;

    for my $RootKey ( sort keys %{$XMLData} ) {
        if ( ref $XMLData->{$RootKey} eq 'ARRAY' ) {
            my @NewXMLParts;
            $NewXMLParts[0] = undef;

            for my $ArrayItem ( @{ $XMLData->{$RootKey} } ) {
                if ( ref $ArrayItem eq 'HASH' ) {

                    # extract the root key from the hash and assign it to content key
                    my $Content = delete $ArrayItem->{$RootKey};

                    # start recursion
                    my $NewXMLDataPart = $Self->FormatXMLData(
                        XMLData => $ArrayItem,
                        Child   => 1,
                    );
                    push @NewXMLParts, {
                        Content => $Content,
                        %{$NewXMLDataPart},
                    };
                }
                elsif ( ref $ArrayItem eq '' ) {
                    push @NewXMLParts, {
                        Content => $ArrayItem,
                    };
                }
            }

            # assamble the final value from the parts array
            $NewXMLData->{$RootKey} = \@NewXMLParts;
        }

        if ( ref $XMLData->{$RootKey} eq 'HASH' ) {

            my @NewXMLParts;
            $NewXMLParts[0] = undef;

            # extract the root key from the hash and assign it to content key
            my $Content = delete $XMLData->{$RootKey}->{$RootKey};

            # start recursion
            my $NewXMLDataPart = $Self->FormatXMLData(
                XMLData => $XMLData->{$RootKey},
                Child   => 1,
            );
            push @NewXMLParts, {
                Content => $Content,
                %{$NewXMLDataPart},
            };

            # assamble the final value from the parts array
            $NewXMLData->{$RootKey} = \@NewXMLParts;
        }

        elsif ( ref $XMLData->{$RootKey} eq '' ) {
            $NewXMLData->{$RootKey} = [
                undef,
                {
                    Content => $XMLData->{$RootKey},
                },
            ];
        }
    }

    # return only the part on recursion
    if ($Child) {
        return $NewXMLData;
    }

    # return the complete XMLData as needed for version add
    return [
        undef,
        {
            Version => [
                undef,
                $NewXMLData
            ],
        },
    ];
}

=head2 InvertFormatXMLData()

Creates a readable XMLData.

    my $NewXMLData = $CommonObject->InvertFormatXMLData(
        XMLData    => $XMLDataHashRef,
    );

    returns:

    $NewXMLData = $XMLDataHashRef,                  # suitable for version add

=cut

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

    my $XMLData = $Param{XMLData};

    my $NewXMLData;
    my $Content;
    ROOTHASH:
    for my $RootHash ( @{$XMLData} ) {
        next ROOTHASH if !defined $RootHash;
        delete $RootHash->{TagKey};

        for my $RootHashKey ( sort keys %{$RootHash} ) {
            if ( ref $RootHash->{$RootHashKey} eq 'ARRAY' ) {
                if ( scalar @{ $RootHash->{$RootHashKey} } > 2 ) {

                    # we are on an array
                    my $Counter = 0;
                    ARRAYITEM:
                    for my $ArrayItem ( @{ $RootHash->{$RootHashKey} } ) {
                        next ARRAYITEM if !defined $ArrayItem;
                        delete $ArrayItem->{TagKey};

                        $Content = delete $ArrayItem->{Content} || '';

                        if ( scalar keys %{$ArrayItem} ) {
                            $NewXMLData->{$RootHashKey}->[$Counter]->{$RootHashKey} = $Content;

                            # start recursion
                            for my $ArrayItemKey ( sort keys %{$ArrayItem} ) {

                                my $NewXMLDataPart = $Self->InvertFormatXMLData(
                                    XMLData => $ArrayItem->{$ArrayItemKey},
                                );
                                $NewXMLData->{$RootHashKey}->[$Counter]->{$ArrayItemKey} = $NewXMLDataPart;
                            }
                        }
                        else {
                            $NewXMLData->{$RootHashKey}->[$Counter] = $Content;
                        }

                        $Counter++;
                    }
                }
                else {

                    # we are on a hash or single value
                    ARRAYITEM:
                    for my $ArrayItem ( @{ $RootHash->{$RootHashKey} } ) {
                        next ARRAYITEM if !defined $ArrayItem;
                        delete $ArrayItem->{TagKey};

                        $Content = delete $ArrayItem->{Content} || '';

                        if ( scalar keys %{$ArrayItem} ) {
                            $NewXMLData->{$RootHashKey}->{$RootHashKey} = $Content;

                            # start recursion
                            for my $ArrayItemKey ( sort keys %{$ArrayItem} ) {

                                my $NewXMLDataPart = $Self->InvertFormatXMLData(
                                    XMLData => $ArrayItem->{$ArrayItemKey},
                                );
                                $NewXMLData->{$RootHashKey}->{$ArrayItemKey} = $NewXMLDataPart;
                            }
                        }
                        else {
                            $NewXMLData->{$RootHashKey} = $Content;
                        }
                    }
                }
            }

            # if we are on a final node
            elsif ( ref $RootHash->{$RootHashKey} eq '' && $RootHashKey eq 'Content' ) {
                $NewXMLData = $RootHash->{$RootHashKey};
            }
        }
    }

    return $NewXMLData;
}

=head2 InvertReplaceXMLData()

replace the XMLData to one that uses user values.

    my $NewXMLData = $CommonObject->InvertReplaceXMLData(
        Definition => $DefinitionArrayRef,          # Config Item Definition ot just part of it
        XMLData    => $XMLDataHashRef,
        Parent     => 'some parent',
    );

    returns:

    $NewXMLData = $XMLDataHashRef,                  # with replaced values

=cut

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

    my $Definition = $Param{Definition};
    my $XMLData    = $Param{XMLData};
    my $Parent     = $Param{Parent} || '';

    my $NewXMLData = $XMLData;

    ITEM:
    for my $DefItem ( @{$Definition} ) {
        my $ItemKey = $DefItem->{Key};

        my $NewValue;

        if (
            IsHashRefWithData($XMLData)
            && $XMLData->{$ItemKey}
            && ref $XMLData->{$ItemKey} eq 'ARRAY'
            )
        {

            for my $ArrayItem ( @{ $XMLData->{$ItemKey} } ) {

                if ( ref $ArrayItem eq 'HASH' ) {

                    # get the new value
                    $NewValue = $Self->_InvertReplaceValue(
                        Value   => $ArrayItem->{$ItemKey},
                        Input   => $DefItem->{Input},
                        ItemKey => $ItemKey,
                        Parent  => $Parent,
                    );

                    # replace the value in the array
                    $ArrayItem->{$ItemKey} = $NewValue;
                }

                elsif ( ref $ArrayItem eq '' ) {
                    $NewValue = $Self->_InvertReplaceValue(
                        Value   => $ArrayItem,
                        Input   => $DefItem->{Input},
                        ItemKey => $ItemKey,
                        Parent  => $Parent,
                    );

                    # replace the value in the array
                    $ArrayItem = $NewValue;
                }
            }
        }
        elsif (
            IsHashRefWithData($XMLData)
            && $XMLData->{$ItemKey}
            && ref $XMLData->{$ItemKey} eq 'HASH'
            )
        {
            $NewValue = $Self->_InvertReplaceValue(
                Value   => $XMLData->{$ItemKey}->{$ItemKey},
                Input   => $DefItem->{Input},
                ItemKey => $ItemKey,
                Parent  => $Parent,
            );

            # replace the value in the hash
            $XMLData->{$ItemKey}->{$ItemKey} = $NewValue;
        }
        else {

            # only perform replace if item really exits in the XMLData
            if ( IsHashRefWithData($XMLData) && $XMLData->{$ItemKey} ) {
                $NewValue = $Self->_InvertReplaceValue(
                    Value   => $XMLData->{$ItemKey},
                    Input   => $DefItem->{Input},
                    ItemKey => $ItemKey,
                    Parent  => $Parent,
                );

                # replace the root value
                $XMLData->{$ItemKey} = $NewValue;
            }
        }

        # replace value in the resulting XMLData
        if ( IsHashRefWithData($XMLData) && $XMLData->{$ItemKey} ) {
            $NewXMLData->{$ItemKey} = $XMLData->{$ItemKey};
        }

        # check if there is a sub and start recursion
        if ( defined $DefItem->{Sub} ) {

            if ( ref $XMLData->{$ItemKey} eq 'ARRAY' ) {
                my $Counter = 0;
                for my $ArrayItem ( @{ $XMLData->{$ItemKey} } ) {

                    # start recursion for each array item
                    my $NewXMLDataPart = $Self->InvertReplaceXMLData(
                        Definition => $DefItem->{Sub},
                        XMLData    => $ArrayItem,
                        Parent     => $Parent . $ItemKey . "[$Counter]->",
                    );
                    $NewXMLData->{$ItemKey}->[$Counter] = $NewXMLDataPart;
                    $Counter++;
                }
            }
            elsif ( ref $XMLData->{$ItemKey} eq 'HASH' ) {

                # start recursion
                my $NewXMLDataPart = $Self->InvertReplaceXMLData(
                    Definition => $DefItem->{Sub},
                    XMLData    => $XMLData->{$ItemKey},
                    Parent     => $Parent . $ItemKey . '->',
                );
                $NewXMLData->{$ItemKey} = $NewXMLDataPart;
            }
            else {
                if ( $XMLData->{$ItemKey} ) {

                    # start recusrsion
                    my $NewXMLDataPart = $Self->InvertReplaceXMLData(
                        Definition => $DefItem->{Sub},
                        XMLData    => {},
                        Parent     => $Parent . $ItemKey . '->',
                    );
                    $NewXMLData->{$ItemKey} = $NewXMLDataPart;
                }
            }
        }
    }
    return $NewXMLData;
}

=head1 INTERNAL INTERFACE

=head2 _CharsetList()

returns a list of all available charsets.

    my $CharsetList = $CommonObject->_CharsetList(
        UserID => 123,
    );

    returns
    $Success = {
        #...
        iso-8859-1  => 1,
        iso-8859-15 => 1,
        MacRoman    => 1,
        utf8        => 1,
        #...
    }

=cut

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

    # get charset array
    use Encode;
    my @CharsetList = Encode->encodings(":all");

    my %CharsetHash;

    # create a charset lookup table
    for my $Charset (@CharsetList) {
        $CharsetHash{$Charset} = 1;
    }

    return \%CharsetHash;
}

=head2 _CheckValue()

checks if the given value is valid.

    my $ValueCheck = $CommonObject->_CheckValue(
        Value   => $Value                        # $Value could be a string, a time stamp,
                                                 #   general catalog class name, or a integer
        Input   => $InputDefinitionHashRef,      # The definition of the element input extracted
                                                 #   from the Configuration Item definition for
                                                 #   for each value
        ItemKey => 'some key',                   # The name of the value as sent in the SOAP
                                                 #   request
        Parent  => 'soem parent key->',          # The name of the parent followed by -> or empty
                                                 #   for root key items
    );

    returns:

    $ValueCheck = {
        Success => 1,                            # if everything is OK
    }

    $ValueCheck = {
        ErrorCode    => 'Function.Error',        # if error
        ErrorMessage => 'Error description',
    }

=cut

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

    my $Parent  = $Param{Parent};
    my $ItemKey = $Param{ItemKey};

    if (
        defined $Param{Input}->{Required} && $Param{Input}->{Required} && !$Param{Value}
        )
    {
        return {
            ErrorCode => "$Self->{OperationName}.MissingParameter",
            ErrorMessage =>
                "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter value"
                . " is required and is missing!",
        };
    }

    if ( $Param{Input}->{Type} eq 'Text' || $Param{Input}->{Type} eq 'TextArea' ) {

        # run Text validations
        if ( !$Self->ValidateInputText(%Param) ) {
            return {
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter value"
                    . " excedes the maxium length!",
            };
        }
    }
    elsif ( $Param{Input}->{Type} eq 'Date' ) {

        # run Date validations
        if ( !$Self->ValidateInputDate(%Param) ) {
            return {
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter value"
                    . " is not a valid Date format!",
            };
        }
    }
    elsif ( $Param{Input}->{Type} eq 'DateTime' ) {

        # run DateTime validations
        if ( !$Self->ValidateInputDateTime(%Param) ) {
            return {
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter value"
                    . " is not a valid DateTime format!",
            };
        }
    }
    elsif ( $Param{Input}->{Type} eq 'Customer' ) {

        # run Customer validations
        if ( !$Self->ValidateInputCustomer(%Param) ) {
            return {
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter value"
                    . " is not a valid customer!",
            };
        }
    }
    elsif ( $Param{Input}->{Type} eq 'CustomerCompany' ) {

        # run CustomerCompany validations
        if ( !$Self->ValidateInputCustomerCompany(%Param) ) {
            return {
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter value"
                    . " is not a valid customer company!",
            };
        }
    }
    elsif ( $Param{Input}->{Type} eq 'Integer' ) {

        # run Integer validations
        if ( !$Self->ValidateInputInteger(%Param) ) {
            return {
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter value"
                    . " is not a valid Integer or out of range!",
            };
        }
    }
    elsif ( $Param{Input}->{Type} eq 'GeneralCatalog' ) {

        # run General Catalog validations
        if ( !$Self->ValidateInputGeneralCatalog(%Param) ) {
            return {
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter value"
                    . " is not a valid for General Catalog '$Param{Input}->{Class}'!",
            };
        }
    }
    else {

        # The type is dummy, do nothing
    }

    return {
        Success => 1,
    };
}

=head2 _ReplaceValue()

replace user values with system ready values.

    my $NewValue = $CommonObject->_ReplaceValue(
        Value   => $Value                        # $Value could be a string, a time stamp,
                                                 #   general catalog class name, or a integer
        Input   => $InputDefinitionHashRef,      # The definition of the element input extracted
                                                 #   from the Configuration Item definition for
                                                 #   for each value
                                                 #   for root key items
    );

    returns:

    $NewValue = $ANewValue

=cut

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

    # set the list of input types that needs to be replaced
    my %ReplaceInputTypes = (
        Date           => 1,
        DateTime       => 1,
        GeneralCatalog => 1,
    );
    if ( !$ReplaceInputTypes{ $Param{Input}->{Type} } ) {
        return $Param{Value};
    }

    my $NewValue;

    if ( $Param{Input}->{Type} eq 'Date' ) {

        # run Date replace
        $NewValue = $Self->ReplaceInputDate(%Param);
    }
    elsif ( $Param{Input}->{Type} eq 'DateTime' ) {

        # run DateTime replace
        $NewValue = $Self->ReplaceInputDateTime(%Param);
    }
    else {
        # run General Catalog replace
        $NewValue = $Self->ReplaceInputGeneralCatalog(%Param);
    }

    return $NewValue;
}

=head2 _InvertReplaceValue()

replace internal values with user values.

    my $NewValue = $OperationObject->_InvertReplaceValue(
        Value   => $Value                        # $Value could be a string, a time stamp,
                                                 #   general catalog class name, or a integer
        Input   => $InputDefinitionHashRef,      # The definition of the element input extracted
                                                 #   from the Configuration Item definition for
                                                 #   for each value
                                                 #   for root key items
    );

    returns:

    $NewValue = $ANewValue

=cut

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

    # set the list of input types that needs to be replaced
    my %ReplaceInputTypes = (
        Date           => 1,
        GeneralCatalog => 1,
    );

    if ( !$ReplaceInputTypes{ $Param{Input}->{Type} } ) {
        return $Param{Value};
    }

    my $NewValue;

    if ( $Param{Input}->{Type} eq 'Date' ) {

        # run Date replace
        $NewValue = $Self->InvertReplaceInputDate(%Param);
    }
    else {
        # run General Catalog replace
        $NewValue = $Self->InvertReplaceInputGeneralCatalog(%Param);
    }

    return $NewValue;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::GenericInterface::Operation::ConfigItem::ConfigItemCreate;

use strict;
use warnings;

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

use parent qw(
    Kernel::GenericInterface::Operation::Common
    Kernel::GenericInterface::Operation::ConfigItem::Common
);

our $ObjectManagerDisabled = 1;

=head1 NAME

Kernel::GenericInterface::Operation::ConfigItem::ConfigItemCreate - GenericInterface ConfigItem ConfigItemCreate Operation backend

=head1 PUBLIC INTERFACE

=head2 new()

usually, you want to create an instance of this
by using Kernel::GenericInterface::Operation->new();

=cut

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

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

    # check needed objects
    for my $Needed (qw( DebuggerObject WebserviceID )) {
        if ( !$Param{$Needed} ) {
            return {
                Success      => 0,
                ErrorMessage => "Got no $Needed!",
            };
        }

        $Self->{$Needed} = $Param{$Needed};
    }

    $Self->{OperationName} = 'ConfigItemCreate';

    $Self->{Config} = $Kernel::OM->Get('Kernel::Config')->Get('GenericInterface::Operation::ConfigItemCreate');

    $Self->{Config}->{DefaultValue} = 'Not Defined';

    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

    # get a list of all config item classes
    $Self->{ClassList} = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    if ( !IsHashRefWithData( $Self->{ClassList} ) ) {
        return $Self->{DebuggerObject}->Error(
            Summary => 'Error when trying to get class listing of ITSM::ConfigItem::Class',
        );
    }

    # get a list of all incistates
    $Self->{InciStateList} = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::Core::IncidentState',
    );

    if ( !IsHashRefWithData( $Self->{InciStateList} ) ) {
        return $Self->{DebuggerObject}->Error(
            Summary => 'Error when trying to get incident state listing of'
                . ' ITSM::Core::IncidentState',
        );
    }

    # get a list of all deplstates
    $Self->{DeplStateList} = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    if ( !IsHashRefWithData( $Self->{DeplStateList} ) ) {
        return $Self->{DebuggerObject}->Error(
            Summary => 'Error when trying to get incident state listing of'
                . ' ITSM::ConfigItem::DeploymentState',
        );
    }

    # also provide the classlist in reversed form for easier reverse lookups
    my %ReverseClassList = reverse %{ $Self->{ClassList} };
    $Self->{ReverseClassList} = \%ReverseClassList;

    # also provide the incistatelist in reversed form for easier reverse lookups
    my %ReverseInciStateList = reverse %{ $Self->{InciStateList} };
    $Self->{ReverseInciStateList} = \%ReverseInciStateList;

    # also provide the deplstatelist in reversed form for easier reverse lookups
    my %ReverseDeplStateList = reverse %{ $Self->{DeplStateList} };
    $Self->{ReverseDeplStateList} = \%ReverseDeplStateList;

    return $Self;
}

=head2 Run()

perform ConfigItemCreate Operation. This will return the created config item number.

    my $Result = $OperationObject->Run(
        Data => {
            UserLogin         => 'some agent login',        # UserLogin or SessionID is
                                                            #   required
            SessionID         => 123,

            Password  => 'some password',                   # if UserLogin is sent then
                                                            #   Password is required
            ConfigItem => {
                Number       => '111',                      # optional
                Class        => 'Config Item Class',
                Name         => 'The Name',
                DeplState    => 'deployment state',
                InciState    => 'incident state',
                CIXMLData    => $ArrayHashRef,              # it depends on the Configuration Item class and definition

                Attachment => [
                    {
                        Content     => 'content'            # base64 encoded
                        ContentType => 'some content type'
                        Filename    => 'some fine name'
                    },
                    # ...
                ],
                #or
                #Attachment => {
                #    Content     => 'content'
                #    ContentType => 'some content type'
                #    Filename    => 'some fine name'
                #},
            },
        },
    );

    $Result = {
        Success         => 1,                       # 0 or 1
        ErrorMessage    => '',                      # in case of error
        Data            => {                        # result data payload after Operation
            ConfigItemID => 123,                    # Configuration Item  ID number in OTRS::ITSM (Service desk system)
            Number       => 2324454323322           # Configuration Item  Number in OTRS::ITSM (Service desk system)
            Error => {                              # should not return errors
                    ErrorCode    => 'ConfigItemCreate.ErrorCode'
                    ErrorMessage => 'Error Description'
            },
        },
    };

=cut

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

    my $Result = $Self->Init(
        WebserviceID => $Self->{WebserviceID},
    );

    if ( !$Result->{Success} ) {

        $Self->ReturnError(
            ErrorCode    => 'Webservice.InvalidConfiguration',
            ErrorMessage => $Result->{ErrorMessage},
        );
    }

    # check needed stuff
    if (
        !$Param{Data}->{UserLogin}
        && !$Param{Data}->{SessionID}
        )
    {
        return $Self->ReturnError(
            ErrorCode => "$Self->{OperationName}.MissingParameter",
            ErrorMessage =>
                "$Self->{OperationName}: UserLogin or SessionID is required!",
        );
    }

    if ( $Param{Data}->{UserLogin} ) {

        if ( !$Param{Data}->{Password} )
        {
            return $Self->ReturnError(
                ErrorCode    => "$Self->{OperationName}.MissingParameter",
                ErrorMessage => "$Self->{OperationName}: Password or SessionID is required!",
            );
        }
    }

    # authenticate user
    my ( $UserID, $UserType ) = $Self->Auth(%Param);

    if ( !$UserID ) {
        return $Self->ReturnError(
            ErrorCode    => "$Self->{OperationName}.AuthFail",
            ErrorMessage => "$Self->{OperationName}: User could not be authenticated!",
        );
    }

    # check needed hashes
    for my $Needed (qw(ConfigItem)) {
        if ( !IsHashRefWithData( $Param{Data}->{$Needed} ) ) {
            return $Self->ReturnError(
                ErrorCode => "$Self->{OperationName}.MissingParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: $Needed parameter is missing or not valid!",
            );
        }
    }

    # isolate config item parameter
    my $ConfigItem = $Param{Data}->{ConfigItem};

    # remove leading and trailing spaces
    for my $Attribute ( sort keys %{$ConfigItem} ) {
        if ( ref $Attribute ne 'HASH' && ref $Attribute ne 'ARRAY' ) {

            #remove leading spaces
            $ConfigItem->{$Attribute} =~ s{\A\s+}{};

            #remove trailing spaces
            $ConfigItem->{$Attribute} =~ s{\s+\z}{};
        }
    }

    # CIXMLData is not mandatory, but it must be HashRef if exists.
    if ( defined $ConfigItem->{CIXMLData} ) {
        if ( !IsHashRefWithData( $ConfigItem->{CIXMLData} ) ) {
            return $Self->ReturnError(
                ErrorCode    => "$Self->{OperationName}.MissingParameter",
                ErrorMessage => "$Self->{OperationName}: ConfigItem->CIXMLData is missing or invalid!",
            );
        }

        # Remove leading and trailing spaces for CIXMLData (must be HashRef).
        $Self->_CleanXMLData( XMLData => $ConfigItem->{CIXMLData} );
    }

    # check ConfigItem attribute values
    my $ConfigItemCheck = $Self->_CheckConfigItem( ConfigItem => $ConfigItem );

    if ( !$ConfigItemCheck->{Success} ) {
        return $Self->ReturnError( %{$ConfigItemCheck} );
    }

    # check create permissions
    my $Permission = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->Permission(
        Scope   => 'Class',
        ClassID => $Self->{ReverseClassList}->{ $ConfigItem->{Class} },
        UserID  => $UserID,
        Type    => $Self->{Config}->{Permission},
    );

    if ( !$Permission ) {
        return $Self->ReturnError(
            ErrorCode    => "$Self->{OperationName}.AccessDenied",
            ErrorMessage => "$Self->{OperationName}: Can not create configuration items!",
        );
    }

    my $Attachment;
    my @AttachmentList;

    if ( defined $Param{Data}->{ConfigItem}->{Attachment} ) {

        # isolate Attachment parameter
        $Attachment = delete $Param{Data}->{ConfigItem}->{Attachment};

        # homologate imput to array
        if ( IsHashRefWithData($Attachment) ) {
            push @AttachmentList, $Attachment;
        }
        elsif ( IsArrayRefWithData($Attachment) ) {
            @AttachmentList = @{$Attachment};
        }
        else {
            return $Self->ReturnError(
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->Attachment parameter is invalid!",
            );
        }

        # check Attachment internal structure
        for my $AttachmentItem (@AttachmentList) {
            if ( !IsHashRefWithData($AttachmentItem) ) {
                return $Self->ReturnError(
                    ErrorCode => "$Self->{OperationName}.InvalidParameter",
                    ErrorMessage =>
                        "$Self->{OperationName}: ConfigItem->Attachment parameter is invalid!",
                );
            }

            # remove leading and trailing spaces
            for my $Attribute ( sort keys %{$AttachmentItem} ) {
                if ( ref $Attribute ne 'HASH' && ref $Attribute ne 'ARRAY' ) {

                    #remove leading spaces
                    $AttachmentItem->{$Attribute} =~ s{\A\s+}{};

                    #remove trailing spaces
                    $AttachmentItem->{$Attribute} =~ s{\s+\z}{};
                }
            }

            # check Attachment attribute values
            my $AttachmentCheck = $Self->_CheckAttachment( Attachment => $AttachmentItem );

            if ( !$AttachmentCheck->{Success} ) {
                return $Self->ReturnError( %{$AttachmentCheck} );
            }
        }
    }

    return $Self->_ConfigItemCreate(
        ConfigItem     => $ConfigItem,
        AttachmentList => \@AttachmentList,
        UserID         => $UserID,
    );
}

=head1 INTERNAL INTERFACE

=head2 _CleanXMLData()

removed trailing and leading white spaces in the XMLData.

    my $XMLDataClean = $OperationObject->_CleanXMLData(
        Definition => $DefinitionArrayRef,          # Config Item Definition ot just part of it
        XMLData    => $XMLDataHashRef,
    );

    returns:

    $XMLDataClean = {
        Success => 1,                               # if everything is OK
    }

    $XMLDataClean = {
        ErrorCode    => 'Function.Error',           # if error
        ErrorMessage => 'Error description',
    }

=cut

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

    my $XMLData = $Param{XMLData};

    KEY:
    for my $Key ( sort keys %{$XMLData} ) {
        if ( ref $XMLData->{$Key} eq 'ARRAY' ) {
            ELEMENT:
            for my $Element ( @{ $XMLData->{$Key} } ) {
                if ( ref $Element eq 'HASH' ) {

                    # start recursion
                    $Self->_CleanXMLData( XMLData => $Element );
                    next ELEMENT;
                }
                elsif ( ref $Element eq '' ) {

                    #remove leading spaces
                    $Element =~ s{\A\s+}{};

                    #remove trailing spaces
                    $Element =~ s{\s+\z}{};
                }
            }
        }
        elsif ( ref $XMLData->{$Key} eq 'HASH' ) {

            # start recursion
            $Self->_CleanXMLData( XMLData => $XMLData->{$Key} );
            next KEY;

        }
        elsif ( ref $XMLData->{$Key} eq '' ) {

            #remove leading spaces
            $XMLData->{$Key} =~ s{\A\s+}{};

            #remove trailing spaces
            $XMLData->{$Key} =~ s{\s+\z}{};
        }
    }

    return 1;
}

=head2 _CheckConfigItem()

checks if the given config item parameters are valid.

    my $ConfigItemCheck = $OperationObject->_CheckConfigItem(
        ConfigItem => $ConfigItem,                  # all config item parameters
    );

    returns:

    $ConfigItemCheck = {
        Success => 1,                               # if everything is OK
    }

    $ConfigItemCheck = {
        ErrorCode    => 'Function.Error',           # if error
        ErrorMessage => 'Error description',
    }

=cut

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

    my $ConfigItem = $Param{ConfigItem};

    # check config item internally
    for my $Needed (qw(Class Name DeplState InciState)) {
        if ( !$ConfigItem->{$Needed} ) {
            return {
                ErrorCode    => "$Self->{OperationName}.MissingParameter",
                ErrorMessage => "$Self->{OperationName}: ConfigItem->$Needed parameter is missing!",
            };
        }
    }

    # check ConfigItem->Class
    if ( !$Self->ValidateClass( %{$ConfigItem} ) ) {
        return {
            ErrorCode => "$Self->{OperationName}.InvalidParameter",
            ErrorMessage =>
                "$Self->{OperationName}: ConfigItem->Class parameter is invalid!",
        };
    }

    # check ConfigItem->DeplState
    if ( !$Self->ValidateDeplState( %{$ConfigItem} ) ) {
        return {
            ErrorCode => "$Self->{OperationName}.InvalidParameter",
            ErrorMessage =>
                "$Self->{OperationName}: ConfigItem->DeplState parameter is invalid!",
        };
    }

    # check ConfigItem->DeplState
    if ( !$Self->ValidateInciState( %{$ConfigItem} ) ) {
        return {
            ErrorCode => "$Self->{OperationName}.InvalidParameter",
            ErrorMessage =>
                "$Self->{OperationName}: ConfigItem->InciState parameter is invalid!",
        };
    }

    # get last config item defintion
    my $DefinitionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->DefinitionGet(
        ClassID => $Self->{ReverseClassList}->{ $ConfigItem->{Class} },
    );

    my $XMLDataCheckResult = $Self->CheckXMLData(
        Definition => $DefinitionData->{DefinitionRef},
        XMLData    => $ConfigItem->{CIXMLData},
    );

    if ( !$XMLDataCheckResult->{Success} ) {
        return $XMLDataCheckResult;
    }

    # if everything is OK then return Success
    return {
        Success => 1,
    };
}

=head2 _CheckAttachment()

checks if the given attachment parameter is valid.

    my $AttachmentCheck = $OperationObject->_CheckAttachment(
        Attachment => $Attachment,                  # all attachment parameters
    );

    returns:

    $AttachmentCheck = {
        Success => 1,                               # if everething is OK
    }

    $AttachmentCheck = {
        ErrorCode    => 'Function.Error',           # if error
        ErrorMessage => 'Error description',
    }

=cut

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

    my $Attachment = $Param{Attachment};

    # check attachment item internally
    for my $Needed (qw(Content ContentType Filename)) {
        if ( !$Attachment->{$Needed} ) {
            return {
                ErrorCode => "$Self->{OperationName}.MissingParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: Attachment->$Needed parameter is missing!",
            };
        }
    }

    # check Article->ContentType
    if ( $Attachment->{ContentType} ) {

        $Attachment->{ContentType} = lc $Attachment->{ContentType};

        # check Charset part
        my $Charset = '';
        if ( $Attachment->{ContentType} =~ /charset=/i ) {
            $Charset = $Attachment->{ContentType};
            $Charset =~ s/.+?charset=("|'|)(\w+)/$2/gi;
            $Charset =~ s/"|'//g;
            $Charset =~ s/(.+?);.*/$1/g;
        }

        if ( $Charset && !$Self->ValidateCharset( Charset => $Charset ) )
        {
            return {
                ErrorCode    => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage => "$Self->{OperationName}: Attachment->ContentType is invalid!",
            };
        }

        # check MimeType part
        my $MimeType = '';
        if ( $Attachment->{ContentType} =~ /^(\w+\/\w+)/i ) {
            $MimeType = $1;
            $MimeType =~ s/"|'//g;
        }

        if ( !$Self->ValidateMimeType( MimeType => $MimeType ) ) {
            return {
                ErrorCode    => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage => "$Self->{OperationName}: Attachment->ContentType is invalid!",
            };
        }
    }

    # if everything is OK then return Success
    return {
        Success => 1,
    };
}

=head2 _ConfigItemCreate()

creates a configuration item with attachments if specified.

    my $Response = $OperationObject->_ConfigItemCreate(
        ConfigItem     => $ConfigItem,             # all configuration item parameters
        AttachmentList => $Attachment,             # a list of all attachments
        UserID         => 123,
    );

    returns:

    $Response = {
        Success => 1,                               # if everething is OK
        Data => {
            ConfigItemID => 123,
            ConfigItemNumber => 'CN123',
        }
    }

    $Response = {
        Success      => 0,                         # if unexpected error
        ErrorMessage => "$Param{ErrorCode}: $Param{ErrorMessage}",
    }

=cut

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

    my $ConfigItem     = $Param{ConfigItem};
    my $AttachmentList = $Param{AttachmentList};

    my $DeplStateID = $Self->{ReverseDeplStateList}->{ $ConfigItem->{DeplState} };
    my $InciStateID = $Self->{ReverseInciStateList}->{ $ConfigItem->{InciState} };

    my $RawXMLData = $ConfigItem->{CIXMLData};

    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    # get last config item defintion
    my $DefinitionData = $ConfigItemObject->DefinitionGet(
        ClassID => $Self->{ReverseClassList}->{ $ConfigItem->{Class} },
    );

    # replace date, date time, customer, company and general catalog values
    my $ReplacedXMLData = $Self->ReplaceXMLData(
        XMLData    => $RawXMLData,
        Definition => $DefinitionData->{DefinitionRef},
    );

    # create an XMLData structure suitable for VersionAdd
    my $XMLData = $Self->FormatXMLData(
        XMLData => $ReplacedXMLData,
    );

    # create new config item
    my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        Number  => $ConfigItem->{Number},
        ClassID => $Self->{ReverseClassList}->{ $ConfigItem->{Class} },
        UserID  => $Param{UserID},
    );

    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ConfigItemID,
        Name         => $ConfigItem->{Name},
        DefinitionID => $DefinitionData->{DefinitionID},
        DeplStateID  => $DeplStateID,
        InciStateID  => $InciStateID,
        XMLData      => $XMLData,
        UserID       => $Param{UserID},
    );

    if ( !$ConfigItemID && !$VersionID ) {
        return {
            Success      => 0,
            ErrorMessage => 'Configuration Item could not be created, please contact the system'
                . 'administrator',
        };
    }

    # set attachments
    if ( IsArrayRefWithData($AttachmentList) ) {

        for my $Attachment ( @{$AttachmentList} ) {
            my $Result = $Self->CreateAttachment(
                Attachment   => $Attachment,
                ConfigItemID => $ConfigItemID,
                UserID       => $Param{UserID},
            );

            if ( !$Result->{Success} ) {
                my $ErrorMessage =
                    $Result->{ErrorMessage} || "Attachment could not be created, please contact"
                    . " the system administrator";

                return {
                    Success      => 0,
                    ErrorMessage => $ErrorMessage,
                };
            }
        }
    }

    # get ConfigItem data
    my $ConfigItemData = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => $ConfigItemID,
    );

    if ( !IsHashRefWithData($ConfigItemData) ) {
        return {
            Success      => 0,
            ErrorMessage => 'Could not get new configuration item information, please contact the'
                . ' system administrator',
        };
    }

    return {
        Success => 1,
        Data    => {
            ConfigItemID => $ConfigItemID,
            Number       => $ConfigItemData->{Number},
        },
    };
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::GenericInterface::Operation::ConfigItem::ConfigItemDelete;

use strict;
use warnings;

use MIME::Base64;
use Kernel::System::VariableCheck qw(:all);

use parent qw(
    Kernel::GenericInterface::Operation::Common
    Kernel::GenericInterface::Operation::ConfigItem::Common
);

our $ObjectManagerDisabled = 1;

=head1 NAME

Kernel::GenericInterface::Operation::ConfigItem::ConfigItemDelete - GenericInterface Configuration Item Delete Operation backend

=head1 PUBLIC INTERFACE

=head2 new()

usually, you want to create an instance of this
by using Kernel::GenericInterface::Operation->new();

=cut

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

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

    # check needed objects
    for my $Needed (qw(DebuggerObject WebserviceID)) {
        if ( !$Param{$Needed} ) {
            return {
                Success      => 0,
                ErrorMessage => "Got no $Needed!",
            };
        }

        $Self->{$Needed} = $Param{$Needed};
    }

    $Self->{OperationName} = 'ConfigItemDelete';

    $Self->{Config} = $Kernel::OM->Get('Kernel::Config')->Get('GenericInterface::Operation::ConfigItemDelete');

    return $Self;
}

=head2 Run()

perform ConfigItemDelete Operation. This function is able to return
one or more ConfigItem entries in one call.

    my $Result = $OperationObject->Run(
        Data => {
            UserLogin         => 'some agent login',                            # UserLogin or CustomerUserLogin or SessionID is
                                                                                #   required
            CustomerUserLogin => 'some customer login',
            SessionID         => 123,

            Password          => 'some password',                               # if UserLogin or customerUserLogin is sent then
                                                                                #   Password is required
            ConfigItemID      => '32,33',                                       # required, could be coma separated IDs or an Array
        },
    );

    $Result = {
        Success         => 1,                       # 0 or 1
        ErrorMessage    => '',                      # in case of error
        Data            => {                        # result data payload after Operation
            ConfigItemID => [123, 456],         # Configuration Item IDs number in OTRS::ITSM (Service desk system)
            Error => {                              # should not return errors
                    ErrorCode    => 'ConfigItemDelete.ErrorCode'
                    ErrorMessage => 'Error Description'
            },
        },
    };

=cut

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

    my $Result = $Self->Init(
        WebserviceID => $Self->{WebserviceID},
    );

    if ( !$Result->{Success} ) {
        $Self->ReturnError(
            ErrorCode    => 'Webservice.InvalidConfiguration',
            ErrorMessage => $Result->{ErrorMessage},
        );
    }

    my ( $UserID, $UserType ) = $Self->Auth(
        %Param
    );

    if ( !$UserID ) {
        return $Self->ReturnError(
            ErrorCode    => '$Self->{OperationName}.AuthFail',
            ErrorMessage => "$Self->{OperationName}: Authorization failing!",
        );
    }

    # check needed stuff
    for my $Needed (qw(ConfigItemID)) {
        if ( !$Param{Data}->{$Needed} ) {
            return $Self->ReturnError(
                ErrorCode    => "$Self->{OperationName}.MissingParameter",
                ErrorMessage => "$Self->{OperationName}: $Needed parameter is missing!",
            );
        }
    }
    my $ErrorMessage = '';

    # all needed variables
    my @ConfigItemIDs;
    if ( IsStringWithData( $Param{Data}->{ConfigItemID} ) ) {
        @ConfigItemIDs = split /\s*,\s*/, $Param{Data}->{ConfigItemID};
    }
    elsif ( IsArrayRefWithData( $Param{Data}->{ConfigItemID} ) ) {
        @ConfigItemIDs = @{ $Param{Data}->{ConfigItemID} };
    }
    else {
        return $Self->ReturnError(
            ErrorCode    => "$Self->{OperationName}.WrongStructure",
            ErrorMessage => "$Self->{OperationName}: Structure for ConfigItemID is not correct!",
        );
    }

    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    my @DeletedConfigItemIDs;

    # start ConfigItem loop
    CONFIGITEM:
    for my $ConfigItemID (@ConfigItemIDs) {

        # check create permissions
        my $Permission = $ConfigItemObject->Permission(
            Scope  => 'Item',
            ItemID => $ConfigItemID,
            UserID => $UserID,
            Type   => $Self->{Config}->{Permission},
        );

        if ( !$Permission ) {
            return $Self->ReturnError(
                ErrorCode    => "$Self->{OperationName}.AccessDenied",
                ErrorMessage => "$Self->{OperationName}: Can not delete configuration item!",
            );
        }

        # delete the configitem
        my $DeleteSuccess = $ConfigItemObject->ConfigItemDelete(
            ConfigItemID => $ConfigItemID,
            UserID       => $UserID,
        );

        if ( !$DeleteSuccess ) {

            $ErrorMessage = 'Could not delete ConfigItem ID ' . $ConfigItemID
                . ' in Kernel::GenericInterface::Operation::ConfigItem::ConfigItemDelete::Run()';

            return $Self->ReturnError(
                ErrorCode    => '$Self->{OperationName}.DeleteError',
                ErrorMessage => "$Self->{OperationName}: $ErrorMessage",
            );
        }

        push @DeletedConfigItemIDs, $ConfigItemID;

    }    # finish ConfigItem loop

    if ( !IsArrayRefWithData( \@DeletedConfigItemIDs ) ) {
        return {
            Success      => 0,
            ErrorMessage => 'Could not delete ConfigItems!',
        };
    }

    return {
        Success => 1,
        Data    => {
            ConfigItemID => \@DeletedConfigItemIDs,
        },
    };
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::GenericInterface::Operation::ConfigItem::ConfigItemGet;

use strict;
use warnings;

use MIME::Base64;
use Kernel::System::VariableCheck qw(:all);

use parent qw(
    Kernel::GenericInterface::Operation::Common
    Kernel::GenericInterface::Operation::ConfigItem::Common
);

our $ObjectManagerDisabled = 1;

=head1 NAME

Kernel::GenericInterface::Operation::ConfigItem::ConfigItemGet - GenericInterface Configuration Item Get Operation backend

=head1 PUBLIC INTERFACE

=head2 new()

usually, you want to create an instance of this
by using Kernel::GenericInterface::Operation->new();

=cut

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

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

    # check needed objects
    for my $Needed (qw( DebuggerObject WebserviceID )) {
        if ( !$Param{$Needed} ) {
            return {
                Success      => 0,
                ErrorMessage => "Got no $Needed!",
            };
        }

        $Self->{$Needed} = $Param{$Needed};
    }

    $Self->{OperationName} = 'ConfigItemGet';

    $Self->{Config} = $Kernel::OM->Get('Kernel::Config')->Get('GenericInterface::Operation::ConfigItemGet');

    return $Self;
}

=head2 Run()

perform ConfigItemGet Operation. This function is able to return
one or more ConfigItem entries in one call.

    my $Result = $OperationObject->Run(
        Data => {
            UserLogin         => 'some agent login',                            # UserLogin or SessionID is
            SessionID         => 123,                                           #   required
            Password          => 'some password',                               # if UserLogin is sent then Password is required
            ConfigItemID      => '32,33',                                       # required, could be coma separated IDs or an Array
            Attachments       => 1,                                             # Optional, 1 as default. If it's set with the value 1,
                                                                                # attachments for articles will be included on ConfigItem data
        },
    );

    $Result = {
        Success      => 1,                                # 0 or 1
        ErrorMessage => '',                               # In case of an error
        Data         => {
            ConfigItem => [
                {

                    Number             => '20101027000001',
                    ConfigItemID       => 123,
                    Name               => 'some name',
                    Class              => 'some class',
                    VersionID          => 123,
                    LastVersionID      => 123,
                    DefinitionID       => 123,
                    InciState          => 'some incident state',
                    InciStateType      => 'some incident state type',
                    DeplState          => 'some deployment state',
                    DeplStateType      => 'some deployment state type',
                    CurInciState       => 'some incident state',
                    CurInciStateType   => 'some incident state type',
                    CurDeplState       => 'some deployment state',
                    CurDeplStateType   => 'some deployment state type',
                    CreateTime         => '2010-10-27 20:15:00'
                    CreateBy           => 123,
                    CIXMLData          => $XMLDataHashRef,

                    Attachment => [
                        {
                            Content            => "xxxx",     # actual attachment contents, base64 enconded
                            ContentType        => "application/pdf",
                            Filename           => "StdAttachment-Test1.pdf",
                            Filesize           => "4.6 KBytes",
                            Preferences        => $PreferencesHashRef,
                        },
                        {
                           # . . .
                        },
                    ],
                },
                {
                    # . . .
                },
            ],
        },
    };

=cut

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

    my $Result = $Self->Init(
        WebserviceID => $Self->{WebserviceID},
    );

    if ( !$Result->{Success} ) {
        $Self->ReturnError(
            ErrorCode    => 'Webservice.InvalidConfiguration',
            ErrorMessage => $Result->{ErrorMessage},
        );
    }

    my ( $UserID, $UserType ) = $Self->Auth(
        %Param
    );

    if ( !$UserID ) {
        return $Self->ReturnError(
            ErrorCode    => '$Self->{OperationName}.AuthFail',
            ErrorMessage => "$Self->{OperationName}: Authorization failing!",
        );
    }

    # check needed stuff
    for my $Needed (qw(ConfigItemID)) {
        if ( !$Param{Data}->{$Needed} ) {
            return $Self->ReturnError(
                ErrorCode    => "$Self->{OperationName}.MissingParameter",
                ErrorMessage => "$Self->{OperationName}: $Needed parameter is missing!",
            );
        }
    }
    my $ErrorMessage = '';

    # all needed variables
    my @ConfigItemIDs;
    if ( IsStringWithData( $Param{Data}->{ConfigItemID} ) ) {
        @ConfigItemIDs = split( /,/, $Param{Data}->{ConfigItemID} );
    }
    elsif ( IsArrayRefWithData( $Param{Data}->{ConfigItemID} ) ) {
        @ConfigItemIDs = @{ $Param{Data}->{ConfigItemID} };
    }
    else {
        return $Self->ReturnError(
            ErrorCode    => "$Self->{OperationName}.WrongStructure",
            ErrorMessage => "$Self->{OperationName}: Structure for ConfigItemID is not correct!",
        );
    }
    my $Attachments = $Param{Data}->{Attachments} || 0;
    my $ReturnData  = {
        Success => 1,
    };

    my @Item;

    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    # start ConfigItem loop
    CONFIGITEM:
    for my $ConfigItemID (@ConfigItemIDs) {

        # check create permissions
        my $Permission = $ConfigItemObject->Permission(
            Scope  => 'Item',
            ItemID => $ConfigItemID,
            UserID => $UserID,
            Type   => $Self->{Config}->{Permission},
        );

        if ( !$Permission ) {
            return $Self->ReturnError(
                ErrorCode    => "$Self->{OperationName}.AccessDenied",
                ErrorMessage => "$Self->{OperationName}: Can not get configuration item!",
            );
        }

        # get the ConfigItem entry
        my $ConfigItem = $ConfigItemObject->ConfigItemGet(
            ConfigItemID => $ConfigItemID,
            UserID       => $UserID,
        );

        # get latest version
        my $Version = $ConfigItemObject->VersionGet(
            ConfigItemID => $ConfigItemID,
            UserID       => $UserID,
        );

        if ( !IsHashRefWithData($Version) ) {

            $ErrorMessage = 'Could not get ConfigItem data'
                . ' in Kernel::GenericInterface::Operation::ConfigItem::ConfigItemGet::Run()';

            return $Self->ReturnError(
                ErrorCode    => '$Self->{OperationName}.InvalidParameter',
                ErrorMessage => "$Self->{OperationName}: $ErrorMessage",
            );
        }

        # remove unneeded items
        delete $Version->{ClassID};
        delete $Version->{CurDeplStateID};
        delete $Version->{CurInciStateID};
        delete $Version->{DeplStateID};
        delete $Version->{InciStateID};
        delete $Version->{XMLDefinitionID};

        my $Definition = delete $Version->{XMLDefinition};

        my $FormatedXMLData = $Self->InvertFormatXMLData(
            XMLData => $Version->{XMLData}->[1]->{Version},
        );

        my $ReplacedXMLData = $Self->InvertReplaceXMLData(
            XMLData    => $FormatedXMLData,
            Definition => $Definition,
        );

        $Version->{XMLData} = $ReplacedXMLData;

        # rename XMLData since SOAP transport complains about XML prefix on names
        $Version->{CIXMLData} = delete $Version->{XMLData};

        # set ConfigItem entry data
        my $ConfigItemBundle = $Version;

        if ($Attachments) {

            my @Attachments = $ConfigItemObject->ConfigItemAttachmentList(
                ConfigItemID => $ConfigItemID,
            );

            my @AttachmentDetails;
            ATTACHMENT:
            for my $Filename (@Attachments) {

                next ATTACHMENT if !$Filename;

                my $Attachment = $ConfigItemObject->ConfigItemAttachmentGet(
                    ConfigItemID => $ConfigItemID,
                    Filename     => $Filename,
                );

                # next if not attachment
                next ATTACHMENT if !IsHashRefWithData($Attachment);

                # convert content to base64
                $Attachment->{Content} = encode_base64( $Attachment->{Content} );
                push @AttachmentDetails, $Attachment;
            }

            # set ConfigItem entry data
            $ConfigItemBundle->{Attachment} = '';
            if ( IsArrayRefWithData( \@AttachmentDetails ) ) {
                $ConfigItemBundle->{Attachment} = \@AttachmentDetails;
            }
        }

        # add
        push @Item, $ConfigItemBundle;

    }    # finish ConfigItem loop

    if ( !scalar @Item ) {
        $ErrorMessage = 'Could not get ConfigItem data'
            . ' in Kernel::GenericInterface::Operation::ConfigItem::ConfigItemGet::Run()';

        return $Self->ReturnError(
            ErrorCode    => '$Self->{OperationName}.NoConfigItemData',
            ErrorMessage => "$Self->{OperationName}: $ErrorMessage",
        );

    }

    # set ConfigItem data into return structure
    $ReturnData->{Data}->{ConfigItem} = '';
    if ( IsArrayRefWithData( \@Item ) ) {
        $ReturnData->{Data}->{ConfigItem} = \@Item;
    }

    # return result
    return $ReturnData;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::GenericInterface::Operation::ConfigItem::ConfigItemSearch;

use strict;
use warnings;

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

use parent qw(
    Kernel::GenericInterface::Operation::Common
    Kernel::GenericInterface::Operation::ConfigItem::Common
);

our $ObjectManagerDisabled = 1;

=head1 NAME

Kernel::GenericInterface::Operation::ConfigItem::ConfigItemSearch - GenericInterface ConfigItem ConfigItemSearch Operation backend

=head1 PUBLIC INTERFACE

=head2 new()

usually, you want to create an instance of this
by using Kernel::GenericInterface::Operation->new();

=cut

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

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

    # check needed objects
    for my $Needed (qw( DebuggerObject WebserviceID )) {
        if ( !$Param{$Needed} ) {
            return {
                Success      => 0,
                ErrorMessage => "Got no $Needed!",
            };
        }

        $Self->{$Needed} = $Param{$Needed};
    }

    $Self->{OperationName} = 'ConfigItemSearch';

    $Self->{Config} = $Kernel::OM->Get('Kernel::Config')->Get('GenericInterface::Operation::ConfigItemSearch');

    $Self->{Config}->{DefaultValue} = 'Not Defined';

    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

    # get a list of all config item classes
    $Self->{ClassList} = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    if ( !IsHashRefWithData( $Self->{ClassList} ) ) {
        return $Self->{DebuggerObject}->Error(
            Summary => 'Error when trying to get class listing of ITSM::ConfigItem::Class',
        );
    }

    # get a list of all incistates
    $Self->{InciStateList} = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::Core::IncidentState',
    );

    if ( !IsHashRefWithData( $Self->{InciStateList} ) ) {
        return $Self->{DebuggerObject}->Error(
            Summary => 'Error when trying to get incident state listing of'
                . ' ITSM::Core::IncidentState',
        );
    }

    # get a list of all deplstates
    $Self->{DeplStateList} = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    if ( !IsHashRefWithData( $Self->{DeplStateList} ) ) {
        return $Self->{DebuggerObject}->Error(
            Summary => 'Error when trying to get incident state listing of'
                . ' ITSM::ConfigItem::DeploymentState',
        );
    }

    # also provide the classlist in reversed form for easier reverse lookups
    my %ReverseClassList = reverse %{ $Self->{ClassList} };
    $Self->{ReverseClassList} = \%ReverseClassList;

    # also provide the incistatelist in reversed form for easier reverse lookups
    my %ReverseInciStateList = reverse %{ $Self->{InciStateList} };
    $Self->{ReverseInciStateList} = \%ReverseInciStateList;

    # also provide the deplstatelist in reversed form for easier reverse lookups
    my %ReverseDeplStateList = reverse %{ $Self->{DeplStateList} };
    $Self->{ReverseDeplStateList} = \%ReverseDeplStateList;

    return $Self;
}

=head2 Run()

perform ConfigItemCreate Operation. This will return the created config item number.

    my $Result = $OperationObject->Run(
        Data => {
            UserLogin         => 'some agent login',                        # UserLogin or SessionID is
            SessionID         => 123,                                       #   required


            Password  => 'some password',                                   # if UserLogin is sent then
                                                                            #   Password is required
            ConfigItem => {
                Class        => 'The ConfigItem Class',                     # (optional)
                Number       => 'The ConfigItem Number',                    # (optional)
                Name         => 'The ConfigItem Name',                      # (optional)
                DeplStates   => ['deployment state'],                       # (optional)
                InciStates   => ['incident state'],                         # (optional)

                # config items with created time after ...
                ConfigItemCreateTimeNewerDate => '2006-01-09 00:00:01',     # (optional)

                # config items with created time before then ....
                ConfigItemCreateTimeOlderDate => '2006-01-19 23:59:59',     # (optional)

                # config items with changed time after ...
                ConfigItemChangeTimeNewerDate => '2006-01-09 00:00:01',     # (optional)

                # config items with changed time before then ....
                ConfigItemChangeTimeOlderDate => '2006-01-19 23:59:59',     # (optional)

                CIXMLData    => $ArrayHashRef,                              # (optional), it depends on the Configuration
                                                                            #   Item class and definition

                PreviousVersionSearch => 1,  # (optional) default 0 (0|1)

                OrderBy => [ 'ConfigItemID', 'Number' ],                    # (optional)
                # default: [ 'ConfigItemID' ]
                # (ConfigItemID, Number, ClassID, DeplStateID, InciStateID,
                # CreateTime, CreateBy, ChangeTime, ChangeBy)

                # 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)
                # default: [ 'Down' ]
                # (Down | Up)

                Limit          => 122,                                      # (optional)
            },
        },
    );

    $Result = {
        Success         => 1,                       # 0 or 1
        ErrorMessage    => '',                      # in case of error
        Data            => {                        # result data payload after Operation
            ConfigItemID     => [123, 456],         # Configuration Item  IDs number in OTRS::ITSM (Service desk system)
            Error => {                              # should not return errors
                    ErrorCode    => 'ConfigItemSearch.ErrorCode'
                    ErrorMessage => 'Error Description'
            },
        },
    };

=cut

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

    my $Result = $Self->Init(
        WebserviceID => $Self->{WebserviceID},
    );

    if ( !$Result->{Success} ) {
        $Self->ReturnError(
            ErrorCode    => 'Webservice.InvalidConfiguration',
            ErrorMessage => $Result->{ErrorMessage},
        );
    }

    # check needed stuff
    if (
        !$Param{Data}->{UserLogin}
        && !$Param{Data}->{SessionID}
        )
    {
        return $Self->ReturnError(
            ErrorCode => "$Self->{OperationName}.MissingParameter",
            ErrorMessage =>
                "$Self->{OperationName}: UserLogin or SessionID is required!",
        );
    }

    if ( $Param{Data}->{UserLogin} ) {

        if ( !$Param{Data}->{Password} )
        {
            return $Self->ReturnError(
                ErrorCode    => "$Self->{OperationName}.MissingParameter",
                ErrorMessage => "$Self->{OperationName}: Password or SessionID is required!",
            );
        }
    }

    # authenticate user
    my ( $UserID, $UserType ) = $Self->Auth(%Param);

    if ( !$UserID ) {
        return $Self->ReturnError(
            ErrorCode    => "$Self->{OperationName}.AuthFail",
            ErrorMessage => "$Self->{OperationName}: User could not be authenticated!",
        );
    }

    # check needed hashes
    for my $Needed (qw(ConfigItem)) {
        if ( !IsHashRefWithData( $Param{Data}->{$Needed} ) ) {
            return $Self->ReturnError(
                ErrorCode => "$Self->{OperationName}.MissingParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: $Needed parameter is missing or not valid!",
            );
        }
    }

    # isolate config item parameter
    my $ConfigItem = $Param{Data}->{ConfigItem};

    # remove leading and trailing spaces
    for my $Attribute ( sort keys %{$ConfigItem} ) {
        if ( ref $Attribute ne 'HASH' && ref $Attribute ne 'ARRAY' ) {

            #remove leading spaces
            $ConfigItem->{$Attribute} =~ s{\A\s+}{};

            #remove trailing spaces
            $ConfigItem->{$Attribute} =~ s{\s+\z}{};
        }
    }

    if ( defined $ConfigItem->{CIXMLData} ) {
        if ( !IsHashRefWithData( $ConfigItem->{CIXMLData} ) ) {
            return $Self->ReturnError(
                ErrorCode    => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage => "$Self->{OperationName}: ConfigItem->CIXMLData is invalid!",
            );
        }

        # remove leading and trailing spaces for CIXMLData
        $Self->_CleanXMLData( XMLData => $ConfigItem->{CIXMLData} );
    }

    if ( !( $ConfigItem->{Class} ) ) {
        return $Self->ReturnError(
            ErrorCode => "$Self->{OperationName}.MissingParameter",
            ErrorMessage =>
                "$Self->{OperationName}: ConfigItem->Class parameter is missing!",
        );
    }

    # convert search params to arrays
    if ( defined $ConfigItem->{InciStates} ) {
        my @InciStates;
        if ( IsStringWithData( $ConfigItem->{InciStates} ) ) {
            @InciStates = split( /,/, $ConfigItem->{InciStates} );
        }
        elsif ( IsArrayRefWithData( $ConfigItem->{InciStates} ) ) {
            @InciStates = @{ $ConfigItem->{InciStates} };
        }
        else {
            return $Self->ReturnError(
                ErrorCode => "$Self->{OperationName}.WrongStructure",
                ErrorMessage =>
                    "$Self->{OperationName}: Structure for ConfigItem->InciStates is not correct!",
            );
        }
        $ConfigItem->{InciStates} = \@InciStates;
    }

    if ( defined $ConfigItem->{DeplStates} ) {
        my @DeplStates;
        if ( IsStringWithData( $ConfigItem->{DeplStates} ) ) {
            @DeplStates = split( /,/, $ConfigItem->{DeplStates} );
        }
        elsif ( IsArrayRefWithData( $ConfigItem->{DeplStates} ) ) {
            @DeplStates = @{ $ConfigItem->{DeplStates} };
        }
        else {
            return $Self->ReturnError(
                ErrorCode    => "$Self->{OperationName}.WrongStructure",
                ErrorMessage => "$Self->{OperationName}: Structure for DeplStates is not correct!",
            );
        }
        $ConfigItem->{DeplStates} = \@DeplStates;
    }

    if ( defined $ConfigItem->{OrderBy} ) {
        my @OrderBy;
        if ( IsStringWithData( $ConfigItem->{OrderBy} ) ) {
            @OrderBy = split( /,/, $ConfigItem->{OrderBy} );
        }
        elsif ( IsArrayRefWithData( $ConfigItem->{OrderBy} ) ) {
            @OrderBy = @{ $ConfigItem->{OrderBy} };
        }
        else {
            return $Self->ReturnError(
                ErrorCode    => "$Self->{OperationName}.WrongStructure",
                ErrorMessage => "$Self->{OperationName}: Structure for OrderBy is not correct!",
            );
        }
        $ConfigItem->{OrderBy} = \@OrderBy;
    }

    if ( defined $ConfigItem->{OrderByDirection} ) {
        my @OrderByDirection;
        if ( IsStringWithData( $ConfigItem->{OrderByDirection} ) ) {
            @OrderByDirection = split( /,/, $ConfigItem->{OrderByDirection} );
        }
        elsif ( IsArrayRefWithData( $ConfigItem->{OrderByDirection} ) ) {
            @OrderByDirection = @{ $ConfigItem->{OrderByDirection} };
        }
        else {
            return $Self->ReturnError(
                ErrorCode => "$Self->{OperationName}.WrongStructure",
                ErrorMessage =>
                    "$Self->{OperationName}: Structure for OrderByDirection is not correct!",
            );
        }
        $ConfigItem->{OrderByDirection} = \@OrderByDirection;
    }

    # check ConfigItem attribute values
    my $ConfigItemCheck = $Self->_CheckConfigItem( ConfigItem => $ConfigItem );

    if ( !$ConfigItemCheck->{Success} ) {
        return $Self->ReturnError( %{$ConfigItemCheck} );
    }

    # check search permissions
    my $Permission = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->Permission(
        Scope   => 'Class',
        ClassID => $Self->{ReverseClassList}->{ $ConfigItem->{Class} },
        UserID  => $UserID,
        Type    => $Self->{Config}->{Permission},
    );

    if ( !$Permission ) {
        return $Self->ReturnError(
            ErrorCode    => "$Self->{OperationName}.AccessDenied",
            ErrorMessage => "$Self->{OperationName}: Can not search configuration items!",
        );
    }

    return $Self->_ConfigItemSearch(
        ConfigItem => $ConfigItem,
        UserID     => $UserID,
    );
}

=head1 INTERNAL INTERFACE

=head2 _CleanXMLData()

removed trailing and leading white spaces in the XMLData.

    my $XMLDataClean = $OperationObject->_CleanXMLData(
        Definition => $DefinitionArrayRef,          # Config Item Definition ot just part of it
        XMLData    => $XMLDataHashRef,
    );

    returns:

    $XMLDataClean = {
        Success => 1,                               # if everything is OK
    }

    $XMLDataClean = {
        ErrorCode    => 'Function.Error',           # if error
        ErrorMessage => 'Error description',
    }

=cut

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

    my $XMLData = $Param{XMLData};

    KEY:
    for my $Key ( sort keys %{$XMLData} ) {
        if ( ref $XMLData->{$Key} eq 'ARRAY' ) {
            ELEMENT:
            for my $Element ( @{ $XMLData->{$Key} } ) {
                if ( ref $Element eq 'HASH' ) {

                    # start recursion
                    $Self->_CleanXMLData( XMLData => $Element );
                    next ELEMENT;
                }
                elsif ( ref $Element eq '' ) {

                    #remove leading spaces
                    $Element =~ s{\A\s+}{};

                    #remove trailing spaces
                    $Element =~ s{\s+\z}{};
                }
            }
        }
        elsif ( ref $XMLData->{$Key} eq 'HASH' ) {

            # start recursion
            $Self->_CleanXMLData( XMLData => $XMLData->{$Key} );
            next KEY;

        }
        elsif ( ref $XMLData->{$Key} eq '' ) {

            #remove leading spaces
            $XMLData->{$Key} =~ s{\A\s+}{};

            #remove trailing spaces
            $XMLData->{$Key} =~ s{\s+\z}{};
        }
    }

    return 1;
}

=head2 _CheckConfigItem()

checks if the given config item parameters are valid.

    my $ConfigItemCheck = $OperationObject->_CheckConfigItem(
        ConfigItem => $ConfigItem,                  # all config item parameters
    );

    returns:

    $ConfigItemCheck = {
        Success => 1,                               # if everything is OK
    }

    $ConfigItemCheck = {
        ErrorCode    => 'Function.Error',           # if error
        ErrorMessage => 'Error description',
    }

=cut

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

    my $ConfigItem = $Param{ConfigItem};

    # check config item internally
    # check needed stuff
    for my $Needed (qw(Class)) {
        if ( !$ConfigItem->{$Needed} ) {
            return {
                ErrorCode    => "$Self->{OperationName}.MissingParameter",
                ErrorMessage => "$Self->{OperationName}: ConfigItem->$Needed parameter is missing!",
            };
        }
    }

    # check ConfigItem->Class
    if ( !$Self->ValidateClass( %{$ConfigItem} ) ) {
        return {
            ErrorCode => "$Self->{OperationName}.InvalidParameter",
            ErrorMessage =>
                "$Self->{OperationName}: ConfigItem->Class parameter is invalid!",
        };
    }

    # check optional stuff
    if ( IsArrayRefWithData( $ConfigItem->{InciStates} ) ) {

        for my $InciState ( @{ $ConfigItem->{InciStates} } ) {

            # check ConfigItem->InciStates
            if ( !$Self->ValidateInciState( InciState => $InciState ) ) {
                return {
                    ErrorCode => "$Self->{OperationName}.InvalidParameter",
                    ErrorMessage =>
                        "$Self->{OperationName}: ConfigItem->InciStates parameter is invalid!",
                };
            }
        }
    }

    if ( IsArrayRefWithData( $ConfigItem->{DeplStates} ) ) {

        for my $DeplState ( @{ $ConfigItem->{DeplStates} } ) {

            # check ConfigItem->InciStates
            if ( !$Self->ValidateDeplState( DeplState => $DeplState ) ) {
                return {
                    ErrorCode => "$Self->{OperationName}.InvalidParameter",
                    ErrorMessage =>
                        "$Self->{OperationName}: ConfigItem->DeplStates parameter is invalid!",
                };
            }
        }
    }

    for my $TimeParam (
        qw(CreateTimeNewerDate CreateTimeOlderDate ChangeTimeNewerDate ChangeTimeOlderDate)
        )
    {
        if ( defined $ConfigItem->{"ConfigItem$TimeParam"} ) {
            if (
                !$Self->ValidateInputDateTime(
                    Value => $ConfigItem->{"ConfigItem$TimeParam"},
                )
                )
            {
                return {
                    ErrorCode => "$Self->{OperationName}.InvalidParameter",
                    ErrorMessage =>
                        "$Self->{OperationName}: ConfigItem->ConfigItem$TimeParam parameter is invalid!",
                };
            }
        }
    }

    if ( defined $ConfigItem->{Limit} ) {
        if ( !IsNumber( $ConfigItem->{Limit} ) ) {
            return {
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->Limit parameter is invalid!",
            };
        }
    }

    if ( IsHashRefWithData( $ConfigItem->{CIXMLData} ) ) {

        # get last config item defintion
        my $DefinitionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->DefinitionGet(
            ClassID => $Self->{ReverseClassList}->{ $ConfigItem->{Class} },
        );

        my $XMLDataCheckResult = $Self->_CheckSearchXMLData(
            Definition => $DefinitionData->{DefinitionRef},
            XMLData    => $ConfigItem->{CIXMLData},
        );

        if ( !$XMLDataCheckResult->{Success} ) {
            return $XMLDataCheckResult;
        }
    }

    # if everything is OK then return Success
    return {
        Success => 1,
    };
}

=head2 _ConfigItemSearch()

search a configuration items.

    my $Response = $OperationObject->_ConfigItemSearch(
        ConfigItem     => $ConfigItem,             # all configuration item parameters
        UserID         => 123,
    );

    returns:

    $Response = {
        Success => 1,                               # if everething is OK
        Data => {
            ConfigItemIDs => [123, 456],
        }
    }

    $Response = {
        Success      => 0,                         # if unexpected error
        ErrorMessage => "$Param{ErrorCode}: $Param{ErrorMessage}",
    }

=cut

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

    my $ConfigItem = $Param{ConfigItem};

    my %SearchParams;

    # set search parameters that does not need any conversion
    for my $PlainParam (
        qw(
        Name Number PreviousVersionSearch OrderBy OrderByDirection Limit
        ConfigItemCreateTimeNewerDate ConfigItemCreateTimeOlderDate ConfigItemChangeTimeNewerDate
        ConfigItemChangeTimeOlderDate
        )
        )
    {
        if ( defined $ConfigItem->{$PlainParam} ) {
            $SearchParams{$PlainParam} = $ConfigItem->{$PlainParam};
        }
    }

    # set seach class
    my $ClassID = $Self->{ReverseClassList}->{ $ConfigItem->{Class} };
    $SearchParams{ClassIDs} = [$ClassID];

    # set search incident states
    if ( defined $ConfigItem->{InciStates} ) {
        my @InciStateIDs;
        for my $InciState ( @{ $ConfigItem->{InciStates} } ) {
            my $InciStateID = $Self->{ReverseInciStateList}->{$InciState};
            push @InciStateIDs, $InciStateID;
        }
        $SearchParams{InciStateIDs} = \@InciStateIDs;
    }

    # set search deployment states
    if ( defined $ConfigItem->{DeplStates} ) {
        my @DeplStateIDs;
        for my $DeplState ( @{ $ConfigItem->{DeplStates} } ) {
            my $DeplStateID = $Self->{ReverseDeplStateList}->{$DeplState};
            push @DeplStateIDs, $DeplStateID;
        }
        $SearchParams{DeplStateIDs} = \@DeplStateIDs;
    }

    my $RawXMLData = $ConfigItem->{CIXMLData};

    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    if ( IsHashRefWithData($RawXMLData) ) {

        # get last config item defintion
        my $DefinitionData = $ConfigItemObject->DefinitionGet(
            ClassID => $Self->{ReverseClassList}->{ $ConfigItem->{Class} },
        );

        # replace date, date time, customer, company and general catalog values
        my $ReplacedXMLData = $Self->ReplaceXMLData(
            XMLData    => $RawXMLData,
            Definition => $DefinitionData->{DefinitionRef},
        );

        # create an XMLData structure suitable for ConfigItemSearch
        my $XMLData = $Self->_FormatSearchXMLData(
            XMLData => $ReplacedXMLData,
        );

        if ( IsArrayRefWithData($XMLData) ) {
            $SearchParams{What} = $XMLData;
        }
    }

    $Self->{DebuggerObject}->Info(
        Summary => "$Self->{OperationName}: Search Parameters",
        Data    => \%SearchParams,
    );

    my $ConfigItemIDs = $ConfigItemObject->ConfigItemSearchExtended(%SearchParams);

    if ( IsArrayRefWithData($ConfigItemIDs) ) {
        return {
            Success => 1,
            Data    => {
                ConfigItemIDs => $ConfigItemIDs,
            },
        };
    }
    return {
        Success => 1,
        Data    => {
            ConfigItemIDs => '',
        },
    };
}

=head2 _CheckSearchXMLData()

checks if the given XMLData value are valid.

    my $XMLDataCheck = $CommonObject->_CheckSearchXMLData(
        Definition => $DefinitionArrayRef,          # Config Item Definition ot just part of it
        XMLData    => $XMLDataHashRef,
        Parent     => 'some parent',
    );

    returns:

    $XMLDataCheck = {
        Success => 1,                               # if everything is OK
    }

    $XMLDataCheck = {
        ErrorCode    => 'Function.Error',           # if error
        ErrorMessage => 'Error description',
    }

=cut

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

    my $Definition = $Param{Definition};
    my $XMLData    = $Param{XMLData};
    my $Parent     = $Param{Parent} || '';

    my $CheckValueResult;
    for my $DefItem ( @{$Definition} ) {
        my $ItemKey = $DefItem->{Key};

        if ( ref $XMLData->{$ItemKey} eq 'ARRAY' ) {
            for my $ArrayItem ( @{ $XMLData->{$ItemKey} } ) {
                if ( ref $ArrayItem eq 'HASH' ) {
                    $CheckValueResult = $Self->_CheckValue(
                        Value   => $ArrayItem->{$ItemKey},
                        Input   => $DefItem->{Input},
                        ItemKey => $ItemKey,
                        Parent  => $Parent,
                    );
                    if ( !$CheckValueResult->{Success} ) {
                        return $CheckValueResult;
                    }
                }
                elsif ( ref $ArrayItem eq '' ) {
                    $CheckValueResult = $Self->_CheckValue(
                        Value   => $ArrayItem,
                        Input   => $DefItem->{Input},
                        ItemKey => $ItemKey,
                        Parent  => $Parent,
                    );
                    if ( !$CheckValueResult->{Success} ) {
                        return $CheckValueResult;
                    }
                }
                else {
                    return {
                        ErrorCode => "$Self->{OperationName}.InvalidParameter",
                        ErrorMessage =>
                            "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter"
                            . " is invalid!",
                    };
                }
            }
        }
        elsif ( ref $XMLData->{$ItemKey} eq 'HASH' ) {
            if ( $XMLData->{$ItemKey}->{$ItemKey} ) {
                $CheckValueResult = $Self->_CheckValue(
                    Value   => $XMLData->{$ItemKey}->{$ItemKey},
                    Input   => $DefItem->{Input},
                    ItemKey => $ItemKey,
                    Parent  => $Parent,
                );
                if ( !$CheckValueResult->{Success} ) {
                    return $CheckValueResult;
                }
            }
        }
        else {

            # only perform checks if item really exits in the XMLData
            # CountNin checks was verified and passed before!, so it is safe to skip if needed
            if ( $XMLData->{$ItemKey} ) {
                $CheckValueResult = $Self->_CheckValue(
                    Value   => $XMLData->{$ItemKey},
                    Input   => $DefItem->{Input},
                    ItemKey => $ItemKey,
                    Parent  => $Parent,
                );
                if ( !$CheckValueResult->{Success} ) {
                    return $CheckValueResult;
                }
            }
        }

        # check if there is a sub and start recursion
        if ( defined $DefItem->{Sub} ) {

            if ( ref $XMLData->{$ItemKey} eq 'ARRAY' ) {
                my $Counter = 0;
                for my $ArrayItem ( @{ $XMLData->{$ItemKey} } ) {

                    # start recursion for each array item
                    my $XMLDataCheck = $Self->_CheckSearchXMLData(
                        Definition => $DefItem->{Sub},
                        XMLData    => $ArrayItem,
                        Parent     => $Parent . $ItemKey . "[$Counter]->",
                    );
                    if ( !$XMLDataCheck->{Success} ) {
                        return $XMLDataCheck;
                    }
                    $Counter++;
                }
            }
            elsif ( ref $XMLData->{$ItemKey} eq 'HASH' ) {

                # start recursion
                my $XMLDataCheck = $Self->_CheckSearchXMLData(
                    Definition => $DefItem->{Sub},
                    XMLData    => $XMLData->{$ItemKey},
                    Parent     => $Parent . $ItemKey . '->',
                );
                if ( !$XMLDataCheck->{Success} ) {
                    return $XMLDataCheck;
                }
            }
            else {

                # start recusrsion
                my $XMLDataCheck = $Self->_CheckSearchXMLData(
                    Definition => $DefItem->{Sub},
                    XMLData    => {},
                    Parent     => $Parent . $ItemKey . '->',
                );
                if ( !$XMLDataCheck->{Success} ) {
                    return $XMLDataCheck;
                }
            }
        }
    }

    return {
        Success => 1,
    };
}

=head2 _CheckValue()

checks if the given value is valid.

    my $ValueCheck = $CommonObject->_CheckValue(
        Value   => $Value                        # $Value could be a string, a time stamp,
                                                 #   general catalog class name, or a integer
        Input   => $InputDefinitionHashRef,      # The definition of the element input extracted
                                                 #   from the Configuration Item definition for
                                                 #   for each value
        ItemKey => 'some key',                   # The name of the value as sent in the SOAP
                                                 #   request
        Parent  => 'soem parent key->',          # The name of the parent followed by -> or empty
                                                 #   for root key items
    );

    returns:

    $ValueCheck = {
        Success => 1,                            # if everything is OK
    }

    $ValueCheck = {
        ErrorCode    => 'Function.Error',        # if error
        ErrorMessage => 'Error description',
    }

=cut

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

    my $Parent  = $Param{Parent};
    my $ItemKey = $Param{ItemKey};

    if ( $Param{Input}->{Type} eq 'Text' || $Param{Input}->{Type} eq 'TextArea' ) {

        # run Text validations
        if ( !$Self->ValidateInputText(%Param) ) {
            return {
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter value"
                    . " excedes the maxium length!",
            };
        }
    }
    elsif ( $Param{Input}->{Type} eq 'Date' ) {

        # run Date validations
        if ( !$Self->ValidateInputDate(%Param) ) {
            return {
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter value"
                    . " is not a valid Date format!",
            };
        }
    }
    elsif ( $Param{Input}->{Type} eq 'DateTime' ) {

        # run DateTime validations
        if ( !$Self->ValidateInputDateTime(%Param) ) {
            return {
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter value"
                    . " is not a valid DateTime format!",
            };
        }
    }
    elsif ( $Param{Input}->{Type} eq 'Customer' ) {

        # run Customer validations
        if ( !$Self->ValidateInputCustomer(%Param) ) {
            return {
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter value"
                    . " is not a valid customer!",
            };
        }
    }
    elsif ( $Param{Input}->{Type} eq 'CustomerCompany' ) {

        # run CustomerCompany validations
        if ( !$Self->ValidateInputCustomerCompany(%Param) ) {
            return {
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter value"
                    . " is not a valid customer company!",
            };
        }
    }
    elsif ( $Param{Input}->{Type} eq 'Integer' ) {

        # run Integer validations
        if ( !$Self->ValidateInputInteger(%Param) ) {
            return {
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter value"
                    . " is not a valid Integer or out of range!",
            };
        }
    }
    elsif ( $Param{Input}->{Type} eq 'GeneralCatalog' ) {

        # run General Catalog validations
        if ( !$Self->ValidateInputGeneralCatalog(%Param) ) {
            return {
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->CIXMLData->$Parent$ItemKey parameter value"
                    . " is not a valid for General Catalog '$Param{Input}->{Class}'!",
            };
        }
    }
    else {

        # The type is dummy, do nothing
    }

    return {
        Success => 1,
    };
}

=head2 _FormatSearchXMLData()

Create a XMLData suitable for ConfigItemSeach.

    my $NewXMLData = $OperationObject->_FormatSearchXMLData(
        XMLData    => $XMLDataHashRef,
        Child      => 1,                    # or 0, optional
    );

    returns:

    $NewXMLData = $XMLDataHashRef,                  # suitable for version add

=cut

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

    my $XMLData = $Param{XMLData};
    my $Parent  = $Param{Parent} || '';
    my $Child   = $Param{Child};

    my $NewXMLData = $Param{NewXMLData} || {};

    for my $RootKey ( sort keys %{$XMLData} ) {
        if ( ref $XMLData->{$RootKey} eq 'ARRAY' ) {

            for my $ArrayItem ( @{ $XMLData->{$RootKey} } ) {
                if ( ref $ArrayItem eq 'HASH' ) {

                    # extract the root key from the hash and assing it to content key
                    my $Content = delete $ArrayItem->{$RootKey};

                    # start recursion
                    my $NewXMLDataPart = $Self->_FormatSearchXMLData(
                        XMLData => $ArrayItem,
                        Parent  => $Parent . $RootKey . '::',
                        Child   => 1,
                    );

                    if ($Content) {
                        push @{ $NewXMLData->{ $Parent . $RootKey } }, $Content;
                    }

                    # assamble the final value from the parts
                    for my $NewKey ( %{$NewXMLDataPart} ) {
                        for my $Item ( @{ $NewXMLDataPart->{$NewKey} } ) {
                            push @{ $NewXMLData->{$NewKey} }, $Item;
                        }
                    }
                }
                elsif ( ref $ArrayItem eq '' ) {
                    push @{ $NewXMLData->{ $Parent . $RootKey } }, $ArrayItem;
                }
            }
        }

        elsif ( ref $XMLData->{$RootKey} eq 'HASH' ) {

            # extract the root key from the hash and assing it to content key
            my $Content = delete $XMLData->{$RootKey}->{$RootKey};

            # start recursion
            my $NewXMLDataPart = $Self->_FormatSearchXMLData(
                XMLData => $XMLData->{$RootKey},
                Parent  => $Parent . $RootKey . "::",
                Child   => 1,
            );

            if ($Content) {
                push @{ $NewXMLData->{ $Parent . $RootKey } }, $Content;
            }

            # assamble the final value from the part
            %{$NewXMLData} = ( %{$NewXMLData}, %{$NewXMLDataPart} );
        }

        elsif ( ref $XMLData->{$RootKey} eq '' ) {
            push @{ $NewXMLData->{ $Parent . $RootKey } }, $XMLData->{$RootKey};
        }
    }

    # return only the part on recursion
    if ($Child) {
        return $NewXMLData;
    }

    # return the complete XMLData as needed for ConfigItemSearch
    my @ReturnStructure;
    for my $SearchParam ( sort keys %{$NewXMLData} ) {
        my $SearchKey = $SearchParam;
        $SearchKey =~ s{ :: }{\'\}[%]\{\'}xmsg;
        $SearchKey = "[1]{'Version'}[1]{'$SearchKey'}[%]{'Content'}";
        push @ReturnStructure, {
            $SearchKey => $NewXMLData->{$SearchParam},
        };
    }
    return \@ReturnStructure;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::GenericInterface::Operation::ConfigItem::ConfigItemUpdate;

use strict;
use warnings;

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

use parent qw(
    Kernel::GenericInterface::Operation::Common
    Kernel::GenericInterface::Operation::ConfigItem::Common
);

our $ObjectManagerDisabled = 1;

=head1 NAME

Kernel::GenericInterface::Operation::ConfigItem::ConfigItemUpdate - GenericInterface ConfigItem ConfigItemUpdate Operation backend

=head1 PUBLIC INTERFACE

=head2 new()

usually, you want to create an instance of this
by using Kernel::GenericInterface::Operation->new();

=cut

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

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

    # check needed objects
    for my $Needed (qw(DebuggerObject WebserviceID)) {
        if ( !$Param{$Needed} ) {
            return {
                Success      => 0,
                ErrorMessage => "Got no $Needed!",
            };
        }

        $Self->{$Needed} = $Param{$Needed};
    }

    # define operation name
    $Self->{OperationName} = 'ConfigItemUpdate';

    $Self->{Config} = $Kernel::OM->Get('Kernel::Config')->Get('GenericInterface::Operation::ConfigItemUpdate');

    $Self->{Config}->{DefaultValue} = 'Not Defined';

    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

    # get a list of all config item classes
    $Self->{ClassList} = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    if ( !IsHashRefWithData( $Self->{ClassList} ) ) {
        return $Self->{DebuggerObject}->Error(
            Summary => 'Error when trying to get class listing of ITSM::ConfigItem::Class',
        );
    }

    # get a list of all incistates
    $Self->{InciStateList} = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::Core::IncidentState',
    );

    if ( !IsHashRefWithData( $Self->{InciStateList} ) ) {
        return $Self->{DebuggerObject}->Error(
            Summary => 'Error when trying to get incident state listing of'
                . ' ITSM::Core::IncidentState',
        );
    }

    # get a list of all deplstates
    $Self->{DeplStateList} = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    if ( !IsHashRefWithData( $Self->{DeplStateList} ) ) {
        return $Self->{DebuggerObject}->Error(
            Summary => 'Error when trying to get incident state listing of'
                . ' ITSM::ConfigItem::DeploymentState',
        );
    }

    # also provide the classlist in reversed form for easier reverse lookups
    my %ReverseClassList = reverse %{ $Self->{ClassList} };
    $Self->{ReverseClassList} = \%ReverseClassList;

    # also provide the incistatelist in reversed form for easier reverse lookups
    my %ReverseInciStateList = reverse %{ $Self->{InciStateList} };
    $Self->{ReverseInciStateList} = \%ReverseInciStateList;

    # also provide the deplstatelist in reversed form for easier reverse lookups
    my %ReverseDeplStateList = reverse %{ $Self->{DeplStateList} };
    $Self->{ReverseDeplStateList} = \%ReverseDeplStateList;

    return $Self;
}

=head2 Run()

perform ConfigItemUpdate Operation. This will return the updated config item number.

    my $Result = $OperationObject->Run(
        Data => {
            UserLogin => 'some agent login',                # UserLogin or SessionID is
            SessionID => 123,                               #   required

            Password  => 'some password',                   # if UserLogin is sent then Password is required

            ReplaceExistingData => 0,                       # optional, 0 or 1, default 0
                                                            # this will replace the existing XML data and attachments
            ConfigItemID => 123,

            ConfigItem   => {
                Class     => 'Config Item Class',
                Name      => 'The Name',
                DeplState => 'deployment state',
                InciState => 'incident state',
                CIXMLData => $ArrayHashRef,                 # it depends on the Configuration Item class and definition

                Attachment => [
                    {
                        Content     => 'content'            # base64 encoded
                        ContentType => 'some content type'
                        Filename    => 'some fine name'
                    },
                    # ...
                ],
                # or
                #Attachment => {
                #   Content     => 'content'
                #   ContentType => 'some content type'
                #   Filename    => 'some fine name'
                #},
            },
        },
    );

    $Result = {
        Success         => 1,                       # 0 or 1
        ErrorMessage    => '',                      # in case of error
        Data            => {                        # result data payload after Operation
            ConfigItemID => 123,                    # Configuration Item  ID number in OTRS::ITSM (Service desk system)
            Number       => 2324454323322           # Configuration Item  Number in OTRS::ITSM (Service desk system)
            Error => {                              # should not return errors
                    ErrorCode    => 'ConfigItemUpdate.ErrorCode'
                    ErrorMessage => 'Error Description'
            },
        },
    };

=cut

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

    my $Result = $Self->Init(
        WebserviceID => $Self->{WebserviceID},
    );

    if ( !$Result->{Success} ) {
        $Self->ReturnError(
            ErrorCode    => 'Webservice.InvalidConfiguration',
            ErrorMessage => $Result->{ErrorMessage},
        );
    }

    # check needed stuff
    if (
        !$Param{Data}->{UserLogin}
        && !$Param{Data}->{SessionID}
        )
    {
        return $Self->ReturnError(
            ErrorCode => "$Self->{OperationName}.MissingParameter",
            ErrorMessage =>
                "$Self->{OperationName}: UserLogin or SessionID is required!",
        );
    }

    if ( $Param{Data}->{UserLogin} ) {

        if ( !$Param{Data}->{Password} )
        {
            return $Self->ReturnError(
                ErrorCode    => "$Self->{OperationName}.MissingParameter",
                ErrorMessage => "$Self->{OperationName}: Password or SessionID is required!",
            );
        }
    }

    # authenticate user
    my ( $UserID, $UserType ) = $Self->Auth(%Param);

    if ( !$UserID ) {
        return $Self->ReturnError(
            ErrorCode    => "$Self->{OperationName}.AuthFail",
            ErrorMessage => "$Self->{OperationName}: User could not be authenticated!",
        );
    }

    # check needed hashes
    for my $Needed (qw(ConfigItem)) {
        if ( !IsHashRefWithData( $Param{Data}->{$Needed} ) ) {
            return $Self->ReturnError(
                ErrorCode => "$Self->{OperationName}.MissingParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: $Needed parameter is missing or not valid!",
            );
        }
    }

    # check needed items
    for my $Needed (qw(ConfigItemID)) {
        if ( !IsPositiveInteger( $Param{Data}->{$Needed} ) ) {
            return $Self->ReturnError(
                ErrorCode => "$Self->{OperationName}.MissingParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: $Needed parameter is missing or not valid!",
            );
        }
    }

    # check for valid ConfigItemID
    my $ConfigItemID = $Param{Data}->{ConfigItemID};

    # get config item object
    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    # get ConfigItem data
    my $ConfigItemData = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => $ConfigItemID,
    );

    if ( !IsHashRefWithData($ConfigItemData) ) {
        return $Self->ReturnError(
            ErrorCode    => "$Self->{OperationName}.InvalidParameter",
            ErrorMessage => "$Self->{OperationName}: ConfigItemID is invalid!",
        );
    }

    # isolate config item parameter
    my $ConfigItem = $Param{Data}->{ConfigItem};

    # remove leading and trailing spaces
    for my $Attribute ( sort keys %{$ConfigItem} ) {
        if ( ref $Attribute ne 'HASH' && ref $Attribute ne 'ARRAY' ) {

            # remove leading spaces
            $ConfigItem->{$Attribute} =~ s{\A\s+}{};

            # remove trailing spaces
            $ConfigItem->{$Attribute} =~ s{\s+\z}{};
        }
    }

    # if the parameter ReplaceExistingData is set to 0 or if it is missing
    # then missing, empty or only partially defined CIXMLData parameter attributes are allowed
    # in this case the existing CIXMLData is used for the missing parts.
    # A missing (undefined) CIXMLData attribute has the same effect
    # the ReplaceExistingData parameter also influences if existing attachments should be replaced or kept
    if ( !$Param{Data}->{ReplaceExistingData} || !defined $ConfigItem->{CIXMLData} ) {

        # set to empty hash reference if empty or not defined
        $ConfigItem->{CIXMLData} ||= {};

        # CIXMLData must be a hash reference
        if ( ref $ConfigItem->{CIXMLData} ne 'HASH' ) {
            return $Self->ReturnError(
                ErrorCode    => "$Self->{OperationName}.MissingParameter",
                ErrorMessage => "$Self->{OperationName}: ConfigItem->CIXMLData is missing or invalid!",
            );
        }

        # get latest version data from configitem
        my $Version = $ConfigItemObject->VersionGet(
            ConfigItemID => $ConfigItemID,
            UserID       => $UserID,
        );

        if ( !IsHashRefWithData($Version) ) {

            my $ErrorMessage = 'Could not get ConfigItem data'
                . ' in Kernel::GenericInterface::Operation::ConfigItem::ConfigItemUpdate::Run()';

            return $Self->ReturnError(
                ErrorCode    => '$Self->{OperationName}.InvalidParameter',
                ErrorMessage => "$Self->{OperationName}: $ErrorMessage",
            );
        }

        # remove unneeded items
        delete $Version->{ClassID};
        delete $Version->{CurDeplStateID};
        delete $Version->{CurInciStateID};
        delete $Version->{DeplStateID};
        delete $Version->{InciStateID};
        delete $Version->{XMLDefinitionID};

        my $Definition = delete $Version->{XMLDefinition};

        my $FormatedXMLData = $Self->InvertFormatXMLData(
            XMLData => $Version->{XMLData}->[1]->{Version},
        );

        my $ReplacedXMLData = $Self->InvertReplaceXMLData(
            XMLData    => $FormatedXMLData,
            Definition => $Definition,
        );

        $Version->{XMLData} = $ReplacedXMLData;

        # rename XMLData since SOAP transport complains about XML prefix on names
        $Version->{CIXMLData} = delete $Version->{XMLData};

        # merge existing data and new data from parameters
        $ConfigItem->{CIXMLData} = {
            %{ $Version->{CIXMLData} },
            %{ $ConfigItem->{CIXMLData} },
        };
    }

    if ( !IsHashRefWithData( $ConfigItem->{CIXMLData} ) ) {
        return $Self->ReturnError(
            ErrorCode    => "$Self->{OperationName}.InvalidParameter",
            ErrorMessage => "$Self->{OperationName}: ConfigItem->CIXMLData is empty or invalid!",
        );
    }

    # remove leading and trailing spaces for CIXMLData
    $Self->_CleanXMLData( XMLData => $ConfigItem->{CIXMLData} );

    # check ConfigItem attribute values
    my $ConfigItemCheck = $Self->_CheckConfigItem( ConfigItem => $ConfigItem );

    if ( !$ConfigItemCheck->{Success} ) {
        return $Self->ReturnError( %{$ConfigItemCheck} );
    }

    # check update permissions
    my $Permission = $ConfigItemObject->Permission(
        Scope   => 'Class',
        ClassID => $Self->{ReverseClassList}->{ $ConfigItem->{Class} },
        UserID  => $UserID,
        Type    => $Self->{Config}->{Permission},
    );

    if ( !$Permission ) {
        return $Self->ReturnError(
            ErrorCode    => "$Self->{OperationName}.AccessDenied",
            ErrorMessage => "$Self->{OperationName}: Can not update configuration items!",
        );
    }

    # handle attachments
    my $Attachment;
    my @AttachmentList;

    if ( defined $Param{Data}->{ConfigItem}->{Attachment} ) {

        # isolate Attachment parameter
        $Attachment = delete $Param{Data}->{ConfigItem}->{Attachment};

        # homologate imput to array
        if ( IsHashRefWithData($Attachment) ) {
            push @AttachmentList, $Attachment;
        }
        elsif ( IsArrayRefWithData($Attachment) ) {
            @AttachmentList = @{$Attachment};
        }
        else {
            return $Self->ReturnError(
                ErrorCode => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage =>
                    "$Self->{OperationName}: ConfigItem->Attachment parameter is invalid!",
            );
        }

        # check Attachment internal structure
        for my $AttachmentItem (@AttachmentList) {
            if ( !IsHashRefWithData($AttachmentItem) ) {
                return $Self->ReturnError(
                    ErrorCode => "$Self->{OperationName}.InvalidParameter",
                    ErrorMessage =>
                        "$Self->{OperationName}: ConfigItem->Attachment parameter is invalid!",
                );
            }

            # remove leading and trailing spaces
            for my $Attribute ( sort keys %{$AttachmentItem} ) {
                if ( ref $Attribute ne 'HASH' && ref $Attribute ne 'ARRAY' ) {

                    #remove leading spaces
                    $AttachmentItem->{$Attribute} =~ s{\A\s+}{};

                    #remove trailing spaces
                    $AttachmentItem->{$Attribute} =~ s{\s+\z}{};
                }
            }

            # check Attachment attribute values
            my $AttachmentCheck = $Self->_CheckAttachment( Attachment => $AttachmentItem );

            if ( !$AttachmentCheck->{Success} ) {
                return $Self->ReturnError( %{$AttachmentCheck} );
            }
        }
    }

    return $Self->_ConfigItemUpdate(
        ConfigItem          => $ConfigItem,
        ConfigItemID        => $ConfigItemID,
        AttachmentList      => \@AttachmentList,
        ReplaceExistingData => $Param{Data}->{ReplaceExistingData},
        UserID              => $UserID,
    );
}

=head1 INTERNAL INTERFACE

=head2 _CleanXMLData()

removed trailing and leading white spaces in the XMLData.

    my $XMLDataClean = $OperationObject->_CleanXMLData(
        Definition => $DefinitionArrayRef,          # Config Item Definition ot just part of it
        XMLData    => $XMLDataHashRef,
    );

    returns:

    $XMLDataClean = {
        Success => 1,                               # if everything is OK
    }

    $XMLDataClean = {
        ErrorCode    => 'Function.Error',           # if error
        ErrorMessage => 'Error description',
    }

=cut

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

    my $XMLData = $Param{XMLData};

    KEY:
    for my $Key ( sort keys %{$XMLData} ) {
        if ( ref $XMLData->{$Key} eq 'ARRAY' ) {
            ELEMENT:
            for my $Element ( @{ $XMLData->{$Key} } ) {
                if ( ref $Element eq 'HASH' ) {

                    # start recursion
                    $Self->_CleanXMLData( XMLData => $Element );
                    next ELEMENT;
                }
                elsif ( ref $Element eq '' ) {

                    #remove leading spaces
                    $Element =~ s{\A\s+}{};

                    #remove trailing spaces
                    $Element =~ s{\s+\z}{};
                }
            }
        }
        elsif ( ref $XMLData->{$Key} eq 'HASH' ) {

            # start recursion
            $Self->_CleanXMLData( XMLData => $XMLData->{$Key} );
            next KEY;

        }
        elsif ( ref $XMLData->{$Key} eq '' ) {

            # TODO: Use StringClean function!

            #remove leading spaces
            $XMLData->{$Key} =~ s{\A\s+}{};

            #remove trailing spaces
            $XMLData->{$Key} =~ s{\s+\z}{};
        }
    }

    return 1;
}

=head2 _CheckConfigItem()

checks if the given config item parameters are valid.

    my $ConfigItemCheck = $OperationObject->_CheckConfigItem(
        ConfigItem => $ConfigItem,                  # all config item parameters
    );

    returns:

    $ConfigItemCheck = {
        Success => 1,                               # if everything is OK
    }

    $ConfigItemCheck = {
        ErrorCode    => 'Function.Error',           # if error
        ErrorMessage => 'Error description',
    }

=cut

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

    my $ConfigItem = $Param{ConfigItem};

    # check config item internally
    NEEDED:
    for my $Needed (qw(Class Name DeplState InciState CIXMLData)) {

        next NEEDED if defined $ConfigItem->{$Needed};

        return {
            ErrorCode    => "$Self->{OperationName}.MissingParameter",
            ErrorMessage => "$Self->{OperationName}: ConfigItem->$Needed parameter is missing!",
        };
    }

    # check ConfigItem->Class
    if ( !$Self->ValidateClass( %{$ConfigItem} ) ) {
        return {
            ErrorCode => "$Self->{OperationName}.InvalidParameter",
            ErrorMessage =>
                "$Self->{OperationName}: ConfigItem->Class parameter is invalid!",
        };
    }

    # check ConfigItem->DeplState
    if ( !$Self->ValidateDeplState( %{$ConfigItem} ) ) {
        return {
            ErrorCode => "$Self->{OperationName}.InvalidParameter",
            ErrorMessage =>
                "$Self->{OperationName}: ConfigItem->DeplState parameter is invalid!",
        };
    }

    # check ConfigItem->DeplState
    if ( !$Self->ValidateInciState( %{$ConfigItem} ) ) {
        return {
            ErrorCode => "$Self->{OperationName}.InvalidParameter",
            ErrorMessage =>
                "$Self->{OperationName}: ConfigItem->InciState parameter is invalid!",
        };
    }

    # get last config item defintion
    my $DefinitionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->DefinitionGet(
        ClassID => $Self->{ReverseClassList}->{ $ConfigItem->{Class} },
    );

    my $XMLDataCheckResult = $Self->CheckXMLData(
        Definition => $DefinitionData->{DefinitionRef},
        XMLData    => $ConfigItem->{CIXMLData},
    );

    if ( !$XMLDataCheckResult->{Success} ) {
        return $XMLDataCheckResult;
    }

    # if everything is OK then return Success
    return {
        Success => 1,
    };
}

=head2 _CheckAttachment()

checks if the given attachment parameter is valid.

    my $AttachmentCheck = $OperationObject->_CheckAttachment(
        Attachment => $Attachment,                  # all attachment parameters
    );

    returns:

    $AttachmentCheck = {
        Success => 1,                               # if everething is OK
    }

    $AttachmentCheck = {
        ErrorCode    => 'Function.Error',           # if error
        ErrorMessage => 'Error description',
    }

=cut

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

    my $Attachment = $Param{Attachment};

    # check attachment item internally
    NEEDED:
    for my $Needed (qw(Content ContentType Filename)) {

        next NEEDED if defined $Attachment->{$Needed};

        return {
            ErrorCode => "$Self->{OperationName}.MissingParameter",
            ErrorMessage =>
                "$Self->{OperationName}: Attachment->$Needed  parameter is missing!",
        };
    }

    # check Article->ContentType
    if ( $Attachment->{ContentType} ) {

        $Attachment->{ContentType} = lc $Attachment->{ContentType};

        # check Charset part
        my $Charset = '';
        if ( $Attachment->{ContentType} =~ /charset=/i ) {
            $Charset = $Attachment->{ContentType};
            $Charset =~ s/.+?charset=("|'|)(\w+)/$2/gi;
            $Charset =~ s/"|'//g;
            $Charset =~ s/(.+?);.*/$1/g;
        }

        if ( $Charset && !$Self->ValidateCharset( Charset => $Charset ) )
        {
            return {
                ErrorCode    => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage => "$Self->{OperationName}: Attachment->ContentType is invalid!",
            };
        }

        # check MimeType part
        my $MimeType = '';
        if ( $Attachment->{ContentType} =~ /^(\w+\/\w+)/i ) {
            $MimeType = $1;
            $MimeType =~ s/"|'//g;
        }

        if ( !$Self->ValidateMimeType( MimeType => $MimeType ) ) {
            return {
                ErrorCode    => "$Self->{OperationName}.InvalidParameter",
                ErrorMessage => "$Self->{OperationName}: Attachment->ContentType is invalid!",
            };
        }
    }

    # if everything is OK then return Success
    return {
        Success => 1,
    };
}

=head2 _ConfigItemUpdate()

updates a configuration item with attachments if specified.

    my $Response = $OperationObject->_ConfigItemUpdate(
        ConfigItemID        => 123,
        ConfigItem          => $ConfigItem,             # all configuration item parameters
        AttachmentList      => $Attachment,             # a list of all attachments
        ReplaceExistingData => 0,                       # if the existing xml attributes and attachments should be replaced or kept
        UserID              => 123,
    );

    returns:

    $Response = {
        Success => 1,                               # if everething is OK
        Data => {
            ConfigItemID => 123,
            ConfigItemNumber => 'CN123',
        }
    }

    $Response = {
        Success      => 0,                         # if unexpected error
        ErrorMessage => "$Param{ErrorCode}: $Param{ErrorMessage}",
    }

=cut

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

    my $ConfigItemID   = $Param{ConfigItemID};
    my $ConfigItem     = $Param{ConfigItem};
    my $AttachmentList = $Param{AttachmentList};

    my $DeplStateID = $Self->{ReverseDeplStateList}->{ $ConfigItem->{DeplState} };
    my $InciStateID = $Self->{ReverseInciStateList}->{ $ConfigItem->{InciState} };

    my $RawXMLData = $ConfigItem->{CIXMLData};

    # get config item object
    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    # get last config item defintion
    my $DefinitionData = $ConfigItemObject->DefinitionGet(
        ClassID => $Self->{ReverseClassList}->{ $ConfigItem->{Class} },
    );

    # replace date, date time, customer, company and general catalog values
    my $ReplacedXMLData = $Self->ReplaceXMLData(
        XMLData    => $RawXMLData,
        Definition => $DefinitionData->{DefinitionRef},
    );

    # create an XMLData structure suitable for VersionAdd
    my $XMLData = $Self->FormatXMLData(
        XMLData => $ReplacedXMLData,
    );

    # get the current config item version data
    my $CurrentVersion = $ConfigItemObject->VersionGet(
        ConfigItemID => $ConfigItemID,
        UserID       => $Param{UserID},
    );

    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ConfigItemID,
        Name         => $ConfigItem->{Name},
        DefinitionID => $DefinitionData->{DefinitionID},
        DeplStateID  => $DeplStateID,
        InciStateID  => $InciStateID,
        XMLData      => $XMLData,
        UserID       => $Param{UserID},
    );

    if ( !$VersionID ) {
        return {
            Success      => 0,
            ErrorMessage => 'Configuration Item could not be updated, please contact the system'
                . 'administrator'
        };
    }

    # get the version ID of the config item before the update
    my $CurrentVersionID = $CurrentVersion->{VersionID} || '';

    # compare old version and new version IDs
    if ( $CurrentVersionID eq $VersionID ) {
        $Self->{DebuggerObject}->Notice(
            Summary => "$Self->{OperationName}: No change in configuration item version",
            Data    => 'The internal structure of the configuration item was indentical to the last'
                . ' one, no update was performed',
        );
    }

    # the ReplaceExistingData flag is set
    if ( $Param{ReplaceExistingData} ) {

        # get a list of all attachments
        my @ExistingAttachments = $ConfigItemObject->ConfigItemAttachmentList(
            ConfigItemID => $ConfigItemID,
        );

        # delete all attachments of this config item
        FILENAME:
        for my $Filename (@ExistingAttachments) {

            # delete the attachment
            my $DeletionSuccess = $ConfigItemObject->ConfigItemAttachmentDelete(
                ConfigItemID => $ConfigItemID,
                Filename     => $Filename,
                UserID       => $Param{UserID},
            );

            next FILENAME if $DeletionSuccess;

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Unknown problem when deleting attachment $Filename of ConfigItem "
                    . "$ConfigItemID. Please check the VirtualFS backend for stale files!",
            );
        }
    }

    # set attachments
    ATTACHMENT:
    if ( IsArrayRefWithData($AttachmentList) ) {

        for my $Attachment ( @{$AttachmentList} ) {
            my $Result = $Self->CreateAttachment(
                Attachment   => $Attachment,
                ConfigItemID => $ConfigItemID,
                UserID       => $Param{UserID}
            );

            next ATTACHMENT if $Result->{Success};

            my $ErrorMessage = $Result->{ErrorMessage}
                || "Attachment could not be created, please contact the system administrator";

            return {
                Success      => 0,
                ErrorMessage => $ErrorMessage,
            };
        }
    }

    # get ConfigItem data
    my $ConfigItemData = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => $ConfigItemID,
    );

    if ( !IsHashRefWithData($ConfigItemData) ) {
        return {
            Success      => 0,
            ErrorMessage => 'Could not get new configuration item information, please contact the system administrator',
        };
    }

    return {
        Success => 1,
        Data    => {
            ConfigItemID => $ConfigItemID,
            Number       => $ConfigItemData->{Number},
        },
    };
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::ar_SA_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'بين';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'المكتب';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::bg_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Управление на конфигурационните единици CI';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = 'Промяна';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = 'Клас';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = 'Последна промяна';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = 'време на създаване';
    $Self->{Translation}->{'Zoom view'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Да търси и в предишните версии?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = 'Последно променен от';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Между';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Модел';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Сериен Номер';
    $Self->{Translation}->{'CPU'} = 'Процесор';
    $Self->{Translation}->{'Ram'} = 'Памет';
    $Self->{Translation}->{'Hard Disk'} = 'Твърд диск';
    $Self->{Translation}->{'Capacity'} = 'Капацитет';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Мрежов адаптер';
    $Self->{Translation}->{'IP over DHCP'} = 'IP от DHCP';
    $Self->{Translation}->{'IP Address'} = 'IP адрес';
    $Self->{Translation}->{'Graphic Adapter'} = 'Графична карта';
    $Self->{Translation}->{'Other Equipment'} = 'Друго оборудване';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Дата на изтичане на гаранцията';
    $Self->{Translation}->{'Install Date'} = 'Дата на инсталация';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'Мрежов адрес';
    $Self->{Translation}->{'Subnet Mask'} = 'Маска на подмрежата';
    $Self->{Translation}->{'Gateway'} = 'Гейт преход';
    $Self->{Translation}->{'Licence Type'} = 'Тип на лиценза';
    $Self->{Translation}->{'Licence Key'} = 'Лицензен ключ';
    $Self->{Translation}->{'Quantity'} = 'Количество';
    $Self->{Translation}->{'Expiration Date'} = 'Дата на изтичане';
    $Self->{Translation}->{'Media'} = 'Медия';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'Изтекли';
    $Self->{Translation}->{'Maintenance'} = 'Поддръжка';
    $Self->{Translation}->{'Pilot'} = 'Пилот-водач';
    $Self->{Translation}->{'Planned'} = 'Планиран';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Ремонт/Възстановяване';
    $Self->{Translation}->{'Retired'} = 'Излязъл от употреба';
    $Self->{Translation}->{'Review'} = 'Преглеждан';
    $Self->{Translation}->{'Test/QA'} = 'Тест/Качествен контрол';
    $Self->{Translation}->{'Laptop'} = 'Преносим';
    $Self->{Translation}->{'Desktop'} = 'Настолен';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Сървър';
    $Self->{Translation}->{'Other'} = 'Други';
    $Self->{Translation}->{'Monitor'} = 'Монитор';
    $Self->{Translation}->{'Printer'} = 'Принтер';
    $Self->{Translation}->{'Switch'} = 'Концентратор';
    $Self->{Translation}->{'Router'} = 'Рутер';
    $Self->{Translation}->{'WLAN Access Point'} = 'WLAN точка за достъп';
    $Self->{Translation}->{'Security Device'} = 'У-во за сигурност';
    $Self->{Translation}->{'Backup Device'} = 'Архивиращо у-во';
    $Self->{Translation}->{'Mouse'} = 'Мишка';
    $Self->{Translation}->{'Keyboard'} = 'Клавиатура';
    $Self->{Translation}->{'Camera'} = 'Камера';
    $Self->{Translation}->{'Beamer'} = 'Бимер';
    $Self->{Translation}->{'Modem'} = 'Модем';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA карта';
    $Self->{Translation}->{'USB Device'} = 'USB Устройство';
    $Self->{Translation}->{'Docking Station'} = 'Докинг станция';
    $Self->{Translation}->{'Scanner'} = 'Скенер';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'Офис';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Потребителски приложения';
    $Self->{Translation}->{'Middleware'} = 'Средно ниво';
    $Self->{Translation}->{'Server Application'} = 'Сървърно приложение';
    $Self->{Translation}->{'Client OS'} = 'Потребителска Опер.с-ма';
    $Self->{Translation}->{'Server OS'} = 'сървърна Опер.с-ма';
    $Self->{Translation}->{'Admin Tool'} = 'Административнен инструмент';
    $Self->{Translation}->{'User Tool'} = 'Клиентски инструмент';
    $Self->{Translation}->{'Embedded'} = 'Вградена';
    $Self->{Translation}->{'Single Licence'} = 'Единичен лиценз';
    $Self->{Translation}->{'Per User'} = 'Лицензиране за потребител';
    $Self->{Translation}->{'Per Processor'} = 'Лицензиране за процесор';
    $Self->{Translation}->{'Per Server'} = 'Лицензиране за сървър';
    $Self->{Translation}->{'Per Node'} = 'Лицензиране на точка';
    $Self->{Translation}->{'Volume Licence'} = 'Специални обемни програми';
    $Self->{Translation}->{'Enterprise Licence'} = 'Ентърпрайс лиценз';
    $Self->{Translation}->{'Developer Licence'} = 'Лиценз за разработчици';
    $Self->{Translation}->{'Demo'} = 'Демо';
    $Self->{Translation}->{'Time Restricted'} = 'Лиценз, ограничен по време';
    $Self->{Translation}->{'Freeware'} = 'Безплатно разпространяван';
    $Self->{Translation}->{'Open Source'} = 'Софтуер с отворен код';
    $Self->{Translation}->{'Unlimited'} = 'Неограничен лиценз';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = 'Направи копие';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Параметри за примерните разрешителни групи от атрибутите на общия каталог.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::ca_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = 'Vista de zoom';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Entre';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'Oficina';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::cs_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Správa Konfiguračních Položek';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = 'Změnit';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Stav Incidentu';
    $Self->{Translation}->{'Deployment State'} = 'Stav Nasazení';
    $Self->{Translation}->{'Class'} = 'Třída';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = 'Aktuální Stav Incidentu';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = 'Poslední změna';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Konfig. Položka';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'Jméno této Konfig. Položky';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = 'Doba vytvoření';
    $Self->{Translation}->{'Zoom view'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Nastavení kontextu';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Hledat také v předchozích verzích?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = 'Aktuální Stav Nasazení';
    $Self->{Translation}->{'Last changed by'} = 'Poslední změna od';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = 'Stav Nasazení této Konfig. Položky';
    $Self->{Translation}->{'The incident state of this config item'} = 'Stav Incidentu této Konfig. Položky';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Mezi';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Max počet jedné Položky';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = 'Přeskočeno';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Model';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Sériové Číslo';
    $Self->{Translation}->{'CPU'} = 'Procesor';
    $Self->{Translation}->{'Ram'} = 'RAM';
    $Self->{Translation}->{'Hard Disk'} = 'Pevný Disk';
    $Self->{Translation}->{'Capacity'} = 'Kapacita';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Síťový Adapter';
    $Self->{Translation}->{'IP over DHCP'} = 'IP z DHCP';
    $Self->{Translation}->{'IP Address'} = 'IP Adresa';
    $Self->{Translation}->{'Graphic Adapter'} = 'Grafická Karta';
    $Self->{Translation}->{'Other Equipment'} = 'Jiné Vybavení';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Konec platnosti záruky';
    $Self->{Translation}->{'Install Date'} = 'Datum Instalace';
    $Self->{Translation}->{'Phone 1'} = 'Telefon 1';
    $Self->{Translation}->{'Phone 2'} = 'Telefon 2';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'Síťová Adresa';
    $Self->{Translation}->{'Subnet Mask'} = 'Maska Podsítě';
    $Self->{Translation}->{'Gateway'} = 'Gateway';
    $Self->{Translation}->{'Licence Type'} = 'Druh licence';
    $Self->{Translation}->{'Licence Key'} = 'Licenční Klíč';
    $Self->{Translation}->{'Quantity'} = 'Množství';
    $Self->{Translation}->{'Expiration Date'} = 'Konec Platnosti';
    $Self->{Translation}->{'Media'} = 'Médium';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'Propadnuté';
    $Self->{Translation}->{'Maintenance'} = 'Údržba';
    $Self->{Translation}->{'Pilot'} = 'Řídící';
    $Self->{Translation}->{'Planned'} = 'Naplánováno';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Oprava/Obnovení';
    $Self->{Translation}->{'Retired'} = 'Mimo provoz';
    $Self->{Translation}->{'Review'} = 'Přehled';
    $Self->{Translation}->{'Test/QA'} = 'Test/Kontrola kvality';
    $Self->{Translation}->{'Laptop'} = 'Mobilní';
    $Self->{Translation}->{'Desktop'} = 'Stolní';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Server';
    $Self->{Translation}->{'Other'} = 'Jiné';
    $Self->{Translation}->{'Monitor'} = 'Monitor';
    $Self->{Translation}->{'Printer'} = 'Tiskárna';
    $Self->{Translation}->{'Switch'} = 'Switch';
    $Self->{Translation}->{'Router'} = 'Router';
    $Self->{Translation}->{'WLAN Access Point'} = 'Přístupový Bod WLAN';
    $Self->{Translation}->{'Security Device'} = 'Bezpečnostní Zařízení';
    $Self->{Translation}->{'Backup Device'} = 'Zálohovácí Zařízení';
    $Self->{Translation}->{'Mouse'} = 'Myš';
    $Self->{Translation}->{'Keyboard'} = 'Klávesnice';
    $Self->{Translation}->{'Camera'} = 'Kamera';
    $Self->{Translation}->{'Beamer'} = 'Beamer';
    $Self->{Translation}->{'Modem'} = 'Modem';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA karta';
    $Self->{Translation}->{'USB Device'} = 'USB Zařízení';
    $Self->{Translation}->{'Docking Station'} = 'Dok Stanice';
    $Self->{Translation}->{'Scanner'} = 'Skener';
    $Self->{Translation}->{'Building'} = 'Budova';
    $Self->{Translation}->{'Office'} = 'Kancelář';
    $Self->{Translation}->{'Floor'} = 'podlaží';
    $Self->{Translation}->{'Room'} = 'Místnost';
    $Self->{Translation}->{'Rack'} = 'Rack';
    $Self->{Translation}->{'Workplace'} = 'Pracoviště';
    $Self->{Translation}->{'Outlet'} = 'Filiálka';
    $Self->{Translation}->{'IT Facility'} = 'IT Příslušenství';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Uživatelské Aplikace';
    $Self->{Translation}->{'Middleware'} = 'Střední úroveň';
    $Self->{Translation}->{'Server Application'} = 'Serverová Aplikace';
    $Self->{Translation}->{'Client OS'} = 'Uživatelský Operační Systém';
    $Self->{Translation}->{'Server OS'} = 'Serverový Operační Systém';
    $Self->{Translation}->{'Admin Tool'} = 'Administrační nástroj';
    $Self->{Translation}->{'User Tool'} = 'Uživatelský Nástroj';
    $Self->{Translation}->{'Embedded'} = 'Vestavěný';
    $Self->{Translation}->{'Single Licence'} = 'Jednoduchá Licence';
    $Self->{Translation}->{'Per User'} = 'na Uživatele';
    $Self->{Translation}->{'Per Processor'} = 'na Procesor';
    $Self->{Translation}->{'Per Server'} = 'na Server';
    $Self->{Translation}->{'Per Node'} = 'na Uzel';
    $Self->{Translation}->{'Volume Licence'} = 'Svazková Licence';
    $Self->{Translation}->{'Enterprise Licence'} = 'Podniková Licence ';
    $Self->{Translation}->{'Developer Licence'} = 'Licence Vývojáře';
    $Self->{Translation}->{'Demo'} = 'Demo';
    $Self->{Translation}->{'Time Restricted'} = 'Časově omezená Licence';
    $Self->{Translation}->{'Freeware'} = 'Freeware';
    $Self->{Translation}->{'Open Source'} = 'Software s Otevřeným Kódem';
    $Self->{Translation}->{'Unlimited'} = 'Neomezená';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = 'Duplikace';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parametry skupiny pro příklad oprávnění obecných atributů katalogu';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::da_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Config Item styring';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = 'Skift';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Incident tilstand';
    $Self->{Translation}->{'Deployment State'} = 'Deployment tilstand';
    $Self->{Translation}->{'Class'} = 'Klasse';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = 'Nuværende Incident tilstand';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = 'Sidst ændret';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Config Item';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'Navnet på denne Config Item';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = 'Historik';
    $Self->{Translation}->{'Createtime'} = 'Oprettelsestid';
    $Self->{Translation}->{'Zoom view'} = 'Zoom-visning';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Søg også i tidligere versioner?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = 'Nuværende deployment status';
    $Self->{Translation}->{'Last changed by'} = 'Sidst ændret af';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = 'Deployment tilstand for denne Config item';
    $Self->{Translation}->{'The incident state of this config item'} = 'Incident tilstand for denne Config Item';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Mellem';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Maximum antal af et element';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Model';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Serienummer';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = 'RAM';
    $Self->{Translation}->{'Hard Disk'} = 'Harddisk';
    $Self->{Translation}->{'Capacity'} = 'Kapacitet';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Netkort';
    $Self->{Translation}->{'IP over DHCP'} = 'IP via DHCP';
    $Self->{Translation}->{'IP Address'} = 'IP adresse';
    $Self->{Translation}->{'Graphic Adapter'} = 'Grafikkort';
    $Self->{Translation}->{'Other Equipment'} = 'Andet udstyr';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Udløbsdato for garanti';
    $Self->{Translation}->{'Install Date'} = 'Installationsdato';
    $Self->{Translation}->{'Phone 1'} = 'Telefon 1';
    $Self->{Translation}->{'Phone 2'} = 'Telefon 2';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'Netværksadresse';
    $Self->{Translation}->{'Subnet Mask'} = 'Subnet maske';
    $Self->{Translation}->{'Gateway'} = 'Gateway';
    $Self->{Translation}->{'Licence Type'} = 'Licenstype';
    $Self->{Translation}->{'Licence Key'} = 'Licensnøgle';
    $Self->{Translation}->{'Quantity'} = 'Mængde';
    $Self->{Translation}->{'Expiration Date'} = 'Udløbsdato';
    $Self->{Translation}->{'Media'} = 'Medie';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'Udløbet';
    $Self->{Translation}->{'Maintenance'} = 'Vedligeholdelse';
    $Self->{Translation}->{'Pilot'} = 'Pilot';
    $Self->{Translation}->{'Planned'} = 'Planlagt';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Reperation';
    $Self->{Translation}->{'Retired'} = 'Persioneret';
    $Self->{Translation}->{'Review'} = 'Anmeldelse';
    $Self->{Translation}->{'Test/QA'} = 'Test/QA';
    $Self->{Translation}->{'Laptop'} = 'Laptop';
    $Self->{Translation}->{'Desktop'} = 'Desktop';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Server';
    $Self->{Translation}->{'Other'} = 'Andet';
    $Self->{Translation}->{'Monitor'} = 'Monitor';
    $Self->{Translation}->{'Printer'} = 'Printer';
    $Self->{Translation}->{'Switch'} = 'Switch';
    $Self->{Translation}->{'Router'} = 'Router';
    $Self->{Translation}->{'WLAN Access Point'} = 'WLAN Access Point';
    $Self->{Translation}->{'Security Device'} = 'Sikkerhedsenhed';
    $Self->{Translation}->{'Backup Device'} = 'Backup enhed';
    $Self->{Translation}->{'Mouse'} = 'Mus';
    $Self->{Translation}->{'Keyboard'} = 'Tastatur';
    $Self->{Translation}->{'Camera'} = 'Kamera';
    $Self->{Translation}->{'Beamer'} = 'Beamer';
    $Self->{Translation}->{'Modem'} = 'Modem';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA kort';
    $Self->{Translation}->{'USB Device'} = 'USB enhed';
    $Self->{Translation}->{'Docking Station'} = 'Docking Station';
    $Self->{Translation}->{'Scanner'} = 'Skanner';
    $Self->{Translation}->{'Building'} = 'Bygning';
    $Self->{Translation}->{'Office'} = 'Kontor';
    $Self->{Translation}->{'Floor'} = 'Etage';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = 'Rack';
    $Self->{Translation}->{'Workplace'} = 'Arbejdsplads';
    $Self->{Translation}->{'Outlet'} = 'Stikkontakt';
    $Self->{Translation}->{'IT Facility'} = 'IT facilitet';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Klient aplikation';
    $Self->{Translation}->{'Middleware'} = 'Middleware';
    $Self->{Translation}->{'Server Application'} = 'Server aplikation';
    $Self->{Translation}->{'Client OS'} = 'Klient OS';
    $Self->{Translation}->{'Server OS'} = 'Server OS';
    $Self->{Translation}->{'Admin Tool'} = 'Admin værktøjer';
    $Self->{Translation}->{'User Tool'} = 'Bruger værktøjer';
    $Self->{Translation}->{'Embedded'} = 'Embedded';
    $Self->{Translation}->{'Single Licence'} = 'Enkeltlicens';
    $Self->{Translation}->{'Per User'} = 'Pr bruger';
    $Self->{Translation}->{'Per Processor'} = 'Pr processor';
    $Self->{Translation}->{'Per Server'} = 'Pr server';
    $Self->{Translation}->{'Per Node'} = 'Pr node';
    $Self->{Translation}->{'Volume Licence'} = 'Volumen Licens';
    $Self->{Translation}->{'Enterprise Licence'} = 'Enterprise licens';
    $Self->{Translation}->{'Developer Licence'} = 'Udvikler licens';
    $Self->{Translation}->{'Demo'} = 'Demo';
    $Self->{Translation}->{'Time Restricted'} = 'Tidsbegrænset';
    $Self->{Translation}->{'Freeware'} = 'Freeware';
    $Self->{Translation}->{'Open Source'} = 'Open Source';
    $Self->{Translation}->{'Unlimited'} = 'Unbegrænset';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = 'Dupliker';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::de_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = 'Dieses Feld ist erforderlich und der Wert muss alphanumerisch sein.';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = 'Muss eindeutig sein und darf nur alphanumerische Zeichen enthalten.';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        'Dies ist die Reihenfolge, in der dieses Feld in den aktiven Dialogen angezeigt wird.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = 'Config-Item-Klasse';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        'Die gewählte Config-Item-Klasse ist ungültig oder existiert nicht.';
    $Self->{Translation}->{'Config item deployment states'} = 'Config-Item-Verwendungsstatus';
    $Self->{Translation}->{'Config item link type'} = 'Config-Item-Linktyp';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        'Wählen Sie einen Linktyp, um die Verknüpfung zwischen Ticket und gewählten Config-Items zu aktivieren. Beachten Sie, dass die Verknüpfung nur für dynamische Felder von Tickets zur Verfügung steht.';
    $Self->{Translation}->{'Config item link source'} = 'Config-Item-Linkquelle';
    $Self->{Translation}->{'Config item link removal'} = 'Config-Item-Links entfernen';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        'Aktivieren Sie diese Option, um Verknüpfungen zwischen Ticket und Config-Items zu entfernen, die aus diesem Feld entfernt wurden.';
    $Self->{Translation}->{'Config item key'} = 'Config-Item-Schlüssel';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        'Diese dynamischen Felder werden mit den Werten aus dem/den gewählten Config-Item(s) befüllt.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'ConfigItem-Verwaltung';
    $Self->{Translation}->{'Change class definition'} = 'Klassen-Definition ändern';
    $Self->{Translation}->{'Config Item Class'} = 'ConfigItem-Klasse';
    $Self->{Translation}->{'Definition'} = 'Definition';
    $Self->{Translation}->{'Change'} = 'Ändern';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Vorfallstatus';
    $Self->{Translation}->{'Deployment State'} = 'Verwendungsstatus';
    $Self->{Translation}->{'Class'} = 'Klasse';
    $Self->{Translation}->{'Deployment State Type'} = 'Verwendungsstatus-Typ';
    $Self->{Translation}->{'Current Incident State'} = 'Aktueller Vorfallsstatus';
    $Self->{Translation}->{'Current Incident State Type'} = 'Aktueller Vorfallsstatus-Typ';
    $Self->{Translation}->{'Last changed'} = 'Zuletzt geändert';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Config Item';
    $Self->{Translation}->{'Filter for Classes'} = 'Filter für Klassen';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Wählen Sie eine Klasse aus der Liste aus um ein neues Config Item zu erstellen.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'ITSM ConfigItem Sammelaktion';
    $Self->{Translation}->{'Deployment state'} = 'Verwendungsstatus';
    $Self->{Translation}->{'Incident state'} = 'Vorfallstatus';
    $Self->{Translation}->{'Link to another'} = 'Zu einem anderen verknüpfen';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = 'Ungültige Configuration Item Nummer!';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'Die Nummer des zu verknüpfenden Configuration Items.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = 'Kundenbezogene Config-Items';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = 'Wollen Sie dieses Configuration Item wirklich löschen?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'Der Name dieses Configuration Items';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'Name wird bereits von den Configuration Items mit den folgenden Nummern verwendet: %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = 'Historie von Configuration Item: %s';
    $Self->{Translation}->{'History Content'} = 'Änderungsverlauf';
    $Self->{Translation}->{'Createtime'} = 'Erstellt am';
    $Self->{Translation}->{'Zoom view'} = 'Detailansicht';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Kontext-Eintstellungen';
    $Self->{Translation}->{'Config Items per page'} = 'Configuration Items pro Seite';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = 'Eine generische ITSM Configuration Item-Tabelle.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Suche ausführen';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Auch in früheren Versionen suchen?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Configuration Item';
    $Self->{Translation}->{'Configuration Item Information'} = 'Configuration Item Information';
    $Self->{Translation}->{'Current Deployment State'} = 'Aktueller Verwendungsstatus';
    $Self->{Translation}->{'Last changed by'} = 'Zuletzt geändert von';
    $Self->{Translation}->{'Show one version'} = 'Nur eine Version anzeigen';
    $Self->{Translation}->{'Show all versions'} = 'Alle Versionen anzeigen';
    $Self->{Translation}->{'Version Incident State'} = 'Vorfallstatus-Version';
    $Self->{Translation}->{'Version Deployment State'} = 'Version des Verwendungsstatus.';
    $Self->{Translation}->{'Version Number'} = 'Versionsnummer';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Configuration Item-Versionsdetails';
    $Self->{Translation}->{'Property'} = 'Eigenschaft';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = 'Es wird kein Zugriff auf die Klasse gewährt!';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = 'Übersicht: ITSM ConfigItem';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = 'Keine ID für das Configuration Item angegeben!';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = 'Sie benötigen wenigstens ein ausgewähltes Configuration Item!';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        'Sie haben keine Schreib-Berechtigungen für dieses Configuration Item: %s.';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = 'Configuration Item "%s" in Datenbank nicht gefunden!';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = 'Es war nicht möglich, die Configuration Item-ID %s zu löschen!';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = 'Keine Version für Configuration Item-ID %s gefunden!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = 'Keine Configuration Item-ID, Duplikats-ID oder Klassen-ID angegeben.';
    $Self->{Translation}->{'No access is given!'} = 'Kein Zugriff gewährt!';
    $Self->{Translation}->{'No definition was defined for class %s!'} = 'Keine Definition für Klasse %s definiert!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = 'Kann Historie nicht anzeigen, keine Configuration Item-ID übermittelt!';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = 'Kann Historie nicht anzeigen, keine Configuration Item-ID übermittelt!';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = 'Neues Configuration Item (ID=%s)';
    $Self->{Translation}->{'New version (ID=%s)'} = 'Neue Version (ID=%s)';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = 'Verwendungsstatus aktualisiert (neu=%s, alt=%s)';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = 'Vorfallstatus aktualisiert (neu=%s, alt=%s)';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = 'ConfigItem (ID=%s) gelöscht';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = 'Verknüpfung zu %s (Typ=%s) hinzugefügt';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = 'Verknüpfung zu %s (Typ=%s) gelöscht';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = 'ConfigItem-Definition aktualisiert (ID=%s)';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = 'Name aktualisiert (neu=%s, alt=%s)';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = 'Attribut %s aktualisiert von "%s" nach "%s"';
    $Self->{Translation}->{'Version %s deleted'} = 'Version %s gelöscht';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = 'Keine Configuration Item-ID oder Versions-ID angegeben.';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = 'Configuration Item kann nicht angezeigt werden, keine Zugriffsrechte vergeben!';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = 'Configuration Item-ID %s in Datenbank nicht gefunden.';
    $Self->{Translation}->{'VersionID %s not found in database!'} = 'Versions-ID %s in Datenbank nicht gefunden!';
    $Self->{Translation}->{'ConfigItem'} = 'ConfigItem';
    $Self->{Translation}->{'printed by %s at %s'} = 'gedruckt von %s am %s';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = 'Ungültige Klassen-ID!';
    $Self->{Translation}->{'No ClassID is given!'} = 'Keine Klassen-ID übermittelt!';
    $Self->{Translation}->{'No access rights for this class given!'} = 'Keine Zugriffsrechte für diese Klasse vergeben!';
    $Self->{Translation}->{'No Result!'} = 'Kein Ergebnis!';
    $Self->{Translation}->{'Config Item Search Results'} = 'Config Item-Suchergebnisse';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        'Kann das Configuration Item nicht anzeigen, keine Zugriffsrechte für Configuration Items vergeben.';
    $Self->{Translation}->{'operational'} = 'Operativ';
    $Self->{Translation}->{'warning'} = 'Warnung';
    $Self->{Translation}->{'incident'} = 'Vorfall';
    $Self->{Translation}->{'The deployment state of this config item'} = 'Der Verwendungsstatus dieses Config Items';
    $Self->{Translation}->{'The incident state of this config item'} = 'Der Vorfallstatus dieses Config Items';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = 'Angezeigte Configuration Items';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Zwischen';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Maximale Anzahl eines Elements';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Leere Felder zeigen an, dass die aktuellen Werte beibehalten werden.';
    $Self->{Translation}->{'Skipped'} = 'Übersprungen';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Modell';
    $Self->{Translation}->{'Customer Company'} = 'Kundenunternehmen';
    $Self->{Translation}->{'Serial Number'} = 'Seriennummer';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = 'RAM';
    $Self->{Translation}->{'Hard Disk'} = 'Festplatte';
    $Self->{Translation}->{'Capacity'} = 'Kapazität';
    $Self->{Translation}->{'FQDN'} = 'FQDN';
    $Self->{Translation}->{'Network Adapter'} = 'Netzwerk-Adapter';
    $Self->{Translation}->{'IP over DHCP'} = 'IP über DHCP';
    $Self->{Translation}->{'IP Address'} = 'IP-Addresse';
    $Self->{Translation}->{'Graphic Adapter'} = 'Grafik-Adapter';
    $Self->{Translation}->{'Other Equipment'} = 'Sonstige Ausstattung';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Garantie Ablaufdatum';
    $Self->{Translation}->{'Install Date'} = 'Installationsdatum';
    $Self->{Translation}->{'Phone 1'} = 'Telefon 1';
    $Self->{Translation}->{'Phone 2'} = 'Telefon 2';
    $Self->{Translation}->{'E-Mail'} = 'E-Mail';
    $Self->{Translation}->{'Network Address'} = 'Netzwerk-Adresse';
    $Self->{Translation}->{'Subnet Mask'} = 'Subnetz-Maske';
    $Self->{Translation}->{'Gateway'} = 'Gateway';
    $Self->{Translation}->{'Licence Type'} = 'Lizenztyp';
    $Self->{Translation}->{'Licence Key'} = 'Lizenzschlüssel';
    $Self->{Translation}->{'Quantity'} = 'Menge';
    $Self->{Translation}->{'Expiration Date'} = 'Ablaufdatum';
    $Self->{Translation}->{'Media'} = 'Medium';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = 'Computer';
    $Self->{Translation}->{'Hardware'} = 'Hardware';
    $Self->{Translation}->{'Network'} = 'Netzwerk';
    $Self->{Translation}->{'Software'} = 'Software';
    $Self->{Translation}->{'Expired'} = 'Abgelaufen';
    $Self->{Translation}->{'Maintenance'} = 'In Wartung';
    $Self->{Translation}->{'Pilot'} = 'Pilotbetrieb';
    $Self->{Translation}->{'Planned'} = 'Geplant';
    $Self->{Translation}->{'Production'} = 'Produktiv';
    $Self->{Translation}->{'Repair'} = 'In Reparatur';
    $Self->{Translation}->{'Retired'} = 'Außer Dienst';
    $Self->{Translation}->{'Review'} = 'Unter Review';
    $Self->{Translation}->{'Test/QA'} = 'Test/QS';
    $Self->{Translation}->{'Laptop'} = 'Laptop';
    $Self->{Translation}->{'Desktop'} = 'Desktop';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Server';
    $Self->{Translation}->{'Other'} = 'Sonstiges';
    $Self->{Translation}->{'Monitor'} = 'Monitor';
    $Self->{Translation}->{'Printer'} = 'Drucker';
    $Self->{Translation}->{'Switch'} = 'Switch';
    $Self->{Translation}->{'Router'} = 'Router';
    $Self->{Translation}->{'WLAN Access Point'} = 'WLAN-Zugangspunkt';
    $Self->{Translation}->{'Security Device'} = 'Sicherheitsgerät';
    $Self->{Translation}->{'Backup Device'} = 'Backup-Gerät';
    $Self->{Translation}->{'Mouse'} = 'Maus';
    $Self->{Translation}->{'Keyboard'} = 'Tastatur';
    $Self->{Translation}->{'Camera'} = 'Kamera';
    $Self->{Translation}->{'Beamer'} = 'Beamer';
    $Self->{Translation}->{'Modem'} = 'Modem';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA-Karte';
    $Self->{Translation}->{'USB Device'} = 'USB-Gerät';
    $Self->{Translation}->{'Docking Station'} = 'Docking-Station';
    $Self->{Translation}->{'Scanner'} = 'Scanner';
    $Self->{Translation}->{'Building'} = 'Gebäude';
    $Self->{Translation}->{'Office'} = 'Büro';
    $Self->{Translation}->{'Floor'} = 'Etage';
    $Self->{Translation}->{'Room'} = 'Raum';
    $Self->{Translation}->{'Rack'} = 'Rack';
    $Self->{Translation}->{'Workplace'} = 'Arbeitsplatz';
    $Self->{Translation}->{'Outlet'} = 'Anschlussdose';
    $Self->{Translation}->{'IT Facility'} = 'IT-Einrichtung';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telko';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Client-Anwendung';
    $Self->{Translation}->{'Middleware'} = 'Middleware';
    $Self->{Translation}->{'Server Application'} = 'Server-Anwendung';
    $Self->{Translation}->{'Client OS'} = 'Client-Betriebssystem';
    $Self->{Translation}->{'Server OS'} = 'Server-Betriebssystem';
    $Self->{Translation}->{'Admin Tool'} = 'Administrator-Tool';
    $Self->{Translation}->{'User Tool'} = 'Benutzer-Tool';
    $Self->{Translation}->{'Embedded'} = 'Embedded';
    $Self->{Translation}->{'Single Licence'} = 'Einzellizenz';
    $Self->{Translation}->{'Per User'} = 'Pro Benutzer';
    $Self->{Translation}->{'Per Processor'} = 'Pro Prozessor';
    $Self->{Translation}->{'Per Server'} = 'Pro Server';
    $Self->{Translation}->{'Per Node'} = 'Pro Knoten';
    $Self->{Translation}->{'Volume Licence'} = 'Volume-Lizenz';
    $Self->{Translation}->{'Enterprise Licence'} = 'Enterprise-Lizenz';
    $Self->{Translation}->{'Developer Licence'} = 'Entwickler-Lizenz';
    $Self->{Translation}->{'Demo'} = 'Demo';
    $Self->{Translation}->{'Time Restricted'} = 'Zeitlich begrenzt';
    $Self->{Translation}->{'Freeware'} = 'Freeware';
    $Self->{Translation}->{'Open Source'} = 'Open Source';
    $Self->{Translation}->{'Unlimited'} = 'Unlimitiert';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = 'Alternativ zu';
    $Self->{Translation}->{'Assigned CIs'} = 'Zugewiesene Configuration Items';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = 'Automatisches Laden von ITSMConfigurationManagement-Erweiterungen.';
    $Self->{Translation}->{'CIs assigned to customer company'} = 'Zum Kunden zugewiesene Configurations Items';
    $Self->{Translation}->{'CIs assigned to customer user'} = 'Zum Kundenbenutzer zugewiesene Configurations Items';
    $Self->{Translation}->{'CMDB Settings'} = 'CMDB-Einstellungen';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        'Prüft Namen auf Eindeutigkeit innerhalb der selben Configuration Item-Klasse oder global, d.h. es werden alle Configuration Items jeglicher Configuration Item-Klasse bei der Prüfung auf einen eindeutigen Namen berücksichtigt.';
    $Self->{Translation}->{'Config Items'} = 'Configuration Items';
    $Self->{Translation}->{'Config item (dropdown)'} = 'Config-Item (Einfachauswahl)';
    $Self->{Translation}->{'Config item (multiselect)'} = 'Config-Item (Mehrfachauswahl)';
    $Self->{Translation}->{'Config item add.'} = 'Configuration Item hinzufügen.';
    $Self->{Translation}->{'Config item edit.'} = 'Configuration Item bearbeiten.';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        'Configuration Item-Ereignismodul, dass die Protokollierung der Historie im Agenten-Interface ermöglicht.';
    $Self->{Translation}->{'Config item history.'} = 'Configuration Item-Historie.';
    $Self->{Translation}->{'Config item print.'} = 'Configuration Item drucken.';
    $Self->{Translation}->{'Config item zoom.'} = 'Configuration Item-Detailansicht.';
    $Self->{Translation}->{'ConfigItemNumber'} = 'Configuration Item-Nummer';
    $Self->{Translation}->{'Configuration Item Limit'} = 'Configuration Item-Limit';
    $Self->{Translation}->{'Configuration Item limit per page.'} = 'Configuration Item-Limit pro Seite.';
    $Self->{Translation}->{'Configuration Management Database.'} = 'Configuration Management Database.';
    $Self->{Translation}->{'Configuration item bulk module.'} = 'Configuration Item-Sammelaktion.';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Die Definitionen von Configuration Item-Limit erstellen und verwalten.';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        'Definieren Sie Actions, in denen im Verknüpfte-Objekte-Widget ein Einstellungen-Knopf verfügbar sein soll (LinkObject::ViewMode = "complex"). Bitte beachten Sie, dass für diese Actions die folgenden JS- und CSS-Dateien registriert sein müssen:  Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js und Core.Agent.LinkObject.js.';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        'Definieren Sie eine Zuordnung zwischen Item-Klassen und Fontawesome-Symbolen, welche im Agenten-Interface angezeigt werden.';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        'Definiert die benötigten Berechtigungen zum Erstellen von ITSM Configuration Items durch das Generic Interface.';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        'Definiert die benötigten Berechtigungen zum Löschen von ITSM Configuration Items durch das Generic Interface.';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        'Definiert die benötigten Berechtigungen, um ITSM-Configuration Items über das Generic Interface abzurufen.';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        'Definiert die benötigten Berechtigungen, um nach ITSM-Configuration Items mit dem Generic Interface zu suchen.';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        'Definiert die benötigten Berechtigungen, um ITSM-Generic Items mit dem Generic Interface zu aktualisieren.';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        'Definiert das Übersichtsmodul um die "Klein"-Ansicht einer Configuration Items-Liste anzuzeigen.';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        'Definiert reguläre Ausdrücke individuell für jedes Configuration Item, um den Namen des Configuration Item zu überprüfen und entsprechende Fehlermeldungen anzuzeigen.';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        'Definiert das Standard-Unterobjekt der Klasse \'ITSMConfigItem\'.';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        'Definiert die Anzahl der Zeilen für den Editor der Configuration Item-Definition im Administrator-Interface.';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        'Definiert die Reihenfolge der Vorfallstatus von hoch (bspw. kritisch) nach niedrig (bspw. funktional).';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        'Definiert die relevanten Verwendungsstaus, bei denen verknüpfte Tickets den Status eines Configuration Items beeinflussen.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        'Definiert das Suchlimit für die ITSM Configuration-Ansicht.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        'Definiert das Suchlimit für die Configuration Items-Suche.';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        'Definiert die angezeigten Tabellenspalten in der Configuration Item-Übersicht. Diese Option hat keinen Effekt auf die Position der Spalten. Hinweis: Klassenspalte ist immer verfügbar, wenn Filter "Alle" gewählt ist.';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        'Definiert die angezeigten Tabellenspalten für die Configuration Item-Suche. Diese Option hat keinen Effekt auf die Position der Spalten.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Definiert die angezeigten Spalten von Configuration Items in der Configuration Item-Übersicht in Abhängigkeit von der Configuration Item-Klasse. Jedem Eintrag müssen der Klassenname und Doppelpunkte vorangestellt werden (z.B. Computer:::). Es gibt einige Configuration Item-Attribute, die allen Configuration Items gemeinsam sind (Beispiel für die Klasse Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). Um einzelne Configuration Item-Attribute, wie sie in der Configuration Item-Definition definiert sind, darzustellen, muss das folgende Schema verwendet werden (Beispiel für die Klasse Computer): Computer::HardDisk::1, Computer::HardDisk::1::Kapazität::1, Computer::HardDisk::2, Computer::HardDisk::2::Kapazität::1. Wenn es keinen Eintrag für eine Configuration Item-Klasse gibt, werden die Standard-Spalten so angezeigt, wie in der Einstellung "ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns" definiert.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Definiert die angezeigten Spalten von Configuration Items in der Configuration Item-Suche in Abhängigkeit von der Configuration Item-Klasse. Jedem Eintrag müssen der Klassenname und Doppelpunkte vorangestellt werden (z.B. Computer:::). Es gibt einige Configuration Item-Attribute, die allen Configuration Items gemeinsam sind (Beispiel für die Klasse Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). Um einzelne Configuration Item-Attribute, wie sie in der Configuration Item-Definition definiert sind, darzustellen, muss das folgende Schema verwendet werden (Beispiel für die Klasse Computer): Computer::HardDisk::1, Computer::HardDisk::1::Kapazität::1, Computer::HardDisk::2, Computer::HardDisk::2::Kapazität::1. Wenn es keinen Eintrag für eine Configuration Item-Klasse gibt, werden die Standard-Spalten so angezeigt, wie in der Einstellung "ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns" definiert.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        'Definiert die angezeigten Spalten von Configuration Items in der Configuration Item-Übersicht in Abhängigkeit von der Configuration Item-Klasse. Jedem Eintrag müssen der Klassenname und Doppelpunkte vorangestellt werden (z.B. Computer:::). Es gibt einige Configuration Item-Attribute, die allen Configuration Items gemeinsam sind (Beispiel für die Klasse Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). Um einzelne Configuration Item-Attribute, wie sie in der Configuration Item-Definition definiert sind, darzustellen, muss das folgende Schema verwendet werden (Beispiel für die Klasse Computer): Computer::HardDisk::1, Computer::HardDisk::1::Kapazität::1, Computer::HardDisk::2, Computer::HardDisk::2::Kapazität::1. Wenn es keinen Eintrag für eine Configuration Item-Klasse gibt, werden die Standard-Spalten so angeuzeigt, wie in der Einstellung "AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany" definiert (Schlüssel "DefaultColumns").';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        'Definiert die angezeigten Spalten in der Komplexansicht der Verknüpfungstabelle für alle Configuration Item-Klassen. Wenn es keinen Eintrag gibt, werden die Standardspalten angezeigt.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        'Definiert die angezeigten Spalten von Configuration Items in der Komplexansicht der Verknüpfungstabelle abhängig von der Configuration Item-Klasse. Definiert die angezeigten Spalten von Configuration Items in der Configuration Item-Übersicht in Abhängigkeit von der Configuration Item-Klasse. Jedem Eintrag müssen der Klassenname und Doppelpunkte vorangestellt werden (z.B. Computer:::). Es gibt einige Configuration Item-Attribute, die allen Configuration Items gemeinsam sind (Beispiel für die Klasse Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). Um einzelne Configuration Item-Attribute, wie sie in der Configuration Item-Definition definiert sind, darzustellen, muss das folgende Schema verwendet werden (Beispiel für die Klasse Computer): Computer::HardDisk::1, Computer::HardDisk::1::Kapazität::1, Computer::HardDisk::2, Computer::HardDisk::2::Kapazität::1. Wenn es keinen Eintrag für eine Configuration Item-Klasse gibt, werden die Standardspalten angezeigt.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        'Legt fest, welche Art von Link (aus der Ticketperspektive benannt) den Status eines verlinkten CI beeinflussen kann.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        'Legt fest, welche Art von Link (aus der Ticket-Perspektive) verwendet wird, um Tickets und Config-Items zu verlinken.';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        'Definiert, welcher Typ von Ticket den Status eines verknüpften Configuration Items beeinflussen kann.';
    $Self->{Translation}->{'Delete Configuration Item'} = 'Configuration Item löschen.';
    $Self->{Translation}->{'DependsOn'} = 'Hängt ab von';
    $Self->{Translation}->{'Deployment State Color'} = 'Verwendungsstatus-Farbe';
    $Self->{Translation}->{'Duplicate'} = 'Duplizieren';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = 'Aktiviert die Suche nach Config-Items basierend auf dem Kundenbenutzer.';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        'Aktiviert die Suche nach Config-Items basierend auf dem Kunden.';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        'Aktiviert die Funktion "Sammelaktion" für das Agenten-Frontend, um zur gleichen Zeit an mehreren Configuration Items arbeiten zu können.';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        'Aktiviert die Funktion "Sammelaktion" nur für die gelisteten Gruppen.';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '(De-)Aktiviert die Funktionalität um ConfigItems auf eindeutige Namen zu überprüfen. Bevor Sie diese Option aktivieren, sollten Sie Ihr System auf bereits vorhandene ConfigItems mit gleichem Namen überprüfen. Sie können dies mit Hilfe des scripts bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates tun.';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = 'Ein Ereignismodul das ein Configuration Item verlinkt.';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        'Ein Ereignismodul das beim Verknüpfen von Tickets mit Configuration Items den Status des Configuration Items verändert.';
    $Self->{Translation}->{'ITSM config item overview.'} = 'ITSM Configuration Item-Übersicht';
    $Self->{Translation}->{'Limit for config item search.'} = 'Limit für die Config-Item-Suche.';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = 'Modul zur Überprüfung der Gruppe, die für eine Klasse verantwortlich ist.';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        'Modul, um die Gruppe zu überprüfen, die für ein Configuration Item verantwortlich ist.';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = 'Modul zum Generieren von ITSM-Configuration Item-Statistiken.';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        'Nicht mehr in Verwendung. Nicht ändern. Notwendig wg. Kompatibilität zu vorherigen Paketversionen.';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        'Objekt-Backend Modul Registration des Import/Export Moduls.';
    $Self->{Translation}->{'Overview.'} = 'Übersicht.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Legt die Parameter für das Dashboard-Backend fest. "Limit" legt die Anzahl an Einträgen fest, die standardmäßig angezeigt werden. "Group" beschränkt den Zugang zum jeweiligen Dashlet (z. B. Group: admin;group1;group2). "Default" bestimmt, ob das Dashlet standardmäßig aktiv ist oder vom Nutzer manuell aktiviert werden muss. "CacheTTLLocal" bestimmt die Cachingdauer für das Dashlet in Minuten.';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        'Parameter für die Farben der Verwendungsstatus in den persönlichen Einstellungen im Agenten-Interface.';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        'Parameter für die Farben der Verwendungsstatus in den persönlichen Einstellungen im Agenten-Interface.';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parameter für die zugriffsberechtigte Gruppe der General-Katalog-Attribute.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        'Parameter für die Seiten, in denen Configuration Items angezeigt werden.';
    $Self->{Translation}->{'Permission Group'} = 'Berechtigungsgruppe';
    $Self->{Translation}->{'RelevantTo'} = 'Relevant für';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = 'Registriert einen Output-Filter, der ein Config-Item-Widget anzeigt.';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
        'Erforderliche Gruppenberechtigungen für die Verwendung des Customer-Config-Item-Widgets in der Agentenoberfläche.';
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} =
        'Registriert ein ITSMConfigItem-EventModule, das zugehörige Anhänge speichert.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        'Erforderliche Berechtigungen, um die Ansicht zur Konfiguration von Configuration Items im Agenten-Interface nutzen zu können.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        'Erforderliche Berechtigungen, um die Ansicht zur Suche von Configuration Items im Agenten-Interface nutzen zu können.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        'Erforderliche Berechtigungen, um die Detailansicht von Configuration Items im Agenten-Interface nutzen zu können.';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        'Erforderliche Berechtigungen, um die Ansicht zum Hinzufügen von Configuration Items im Agenten-Interface nutzen zu können.';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        'Erforderliche Berechtigungen für die Verwendung des Config-Item-Dialogs in der Agentenschnittstelle.';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        'Erforderliche Berechtigungen, um die Ansicht zum Bearbeiten von Configuration Items im Agenten-Interface nutzen zu können.';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        'Erforderliche Berechtigungen, um die Historien-Ansicht von Configuration Items im Agenten-Interface nutzen zu können.';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        'Erforderliche Berechtigungen, um die Ansicht zum Drucken von Configuration Items im Agenten-Interface nutzen zu können.';
    $Self->{Translation}->{'Required privileges to delete config items.'} = 'Erforderliche Berechtigungen, um Configuration Items löschen zu können.';
    $Self->{Translation}->{'Search config items.'} = 'Configuration Items suchen.';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        'Wählt das Modul zum Erzeugen der Configuration Item-Nummer. "AutoIncrement" erhöht die Nummer des Configuration Items, die System-ID, die Klassen-ID und des benutzen Zählers. Das Format ist "SystemID.ConfigItemClassID.Counter", bspw. 1205000004, 1205000005.';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        'Settzt den Vorfallstatus eines Configuration Items automatisch, wenn ein Ticket mit einem Configuration Item verknüpft wird.';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        'Setzt den Verwendungsstatus für die Funktion "Sammelaktion" für Configuration Items im Agenten-Interace.';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        'Setzt den Verwendungsstatus für die Funktion "Sammelaktion" für Configuration Items im Agenten-Interace.';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        'Zeigt eine Verknüpfung im Menü, die es erlaubt ein Configuration Item mit einem anderen Objekt in der Detailansicht des Configuration Items des Agenten-Interface zu verknüpfen.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        'Zeigt im Menü der Configuration Item-Übersicht die Verknüpfung zur Historie eines Configuration Items.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        'Zeigt im Menü der Configuration Item-Detailansicht die Verknüpfung zur Historie eines Configuration Items.';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        'Zeigt im Menü der Detailansicht eines Configuration Items die Verknüpfung zum Löschen eines Configuration Items.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        'Zeigt im Menü der Configuration Item-Übersicht die Verknüpfung zum Duplizieren eines Configuration Items.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        'Zeigt im Menü der Detailansicht eines Configuration Items die Verknüpfung zum Duplizieren eines Configuration Items.';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        'Zeigt im Menü der Detailansicht eines Configuration Items die Verknüpfung zum Bearbeiten eines Configuration Items.';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        'Zeigt im Menü der Detailansicht eines Configuration Item eine Verknüpfung zum Zurückgehen auf die vorherige Ansicht.';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        'Zeigt im Menü der Detailansicht eines Configuration Items die Verknüpfung zum Drucken eines Configuration Items.';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        'Zeigt eine Verknüpfung im Menü der Ansicht "Configuration Item-Übersicht", um zur Detailansicht des Configuration Items zu gelangen.';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        'Zeigt die Configuration Item-Historie (umgekehrte Reihenfolge) im Agenten-Interface an.';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        'Der Identifikator für ein Configuration Item, bspw. "ConfigItem#" , "MeinConfigItem#". Der Standard ist "ConfigItem#".';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        'Ticket-Event-Modul, das Links zwischen Tickets und Config-Items erstellt und entfernt.';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = 'Ticket-Event-Modul, das Dynamische Felter aktualisiert.';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        'Diese Konfiguration registriert ein Frontend-Modul für die Agenten-Interface, das die AJAX-Schnittstelle für AgentITSMConfigItemCustomerCIsWidget bereitstellt.';
    $Self->{Translation}->{'class'} = 'Klasse';
    $Self->{Translation}->{'global'} = 'Global';
    $Self->{Translation}->{'postproductive'} = 'Post-produktiv';
    $Self->{Translation}->{'preproductive'} = 'Pre-produktiv';
    $Self->{Translation}->{'productive'} = 'Produktiv';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::el_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Μεταξύ';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'Γραφείο';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::en_CA_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = '';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = 'License Type';
    $Self->{Translation}->{'Licence Key'} = 'License Key';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = 'Keyboard';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = '';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = 'Single License';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = 'Volume License';
    $Self->{Translation}->{'Enterprise Licence'} = 'Enterprise License';
    $Self->{Translation}->{'Developer Licence'} = 'Developer License';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::en_GB_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = 'Change';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = 'Last changed';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = 'History Content';
    $Self->{Translation}->{'Createtime'} = 'Createtime';
    $Self->{Translation}->{'Zoom view'} = 'Zoom view';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Context Settings';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = 'Last changed by';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = 'No Result!';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Between';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = 'Server';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = 'Keyboard';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'Office';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parameters for the example permission groups of the general catalogue attributes.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::es_CO_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = 'Vista detallada';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Entre';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'Oficina';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::es_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Gestión de Elementos de Configuración';
    $Self->{Translation}->{'Change class definition'} = 'Cambiar definición de la Clase';
    $Self->{Translation}->{'Config Item Class'} = 'Clase de elemento de configuración';
    $Self->{Translation}->{'Definition'} = 'Definición';
    $Self->{Translation}->{'Change'} = 'Modificar';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Estado del Incidente';
    $Self->{Translation}->{'Deployment State'} = 'Estado de la Implementación';
    $Self->{Translation}->{'Class'} = 'Clase';
    $Self->{Translation}->{'Deployment State Type'} = 'Tipo de estado de implementación';
    $Self->{Translation}->{'Current Incident State'} = 'Estado Actual del Incidente';
    $Self->{Translation}->{'Current Incident State Type'} = 'Estado actual del Tipo de incidente';
    $Self->{Translation}->{'Last changed'} = 'Último cambio';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Elemento de Configuración';
    $Self->{Translation}->{'Filter for Classes'} = 'Filtro para Clases';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Seleccione una Clase de la lista para crear un nuevo Item de Configuración';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'ITSM ConfigItem Acción Masiva';
    $Self->{Translation}->{'Deployment state'} = 'Estado de despliegue';
    $Self->{Translation}->{'Incident state'} = 'Estado del incidente';
    $Self->{Translation}->{'Link to another'} = 'Enlazar con otro';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '¡Número de Ítem de Configuración no valido!';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'El número de otro elemento de configuración con el que vincular.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '¿Realmente desea eliminar este elemento de configuración?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'El nombre de este elemento de configuración';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'El nombre ya esta en uso por Ítems de Configuración con los siguientes número(s): %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = 'Historial del Elemento de configuración: %s';
    $Self->{Translation}->{'History Content'} = 'Contenido del historial';
    $Self->{Translation}->{'Createtime'} = 'Fecha de Creación';
    $Self->{Translation}->{'Zoom view'} = 'Vista detallada ';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Ajustes de Contexto';
    $Self->{Translation}->{'Config Items per page'} = 'CIs por página';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = 'Una tabla genérica de ítems de configuración de ITSM';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Ejecutar la búsqueda';
    $Self->{Translation}->{'Also search in previous versions?'} = '¿Desea buscar en versiones anteriores?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Elemento de configuración';
    $Self->{Translation}->{'Configuration Item Information'} = 'Información del elemento de configuración';
    $Self->{Translation}->{'Current Deployment State'} = 'Estado Actual de la Implementación';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = 'Mostrar una versión';
    $Self->{Translation}->{'Show all versions'} = 'Mostrar todas las versiones';
    $Self->{Translation}->{'Version Incident State'} = 'Estado del incidente de la versión';
    $Self->{Translation}->{'Version Deployment State'} = 'Estado de la versión de implementación';
    $Self->{Translation}->{'Version Number'} = 'Número de versión';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Elemento de configuración Detalles de la versión';
    $Self->{Translation}->{'Property'} = 'Propiedad';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '¡No se ha proporcionado acceso a Clase!';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = 'Resumen: ITSM ConfigItem';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '¡No se ha proporcionado ConfigItemID!';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '¡Necesita al menos un elemento de configuración seleccionado!';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        'No tiene permiso de escritura para este elemento de configuración: %s';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '¡Elemento de configuración "%s"  no se encontró en la base de datos!';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '¡No fue posible eliminar el configitem ID %s!';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '¡No se encontró Versión para ConfigItemID %s!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '¡No se ha proporcionado ConfigItemID, DuplicateID o ClassID!';
    $Self->{Translation}->{'No access is given!'} = '¡No se ha proporcionado acceso!';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '¡Ninguna definición fue definida para clase %s!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '¡No se puede mostrar el historial, no se ha proporcionado ConfigItemID!';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '¡No se puede mostrar el historial, no se han proporcionado permisos de acceso!';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = 'Nuevo ConfigItem (ID=%s)';
    $Self->{Translation}->{'New version (ID=%s)'} = 'Nueva versión (ID=%s)';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = 'Actualizado estado de despliegue (nuevo=%s, viejo=%s)';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = 'Actualizado estado de incidencia (nuevo=%s,viejo=%s)';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = 'ConfigItem (ID=%s) eliminado';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = 'Enlace a %s (tipo=%s) añadido';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = 'Enlace a %s (tipo=%s) eliminado';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = 'Actualizada definición de ConfigItem (ID=%s)';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = 'Nombre actualizado (nuevo=%s,viejo=%s)';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = 'Atributo %s actualizado de "%s" a "%s"';
    $Self->{Translation}->{'Version %s deleted'} = 'Versión %s eliminada';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '¡No se ha proporcionado ConfigItemID o VersionID!';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '¡No se puede mostrar el elemento de configuración, no se han proporcionado permisos de acceso!';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '¡ConfigItemID %s no se encontró en la base de datos!';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '¡VersionID %s no se encontró en la base de datos!';
    $Self->{Translation}->{'ConfigItem'} = 'ConfigItem';
    $Self->{Translation}->{'printed by %s at %s'} = 'impreso por %s en %s';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '¡ClassID no es válida!';
    $Self->{Translation}->{'No ClassID is given!'} = '¡No se ha proporcionado ClassID!';
    $Self->{Translation}->{'No access rights for this class given!'} = '¡No se ha proporcionado permisos de acceso para esta clase!';
    $Self->{Translation}->{'No Result!'} = '¡Sin resultados!';
    $Self->{Translation}->{'Config Item Search Results'} = 'Resultados de búsqueda de elemento de configuración';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '¡No se puede mostrar el elemento, no se han proporcionado permisos de acceso para ConfigItem!';
    $Self->{Translation}->{'operational'} = 'operacional';
    $Self->{Translation}->{'warning'} = 'advertencia';
    $Self->{Translation}->{'incident'} = 'incidente';
    $Self->{Translation}->{'The deployment state of this config item'} = 'El estado de implementación de este elemento de configuración';
    $Self->{Translation}->{'The incident state of this config item'} = 'El estado de este elemento de configuración';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = 'Se muestran elementos de configuración';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Entre';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Número máximo de un elemento ';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Los campos vacíos indican que los valores actuales se mantienen';
    $Self->{Translation}->{'Skipped'} = 'Saltado';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Modelo';
    $Self->{Translation}->{'Customer Company'} = 'Empresa del Cliente';
    $Self->{Translation}->{'Serial Number'} = 'Número Serial';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = 'RAM';
    $Self->{Translation}->{'Hard Disk'} = 'Disco Duro';
    $Self->{Translation}->{'Capacity'} = 'Capacidad';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Adaptador de Red';
    $Self->{Translation}->{'IP over DHCP'} = 'IP sobre DHCP';
    $Self->{Translation}->{'IP Address'} = 'Dirección IP';
    $Self->{Translation}->{'Graphic Adapter'} = 'Tarjeta Gráfica';
    $Self->{Translation}->{'Other Equipment'} = 'Otro Equipo';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Fecha de Expiración de la Garantía';
    $Self->{Translation}->{'Install Date'} = 'Fecha de Instalación';
    $Self->{Translation}->{'Phone 1'} = 'Teléfono 1';
    $Self->{Translation}->{'Phone 2'} = 'Teléfono 2';
    $Self->{Translation}->{'E-Mail'} = 'E-Mail';
    $Self->{Translation}->{'Network Address'} = 'Dirección de Red';
    $Self->{Translation}->{'Subnet Mask'} = 'Máscara de Subred';
    $Self->{Translation}->{'Gateway'} = 'Puerta de enlace';
    $Self->{Translation}->{'Licence Type'} = 'Tipo de Licencia';
    $Self->{Translation}->{'Licence Key'} = 'Clave de Licencia';
    $Self->{Translation}->{'Quantity'} = 'Cantidad';
    $Self->{Translation}->{'Expiration Date'} = 'Fecha de Expiración';
    $Self->{Translation}->{'Media'} = 'Medio';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = 'Ordenador';
    $Self->{Translation}->{'Hardware'} = 'Hardware';
    $Self->{Translation}->{'Network'} = 'Red';
    $Self->{Translation}->{'Software'} = 'Software';
    $Self->{Translation}->{'Expired'} = 'Expirado';
    $Self->{Translation}->{'Maintenance'} = 'Mantenimiento';
    $Self->{Translation}->{'Pilot'} = 'Piloto';
    $Self->{Translation}->{'Planned'} = 'Planificado';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Reparación';
    $Self->{Translation}->{'Retired'} = 'Retirado';
    $Self->{Translation}->{'Review'} = 'Revisado';
    $Self->{Translation}->{'Test/QA'} = 'Prueba / Control de calidad';
    $Self->{Translation}->{'Laptop'} = 'Ordenador portátil';
    $Self->{Translation}->{'Desktop'} = 'Escritorio';
    $Self->{Translation}->{'PDA'} = 'Agenda Personal';
    $Self->{Translation}->{'Server'} = 'Servidor';
    $Self->{Translation}->{'Other'} = 'Otro';
    $Self->{Translation}->{'Monitor'} = 'Monitor';
    $Self->{Translation}->{'Printer'} = 'Impresora';
    $Self->{Translation}->{'Switch'} = 'Switch';
    $Self->{Translation}->{'Router'} = 'Enrutador';
    $Self->{Translation}->{'WLAN Access Point'} = 'Punto de Acceso WLAN';
    $Self->{Translation}->{'Security Device'} = 'Dispositivo de Seguridad';
    $Self->{Translation}->{'Backup Device'} = 'Dispositivo de Respaldo';
    $Self->{Translation}->{'Mouse'} = 'Ratón';
    $Self->{Translation}->{'Keyboard'} = 'Teclado';
    $Self->{Translation}->{'Camera'} = 'Cámara';
    $Self->{Translation}->{'Beamer'} = 'Proyector';
    $Self->{Translation}->{'Modem'} = 'Módem';
    $Self->{Translation}->{'PCMCIA Card'} = 'Tarjeta PCMCIA';
    $Self->{Translation}->{'USB Device'} = 'Dispositivo USB';
    $Self->{Translation}->{'Docking Station'} = 'Estación de acoplamiento';
    $Self->{Translation}->{'Scanner'} = 'Escáner';
    $Self->{Translation}->{'Building'} = 'Edificio';
    $Self->{Translation}->{'Office'} = 'Oficina';
    $Self->{Translation}->{'Floor'} = 'Piso';
    $Self->{Translation}->{'Room'} = 'Habitación';
    $Self->{Translation}->{'Rack'} = 'Rack';
    $Self->{Translation}->{'Workplace'} = 'Lugar de trabajo';
    $Self->{Translation}->{'Outlet'} = 'Outlet';
    $Self->{Translation}->{'IT Facility'} = 'Instalaciones de TI';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Conferencia telefónica';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Aplicación Cliente';
    $Self->{Translation}->{'Middleware'} = 'Middleware';
    $Self->{Translation}->{'Server Application'} = 'Aplicación Servidor';
    $Self->{Translation}->{'Client OS'} = 'Sistema Operativo del Cliente';
    $Self->{Translation}->{'Server OS'} = 'Sistema operativo del Servidor';
    $Self->{Translation}->{'Admin Tool'} = 'Herramienta de Administración';
    $Self->{Translation}->{'User Tool'} = 'Herramienta de Usuario';
    $Self->{Translation}->{'Embedded'} = 'Embebido';
    $Self->{Translation}->{'Single Licence'} = 'Licencia Única';
    $Self->{Translation}->{'Per User'} = 'Por Usuario';
    $Self->{Translation}->{'Per Processor'} = 'Por Procesador';
    $Self->{Translation}->{'Per Server'} = 'Por Servidor';
    $Self->{Translation}->{'Per Node'} = 'Por Nodo';
    $Self->{Translation}->{'Volume Licence'} = 'Licencia por Volumen';
    $Self->{Translation}->{'Enterprise Licence'} = 'Licencia Corporativa';
    $Self->{Translation}->{'Developer Licence'} = 'Licencia de Desarrollador';
    $Self->{Translation}->{'Demo'} = 'Demostración';
    $Self->{Translation}->{'Time Restricted'} = 'Tiempo Restringido';
    $Self->{Translation}->{'Freeware'} = 'Freeware';
    $Self->{Translation}->{'Open Source'} = 'Código Abierto';
    $Self->{Translation}->{'Unlimited'} = 'Ilimitada';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = 'CIs asignados';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = 'Cis asignados a la Empresa del Cliente';
    $Self->{Translation}->{'CIs assigned to customer user'} = 'Cis asignados al usuario del cliente';
    $Self->{Translation}->{'CMDB Settings'} = 'Configuraciones CMDB';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        'Busque un nombre único solo dentro de la misma clase de ConfigItem (\'clase\') o globalmente (\'global\'), lo que significa que cada ConfigItem existente se tiene en cuenta cuando se buscan duplicados.';
    $Self->{Translation}->{'Config Items'} = 'Elementos de configuración';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = 'Añadir elemento de configuración';
    $Self->{Translation}->{'Config item edit.'} = 'Editar elemento de configuración.';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        'Módulo de evento de elementos de configuración que habilita el acceso a la historia en la interfaz del agente.';
    $Self->{Translation}->{'Config item history.'} = 'Historial del elemento de configuración.';
    $Self->{Translation}->{'Config item print.'} = 'Imprimir elemento de configuración.';
    $Self->{Translation}->{'Config item zoom.'} = 'Detallez del elemento de configuración.';
    $Self->{Translation}->{'ConfigItemNumber'} = 'ConfigItemNumber';
    $Self->{Translation}->{'Configuration Item Limit'} = 'Límite de Elementos de Configuración';
    $Self->{Translation}->{'Configuration Item limit per page.'} = 'Límite de elemento de configuración por página.';
    $Self->{Translation}->{'Configuration Management Database.'} = 'Gestión de la Configuración de Bases de Datos';
    $Self->{Translation}->{'Configuration item bulk module.'} = 'Elemento de configuración del módulo masivo';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        'Router backend de búsqueda de elementos de configuración de la interfaz del agente.';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Crea y gestiona las definiciones de los elementos de configuración.';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        'Definir acciones donde está disponible un botón de configuración en el widget de objetos vinculados (LinkObject::ViewMode = "complex"). Tenga en cuenta que estas acciones deben haber registrado los siguientes archivos JS y CSS: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js y Core.Agent.LinkObject.js.';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        'Define los permisos necesarios para crear elementos de configuración de ITSM utilizando la interfaz genérica.';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        'Define los permisos necesarios para eliminar elementos de configuración de ITSM mediante la interfaz genérica.';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        'Define los permisos necesarios para obtener elementos de configuración de ITSM utilizando la interfaz genérica.';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        'Define los permisos necesarios para buscar elementos de configuración de ITSM utilizando la interfaz genérica.';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        'Define los permisos necesarios para actualizar los elementos de configuración de ITSM mediante la interfaz genérica.';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        'Define un módulo de resumen para mostrar la pequeña vista de una lista de elemento de configuración.';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        'Define expresiones regulares individualmente para cada clase ConfigItem para verificar el nombre de ConfigItem y para mostrar los mensajes de error correspondientes.';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        'Define el subobjeto por default de la clase \'ITSMConfigItem\'.';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        'Define el número de filas para el editor de definiciones de CI en la interfaz de administración.';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        'Define el orden de los estados de incidencia desde alto (por ejemplo, crítico) a bajo (por ejemplo, funcional).';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        'Define los estados de implementación relevantes donde los tickets vinculados pueden afectar el estado de un CI.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        'Define el límite de búsqueda para la pantalla AgentITSMConfigItem.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        'Define el límite de búsqueda para la pantalla AgentITSMConfigItemSearch.';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        'Define las columnas mostradas en la descripción general del elemento de configuración. Esta opción no tiene efecto en la posición de la columna. Nota: La columna de clase siempre está disponible si se selecciona el filtro \'Todos\'.';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        'Define las columnas mostradas en la búsqueda de elementos de configuración. Esta opción no tiene efecto en la posición de la columna.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Define las columnas de los elementos de configuración que se muestran en la vista general del elemento de configuración, en función de la clase CI. Cada entrada debe ir precedida del nombre de la clase y dos puntos dobles (es decir, computadora: :). Hay algunos atributos de CI que son comunes a todos los elementos de configuración (ejemplo para la clase Computadora: Computadora :: Nombre, Computadora :: CurDeplState, Computadora :: CreateTime). Para mostrar los IC-Atributos individuales definidos en la Definición-CI, se debe utilizar el siguiente esquema (ejemplo para la clase Computadora): Computadora :: Disco Duro :: 1, Computadora :: Disco Duro :: 1 :: Capacidad :: 1, Computadora :: Disco Duro :: 2, Computadora :: Disco Duro :: 2 :: Capacidad :: 1. Si no hay ninguna entrada para una clase CI, las columnas predeterminadas se muestran como se define en la configuración ITSMConfigItem :: Frontend :: AgentITSMConfigItem ### ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Define las columnas de los elementos de configuración mostrados en la búsqueda del elemento config dependiendo de la clase CI. Cada entrada debe ir precedida del nombre de la clase y dos puntos dobles (es decir, computadora: :). Hay algunos atributos de CI que son comunes a todos los elementos de configuración (ejemplo para la clase Computadora: Computadora :: Nombre, Computadora :: CurDeplState, Computadora :: CreateTime). Para mostrar los IC-Atributos individuales definidos en la Definición-CI, se debe utilizar el siguiente esquema (ejemplo para la clase Computadora): Computadora :: Disco Duro :: 1, Computadora :: Disco Duro :: 1 :: Capacidad :: 1, Computadora :: Disco Duro :: 2, Computadora :: Disco Duro :: 2 :: Capacidad :: 1. Si no hay ninguna entrada para una clase CI, las columnas predeterminadas se muestran como se define en la configuración ITSMConfigItem :: Frontend :: AgentITSMConfigItem ### ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        'De las columnas de elemento de configuración mostradas en el widget según la clase de CI. Cada entrada debe ir precedida con el nombre de la clase y dos puntos dobles( es decir; Ordenador::). Hay algunos atributos de CI que son comunes a todos los CIs (ejemplo para la clase Computer: Ordenador::Nombre, Ordenador::CurDeplState, Ordenador::FechaCreación). Para mostrar los atributos de CI individuales definidos en la definición-CI, se debe utilizar el siguiente esquema (ejemplo para la clase Ordenador) Ordenador::DiscoDuro::1, Ordenador::DiscoDuro::1::Capacidad::1, Ordenador::DiscoDuro::2, Ordenador::DiscoDuro::2::Capacidad::1. Si no hay ninguna entrada para una clase CI, las columnas por defecto se muestran en la siguiente configuración AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        'Define las columnas de los elementos de configuración mostrados en la vista compleja de la tabla de enlaces, según la clase de CI. Cada entrada debe ir precedida del nombre de la clase y dos puntos dobles (es decir, computadora: :). Hay algunos atributos de CI que son comunes a todos los elementos de configuración (ejemplo para la clase Computadora: Computadora :: Nombre, Computadora :: CurDeplState, Computadora :: CreateTime). Para mostrar los IC-Atributos individuales definidos en la Definición-CI, se debe utilizar el siguiente esquema (ejemplo para la clase Computadora): Computadora :: Disco Duro :: 1, Computadora :: Disco Duro :: 1 :: Capacidad :: 1, Computadora :: Disco Duro :: 2, Computadora :: Disco Duro :: 2 :: Capacidad :: 1. Si no hay una entrada para una clase CI, se muestran las columnas predeterminadas.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        'Define qué tipo de enlace (nombrado desde la perspectiva del ticket) puede afectar el estado de un CI vinculado.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        'Define qué tipo de ticket puede afectar el estado de un CI vinculado.';
    $Self->{Translation}->{'Delete Configuration Item'} = 'Eliminar elemento de configuración';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = 'Color del estado de despliegue';
    $Self->{Translation}->{'Duplicate'} = 'Duplicado';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        'Permite que la característica de acción masiva del elemento de configuración para la interfaz del agente funcione en más de un elemento de configuración a la vez.';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        'Habilita la característica de acción masiva del elemento de configuración solo para los grupos enumerados.';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        'Habilita / deshabilita la funcionalidad para verificar ConfigItems para nombres únicos. Antes de habilitar esta opción, debe verificar en su sistema los elementos de configuración ya existentes con nombres duplicados. Puede hacer esto con el script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        'Módulo de eventos para configurar configitem-status en ticket-configitem-link.';
    $Self->{Translation}->{'ITSM config item overview.'} = 'Descripción general del elemento de configuración de ITSM.';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = 'Módulo para verificar el grupo responsable de una clase.';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        'Módulo para verificar el grupo responsable de un elemento de configuración.';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = 'Módulo para generar estadísticas de los elementos de configuración ITSM.';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        'Registro del módulo de objeto backend para el módulo de importación/exportación.';
    $Self->{Translation}->{'Overview.'} = 'Vista general.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parámetros para el backend del Panel Principal del resumen de elemento de configuración de la empresa del cliente de la interfaz del agente. "Límite" es el número de entradas mostradas por defecto. "Grupo" se usa para restringir el accedo al complemento (por ejemplo, Grupo: admin;grupo1;grupo2). "Por defecto" determina si el complemento está habilitado por defecto o si el usuaro necesita habilitarlo manualmente. "CacheTTLLocal" es el tiempo de caché en minutos para el complemento.';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        'Los parámetros para la implementación indican el color en la vista de preferencias de la interfaz de agente.';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        'Parámetros para los estados de implementación en la vista de preferencias de la interfaz del agente.';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parámetros para los permisos de ejemplo de los atributos del catálogo general.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        'Parámetros para las páginas (donde se muestran los elementos de configuración)';
    $Self->{Translation}->{'Permission Group'} = 'Grupo de permisos';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        'Permisos necesarios para usar la ventana de los elementos de configuración en la interfaz del agente.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        'Permisos necesarios para usar la ventana de búsqueda de elementos de configuración en la interfaz del agente.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        'Permisos necesarios para usar la ventana de detalle de los elementos de configuración en la interfaz del agente.';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        'Permisos necesarios para usar la ventana para agregar elementos de configuración en la interfaz del agente.';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        'Permisos necesarios para usar la ventana de edición de los elementos de configuración en la interfaz del agente.';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        'Permisos necesarios para usar la ventana de historial de los elementos de configuración en la interfaz del agente.';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        'Permisos necesarios para usar la ventana de impresión de elementos de configuración en la interfaz del agente.';
    $Self->{Translation}->{'Required privileges to delete config items.'} = 'Privilegios necesarios para eliminar los elementos de configuración.';
    $Self->{Translation}->{'Search config items.'} = 'Buscar elementos de configuración.';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        'Selecciona el módulo generador de números de los elementos de configuración. "AutoIncrementar" incrementa el número del elemento de configuración; el SystemID, el ConfigItemClassID y el contador son usados. El formato es "SystemID.ConfigItemClassID.Counter", por ejemplo: 1205000004, 1205000005.';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        'Establezca el estado de incidente de un CI automáticamente cuando un ticket esté vinculado a un CI.';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        'Establece el estado de implementación en la pantalla masiva del elemento de configuración de la interfaz de agente.';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        'Establece el estado del incidente en la pantalla masiva del elemento de configuración de la interfaz de agente.';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        'Muestra un link en el menú para vincular un elemento de configuración con otro objeto, en la vista detallada de dicho elemento de configuración de la interfaz del agente.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        'Muestra un enlace en el menú para acceder al historial de un elemento de configuración en la vista general del elemento de configuración de la interfaz de agente.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        'Muestra un link en el menú para acceder a la historia de un elemento de configuracion en su vista detallada, en la interfaz del agente.';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        'Muestra un enlace en el menú para eliminar un elemento de configuración en su vista ampliada de la interfaz de agente.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        'Muestra un enlace en el menú para duplicar un elemento de configuración en la descripción general del elemento de configuración de la interfaz de agente.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        'Muestra un link en el menú para duplicar un elemento de configuracion en su vista detallada, en la interfaz del agente.';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        'Muestra un link en el menú para editar un elemento de configuracion en su vista detallada, en la interfaz del agente.';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        'Muestra un enlace en el menú para volver a la vista de zoom de elemento de configuración de la interfaz de agente.';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        'Muestra un link en el menú para imprimir un elemento de configuracion en su vista detallada, en la interfaz del agente.';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        'Muestra un enlace en el menú para acercarse a un elemento de configuración en la vista general del elemento de configuración de la interfaz de agente.';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        'Muestra la historia de los elementos de configuración (ordenados inversamente) en la interfaz del agente.';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        'El identificador de un elemento de configuración, por ejemplo. ConfigItem #, MyConfigItem #. El valor por defecto es ConfigItem #.';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = 'clase';
    $Self->{Translation}->{'global'} = 'global';
    $Self->{Translation}->{'postproductive'} = 'postproductivo';
    $Self->{Translation}->{'preproductive'} = 'preproductivo';
    $Self->{Translation}->{'productive'} = 'productivo';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::es_MX_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Estado del Incidente';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = 'Clase';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = 'Estado del Incidente Actual';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = 'Último cambio';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = 'Vista detallada';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Entre';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = 'Omitido';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Módelo';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = 'Disco Duro';
    $Self->{Translation}->{'Capacity'} = 'Capacidad';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Adaptador de Red';
    $Self->{Translation}->{'IP over DHCP'} = 'IP sobre DHCP';
    $Self->{Translation}->{'IP Address'} = 'Dirección IP';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = 'Otro Equipamiento';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = 'Fecha de Instalación';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'Dirección de Red';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = 'Cantidad';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'Expirado';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Reparar';
    $Self->{Translation}->{'Retired'} = 'Retirado';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = 'Laptop';
    $Self->{Translation}->{'Desktop'} = 'Escritorio';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Servidor';
    $Self->{Translation}->{'Other'} = 'Otro';
    $Self->{Translation}->{'Monitor'} = 'Monitor';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = 'Switch';
    $Self->{Translation}->{'Router'} = 'Ruteador';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = 'Dispositivo de Respaldo';
    $Self->{Translation}->{'Mouse'} = 'Mouse';
    $Self->{Translation}->{'Keyboard'} = 'Teclado';
    $Self->{Translation}->{'Camera'} = 'Cámara';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = 'Modem';
    $Self->{Translation}->{'PCMCIA Card'} = 'Tarjeta PCMCIA';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = 'Escaner ';
    $Self->{Translation}->{'Building'} = 'Edificio';
    $Self->{Translation}->{'Office'} = 'Oficina';
    $Self->{Translation}->{'Floor'} = 'Piso';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = 'Rack';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Aplicación del Cliente';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = 'Sistema Operativo del Cliente';
    $Self->{Translation}->{'Server OS'} = 'Sistema Operativo de Servidor';
    $Self->{Translation}->{'Admin Tool'} = 'Herramienta del Administración';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = 'Por Usuario';
    $Self->{Translation}->{'Per Processor'} = 'Por Procesador';
    $Self->{Translation}->{'Per Server'} = 'Por Servidor';
    $Self->{Translation}->{'Per Node'} = 'Por Nodo';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = 'Licencia del Desarrollador';
    $Self->{Translation}->{'Demo'} = 'Demo';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = 'Freeware';
    $Self->{Translation}->{'Open Source'} = 'Open Source';
    $Self->{Translation}->{'Unlimited'} = 'Ilimitado';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = 'Duplicado';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = 'Vista General.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parámetros para el ejemplo de permisos de grupo del catálogo general de atributos';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::et_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Between';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'Kontor';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::fa_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'مدیریت CI';
    $Self->{Translation}->{'Change class definition'} = 'تغییر تعریف کلاس';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = 'تعریف';
    $Self->{Translation}->{'Change'} = 'تغییر';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'وضعیت رخداد';
    $Self->{Translation}->{'Deployment State'} = 'وضعیت نصب';
    $Self->{Translation}->{'Class'} = 'کلاس';
    $Self->{Translation}->{'Deployment State Type'} = 'نوع وضعیت نصب';
    $Self->{Translation}->{'Current Incident State'} = 'وضعیت جاری رخداد';
    $Self->{Translation}->{'Current Incident State Type'} = 'نوع وضعیت جاری رخداد';
    $Self->{Translation}->{'Last changed'} = 'آخرین تغییر';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'آیتم پیکربندی';
    $Self->{Translation}->{'Filter for Classes'} = 'فیلتر برای کلاس‌ها';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'کلاسی را برای ساخت یک آیتم پبکربندی جدید انتخا کنید.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'نام این آیتم‌های پیکربندی';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = 'محتوای سابقه';
    $Self->{Translation}->{'Createtime'} = 'زمان ایجاد ';
    $Self->{Translation}->{'Zoom view'} = 'نمای کامل';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'تنظیمات مفاد';
    $Self->{Translation}->{'Config Items per page'} = 'تعداد آیتم‌های پیکربندی در هر صفحه';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = 'در ویرایش‌های قبلی هم جستجو انجام شود؟';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'آیتم پیکربندی';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = 'وضعیت فعلی نصب';
    $Self->{Translation}->{'Last changed by'} = 'آخرین تغییر به وسیله';
    $Self->{Translation}->{'Show one version'} = 'نمایش یک نسخه';
    $Self->{Translation}->{'Show all versions'} = 'نمایش تمام نسخه‌ها';
    $Self->{Translation}->{'Version Incident State'} = 'وضعیت رخداد نسخه';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = 'شماره نسخه';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'جزئیات نسخه آیتم پیکربندی';
    $Self->{Translation}->{'Property'} = 'ویژگی';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = 'آیتم پیکربندی';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = 'وضعیت نصب این CI';
    $Self->{Translation}->{'The incident state of this config item'} = 'وضعیت رخداد این CI';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'بین';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'حداکثر شماره یک عنصر';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'فیلدهای خالی مشخص می‌کند که مقادیر کنونی باقی خواهند ماند';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'مدل';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'شماره سریال';
    $Self->{Translation}->{'CPU'} = 'سی‌پی‌یو';
    $Self->{Translation}->{'Ram'} = 'رم';
    $Self->{Translation}->{'Hard Disk'} = 'هارد دیسک';
    $Self->{Translation}->{'Capacity'} = 'ظرفیت';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'کارت شبکه';
    $Self->{Translation}->{'IP over DHCP'} = 'IP از طریق DHCP';
    $Self->{Translation}->{'IP Address'} = 'آدرس IP';
    $Self->{Translation}->{'Graphic Adapter'} = 'کارت گرافیک';
    $Self->{Translation}->{'Other Equipment'} = 'تجهیزات متفرقه';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'تاریخ انقضای ضمانت‌نامه';
    $Self->{Translation}->{'Install Date'} = 'تاریخ نصب';
    $Self->{Translation}->{'Phone 1'} = 'تلفن ۱';
    $Self->{Translation}->{'Phone 2'} = 'تلفن ۲';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'آدرس شبکه';
    $Self->{Translation}->{'Subnet Mask'} = 'Subnet Mask';
    $Self->{Translation}->{'Gateway'} = 'Gateway';
    $Self->{Translation}->{'Licence Type'} = 'نوع اجازه‌نامه';
    $Self->{Translation}->{'Licence Key'} = 'کلید اجازه‌نامه';
    $Self->{Translation}->{'Quantity'} = 'تعداد';
    $Self->{Translation}->{'Expiration Date'} = 'تاریخ انقضا';
    $Self->{Translation}->{'Media'} = 'رسانه';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'منقضی شده';
    $Self->{Translation}->{'Maintenance'} = 'پشتیبانی و نگهداری';
    $Self->{Translation}->{'Pilot'} = 'نمونه آزمایشی';
    $Self->{Translation}->{'Planned'} = 'برنامه ریزی شده';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'تعمیر';
    $Self->{Translation}->{'Retired'} = 'بازنشسته شده';
    $Self->{Translation}->{'Review'} = 'بازبینی';
    $Self->{Translation}->{'Test/QA'} = 'تست/تضمین کیفیت';
    $Self->{Translation}->{'Laptop'} = 'کیفی';
    $Self->{Translation}->{'Desktop'} = 'رومیزی';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'سرور';
    $Self->{Translation}->{'Other'} = 'غیره';
    $Self->{Translation}->{'Monitor'} = 'صفحه نمایش';
    $Self->{Translation}->{'Printer'} = 'چاپگر';
    $Self->{Translation}->{'Switch'} = 'سویچ';
    $Self->{Translation}->{'Router'} = 'مسیریاب';
    $Self->{Translation}->{'WLAN Access Point'} = 'نقاط دسترسی WLAN';
    $Self->{Translation}->{'Security Device'} = 'وسایل امنیتی';
    $Self->{Translation}->{'Backup Device'} = 'وسایل پشتیبان‌گیری';
    $Self->{Translation}->{'Mouse'} = 'ماوس';
    $Self->{Translation}->{'Keyboard'} = 'صفحه کلید';
    $Self->{Translation}->{'Camera'} = 'دوربین';
    $Self->{Translation}->{'Beamer'} = 'پرتو افکن';
    $Self->{Translation}->{'Modem'} = 'مودم';
    $Self->{Translation}->{'PCMCIA Card'} = 'کارت PCMCIA';
    $Self->{Translation}->{'USB Device'} = 'تجهیزات USB';
    $Self->{Translation}->{'Docking Station'} = 'ایستگاه اتصال';
    $Self->{Translation}->{'Scanner'} = 'اسکنر';
    $Self->{Translation}->{'Building'} = 'ساختمان';
    $Self->{Translation}->{'Office'} = 'محل کار';
    $Self->{Translation}->{'Floor'} = 'طبقه';
    $Self->{Translation}->{'Room'} = 'اتاق';
    $Self->{Translation}->{'Rack'} = 'قفسه';
    $Self->{Translation}->{'Workplace'} = 'محل کار';
    $Self->{Translation}->{'Outlet'} = 'پریز';
    $Self->{Translation}->{'IT Facility'} = 'تسهیلات IT';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'برنامه‌های کلاینت';
    $Self->{Translation}->{'Middleware'} = 'میان افزار';
    $Self->{Translation}->{'Server Application'} = 'نرم افزار سرور';
    $Self->{Translation}->{'Client OS'} = 'سیستم عامل کلاینت';
    $Self->{Translation}->{'Server OS'} = 'سیستم عامل سرور';
    $Self->{Translation}->{'Admin Tool'} = 'ابزار مدیریتی';
    $Self->{Translation}->{'User Tool'} = 'ابزار کاربر';
    $Self->{Translation}->{'Embedded'} = 'داخلی';
    $Self->{Translation}->{'Single Licence'} = 'اجازه‌نامه تکی';
    $Self->{Translation}->{'Per User'} = 'به ازای کاربر';
    $Self->{Translation}->{'Per Processor'} = 'به ازای پردازشگر';
    $Self->{Translation}->{'Per Server'} = 'به ازای سرور';
    $Self->{Translation}->{'Per Node'} = 'به ازای کامپیوتر';
    $Self->{Translation}->{'Volume Licence'} = 'اجازه‌نامه حجمی';
    $Self->{Translation}->{'Enterprise Licence'} = 'اجازه‌نامه سازمانی';
    $Self->{Translation}->{'Developer Licence'} = 'اجازه‌نامه توسعه دهنده';
    $Self->{Translation}->{'Demo'} = 'نسخه نمایشی';
    $Self->{Translation}->{'Time Restricted'} = 'دارای محدوده زمانی';
    $Self->{Translation}->{'Freeware'} = 'نرم افزار مجانی';
    $Self->{Translation}->{'Open Source'} = 'کد باز';
    $Self->{Translation}->{'Unlimited'} = 'نامحدود';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'تایید';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        'ماژول رویداد آیتم پیکربندی که ثبت وقایع در تاریخچه را در واسط کاربری کارشناس فعال می‌سازد.';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = 'محدودیت آیتم پیکربندی';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'ساخت و مدیریت تعاریف آیتم‌های پیکربندی';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = 'تکثیر';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'پارامترهایی برای گروه‌های دسترسی نمونه مربوط به ویژگی‌های فهرست عمومی';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::fi_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Välillä';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'Toimisto';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::fr_CA_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = 'Vue de la synthèse';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'entre le';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'Bureau';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::fr_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Gestion des éléments de configuration';
    $Self->{Translation}->{'Change class definition'} = 'Modifier la définition d\'une classe';
    $Self->{Translation}->{'Config Item Class'} = 'Classe d\'élément de configuration';
    $Self->{Translation}->{'Definition'} = 'Définition';
    $Self->{Translation}->{'Change'} = 'Modifier';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'État d\'incident';
    $Self->{Translation}->{'Deployment State'} = 'État de déploiement';
    $Self->{Translation}->{'Class'} = 'Classe';
    $Self->{Translation}->{'Deployment State Type'} = 'Type d\'état du déploiement';
    $Self->{Translation}->{'Current Incident State'} = 'État actuel de l\'incident';
    $Self->{Translation}->{'Current Incident State Type'} = 'Type d\'état actuel de l\'incident';
    $Self->{Translation}->{'Last changed'} = 'Dernières modifications';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Élément de configuration';
    $Self->{Translation}->{'Filter for Classes'} = 'Filtres pour les classes';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Choisissez une classe pour créer un nouvel élément de configuration.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = 'État du déploiement';
    $Self->{Translation}->{'Incident state'} = 'État de l\'incident';
    $Self->{Translation}->{'Link to another'} = 'Lier à un autre';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = 'Numéro d\'élément de configuration invalide !';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'Le numéro d\'un autre élément de configuration à lier.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = 'Voulez-vous vraiment supprimer cet élément de configuration ?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'Le nom de cet élément de configuration';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'Le nom est déjà utilisé par l\'élément de configuration avec le numéro suivant : %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = 'Historique de l\'élément de configuration %s';
    $Self->{Translation}->{'History Content'} = 'Contenu de l\'historique';
    $Self->{Translation}->{'Createtime'} = 'Date de création';
    $Self->{Translation}->{'Zoom view'} = 'Vue détaillée';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Réglages de contexte';
    $Self->{Translation}->{'Config Items per page'} = 'Éléments de configuration par page';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Lancer la recherche';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Chercher également dans les versions précédentes ?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Élément de configuration';
    $Self->{Translation}->{'Configuration Item Information'} = 'Informations de l\'élément de configuration';
    $Self->{Translation}->{'Current Deployment State'} = 'État actuel de déploiement';
    $Self->{Translation}->{'Last changed by'} = 'Dernières modifications effectuées par';
    $Self->{Translation}->{'Show one version'} = 'Afficher une seule version';
    $Self->{Translation}->{'Show all versions'} = 'Afficher toutes les versions';
    $Self->{Translation}->{'Version Incident State'} = 'État de la version de l\'incident';
    $Self->{Translation}->{'Version Deployment State'} = 'État de la version du déploiement';
    $Self->{Translation}->{'Version Number'} = 'Numéro de version';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Détails de la version de l\'élément de configuration';
    $Self->{Translation}->{'Property'} = 'Propriété';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = 'Aucun accès à la classe n\'a été donné !';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = 'Vue d\'ensemble: éléments de configuration';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = 'Aucun ID d\'élément de configuration n\'a été donné !';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = 'Vous devez sélectionner au moins un élément de configuration !';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        'Vous n\'avez pas le droit de modifier l\'élément de configuration %s.';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = 'L\'élément de configuration "%s" n\'a pas été trouvé dans la base de données !';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = 'Impossible de supprimer l\'ID de l\'élément de configuration %s !';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = 'Aucune version trouvée pour l\'ID de l\'élément de configuration %s !';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = 'Aucun ID d\'élément de configuration, de duplication ou de classe n\'a été donné !';
    $Self->{Translation}->{'No access is given!'} = 'Aucun accès n\'a été donné !';
    $Self->{Translation}->{'No definition was defined for class %s!'} = 'Aucune définition définie pour la classe %s !';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = 'Impossible d\'afficher l\'historique, aucun ID d\'élément de configuration n\'a été donné !';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = 'Impossible d\'afficher l\'historique, aucun accès n\'a été donné !';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = 'Nouvel élément de configuration (ID=%s)';
    $Self->{Translation}->{'New version (ID=%s)'} = 'Nouvelle version (ID=%s)';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = 'État de déploiement mis à jour (nouveau=%s, ancien=%s)';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = 'État de l\'incident mis à jour (nouveau=%s, ancien=%s)';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = 'Élément de configuration (ID=%s) supprimé';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = 'Lier à %s (type=%s) ajouté';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = 'Lier à %s (type=%s) supprimé';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = 'Définition de l\'élément de configuration mise à jour (ID=%s)';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = 'Nom mis à jour (nouveau=%s, ancien=%s)';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = 'Attribue %s mis à jour de "%s" à "%s"';
    $Self->{Translation}->{'Version %s deleted'} = 'Version %s supprimée';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = 'Aucun ID d\'élément de configuration ou de version n\'a été donné !';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = 'Impossible d\'afficher l\'élément de configuration, aucun accès n\'a été donné !';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = 'L\'ID de l\'élément de configuration %s n\'a pas été trouvé dans la base de données !';
    $Self->{Translation}->{'VersionID %s not found in database!'} = 'L\'ID de version %s n\'a pas été trouvé dans la base de données !';
    $Self->{Translation}->{'ConfigItem'} = 'Élément de configuration';
    $Self->{Translation}->{'printed by %s at %s'} = 'Imprimé par %s à %s';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = 'ID de classe invalide !';
    $Self->{Translation}->{'No ClassID is given!'} = 'Aucun ID de classe n\'a été donné !';
    $Self->{Translation}->{'No access rights for this class given!'} = 'Aucuns droits d\'accès pour cette classe n\'a été donné !';
    $Self->{Translation}->{'No Result!'} = 'Aucuns résultats !';
    $Self->{Translation}->{'Config Item Search Results'} = 'Résultat de la recherche des éléments de configuration';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        'Impossible d\'afficher l\'élément, aucuns droits d\'accès pour cet élément de configuration n\'a été donné !';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = 'L\'état de déploiement de cet élément de configuration';
    $Self->{Translation}->{'The incident state of this config item'} = 'L\'état d\'incident de cet élément de configuration';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = 'Afficher les éléments de configuration';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Entre';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Quantité maximale pour un élément';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Les champs vides indiquent que les valeurs par défauts sont gardées';
    $Self->{Translation}->{'Skipped'} = 'Passé(s)';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Modèle';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Numéro de série';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = 'RAM';
    $Self->{Translation}->{'Hard Disk'} = 'Disque dur';
    $Self->{Translation}->{'Capacity'} = 'Capacité';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Adaptateur réseau';
    $Self->{Translation}->{'IP over DHCP'} = 'IP sur DHCP';
    $Self->{Translation}->{'IP Address'} = 'Adresse IP';
    $Self->{Translation}->{'Graphic Adapter'} = 'Adaptateur graphique';
    $Self->{Translation}->{'Other Equipment'} = 'Autre équipement';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Date d\'expiration de la garantie';
    $Self->{Translation}->{'Install Date'} = 'Date d\'installation';
    $Self->{Translation}->{'Phone 1'} = 'Téléphone 1';
    $Self->{Translation}->{'Phone 2'} = 'Téléphone 2';
    $Self->{Translation}->{'E-Mail'} = 'Adresse e-mail';
    $Self->{Translation}->{'Network Address'} = 'Adresse réseau';
    $Self->{Translation}->{'Subnet Mask'} = 'Masque du sous réseau';
    $Self->{Translation}->{'Gateway'} = 'Passerelle';
    $Self->{Translation}->{'Licence Type'} = 'Type de la licence';
    $Self->{Translation}->{'Licence Key'} = 'Clé de la licence';
    $Self->{Translation}->{'Quantity'} = 'Quantité';
    $Self->{Translation}->{'Expiration Date'} = 'Date d\'expiration';
    $Self->{Translation}->{'Media'} = 'Média';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = 'Ordinateur';
    $Self->{Translation}->{'Hardware'} = 'Matériel';
    $Self->{Translation}->{'Network'} = 'Réseau';
    $Self->{Translation}->{'Software'} = 'Logiciel';
    $Self->{Translation}->{'Expired'} = 'Expiré';
    $Self->{Translation}->{'Maintenance'} = 'Maintenance';
    $Self->{Translation}->{'Pilot'} = 'Pilote';
    $Self->{Translation}->{'Planned'} = 'Planifié';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'En réparation';
    $Self->{Translation}->{'Retired'} = 'Retiré';
    $Self->{Translation}->{'Review'} = 'Revue';
    $Self->{Translation}->{'Test/QA'} = 'Test/QA';
    $Self->{Translation}->{'Laptop'} = 'Ordinateur portable';
    $Self->{Translation}->{'Desktop'} = 'Ordinateur';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Serveur';
    $Self->{Translation}->{'Other'} = 'Autre';
    $Self->{Translation}->{'Monitor'} = 'Moniteur';
    $Self->{Translation}->{'Printer'} = 'Imprimante';
    $Self->{Translation}->{'Switch'} = 'Commutateur';
    $Self->{Translation}->{'Router'} = 'Routeur';
    $Self->{Translation}->{'WLAN Access Point'} = 'Point d\'access WLAN';
    $Self->{Translation}->{'Security Device'} = 'Périphérique de sécurité';
    $Self->{Translation}->{'Backup Device'} = 'Élément de sauvegarde';
    $Self->{Translation}->{'Mouse'} = 'Souris';
    $Self->{Translation}->{'Keyboard'} = 'Clavier';
    $Self->{Translation}->{'Camera'} = 'Caméra';
    $Self->{Translation}->{'Beamer'} = 'Vidéoprojecteur';
    $Self->{Translation}->{'Modem'} = 'Modem';
    $Self->{Translation}->{'PCMCIA Card'} = 'Carte PCMCIA';
    $Self->{Translation}->{'USB Device'} = 'Périphérique USB';
    $Self->{Translation}->{'Docking Station'} = 'Station d\'accueil';
    $Self->{Translation}->{'Scanner'} = 'Scanner';
    $Self->{Translation}->{'Building'} = 'Bâtiment';
    $Self->{Translation}->{'Office'} = 'Bureau';
    $Self->{Translation}->{'Floor'} = 'Étage';
    $Self->{Translation}->{'Room'} = 'Pièce';
    $Self->{Translation}->{'Rack'} = 'Rack';
    $Self->{Translation}->{'Workplace'} = 'Emplacement';
    $Self->{Translation}->{'Outlet'} = 'Prise';
    $Self->{Translation}->{'IT Facility'} = 'Département "Technologies de l\'Information"';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Application cliente';
    $Self->{Translation}->{'Middleware'} = 'Intergiciel (middleware)';
    $Self->{Translation}->{'Server Application'} = 'Application serveur';
    $Self->{Translation}->{'Client OS'} = 'OS client';
    $Self->{Translation}->{'Server OS'} = 'OS serveur';
    $Self->{Translation}->{'Admin Tool'} = 'Outil d\'administration';
    $Self->{Translation}->{'User Tool'} = 'Outil utilisateur';
    $Self->{Translation}->{'Embedded'} = 'Embarqué';
    $Self->{Translation}->{'Single Licence'} = 'License unique';
    $Self->{Translation}->{'Per User'} = 'Par utilisateur';
    $Self->{Translation}->{'Per Processor'} = 'Par processeur';
    $Self->{Translation}->{'Per Server'} = 'Par serveur';
    $Self->{Translation}->{'Per Node'} = 'Par noeud';
    $Self->{Translation}->{'Volume Licence'} = 'License par volume';
    $Self->{Translation}->{'Enterprise Licence'} = 'License entreprise';
    $Self->{Translation}->{'Developer Licence'} = 'License développeur';
    $Self->{Translation}->{'Demo'} = 'Démo';
    $Self->{Translation}->{'Time Restricted'} = 'Limitée dans le temps';
    $Self->{Translation}->{'Freeware'} = 'Logiciel gratuit (freeware)';
    $Self->{Translation}->{'Open Source'} = 'Open-source';
    $Self->{Translation}->{'Unlimited'} = 'Illimité';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = 'Éléments de configuration attribués';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = 'Éléments de configuration attribués à un client';
    $Self->{Translation}->{'CIs assigned to customer user'} = 'Éléments de configuration attribués à un utilisateur client';
    $Self->{Translation}->{'CMDB Settings'} = 'Paramètres de la CMDB (base de données des éléments de configuration)';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = 'Éléments de configuration';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = 'Élément de configuration ajouté';
    $Self->{Translation}->{'Config item edit.'} = 'Éléments de configuration édité';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = 'Historique de l\'élément de configuration.';
    $Self->{Translation}->{'Config item print.'} = 'Impression de l\'élément de configuration.';
    $Self->{Translation}->{'Config item zoom.'} = 'Vue détaillée de l\'élément de configuration.';
    $Self->{Translation}->{'ConfigItemNumber'} = 'Numéro de l\'élément de configuration';
    $Self->{Translation}->{'Configuration Item Limit'} = 'Limite d\'éléments de configuration';
    $Self->{Translation}->{'Configuration Item limit per page.'} = 'Limite d\'éléments de configuration par page.';
    $Self->{Translation}->{'Configuration Management Database.'} = 'Base de données des éléments de configuration.';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Créer et gérer les définitions des éléments de configuration.';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = 'Supprimer un élément de configuration';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = 'Couleur de l\'état de déploiement';
    $Self->{Translation}->{'Duplicate'} = 'Dupliquer';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = 'Vue d\'ensemble de l\'élément de configuration';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = 'Vue d\'ensemble.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Paramètres pour l\'exemple des permissions de groupes des attributs du catalogue général.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = 'Rechercher les éléments de configuration.';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = 'classe';
    $Self->{Translation}->{'global'} = 'global';
    $Self->{Translation}->{'postproductive'} = 'post-productif';
    $Self->{Translation}->{'preproductive'} = 'pré-productif';
    $Self->{Translation}->{'productive'} = 'productif';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::gl_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = 'Definición';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Estado do Incidente';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = 'Clase';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = 'Estado do Incidente actual';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = 'Último cambio';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = 'Filtrar por clases';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = 'Estado do incidente';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = 'Vista zoom';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = 'Número de versión';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = 'Propiedade';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Entre';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Número máximo dun elemento';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Campos baleiros indican que os valores actuais son mantidos';
    $Self->{Translation}->{'Skipped'} = 'Omitido';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Modelo';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Número de serie';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = 'Disco duro';
    $Self->{Translation}->{'Capacity'} = 'Capacidade';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Adaptador de rede';
    $Self->{Translation}->{'IP over DHCP'} = 'IP por DHCP';
    $Self->{Translation}->{'IP Address'} = 'Enderezo IP';
    $Self->{Translation}->{'Graphic Adapter'} = 'Adaptador gráfico';
    $Self->{Translation}->{'Other Equipment'} = 'Outro equipamento';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = 'Data de instalación';
    $Self->{Translation}->{'Phone 1'} = 'Teléfono 1';
    $Self->{Translation}->{'Phone 2'} = 'Teléfono 2';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'Enderezo de rede';
    $Self->{Translation}->{'Subnet Mask'} = 'Máscara de subrede';
    $Self->{Translation}->{'Gateway'} = 'Pasarela';
    $Self->{Translation}->{'Licence Type'} = 'Tipo Licenza';
    $Self->{Translation}->{'Licence Key'} = 'Chave Licenza';
    $Self->{Translation}->{'Quantity'} = 'Cantidade';
    $Self->{Translation}->{'Expiration Date'} = 'Data de caducidade';
    $Self->{Translation}->{'Media'} = 'Recursos multimedia';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'Caducado';
    $Self->{Translation}->{'Maintenance'} = 'Mantemento';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = 'Portátil';
    $Self->{Translation}->{'Desktop'} = 'Escritorio';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Servidor';
    $Self->{Translation}->{'Other'} = 'Outros';
    $Self->{Translation}->{'Monitor'} = 'Monitor';
    $Self->{Translation}->{'Printer'} = 'Impresora';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = 'Encamiñador';
    $Self->{Translation}->{'WLAN Access Point'} = 'Punto de acceso a rede sen fíos';
    $Self->{Translation}->{'Security Device'} = 'Dispositivo de seguranza';
    $Self->{Translation}->{'Backup Device'} = 'Dispositivo Backup';
    $Self->{Translation}->{'Mouse'} = 'Rato';
    $Self->{Translation}->{'Keyboard'} = 'Teclado';
    $Self->{Translation}->{'Camera'} = 'Cámara';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = 'Módem';
    $Self->{Translation}->{'PCMCIA Card'} = 'Placa PCMCIA';
    $Self->{Translation}->{'USB Device'} = 'Dispositivo USB';
    $Self->{Translation}->{'Docking Station'} = 'Estación de Acoplamento';
    $Self->{Translation}->{'Scanner'} = 'Escáner';
    $Self->{Translation}->{'Building'} = 'Edificio';
    $Self->{Translation}->{'Office'} = 'Oficina';
    $Self->{Translation}->{'Floor'} = 'Andar';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = 'Bastidor';
    $Self->{Translation}->{'Workplace'} = 'Lugar de traballo';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = 'Instalación IT';
    $Self->{Translation}->{'LAN'} = 'Rede local';
    $Self->{Translation}->{'WLAN'} = 'Rede sen fíos';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Aplicativo cliente';
    $Self->{Translation}->{'Middleware'} = 'Middleware';
    $Self->{Translation}->{'Server Application'} = 'Aplicativo servidor';
    $Self->{Translation}->{'Client OS'} = 'Sistema operativo cliente';
    $Self->{Translation}->{'Server OS'} = 'Sistema operativo do servidor';
    $Self->{Translation}->{'Admin Tool'} = 'Ferramenta de administración';
    $Self->{Translation}->{'User Tool'} = 'Ferramenta de usuario';
    $Self->{Translation}->{'Embedded'} = 'Incrustado';
    $Self->{Translation}->{'Single Licence'} = 'Licenza única';
    $Self->{Translation}->{'Per User'} = 'Por usuario';
    $Self->{Translation}->{'Per Processor'} = 'Por procesador';
    $Self->{Translation}->{'Per Server'} = 'Por servidor';
    $Self->{Translation}->{'Per Node'} = 'Por nodo';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = 'Licenza Enterprise';
    $Self->{Translation}->{'Developer Licence'} = 'Licenza do desenvolvedor';
    $Self->{Translation}->{'Demo'} = 'Demo';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = 'Freeware';
    $Self->{Translation}->{'Open Source'} = 'Código aberto';
    $Self->{Translation}->{'Unlimited'} = 'Ilimitado';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = 'Duplicado';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parámetros para os grupos de permisos de exemplo dos atributos do catálogo xeral.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::he_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = 'מבט זום';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'בין';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'משרד';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::hi_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = 'ज़ूम दृश्य';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'के बीच';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'कार्यालय';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::hr_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = 'Detaljni pregled';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Između';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'Ured';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::hu_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Konfigurációelem-menedzsment';
    $Self->{Translation}->{'Change class definition'} = 'Osztály-meghatározás módosítása';
    $Self->{Translation}->{'Config Item Class'} = 'Konfigurációelem-osztály';
    $Self->{Translation}->{'Definition'} = 'Meghatározás';
    $Self->{Translation}->{'Change'} = 'Változás';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Incidensállapot';
    $Self->{Translation}->{'Deployment State'} = 'Üzembe állítási állapot';
    $Self->{Translation}->{'Class'} = 'Osztály';
    $Self->{Translation}->{'Deployment State Type'} = 'Üzembe állítási állapottípus';
    $Self->{Translation}->{'Current Incident State'} = 'Jelenlegi incidensállapot';
    $Self->{Translation}->{'Current Incident State Type'} = 'Jelenlegi incidensállapot típus';
    $Self->{Translation}->{'Last changed'} = 'Utolsó módosítás';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Konfigurációelem';
    $Self->{Translation}->{'Filter for Classes'} = 'Szűrő az osztályokhoz';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Válasszon egy osztályt a listából egy új konfigurációelem létrehozásához.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'ITSM konfigurációelem tömeges művelet';
    $Self->{Translation}->{'Deployment state'} = 'Üzembe állítási állapot';
    $Self->{Translation}->{'Incident state'} = 'Incidensállapot';
    $Self->{Translation}->{'Link to another'} = 'Hivatkozás egy másik';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = 'Érvénytelen konfigurációelem-szám!';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'Egy másik konfigurációelem száma, amellyel össze kell kötni.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = 'Valóban törölni szeretné ezt a konfigurációelemet?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'A konfigurációelem neve';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'A nevet már használják a konfigurációelemek a következő számokkal: %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = 'Konfigurációelem előzményei: %s';
    $Self->{Translation}->{'History Content'} = 'Előzménytartalom';
    $Self->{Translation}->{'Createtime'} = 'Létrehozás ideje';
    $Self->{Translation}->{'Zoom view'} = 'Nagyítási nézet';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Környezet beállítások';
    $Self->{Translation}->{'Config Items per page'} = 'Konfigurációelemek oldalanként';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = 'Egy általános ITSM konfigurációelem táblázat';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Keresés futtatása';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Keressen az előző verziókban is?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Konfigurációelem';
    $Self->{Translation}->{'Configuration Item Information'} = 'Konfigurációelem-információk';
    $Self->{Translation}->{'Current Deployment State'} = 'Jelenlegi üzembe állítási állapot';
    $Self->{Translation}->{'Last changed by'} = 'Utoljára módosította';
    $Self->{Translation}->{'Show one version'} = 'Egy verzió megjelenítése';
    $Self->{Translation}->{'Show all versions'} = 'Összes verzió megjelenítése';
    $Self->{Translation}->{'Version Incident State'} = 'Verzió incidensállapot';
    $Self->{Translation}->{'Version Deployment State'} = 'Verzió üzembe állítási állapot';
    $Self->{Translation}->{'Version Number'} = 'Verziószám';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Konfigurációelem-verzió részletek';
    $Self->{Translation}->{'Property'} = 'Tulajdonság';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = 'Nincs hozzáférés megadva az osztályhoz!';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = 'Áttekintés: ITSM konfigurációelem';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = 'Nincs konfigurációelem-azonosító megadva!';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = 'Szüksége van legalább egy kiválasztott konfigurációelemre!';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        'Nincs írási hozzáférése ehhez a konfigurációelemhez: %s.';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = 'A(z) „%s” konfigurációelem nem található az adatbázisban!';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = 'Nem volt lehetséges a(z) %s konfigurációelem-azonosító törlése!';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = 'Nem található verzió a(z) %s konfigurációelem-azonosítónál!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = 'Nincs konfigurációelem-azonosító, kettőzésazonosító vagy osztályazonosító megadva!';
    $Self->{Translation}->{'No access is given!'} = 'Nincs hozzáférés megadva!';
    $Self->{Translation}->{'No definition was defined for class %s!'} = 'Nincs meghatározás meghatározva a(z) %s osztályhoz!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = 'Nem lehet megjeleníteni az előzményeket, nincs konfigurációelem-azonosító megadva!';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = 'Nem lehet megjeleníteni az előzményeket, nincsenek hozzáférési jogok megadva!';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = 'Új konfigurációelem (azonosító = %s)';
    $Self->{Translation}->{'New version (ID=%s)'} = 'Új verzió (azonosító = %s)';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = 'Üzembe állítási állapot frissítve (új = %s, régi = %s)';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = 'Incidensállapot frissítve (új = %s, régi = %s)';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = 'Konfigurációelem (azonosító = %s) törölve';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = 'Kapcsolat ehhez: %s (típus = %s) hozzáadva';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = 'Kapcsolat ehhez: %s (típus = %s) törölve';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = 'Konfigurációelem-meghatározás frissítve (azonosító = %s)';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = 'Név frissítve (új = %s, régi = %s)';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = 'Attribútum (%s) frissítve erről: „%s”, erre: „%s”';
    $Self->{Translation}->{'Version %s deleted'} = '%s. verzió törölve';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = 'Nincs konfigurációelem-azonosító vagy verzióazonosító megadva!';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = 'Nem lehet megjeleníteni a konfigurációelemet, nincsenek hozzáférési jogok megadva!';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = 'A(z) %s konfigurációelem-azonosító nem található az adatbázisban!';
    $Self->{Translation}->{'VersionID %s not found in database!'} = 'A(z) %s verzióazonosító nem található az adatbázisban!';
    $Self->{Translation}->{'ConfigItem'} = 'Konfigurációelem';
    $Self->{Translation}->{'printed by %s at %s'} = 'nyomtatta: %s – %s';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = 'Érvénytelen osztályazonosító!';
    $Self->{Translation}->{'No ClassID is given!'} = 'Nincs osztályazonosító megadva!';
    $Self->{Translation}->{'No access rights for this class given!'} = 'Nincsenek hozzáférési jogok megadva ehhez az osztályhoz!';
    $Self->{Translation}->{'No Result!'} = 'Nincs találat!';
    $Self->{Translation}->{'Config Item Search Results'} = 'Konfigurációelem keresési eredmények';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        'Nem lehet megjeleníteni az elemet, nincsenek hozzáférési jogok megadva a konfigurációelemhez!';
    $Self->{Translation}->{'operational'} = 'üzemképes';
    $Self->{Translation}->{'warning'} = 'figyelmeztetés';
    $Self->{Translation}->{'incident'} = 'incidens';
    $Self->{Translation}->{'The deployment state of this config item'} = 'A konfigurációelem üzembe állítási állapota';
    $Self->{Translation}->{'The incident state of this config item'} = 'A konfigurációelem incidensállapota';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = 'Megjelenített konfigurációelemek';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Között';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Egy elem legnagyobb száma';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Az üres mezők azt jelzik, hogy az aktuális mezők megtartásra kerülnek';
    $Self->{Translation}->{'Skipped'} = 'Kihagyva';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Modell';
    $Self->{Translation}->{'Customer Company'} = 'Ügyfél-vállalat';
    $Self->{Translation}->{'Serial Number'} = 'Sorozatszám';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = 'Ram';
    $Self->{Translation}->{'Hard Disk'} = 'Merevlemez';
    $Self->{Translation}->{'Capacity'} = 'Kapacitás';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Hálózati csatoló';
    $Self->{Translation}->{'IP over DHCP'} = 'IP a DHCP fölött';
    $Self->{Translation}->{'IP Address'} = 'IP-cím';
    $Self->{Translation}->{'Graphic Adapter'} = 'Grafikus csatoló';
    $Self->{Translation}->{'Other Equipment'} = 'Egyéb berendezés';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Garancia lejárati idő';
    $Self->{Translation}->{'Install Date'} = 'Telepítés dátuma';
    $Self->{Translation}->{'Phone 1'} = '1. telefon';
    $Self->{Translation}->{'Phone 2'} = '2. telefon';
    $Self->{Translation}->{'E-Mail'} = 'E-mail';
    $Self->{Translation}->{'Network Address'} = 'Hálózati cím';
    $Self->{Translation}->{'Subnet Mask'} = 'Alhálózati maszk';
    $Self->{Translation}->{'Gateway'} = 'Átjáró';
    $Self->{Translation}->{'Licence Type'} = 'Licenctípus';
    $Self->{Translation}->{'Licence Key'} = 'Licenc kulcs';
    $Self->{Translation}->{'Quantity'} = 'Mennyiség';
    $Self->{Translation}->{'Expiration Date'} = 'Lejárati idő';
    $Self->{Translation}->{'Media'} = 'Adathordozó';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = 'Számítógép';
    $Self->{Translation}->{'Hardware'} = 'Hardver';
    $Self->{Translation}->{'Network'} = 'Hálózat';
    $Self->{Translation}->{'Software'} = 'Szoftver';
    $Self->{Translation}->{'Expired'} = 'Lejárt';
    $Self->{Translation}->{'Maintenance'} = 'Karbantartás';
    $Self->{Translation}->{'Pilot'} = 'Próbaüzem';
    $Self->{Translation}->{'Planned'} = 'Tervezett';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Javítás';
    $Self->{Translation}->{'Retired'} = 'Visszavont';
    $Self->{Translation}->{'Review'} = 'Vizsgálat';
    $Self->{Translation}->{'Test/QA'} = 'Tesztelés/QA';
    $Self->{Translation}->{'Laptop'} = 'Laptop';
    $Self->{Translation}->{'Desktop'} = 'Asztali';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Kiszolgáló';
    $Self->{Translation}->{'Other'} = 'Egyéb';
    $Self->{Translation}->{'Monitor'} = 'Monitor';
    $Self->{Translation}->{'Printer'} = 'Nyomtató';
    $Self->{Translation}->{'Switch'} = 'Kapcsoló';
    $Self->{Translation}->{'Router'} = 'Útválasztó';
    $Self->{Translation}->{'WLAN Access Point'} = 'WLAN hozzáférési pont';
    $Self->{Translation}->{'Security Device'} = 'Biztonsági eszköz';
    $Self->{Translation}->{'Backup Device'} = 'Biztonsági mentés eszköz';
    $Self->{Translation}->{'Mouse'} = 'Egér';
    $Self->{Translation}->{'Keyboard'} = 'Billentyűzet';
    $Self->{Translation}->{'Camera'} = 'Kamera';
    $Self->{Translation}->{'Beamer'} = 'Kivetítő';
    $Self->{Translation}->{'Modem'} = 'Modem';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA kártya';
    $Self->{Translation}->{'USB Device'} = 'USB eszköz';
    $Self->{Translation}->{'Docking Station'} = 'Dokkoló állomás';
    $Self->{Translation}->{'Scanner'} = 'Lapolvasó';
    $Self->{Translation}->{'Building'} = 'Épület';
    $Self->{Translation}->{'Office'} = 'Iroda';
    $Self->{Translation}->{'Floor'} = 'Emelet';
    $Self->{Translation}->{'Room'} = 'Szoba';
    $Self->{Translation}->{'Rack'} = 'Rack';
    $Self->{Translation}->{'Workplace'} = 'Munkahely';
    $Self->{Translation}->{'Outlet'} = 'Elosztó doboz';
    $Self->{Translation}->{'IT Facility'} = 'IT-létesítmény';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telekommunikáció';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Kliens alkalmazás';
    $Self->{Translation}->{'Middleware'} = 'Köztes réteg';
    $Self->{Translation}->{'Server Application'} = 'Kiszolgáló alkalmazás';
    $Self->{Translation}->{'Client OS'} = 'Kliens operációs rendszer';
    $Self->{Translation}->{'Server OS'} = 'Kiszolgáló operációs rendszer';
    $Self->{Translation}->{'Admin Tool'} = 'Adminisztrációs eszköz';
    $Self->{Translation}->{'User Tool'} = 'Felhasználói eszköz';
    $Self->{Translation}->{'Embedded'} = 'Beágyazott';
    $Self->{Translation}->{'Single Licence'} = 'Egyedüli licenc';
    $Self->{Translation}->{'Per User'} = 'Felhasználónként';
    $Self->{Translation}->{'Per Processor'} = 'Processzoronként';
    $Self->{Translation}->{'Per Server'} = 'Kiszolgálónként';
    $Self->{Translation}->{'Per Node'} = 'Csomópontonként';
    $Self->{Translation}->{'Volume Licence'} = 'Mennyiségi licenc';
    $Self->{Translation}->{'Enterprise Licence'} = 'Vállalati licenc';
    $Self->{Translation}->{'Developer Licence'} = 'Fejlesztői licenc';
    $Self->{Translation}->{'Demo'} = 'Bemutató';
    $Self->{Translation}->{'Time Restricted'} = 'Időben korlátozott';
    $Self->{Translation}->{'Freeware'} = 'Freeware';
    $Self->{Translation}->{'Open Source'} = 'Nyílt forrású';
    $Self->{Translation}->{'Unlimited'} = 'Korlátlan';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'OK';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = 'Hozzárendelt konfigurációelemek';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = 'Ügyfél-vállalathoz hozzárendelt konfigurációelemek';
    $Self->{Translation}->{'CIs assigned to customer user'} = 'Ügyfél-felhasználóhoz hozzárendelt konfigurációelemek';
    $Self->{Translation}->{'CMDB Settings'} = 'CMDB beállítások';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        'Egyedi név ellenőrzése kizárólag ugyanabban a konfigurációelem osztályban („osztály”) vagy globálisan („globális”), amely azt jelenti, hogy minden meglévő konfigurációelem figyelembe lesz véve a kettőzések keresésekor.';
    $Self->{Translation}->{'Config Items'} = 'Konfigurációelemek';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = 'Konfigurációelem hozzáadása.';
    $Self->{Translation}->{'Config item edit.'} = 'Konfigurációelem szerkesztése.';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        'Konfigurációelem esemény modul, amely engedélyezi a naplózást az előzményekbe az ügyintézői felületen.';
    $Self->{Translation}->{'Config item history.'} = 'Konfigurációelem előzményei.';
    $Self->{Translation}->{'Config item print.'} = 'Konfigurációelem nyomtatása.';
    $Self->{Translation}->{'Config item zoom.'} = 'Konfigurációelem nagyítása.';
    $Self->{Translation}->{'ConfigItemNumber'} = 'Konfigurációelem-szám';
    $Self->{Translation}->{'Configuration Item Limit'} = 'Konfigurációelem-korlát';
    $Self->{Translation}->{'Configuration Item limit per page.'} = 'Konfigurációelem-korlát oldalanként.';
    $Self->{Translation}->{'Configuration Management Database.'} = 'Konfigurációmenedzsment-adatbázis.';
    $Self->{Translation}->{'Configuration item bulk module.'} = 'Konfigurációelem tömeges modul.';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        'Az ügyintézői felület konfigurációelem keresési háttérprogram útválasztója.';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Meghatározások létrehozása és kezelése a konfigurációelemeknél.';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        'Műveletek meghatározása, ahol egy beállítások gomb érhető el a kapcsolt objektumok felületi elemen (LinkObject::ViewMode = „összetett”). Ne feledje, hogy ezeknek a műveleteknek rendelkezniük kell a következő JS és CSS fájlok regisztrálásával: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js és Core.Agent.LinkObject.js.';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        'Meghatározza a szükséges jogosultságokat az ITSM konfigurációelemek létrehozásához az általános felület használatával.';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        'Meghatározza a szükséges jogosultságokat az ITSM konfigurációelemek törléséhez az általános felület használatával.';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        'Meghatározza a szükséges jogosultságokat az ITSM konfigurációelemek lekéréséhez az általános felület használatával.';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        'Meghatározza a szükséges jogosultságokat az ITSM konfigurációelemek kereséséhez az általános felület használatával.';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        'Meghatározza a szükséges jogosultságokat az ITSM konfigurációelemek frissítéséhez az általános felület használatával.';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        'Egy áttekintő modult határoz meg egy konfigurációelem lista kis nézetének megjelenítéséhez.';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        'Reguláris kifejezéseket határoz meg egyénileg minden egyes konfigurációelem osztálynál a konfigurációelem nevének ellenőrzéséhez, valamint a megfelelő hibaüzenetek megjelenítéséhez.';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        'Meghatározza az „ITSMConfigItem” osztály alapértelmezett alobjektumát.';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        'Meghatározza a CI meghatározás-szerkesztő sorainak számát az adminisztrációs felületen.';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        'Meghatározza az incidensállapotok sorrendjét a legmagasabbtól (például kritikus) a legalacsonyabbig (például funkcionális).';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        'Meghatározza a kapcsolódó üzembe állítási állapotokat, ahol a kapcsolt jegyek befolyásolhatják egy konfigurációelem állapotát.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        'Meghatározza a keresési korlátot az AgentITSMConfigItem képernyőn.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        'Meghatározza a keresési korlátot az AgentITSMConfigItemSearch képernyőn.';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        'Meghatározza a megjelenített oszlopokat a konfigurációelem áttekintőben. Ennek a beállításnak nincs hatása az oszlop helyzetére. Megjegyzés: az osztály oszlop mindig látható, ha az „Összes” szűrő ki van választva.';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        'Meghatározza a megjelenített oszlopokat a konfigurációelem keresőben. Ennek a beállításnak nincs hatása az oszlop helyzetére.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Meghatározza a CI-k megjelenített oszlopait a konfigurációelem áttekintőben a CI osztálytól függően. Minden bejegyzést az oszlop neve és dupla kettőspont előtaggal kell ellátni (azaz Computer::). Van néhány olyan CI-attribútum, amely közös minden CI-vel (például a Computer osztálynál: Computer::Name, Computer::CurDeplState, Computer::CreateTime). Az egyéni CI-attribútumok megjelenítéséhez, ahogy azok a CI-meghatározásban meg vannak adva, a következő sémát kell használni (például a Computer osztálynál): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Ha egy CI osztályhoz nincs bejegyzés, akkor az alapértelmezett oszlopok lesznek megjelenítve, ahogy az ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns beállításban meg van határozva.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Meghatározza a CI-k megjelenített oszlopait a konfigurációelem keresőben a CI osztálytól függően. Minden bejegyzést az oszlop neve és dupla kettőspont előtaggal kell ellátni (azaz Computer::). Van néhány olyan CI-attribútum, amely közös minden CI-vel (például a Computer osztálynál: Computer::Name, Computer::CurDeplState, Computer::CreateTime). Az egyéni CI-attribútumok megjelenítéséhez, ahogy azok a CI-meghatározásban meg vannak adva, a következő sémát kell használni (például a Computer osztálynál): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Ha egy CI osztályhoz nincs bejegyzés, akkor az alapértelmezett oszlopok lesznek megjelenítve, ahogy az ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns beállításban meg van határozva.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        'Meghatározza a konfigurációelemek megjelenített oszlopait a konfigurációelem áttekintőben a konfigurációelem-osztálytól függően. Minden bejegyzést az oszlop neve és dupla kettőspont előtaggal kell ellátni (azaz Computer::). Van néhány olyan konfigurációelem-attribútum, amely közös minden konfigurációelemmel (például a Computer osztálynál: Computer::Name, Computer::CurDeplState, Computer::CreateTime). Az egyéni konfigurációelem-attribútumok megjelenítéséhez, ahogy azok a konfigurációelem-meghatározásban meg vannak adva, a következő sémát kell használni (például a Computer osztálynál): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Ha egy konfigurációelem osztályhoz nincs bejegyzés, akkor az alapértelmezett oszlopok lesznek megjelenítve, ahogy az ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns beállításban meg van határozva.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        'Meghatározza a konfigurációelemek megjelenített oszlopait a kapcsolati tábla komplex nézetében az összes konfigurációelem-osztálynál. Ha nincs bejegyzés, akkor az alapértelmezett oszlopok lesznek megjelenítve.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        'Meghatározza a CI-k megjelenített oszlopait a kapcsolati tábla komplex nézetében a CI osztálytól függően. Minden bejegyzést az oszlop neve és dupla kettőspont előtaggal kell ellátni (azaz Computer::). Van néhány olyan CI-attribútum, amely közös minden CI-vel (például a Computer osztálynál: Computer::Name, Computer::CurDeplState, Computer::CreateTime). Az egyéni CI-attribútumok megjelenítéséhez, ahogy azok a CI-meghatározásban meg vannak adva, a következő sémát kell használni (például a Computer osztálynál): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Ha egy CI osztályhoz nincs bejegyzés, akkor az alapértelmezett oszlopok lesznek megjelenítve.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        'Meghatározza, hogy mely kapcsolattípusok (a jegy nézőpontjából elnevezve) befolyásolhatják egy kapcsolt konfigurációelem állapotát.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        'Meghatározza, hogy mely jegytípus befolyásolhatja egy kapcsolt konfigurációelem állapotát.';
    $Self->{Translation}->{'Delete Configuration Item'} = 'Konfigurációelem törlése';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = 'Üzembe állítási állapot színe';
    $Self->{Translation}->{'Duplicate'} = 'Kettőzés';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        'Engedélyezi a konfigurációelem tömeges művelet funkciót az ügyintézői előtétprogramnál, hogy egyszerre egynél több konfigurációelemmel dolgozhasson.';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        'Csak a felsorolt csoportoknak engedélyezi a konfigurációelem tömeges művelet funkciót.';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        'Engedélyezi vagy letiltja azt a funkcionalitást, amely a konfigurációelemek egyedi neveit ellenőrzi. A lehetőség engedélyezése előtt ellenőriznie kell a rendszerét, hogy vannak-e már létező konfigurációelemek kettőzött névvel. Ezt megteheti a bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates parancsfájllal.';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        'Eseménymodul a konfigurációelem állapotának beállításához egy jegy konfigurációelem hivatkozásán.';
    $Self->{Translation}->{'ITSM config item overview.'} = 'ITSM konfigurációelem áttekintés.';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = 'Egy modul egy osztályért felelős csoport ellenőrzéséhez.';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        'Egy modul egy konfigurációelemért felelős csoport ellenőrzéséhez.';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = 'Egy modul ITSM konfigurációelem statisztikák előállításához.';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        'Objektum háttérprogram modul regisztráció az importálás/exportálás modulhoz.';
    $Self->{Translation}->{'Overview.'} = 'Áttekintés.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Paraméterek az ügyintézői felület ügyfél-vállalat konfigurációelem áttekintőjének vezérlőpult háttérprogramjához. A „Limit” az alapértelmezetten megjelenített bejegyzések száma. A „Group” használható a hozzáférés korlátozásához a bővítményre (például Group: admin;csoport1;csoport2;). A „Default” azt határozza meg, hogy a bővítmény alapértelmezetten engedélyezve van, vagy hogy a felhasználónak kézzel kell engedélyeznie azt. A „CacheTTLLocal” a bővítmény gyorsítótár ideje percben.';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        'Paraméterek az üzembe állítási állapotok színeihez az ügyintézői felület beállítások nézetében.';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        'Paraméterek az üzembe állítási állapotokhoz az ügyintézői felület beállítások nézetében.';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Paraméterek az általános katalógus attribútumainak példa jogosultság csoportjaihoz.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        'Paraméterek az oldalakhoz (amelyeken a konfigurációelemek megjelennek).';
    $Self->{Translation}->{'Permission Group'} = 'Jogosultsági csoport';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        'A szükséges jogosultságok az ITSM konfigurációelem képernyőjének használatához az ügyintézői felületen.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        'A szükséges jogosultságok az ITSM konfigurációelem keresési képernyőjének használatához az ügyintézői felületen.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        'A szükséges jogosultságok az ITSM konfigurációelem nagyítási képernyőjének használatához az ügyintézői felületen.';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        'A szükséges jogosultságok az ITSM konfigurációelem hozzáadása képernyőjének használatához az ügyintézői felületen.';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        'A szükséges jogosultságok az ITSM konfigurációelem szerkesztése képernyőjének használatához az ügyintézői felületen.';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        'A szükséges jogosultságok az ITSM konfigurációelem előzmények képernyőjének használatához az ügyintézői felületen.';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        'A szükséges jogosultságok az ITSM konfigurációelem nyomtatása képernyőjének használatához az ügyintézői felületen.';
    $Self->{Translation}->{'Required privileges to delete config items.'} = 'A szükséges jogosultságok a konfigurációelemek törléséhez.';
    $Self->{Translation}->{'Search config items.'} = 'Konfigurációelemek keresése.';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        'Kiválasztja a konfigurációelem számelőállító modulját. Az „AutoIncrement” növeli a konfigurációelem számát, a használt SystemID, ConfigItemClassID és számláló értékét. A formátum: „SystemID.ConfigItemClassID.Counter”, például: 1205000004, 1205000005.';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        'Egy konfigurációelem incidensállapotának automatikus beállítása, amikor egy jegyet kapcsolnak egy konfigurációelemhez.';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        'Beállítja az üzembe állítási állapotot az ügyintézői felület konfigurációelem tömeges művelet képernyőjén.';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        'Beállítja az incidensállapotot az ügyintézői felület konfigurációelem tömeges művelet képernyőjén.';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        'Egy hivatkozást jelenít meg a menüben, amely lehetővé teszi egy konfigurációelem összekapcsolását egy másik objektummal az ügyintézői felület konfigurációelem nagyítási nézetén.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        'Egy hivatkozást jelenít meg a menüben egy konfigurációelem előzményeinek eléréséhez az ügyintézői felület konfigurációelem áttekintőjében.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        'Egy hivatkozást jelenít meg a menüben egy konfigurációelem előzményeinek eléréséhez az ügyintézői felületen az elem nagyítási nézetében.';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        'Egy hivatkozást jelenít meg a menüben egy konfigurációelem törléséhez az ügyintézői felületen az elem nagyítási nézetében.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        'Egy hivatkozást jelenít meg a menüben egy konfigurációelem kettőzéséhez az ügyintézői felület konfigurációelem áttekintőjében.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        'Egy hivatkozást jelenít meg a menüben egy konfigurációelem kettőzéséhez az ügyintézői felületen az elem nagyítási nézetében.';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        'Egy hivatkozást jelenít meg a menüben egy konfigurációelem szerkesztéséhez az ügyintézői felületen az elem nagyítási nézetében.';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        'Egy hivatkozást jelenít meg a menüben az ügyintézői felület konfigurációelem nagyítási nézethez való visszatéréshez.';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        'Egy hivatkozást jelenít meg a menüben egy konfigurációelem nyomtatásához az ügyintézői felületen az elem nagyítási nézetében.';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        'Egy hivatkozást jelenít meg a menüben egy konfigurációelembe történő nagyításhoz az ügyintézői felület konfigurációelem áttekintőjében.';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        'Megjeleníti a konfigurációelem előzményeit (fordított sorrendben) az ügyintézői felületen.';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        'Egy konfigurációelem azonosítója, például: ConfigItem#, MyConfigItem#. Az alapértelmezett: ConfigItem#.';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = 'osztály';
    $Self->{Translation}->{'global'} = 'globális';
    $Self->{Translation}->{'postproductive'} = 'beüzemelés után';
    $Self->{Translation}->{'preproductive'} = 'beüzemelés előtt';
    $Self->{Translation}->{'productive'} = 'beüzemelve';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::id_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Pengelolaan item konfigurasi';
    $Self->{Translation}->{'Change class definition'} = 'ubah definisi kelas';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = 'Definisi';
    $Self->{Translation}->{'Change'} = 'Ubah';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Kondisi insiden';
    $Self->{Translation}->{'Deployment State'} = 'Status penyebaran';
    $Self->{Translation}->{'Class'} = 'Kelas';
    $Self->{Translation}->{'Deployment State Type'} = 'Tipe status penyebaran';
    $Self->{Translation}->{'Current Incident State'} = 'Kondisi insiden saat ini';
    $Self->{Translation}->{'Current Incident State Type'} = 'Tipe Status insiden saat ini';
    $Self->{Translation}->{'Last changed'} = 'Terakhir diubah';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Item konfigurasi';
    $Self->{Translation}->{'Filter for Classes'} = 'Saringan untuk kelas-kelas';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Pilih satu kelas dari datar untuk membuat sebuah item konfigurasi baru';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'Tindakan masal ItemKonfigurasi ITSM';
    $Self->{Translation}->{'Deployment state'} = 'Status penyebaran';
    $Self->{Translation}->{'Incident state'} = 'Status insiden';
    $Self->{Translation}->{'Link to another'} = 'Tautan untuk ke lainnya';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = 'Nomor item konfigurasi tidak sah';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'Jumlah item konfigurasi lainnya yang akan di hubungkan';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'Nama item konfigurasi ini';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'Nama ini telah di gunakan oleh ItemKonfigurasi dengan nomor(nomor-nomor) berikut: %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = 'Isi sejarah';
    $Self->{Translation}->{'Createtime'} = 'BuatWaktu';
    $Self->{Translation}->{'Zoom view'} = 'Pandangan dekat';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Setting Konteks';
    $Self->{Translation}->{'Config Items per page'} = 'Item konfigurasi per halaman';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Mulai Pencarian';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Cari di versi sebelumnya juga ?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Item Konfigurasi';
    $Self->{Translation}->{'Configuration Item Information'} = 'Informasi item konfigurasi';
    $Self->{Translation}->{'Current Deployment State'} = 'Status penyebaraan saat ini';
    $Self->{Translation}->{'Last changed by'} = 'Terakhir dirubah oleh';
    $Self->{Translation}->{'Show one version'} = 'Tunjukan satu versi';
    $Self->{Translation}->{'Show all versions'} = 'Tunjukan semua versi';
    $Self->{Translation}->{'Version Incident State'} = 'Versi status insiden';
    $Self->{Translation}->{'Version Deployment State'} = 'Versi status penyebaran';
    $Self->{Translation}->{'Version Number'} = 'Nomor versi';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Rincian versi item konfigurasi';
    $Self->{Translation}->{'Property'} = 'Milik';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = 'ItemKonfigurasi';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = 'Tidak ada hasil!';
    $Self->{Translation}->{'Config Item Search Results'} = 'Konfigurasi pencarian nilai';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = 'Status penyebaran item konfigurasi ini';
    $Self->{Translation}->{'The incident state of this config item'} = 'Status insiden item konfigurasi ini';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Diantara';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Jumlah maksimum dari 1 elemen';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Bidang kosong mengindikasikan bahwa nilai saat ini disimpan';
    $Self->{Translation}->{'Skipped'} = 'Dilewati';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Model';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Nomor seri';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = 'Ram';
    $Self->{Translation}->{'Hard Disk'} = 'Hard Disk';
    $Self->{Translation}->{'Capacity'} = 'Kapasitas';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Adapter Jaringan';
    $Self->{Translation}->{'IP over DHCP'} = 'IP lebih dari DHCP';
    $Self->{Translation}->{'IP Address'} = 'Alamat IP';
    $Self->{Translation}->{'Graphic Adapter'} = 'Adapter grafis';
    $Self->{Translation}->{'Other Equipment'} = 'Peralatan lainnya';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Tanggal kadaluarsa garansi';
    $Self->{Translation}->{'Install Date'} = 'Tanggal instalasi';
    $Self->{Translation}->{'Phone 1'} = 'Telepon 1';
    $Self->{Translation}->{'Phone 2'} = 'Telepon 2';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'Alamat Jaringan';
    $Self->{Translation}->{'Subnet Mask'} = 'Subnet Mask';
    $Self->{Translation}->{'Gateway'} = 'Gateway';
    $Self->{Translation}->{'Licence Type'} = 'Tipe Lisensi';
    $Self->{Translation}->{'Licence Key'} = 'Kunci lisensi';
    $Self->{Translation}->{'Quantity'} = 'Kuantitas';
    $Self->{Translation}->{'Expiration Date'} = 'Tanggal Kadaluarsa';
    $Self->{Translation}->{'Media'} = 'Media';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'Telah kadaluarsa';
    $Self->{Translation}->{'Maintenance'} = 'Pemeliharaan';
    $Self->{Translation}->{'Pilot'} = 'Pilot';
    $Self->{Translation}->{'Planned'} = 'Telah direncanakan';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Perbaiki';
    $Self->{Translation}->{'Retired'} = 'Pensiun';
    $Self->{Translation}->{'Review'} = 'Ulasan';
    $Self->{Translation}->{'Test/QA'} = 'Uji/QA';
    $Self->{Translation}->{'Laptop'} = 'Laptop';
    $Self->{Translation}->{'Desktop'} = 'Desktop';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Server';
    $Self->{Translation}->{'Other'} = 'Lain-lain';
    $Self->{Translation}->{'Monitor'} = 'Monitor';
    $Self->{Translation}->{'Printer'} = 'Pencetak';
    $Self->{Translation}->{'Switch'} = 'Tukar';
    $Self->{Translation}->{'Router'} = 'Router';
    $Self->{Translation}->{'WLAN Access Point'} = 'Titik akses WLAN';
    $Self->{Translation}->{'Security Device'} = 'Perangkat keamanan';
    $Self->{Translation}->{'Backup Device'} = 'Perangkat Backup';
    $Self->{Translation}->{'Mouse'} = 'Mouse';
    $Self->{Translation}->{'Keyboard'} = 'Keyboard';
    $Self->{Translation}->{'Camera'} = 'Kamera';
    $Self->{Translation}->{'Beamer'} = 'Beamer';
    $Self->{Translation}->{'Modem'} = 'Modem';
    $Self->{Translation}->{'PCMCIA Card'} = 'Kartu PCMCIA';
    $Self->{Translation}->{'USB Device'} = 'Perangkat USB';
    $Self->{Translation}->{'Docking Station'} = 'Stasiun dok';
    $Self->{Translation}->{'Scanner'} = 'Pemindai';
    $Self->{Translation}->{'Building'} = 'Membangun';
    $Self->{Translation}->{'Office'} = 'Kantor';
    $Self->{Translation}->{'Floor'} = 'Lantai';
    $Self->{Translation}->{'Room'} = 'Ruangan';
    $Self->{Translation}->{'Rack'} = 'Rak';
    $Self->{Translation}->{'Workplace'} = 'Tempat kerja';
    $Self->{Translation}->{'Outlet'} = 'Outlet';
    $Self->{Translation}->{'IT Facility'} = 'Fasilitas IT';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Aplikasi pelanggan';
    $Self->{Translation}->{'Middleware'} = 'Middleware';
    $Self->{Translation}->{'Server Application'} = 'Aplikasi server';
    $Self->{Translation}->{'Client OS'} = 'OS pelanggan';
    $Self->{Translation}->{'Server OS'} = 'OS Server';
    $Self->{Translation}->{'Admin Tool'} = 'Peralatan Admin';
    $Self->{Translation}->{'User Tool'} = 'Peralatan pengguna';
    $Self->{Translation}->{'Embedded'} = 'Tertanam';
    $Self->{Translation}->{'Single Licence'} = 'Lisensi tunggal';
    $Self->{Translation}->{'Per User'} = 'Per Pengguna';
    $Self->{Translation}->{'Per Processor'} = 'Per Prosesor';
    $Self->{Translation}->{'Per Server'} = 'Per server';
    $Self->{Translation}->{'Per Node'} = 'Per Node';
    $Self->{Translation}->{'Volume Licence'} = 'Lisensi Jilid';
    $Self->{Translation}->{'Enterprise Licence'} = 'Lisensi perusahaan';
    $Self->{Translation}->{'Developer Licence'} = 'Lisensi pengembang';
    $Self->{Translation}->{'Demo'} = 'Demo';
    $Self->{Translation}->{'Time Restricted'} = 'Dibatasi waktu';
    $Self->{Translation}->{'Freeware'} = 'Freeware';
    $Self->{Translation}->{'Open Source'} = 'Sumber Terbuka';
    $Self->{Translation}->{'Unlimited'} = 'Tidak terbatas';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        'Periksa hanya nama unik saja di dalam kelas(\'kelas\') itemkonfigurasi yang sama atau secara global (\'global), yang berarti setiap ItemKonfigurasi yang ada akan diperhitungkan ketika mencari duplikat';
    $Self->{Translation}->{'Config Items'} = 'item konfigurasi';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        'Modul peristiwa item konfigurasi yang mengaktifkan logging ke riwayat pada antarmuka agen';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = 'Batas item konfigurasi';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        'Router pencarian backend item konfigurasi pada antarmuka agen';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Buat dan atur definisi - definisi untuk item konfigurasi';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        'Definisikan izin yang di butuhkan untuk menciptakan item konfigurasi ITSM dengan menggunakan antarmuka umum.';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        'Definisikan izin yang di butuhkan untuk mendapatkan item konfigurasi ITSM dengan menggunakan antarmuka umum.';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        'Definisikan izin yang di butuhkan untuk mencari item konfigurasi ITSM dengan menggunakan antarmuka umum.';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        'Definisikan izin yang di butuhkan untuk memperbarui item konfigurasi ITSM dengan menggunakan antarmuka umum.';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        'Definisikan modul gambaran untuk menunjukan tampilan kecil dari sebuah daftar item konfigurasi';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        'Definisikan ekspresi regular secara individual untuk setiap kelas ItemKonfigurasi untuk memeriksa nama ItemKonfigurasi untuk menunjukan pesan error yang sesuai';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        'Definisikan subobjek default dari kelas \'ITSMConfigItem\'.';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        'Definisikan jumlah baris untuk pengubah definisiCI pada antarmuka Admin';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        'Definisikan batas pencarian untuk layar AgentITSMConfigItem ';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        'Definisikan batas pencarian untuk Layar AgentITSMConfigItemSearch ';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        'Definisikan kolom yang di tunjukan pada gambaran item konfigurasi. Pilihan ini tidak mempengaruhi posisi kolom. Catatan: kolom kelas selalu tersedia jika saringan \'Semua\' dipilih.';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        'Definisikan kolom yang ditunjukan didalam pencarian item konfigurasi. Pilihan ini tidak mempengaruhi posisi kolom';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Definisikan kolom yang ditunjukan oleh CI di dalam gambaran item konfigurasi tergantung pada kelas CI. Setiap entri harus diawali dengan nama kelas dan dua tanda titik dua (contohnya komputer::). Ada beberapa CI-Atribut yang umum pada semua CI (Contoh untuk kelas Komputer: Komputer::Nama, Komputer::CurDeplState, Komputer::Waktupembuatan). Untuk menunjukan CI-Atribut secara individual seperti yang didefinisikan di dalam CI-Definisi, Skema berikut ini harus di gunakan (Contoh untuk kelas Komputer): Komputer::HardDisk::1, komputer::HardDisk::1::kapasitas::1, komputer::HardDisk::2, Komputer::HardDisk::2::Kapasitas::1. Jika tidak ada entri untuk sebuah kelas CI, Maka kolom default yang akan ditunjukan sesuai dengan yang di tentukan pada pengaturanITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Mendefinisikan kolom yang ditampilkan dari CI dalam pencarian config barang tergantung pada kelas CI. Setiap entri harus diawali dengan nama kelas dan titik dua ganda (i.e. Computer::). Ada beberapa CI-Atribut yang umum untuk semua CI (Contoh untuk Komputer kelas: Komputer :: Nama, Komputer :: CurDeplState, Komputer :: CreateTime). Untuk menunjukkan individu CI-Atribut sebagaimana didefinisikan dalam CI-Definition, skema berikut harus digunakan (misalnya untuk Komputer kelas):  Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Jika tidak ada entri untuk kelas CI, maka kolom standar ditampilkan sebagaimana didefinisikan dalam pengaturan ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        'Mendefinisikan kolom yang ditampilkan dari CI di link tabel tampilan yang kompleks, tergantung pada kelas CI. Setiap entri harus diawali dengan nama kelas dan titik dua ganda (i.e. Computer::). Ada beberapa CI-Atribut yang umum untuk semua CI (Contoh untuk Komputer kelas: Komputer :: Nama, Komputer :: CurDeplState, Komputer :: CreateTime). Untuk menunjukkan individu CI-Atribut sebagaimana didefinisikan dalam CI-Definition, skema berikut harus digunakan (misalnya untuk Komputer kelas):  Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Jika tidak ada entri untuk kelas CI, maka kolom standar ditampilkan sebagaimana didefinisikan dalam pengaturan ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = 'Duplikat';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        'Memungkinkan fitur item konfigurasi tindakan massal untuk agen frontend untuk bekerja pada lebih dari satu item konfigurasi pada suatu waktu.';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        'Memungkinkan fitur tindakan massal item konfigurasi hanya untuk kelompok terdaftar.';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        'Mengaktifkan / menonaktifkan fungsi untuk memeriksa ConfigItems untuk nama yang unik. Sebelum mengaktifkan opsi ini Anda harus memeriksa sistem anda untuk produk-config yang sudah ada dengan nama duplikat. Anda dapat melakukan ini dengan script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = 'Modul untuk memeriksa kelompok yang bertanggung jawab untuk kelas.';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        'Modul untuk memeriksa kelompok yang bertanggung jawab untuk item konfigurasi.';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = 'Modul untuk menghasilkan statistik config barang ITSM.';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        'Objek backend pendaftaran modul untuk modul impor / ekspor.';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        'Parameter untuk warna state penyebaran dalam preferensi melihat antarmuka agen.';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        'Parameter untuk state penyebaran dalam preferensi melihat antarmuka agen.';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parameter untuk grup conoth perizinan dari atribut katalog umum';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        'Parameter untuk halaman (di mana item konfigurasi akan ditampilkan).';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        'izin yang diperlukan untuk menggunakan layar item konfigurasi ITSM di antarmuka agen.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        'izin yang diperlukan untuk menggunakan konfigurasi ITSM layar pencarian item dalam antarmuka agen.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        'izin yang diperlukan untuk menggunakan konfigurasi ITSM item layar zoom di antarmuka agen.';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        'izin yang diperlukan untuk menggunakan add ITSM layar item konfigurasi dalam antarmuka agen.';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        'izin yang diperlukan untuk menggunakan mengedit ITSM layar item konfigurasi dalam antarmuka agen.';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        'izin yang diperlukan untuk menggunakan sejarah ITSM layar item konfigurasi dalam antarmuka agen.';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        'Izin Yang diperlukan untuk review using Sejarah ITSM Layar Item Konfigurasi Antarmuka agen.';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        'Memilih item konfigurasi modul generator nomor. "AutoIncrement" increment nomor item konfigurasi, SystemID, yang ConfigItemClassID dan meja yang digunakan. Formatnya adalah "SystemID.ConfigItemClassID.Counter", misalnya 1205000004, 1205000005.';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        'Atur status penyebaran di item konfigurasi layar terbesar dari antarmuka agen.';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        'Atur status kejadian pada item konfigurasi layar terbesar dari antarmuka agen.';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        'Menunjukkan link dalam menu yang memungkinkan menghubungkan item konfigurasi dengan objek lain di config barang pandangan zoom dari antarmuka agen.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        'Menunjukkan link dalam menu untuk mengakses sejarah item konfigurasi pada item konfigurasi gambaran dari antarmuka agen.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        'Menunjukkan link dalam menu untuk mengakses sejarah item konfigurasi dalam tampilan zoom-nya dari antarmuka agen.';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        'Menunjukkan link dalam menu untuk menduplikasi item konfigurasi pada item konfigurasi gambaran dari antarmuka agen.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        'Menunjukkan link dalam menu untuk menduplikasi item konfigurasi dalam tampilan zoom-nya dari antarmuka agen.';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        'Menunjukkan link dalam menu untuk mengedit item konfigurasi dalam tampilan zoom-nya dari antarmuka agen.';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        'Menunjukkan link dalam menu untuk mencetak item konfigurasi dalam tampilan zoom-nya dari antarmuka agen.';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        'Menunjukkan sejarah config item (urutan terbalik) di antarmuka agen.';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        'Pengenal untuk item konfigurasi, misalnya ConfigItem #, MyConfig Item #. default adalah ConfigItem #.';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::it_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Gestione degli elementi di configurazione';
    $Self->{Translation}->{'Change class definition'} = 'Modifica la definizione della classe';
    $Self->{Translation}->{'Config Item Class'} = 'Config Item Class';
    $Self->{Translation}->{'Definition'} = 'Definizione';
    $Self->{Translation}->{'Change'} = 'Modifica';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Stato dell\'incidente';
    $Self->{Translation}->{'Deployment State'} = 'Stato di implementazione';
    $Self->{Translation}->{'Class'} = 'Classe';
    $Self->{Translation}->{'Deployment State Type'} = 'Tipo di stato di distribuzione';
    $Self->{Translation}->{'Current Incident State'} = 'Stato attuale dell\'Incidente';
    $Self->{Translation}->{'Current Incident State Type'} = 'Tipo di stato dell\'incidente corrente';
    $Self->{Translation}->{'Last changed'} = 'Ultima modifica';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Elemento di configurazione';
    $Self->{Translation}->{'Filter for Classes'} = 'Filtro per classi';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Selezionare una classe dall\'elenco per creare un nuovo elemento di configurazione.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'Azione in blocco di ITSM ConfigItem';
    $Self->{Translation}->{'Deployment state'} = 'Stato di distribuzione';
    $Self->{Translation}->{'Incident state'} = 'Stato dell\'incidente';
    $Self->{Translation}->{'Link to another'} = 'Collega a un altro';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = 'Numero articolo di configurazione non valido!';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'Il numero di un altro elemento di configurazione con cui collegarsi.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = 'Vuoi davvero eliminare questo elemento di configurazione?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'Il nome di questo elemento di configurazione';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'Il nome è già in uso da ConfigItems con i seguente/i numer(o)i: %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = 'Storia della voce di configurazione: %s';
    $Self->{Translation}->{'History Content'} = 'Contenuto dello storico';
    $Self->{Translation}->{'Createtime'} = 'Istante di creazione';
    $Self->{Translation}->{'Zoom view'} = 'Vista di Dettaglio';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Impostazioni di contesto';
    $Self->{Translation}->{'Config Items per page'} = 'Config. articoli per pagina';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = 'Una tabella di elementi di configurazione ITSM generica';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Esegui ricerca';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Cercare anche in versioni precedenti?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Elemento di configurazione';
    $Self->{Translation}->{'Configuration Item Information'} = 'Informazioni sull\'elemento di configurazione';
    $Self->{Translation}->{'Current Deployment State'} = 'Stato attuale di implementazione';
    $Self->{Translation}->{'Last changed by'} = 'Ultima modifica effettuata da';
    $Self->{Translation}->{'Show one version'} = 'Mostra una versione';
    $Self->{Translation}->{'Show all versions'} = 'Mostra tutte le versioni';
    $Self->{Translation}->{'Version Incident State'} = 'Stato incidente versione';
    $Self->{Translation}->{'Version Deployment State'} = 'Stato di distribuzione della versione';
    $Self->{Translation}->{'Version Number'} = 'Numero di versione';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Dettagli della versione dell\'elemento di configurazione';
    $Self->{Translation}->{'Property'} = 'Proprietà';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = 'Non è concesso l\'accesso alla classe!';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = 'Panoramica: ITSM ConfigItem';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = 'Non viene fornito ConfigItemID!';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = 'È necessario almeno un elemento di configurazione selezionato!';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        'Non hai accesso in scrittura a questo elemento di configurazione: %s.';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = 'Voce di configurazione "%s" non trovata nel database!';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = 'Impossibile eliminare l\'ID di configurazione %s!';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = 'Nessuna versione trovata per ConfigItemID %s!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = 'Non viene fornito ConfigItemID, DuplicateID o ClassID!';
    $Self->{Translation}->{'No access is given!'} = 'Nessun accesso è concesso!';
    $Self->{Translation}->{'No definition was defined for class %s!'} = 'Nessuna definizione è stata definita per la classe %s!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = 'Impossibile mostrare la cronologia, non viene fornito ConfigItemID!';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = 'Impossibile mostrare la cronologia, nessun diritto di accesso concesso!';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = 'Nuovo ConfigItem (ID =%s)';
    $Self->{Translation}->{'New version (ID=%s)'} = 'Nuova versione (ID =%s)';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = 'Stato di distribuzione aggiornato (nuovo=%s, vecchio=%s)';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = 'Stato dell\'incidente aggiornato (nuovo=%s, vecchio= %s)';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = 'ConfigItem (ID=%s) elimina';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = 'Collegamento a %s (tipo=%s) aggiunta';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = 'Collegamento a %s (tipo=%s) eliminata';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = 'Definizione di ConfigItem aggiornata (ID=%s)';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = 'Nome aggiornato (nuovo=%s, vecchio=%s)';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = 'Attributo %s aggiornato da "%s" a "%s"';
    $Self->{Translation}->{'Version %s deleted'} = 'Versione %s eliminata';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = 'Non viene fornito ConfigItemID o VersionID!';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = 'Impossibile mostrare l\'elemento di configurazione, nessun diritto di accesso concesso!';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = 'ConfigItemID %s non trovato nel database!';
    $Self->{Translation}->{'VersionID %s not found in database!'} = 'VersionID %s non trovato nel database!';
    $Self->{Translation}->{'ConfigItem'} = 'ConfigItem';
    $Self->{Translation}->{'printed by %s at %s'} = 'stampato da %s a %s';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = 'ClassID non valido!';
    $Self->{Translation}->{'No ClassID is given!'} = 'Nessun ClassID è dato!';
    $Self->{Translation}->{'No access rights for this class given!'} = 'Nessun diritto di accesso per questa classe dato!';
    $Self->{Translation}->{'No Result!'} = 'Nessun risultato!';
    $Self->{Translation}->{'Config Item Search Results'} = 'Configura i risultati della ricerca degli elementi';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        'Impossibile mostrare l\'elemento, non viene concesso alcun diritto di accesso per ConfigItem!';
    $Self->{Translation}->{'operational'} = 'operativo';
    $Self->{Translation}->{'warning'} = 'avvertimento';
    $Self->{Translation}->{'incident'} = 'incidente';
    $Self->{Translation}->{'The deployment state of this config item'} = 'Lo stato di implementazione di questo elemento di configurazione';
    $Self->{Translation}->{'The incident state of this config item'} = 'Lo stato di incidente di questo elemento di configurazione';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = 'Elementi di configurazione mostrati';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'fra';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Numero massimo di un elemento';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'I campi vuoti indicano che i valori correnti sono mantenuti';
    $Self->{Translation}->{'Skipped'} = 'Saltato';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Modello';
    $Self->{Translation}->{'Customer Company'} = 'Società cliente';
    $Self->{Translation}->{'Serial Number'} = 'Numero seriale';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = 'Ram';
    $Self->{Translation}->{'Hard Disk'} = 'Disco fisso';
    $Self->{Translation}->{'Capacity'} = 'Capacità';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Scheda di rete';
    $Self->{Translation}->{'IP over DHCP'} = 'IP su DHCP';
    $Self->{Translation}->{'IP Address'} = 'Indirizzo IP';
    $Self->{Translation}->{'Graphic Adapter'} = 'Scheda grafica';
    $Self->{Translation}->{'Other Equipment'} = 'Altri apparati';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Data di scadenza della garanzia';
    $Self->{Translation}->{'Install Date'} = 'Data di installazione';
    $Self->{Translation}->{'Phone 1'} = 'Telefono 1';
    $Self->{Translation}->{'Phone 2'} = 'Telefono 2';
    $Self->{Translation}->{'E-Mail'} = 'E-Mail';
    $Self->{Translation}->{'Network Address'} = 'Indirizzo di rete';
    $Self->{Translation}->{'Subnet Mask'} = 'Maschera di rete';
    $Self->{Translation}->{'Gateway'} = 'Gateway';
    $Self->{Translation}->{'Licence Type'} = 'Tipo di licenza';
    $Self->{Translation}->{'Licence Key'} = 'Chiave di licenza';
    $Self->{Translation}->{'Quantity'} = 'Quantità';
    $Self->{Translation}->{'Expiration Date'} = 'Data di scadenza';
    $Self->{Translation}->{'Media'} = 'Supporto';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = 'Computer';
    $Self->{Translation}->{'Hardware'} = 'Hardware';
    $Self->{Translation}->{'Network'} = 'Rete';
    $Self->{Translation}->{'Software'} = 'Software';
    $Self->{Translation}->{'Expired'} = 'Scaduto';
    $Self->{Translation}->{'Maintenance'} = 'Manutenzione';
    $Self->{Translation}->{'Pilot'} = 'Pilot';
    $Self->{Translation}->{'Planned'} = 'Pianificato';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Riparazione';
    $Self->{Translation}->{'Retired'} = 'Ritirato';
    $Self->{Translation}->{'Review'} = 'Revisionato';
    $Self->{Translation}->{'Test/QA'} = 'Test';
    $Self->{Translation}->{'Laptop'} = 'Portatile';
    $Self->{Translation}->{'Desktop'} = 'Desktop';
    $Self->{Translation}->{'PDA'} = 'Palmare';
    $Self->{Translation}->{'Server'} = 'Server';
    $Self->{Translation}->{'Other'} = 'Altro';
    $Self->{Translation}->{'Monitor'} = 'Monitor';
    $Self->{Translation}->{'Printer'} = 'Stampante';
    $Self->{Translation}->{'Switch'} = 'Switch';
    $Self->{Translation}->{'Router'} = 'Router';
    $Self->{Translation}->{'WLAN Access Point'} = 'WLAN Access Point';
    $Self->{Translation}->{'Security Device'} = 'Dispositivo di sicurezza';
    $Self->{Translation}->{'Backup Device'} = 'Dispositivo di copia';
    $Self->{Translation}->{'Mouse'} = 'Mouse';
    $Self->{Translation}->{'Keyboard'} = 'Tastiera';
    $Self->{Translation}->{'Camera'} = 'Videocamera';
    $Self->{Translation}->{'Beamer'} = 'Beamer';
    $Self->{Translation}->{'Modem'} = 'Modem';
    $Self->{Translation}->{'PCMCIA Card'} = 'Scheda PCMCIA';
    $Self->{Translation}->{'USB Device'} = 'Dispositivo USB';
    $Self->{Translation}->{'Docking Station'} = 'Docking Station';
    $Self->{Translation}->{'Scanner'} = 'Scanner';
    $Self->{Translation}->{'Building'} = 'Edificio';
    $Self->{Translation}->{'Office'} = 'Ufficio';
    $Self->{Translation}->{'Floor'} = 'Piano';
    $Self->{Translation}->{'Room'} = 'Stanza';
    $Self->{Translation}->{'Rack'} = 'Armadio';
    $Self->{Translation}->{'Workplace'} = 'Posto di lavoro';
    $Self->{Translation}->{'Outlet'} = 'Presa';
    $Self->{Translation}->{'IT Facility'} = 'Strumento IT';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Applicazione client';
    $Self->{Translation}->{'Middleware'} = 'Middleware';
    $Self->{Translation}->{'Server Application'} = 'Applicazione server';
    $Self->{Translation}->{'Client OS'} = 'SO Cliente';
    $Self->{Translation}->{'Server OS'} = 'SO server';
    $Self->{Translation}->{'Admin Tool'} = 'Strumenti di amministrazione';
    $Self->{Translation}->{'User Tool'} = 'Strumenti utente';
    $Self->{Translation}->{'Embedded'} = 'Integrato';
    $Self->{Translation}->{'Single Licence'} = 'Licenza singola';
    $Self->{Translation}->{'Per User'} = 'Per utente';
    $Self->{Translation}->{'Per Processor'} = 'Per processore';
    $Self->{Translation}->{'Per Server'} = 'Per server';
    $Self->{Translation}->{'Per Node'} = 'Per nodo';
    $Self->{Translation}->{'Volume Licence'} = 'Licenza per volume';
    $Self->{Translation}->{'Enterprise Licence'} = 'Licenza aziendale';
    $Self->{Translation}->{'Developer Licence'} = 'Licenza di sviluppo';
    $Self->{Translation}->{'Demo'} = 'Dimostrativo';
    $Self->{Translation}->{'Time Restricted'} = 'Limitata nel tempo';
    $Self->{Translation}->{'Freeware'} = 'Freeware';
    $Self->{Translation}->{'Open Source'} = 'Open Source';
    $Self->{Translation}->{'Unlimited'} = 'Illimitata';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = 'CIs assegnati';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = 'CIs assegnati alla società cliente';
    $Self->{Translation}->{'CIs assigned to customer user'} = 'CIs assegnati all\'utenza cliente';
    $Self->{Translation}->{'CMDB Settings'} = 'Impostazioni CMDB';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        'Cerca un nome univoco solo all\'interno della stessa classe ConfigItem (\'class\') o globalmente (\'global\'), il che significa che ogni ConfigItem esistente viene preso in considerazione quando si cercano duplicati.';
    $Self->{Translation}->{'Config Items'} = 'Elementi di configurazione';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = 'Aggiungi elemento di configurazione.';
    $Self->{Translation}->{'Config item edit.'} = 'Modifica elemento di configurazione.';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        'Modulo eventi elemento di configurazione che consente la registrazione nella cronologia nell\'interfaccia agenti.';
    $Self->{Translation}->{'Config item history.'} = 'Configura cronologia articoli.';
    $Self->{Translation}->{'Config item print.'} = 'Stampa elemento di configurazione.';
    $Self->{Translation}->{'Config item zoom.'} = 'Configura lo zoom dell\'oggetto.';
    $Self->{Translation}->{'ConfigItemNumber'} = 'ConfigItemNumber';
    $Self->{Translation}->{'Configuration Item Limit'} = 'Limite elemento di configurazione';
    $Self->{Translation}->{'Configuration Item limit per page.'} = 'Limite elemento di configurazione per pagina.';
    $Self->{Translation}->{'Configuration Management Database.'} = 'Database di gestione della configurazione.';
    $Self->{Translation}->{'Configuration item bulk module.'} = 'Modulo di massa elemento di configurazione.';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        'Elemento di configurazione del motore di ricerca dell\'interfaccia utente.';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Crea e gestisci le definizioni per gli elementi di configurazione.';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        'Definisce l\'azione con cui il bottone dei settaggi è disponibile in un oggetto widget collegato (LinkObject::ViewMode = "complex"). Notare che queste Actions devono essere registrate nei seguenti file JS e CSS: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        'Definisce i permessi necessari per creare elementi di configurazione ITSM utilizzando l\'interfaccia generica.';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        'Definisce i permessi necessari per eliminare gli elementi di configurazione ITSM utilizzando l\'interfaccia generica.';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        'Definisce i permessi necessari per ottenere gli elementi di configurazione ITSM utilizzando l\'interfaccia generica.';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        'Definisce i permessi necessari per la ricerca degli elementi di configurazione ITSM utilizzando l\'interfaccia generica.';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        'Definisce i permessi necessari per aggiornare gli elementi di configurazione ITSM utilizzando l\'interfaccia generica.';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        'Definisce un modulo di panoramica per mostrare la vista ridotta di un elenco di elementi di configurazione.';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        'Definisce le espressioni regolari singolarmente per ciascuna classe ConfigItem per controllare il nome ConfigItem e mostrare i messaggi di errore corrispondenti.';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        'Definisce il sotto-oggetto predefinito della classe \'ITSMConfigItem\'.';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        'Definisce il numero di righe per CI l\'editor delle definizioni degli elementi della configurazione nell\'interfaccia di amministrazione.';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        'Definisce l\'ordine degli stati degli incidenti da alto (es. critico) a basso (es. funzionale).';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        'Definisce gli stati di distribuzione pertinenti in cui i ticket collegati possono influire sullo stato di un CI.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        'Definisce il limite di ricerca per la schermata AgentITSMConfigItem.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        'Definisce il limite di ricerca per la schermata AgentITSMConfigItemSearch.';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        'Definisce le colonne visualizzate nella panoramica degli elementi di configurazione. Questa opzione non ha alcun effetto sulla posizione della colonna. Nota: la colonna Classe è sempre disponibile se è selezionato il filtro "Tutto".';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        'Definisce le colonne visualizzate nella ricerca degli elementi di configurazione. Questa opzione non ha alcun effetto sulla posizione della colonna.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Definisce le colonne mostrate degli elementi della configurazione nella panoramica degli elementi di configurazione in base alla classe CI. Ogni voce deve essere preceduta da un nome di classe e due punti (ad esempio Computer::). Esistono alcuni attributi CI comuni a tutti gli elementi della configurazione (esempio per la classe Computer:Computer::Nome, Computer::CurDeplState, Computer::CreateTime). Per mostrare singoli attributi CI come definiti nella definizione CI, è necessario utilizzare il seguente schema (esempio per la classe Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacità::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacità::1. Se non è presente alcuna voce per una classe CI, le colonne predefinite vengono visualizzate come definito nell\'impostazione ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Definisce le colonne mostrate degli elementi della configurazione nella ricerca degli elementi di configurazione in base alla classe CI. Ogni voce deve essere preceduta da un nome di classe e due punti (ad esempio Computer::). Esistono alcuni attributi CI comuni a tutti gli elementi della configurazione (esempio per la classe Computer:Computer::Nome, Computer::CurDeplState, Computer::CreateTime). Per mostrare singoli attributi CI come definiti nella definizione CI, è necessario utilizzare il seguente schema (esempio per la classe Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Se non è presente alcuna voce per una classe CI, le colonne predefinite vengono visualizzate come definito nell\'impostazione ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        'Definisce le colonne visualizzate degli elementi della configurazione nel widget dell\'elemento di configurazione in base alla classe CI. Ogni voce deve essere preceduta da un nome di classe e due punti (ad esempio Computer::). Esistono alcuni attributi CI comuni a tutti gli elementi della configurazione (esempio per la classe Computer: Computer::Nome, Computer::CurDeplState, Computer::CreateTime). Per mostrare singoli attributi CI come definiti nella definizione CI, è necessario utilizzare il seguente schema (esempio per la classe Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer:::HardDisk::2, Computer::HardDisk::2::Capacity::1. Se non è presente alcuna voce per una classe CI, le colonne predefinite vengono visualizzate come definito nell\'impostazione AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (chiave DefaultColumns).';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        'Definisce le colonne di visualizzate degli elementi della configurazione di CIs nella vista complessa della tabella dei collegamenti per tutte le classi CI. Se non è presente alcuna voce, vengono visualizzate le colonne predefinite.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        'Definisce le colonne visualizzate degli elementi della configurazione nella vista complessa della tabella dei collegamenti, a seconda della classe CI. Ogni voce deve essere preceduta da un nome di classe e due punti (ad esempio Computer::). Esistono alcuni attributi CI comuni a tutti gli elementi della configurazione (esempio per la classe Computer: Computer::Nome, Computer::CurDeplState, Computer::CreateTime). Per mostrare singoli attributi CI come definiti nella definizione CI, è necessario utilizzare il seguente schema (esempio per la classe Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Se non è presente alcuna voce per una classe CI, vengono visualizzate le colonne predefinite.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        'Definisce quale tipo di collegamento (denominato dalla prospettiva del ticket) può influire sullo stato di un elemento della configurazione collegato CI.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        'Definisce quale tipo di ticket può influire sullo stato di un elemento della configurazione collegato CI.';
    $Self->{Translation}->{'Delete Configuration Item'} = 'Elimina elemento di configurazione';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = 'Colore dello stato di distribuzione';
    $Self->{Translation}->{'Duplicate'} = 'Duplicato';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        'Abilita la funzione di azione in blocco degli elementi di configurazione affinché il frontend dell\'agente funzioni su più di un elemento di configurazione alla volta.';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        'Abilita la funzione di azione in blocco dell\'elemento di configurazione solo per i gruppi elencati.';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        'Abilita / disabilita la funzionalità per controllare ConfigItems per nomi univoci. Prima di abilitare questa opzione, è necessario controllare il sistema per elementi di configurazione già esistenti con nomi duplicati. Puoi farlo con lo script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        'Modulo evento per impostare lo stato di configurazione su ticket-configitem-link.';
    $Self->{Translation}->{'ITSM config item overview.'} = 'Panoramica degli elementi di configurazione ITSM.';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = 'Modulo per controllare il gruppo responsabile per una classe.';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        'Modulo per verificare il gruppo responsabile di un elemento di configurazione.';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = 'Modulo per generare le statistiche degli elementi di configurazione ITSM.';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        'Registrazione del modulo back-end oggetto per il modulo di importazione / esportazione.';
    $Self->{Translation}->{'Overview.'} = 'Vista Globale.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parametri per il back-end del dashboard della panoramica degli articoli di configurazione della società cliente dell\'interfaccia agenti. "Limite" è il numero di voci visualizzate per impostazione predefinita. "Gruppo" viene utilizzato per limitare l\'accesso al plug-in (ad es. Gruppo: admin; gruppo1; gruppo2;). "Predefinito" determina se il plug-in è abilitato per impostazione predefinita o se l\'utente deve abilitarlo manualmente. "CacheTTLLocal" è il tempo di cache in minuti per il plugin.';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        'I parametri per gli stati di distribuzione colorano nella vista delle preferenze dell\'interfaccia agenti.';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        'Parametri per gli stati di distribuzione nella vista delle preferenze dell\'interfaccia agenti.';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parametri per i gruppi di permessi di esempio degli attributi del catalogo generale.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        'Parametri per le pagine (nelle quali gli elementi di configurazione sono mostrati).';
    $Self->{Translation}->{'Permission Group'} = 'Gruppo di autorizzazioni';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        'Permessi necessari per utilizzare la schermata dell\'elemento di configurazione ITSM nell\'interfaccia agenti.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        'Permessi necessari per utilizzare la schermata di ricerca degli elementi della configurazione ITSM nell\'interfaccia agenti.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        'Permessi necessari per utilizzare la schermata di zoom dell\'elemento di configurazione ITSM nell\'interfaccia agenti.';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        'Permessi necessari per utilizzare la schermata Aggiungi elemento di configurazione ITSM nell\'interfaccia agenti.';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        'Permessi necessari per utilizzare la schermata di modifica dell\'elemento di configurazione ITSM nell\'interfaccia agenti.';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        'Permessi necessari per utilizzare la schermata dell\'elemento di configurazione ITSM della cronologia nell\'interfaccia agenti.';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        'Permessi necessari per utilizzare la schermata di stampa dell\'elemento di configurazione ITSM nell\'interfaccia agenti.';
    $Self->{Translation}->{'Required privileges to delete config items.'} = 'Privilegi richiesti per eliminare gli elementi di configurazione.';
    $Self->{Translation}->{'Search config items.'} = 'Cerca elementi di configurazione.';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        'Seleziona il modulo generatore del numero di elemento di configurazione. "AutoIncrement" incrementa il numero della voce di configurazione, vengono utilizzati SystemID, ConfigItemClassID e il contatore. Il formato è "SystemID.ConfigItemClassID.Counter", ad es. 1205000004, 1205000005.';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        'Imposta automaticamente lo stato dell\'incidente di un elemento della configurazione quando un ticket è collegato a un elemento della configurazione.';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        'Imposta lo stato di distribuzione nella schermata di massa degli elementi di configurazione dell\'interfaccia agenti.';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        'Imposta lo stato dell\'incidente nella schermata in blocco dell\'articolo di configurazione dell\'interfaccia agenti.';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        'Mostra un collegamento nel menu che consente di collegare un elemento di configurazione con un altro oggetto nella vista zoom elemento di configurazione dell\'interfaccia agenti.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        'Mostra un collegamento nel menu per accedere alla cronologia di un elemento di configurazione nella panoramica degli elementi di configurazione dell\'interfaccia agenti.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        'Mostra un collegamento nel menu per accedere alla cronologia di un elemento di configurazione nella vista zoom dell\'interfaccia agenti.';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        'Mostra un collegamento nel menu per eliminare un elemento di configurazione nella sua vista zoom dell\'interfaccia agenti.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        'Mostra un collegamento nel menu per duplicare un elemento di configurazione nella panoramica degli elementi di configurazione dell\'interfaccia agenti.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        'Mostra un collegamento nel menu per duplicare un elemento di configurazione nella sua vista zoom dell\'interfaccia agenti.';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        'Mostra un collegamento nel menu per modificare un elemento di configurazione nella sua vista zoom dell\'interfaccia agenti.';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        'Mostra un collegamento nel menu per tornare indietro nella vista zoom della voce di configurazione dell\'interfaccia agenti.';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        'Mostra un collegamento nel menu per stampare un elemento di configurazione nella sua vista zoom dell\'interfaccia agenti.';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        'Mostra un collegamento nel menu per ingrandire una voce di configurazione nella panoramica delle voci di configurazione dell\'interfaccia agenti.';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        'Mostra la cronologia degli articoli di configurazione (ordine inverso) nell\'interfaccia agenti.';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        'L\'identificatore per un elemento di configurazione, ad es. ConfigItem#, MyConfigItem#. L\'impostazione predefinita è ConfigItem#.';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = 'classe';
    $Self->{Translation}->{'global'} = 'globale';
    $Self->{Translation}->{'postproductive'} = 'postproductive';
    $Self->{Translation}->{'preproductive'} = 'preproductive';
    $Self->{Translation}->{'productive'} = 'produttiva';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::ja_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '構成アイテム管理';
    $Self->{Translation}->{'Change class definition'} = 'クラス定義を変更';
    $Self->{Translation}->{'Config Item Class'} = '構成アイテムのクラス';
    $Self->{Translation}->{'Definition'} = '記述';
    $Self->{Translation}->{'Change'} = '変更';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'インシデント状況';
    $Self->{Translation}->{'Deployment State'} = 'デプロイ・ステータス';
    $Self->{Translation}->{'Class'} = 'クラス';
    $Self->{Translation}->{'Deployment State Type'} = 'デプロイ・ステータスのタイプ';
    $Self->{Translation}->{'Current Incident State'} = '現在のインシデント・ステータス';
    $Self->{Translation}->{'Current Incident State Type'} = '現在のインシデント・ステータスのタイプ';
    $Self->{Translation}->{'Last changed'} = '最終変更日時';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '構成アイテム';
    $Self->{Translation}->{'Filter for Classes'} = 'クラスでフィルタ';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '新規の構成アイテムを作成するには、リストからクラスを選択してください。';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '構成アイテム一括アクション';
    $Self->{Translation}->{'Deployment state'} = 'デプロイ・ステータス';
    $Self->{Translation}->{'Incident state'} = 'インシデント状況';
    $Self->{Translation}->{'Link to another'} = '他とリンクする';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '構成アイテムの値が不正です!';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '他の構成アイテムへのリンク数';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '本当にこの構成アイテムを削除しますか？';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'この 構成アイテム の名称';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'この名称はすでに Number(s): %s の構成アイテムで使用中です。';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '構成アイテムの履歴: %s';
    $Self->{Translation}->{'History Content'} = '履歴内容';
    $Self->{Translation}->{'Createtime'} = '作成日時';
    $Self->{Translation}->{'Zoom view'} = 'ズーム・ビュー';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'コンテキストの設定';
    $Self->{Translation}->{'Config Items per page'} = '1ページ毎の設定項目';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = 'ジェネリック構成アイテムのテーブル';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '検索を実行';
    $Self->{Translation}->{'Also search in previous versions?'} = 'プレビュー時、以前のバージョンも検索対象としますか？';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '構成アイテム';
    $Self->{Translation}->{'Configuration Item Information'} = '構成アイテム情報';
    $Self->{Translation}->{'Current Deployment State'} = '現在のデプロイ・ステータス';
    $Self->{Translation}->{'Last changed by'} = '最終変更者';
    $Self->{Translation}->{'Show one version'} = '説明を表示';
    $Self->{Translation}->{'Show all versions'} = 'すべての説明を表示';
    $Self->{Translation}->{'Version Incident State'} = 'インシデント状態';
    $Self->{Translation}->{'Version Deployment State'} = 'バージョンのデプロイ・ステータス';
    $Self->{Translation}->{'Version Number'} = 'バージョンナンバー';
    $Self->{Translation}->{'Configuration Item Version Details'} = '構成アイテムの詳細';
    $Self->{Translation}->{'Property'} = '項目';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = 'クラスへのアクセス権はありません！';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '一覧: 構成アイテム';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = 'ConfigItemIDが指定されていません！';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '少なくとも1つの選択された構成アイテムが必要です！';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        'この構成アイテムへの書き込みアクセス権がありません: %s';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '構成アイテム "%s"がデータベースに見つかりません！';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '構成アイテムID%sを削除できませんでした！';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = 'ConfigItemID%sのバージョンが見つかりません！';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = 'ConfigItemID、DuplicateID、ClassIDは指定されていません！';
    $Self->{Translation}->{'No access is given!'} = 'アクセス権はありません！';
    $Self->{Translation}->{'No definition was defined for class %s!'} = 'クラス%sの定義が定義されていません！';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '履歴を表示することはできませんが、ConfigItemIDは指定されていません！';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '履歴を表示することはできません。アクセス権は与えられていません！';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '新しい構成アイテム(ID=%s)';
    $Self->{Translation}->{'New version (ID=%s)'} = '新しいバージョン（ID =%s）';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = 'デプロイ・ステータスが更新されました。（新規=%s、前=%s）';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = 'インシデント状態が更新されました（新規=%s、古い=%s）';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '構成アイテムを削除 (ID=%s)';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '%s（タイプ=%s）へのリンクが追加されました';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '%s（タイプ=%s）へのリンクが削除されました';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = 'ConfigItem定義が更新されました（ID =%s）';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '名前が更新されました（新規=%s、古い=%s）';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '属性%sが "%s"から "%s"に更新されました';
    $Self->{Translation}->{'Version %s deleted'} = 'バージョン%sが削除されました';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = 'ConfigItemIDまたはVersionIDは指定されていません！';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '構成アイテムを表示できません。アクセス権はありません！';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = 'ConfigItemID%sはデータベースに見つかりません！';
    $Self->{Translation}->{'VersionID %s not found in database!'} = 'VersionID%sはデータベースに見つかりません！';
    $Self->{Translation}->{'ConfigItem'} = '構成アイテム';
    $Self->{Translation}->{'printed by %s at %s'} = '%sで%sで印刷されました';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = 'ClassIDが無効です！';
    $Self->{Translation}->{'No ClassID is given!'} = 'ClassIDは指定されていません！';
    $Self->{Translation}->{'No access rights for this class given!'} = 'このクラスのアクセス権はありません！';
    $Self->{Translation}->{'No Result!'} = '結果がありません。';
    $Self->{Translation}->{'Config Item Search Results'} = '構成アイテムの検索結果';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '構成アイテムを表示できません。構成アイテムのアクセス権は与えられていません！';
    $Self->{Translation}->{'operational'} = '通常運用';
    $Self->{Translation}->{'warning'} = '警告';
    $Self->{Translation}->{'incident'} = 'インシデント';
    $Self->{Translation}->{'The deployment state of this config item'} = 'この構成アイテムのデプロイ・ステータス';
    $Self->{Translation}->{'The incident state of this config item'} = 'この構成アイテムのインシデント状態';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'この間';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '一エレメントあたりの最大数';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '空欄の項目は現在の値が保持されること意味します';
    $Self->{Translation}->{'Skipped'} = 'スキップされました';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'モデル';
    $Self->{Translation}->{'Customer Company'} = '顧客企業';
    $Self->{Translation}->{'Serial Number'} = 'シリアルナンバー';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = 'RAM';
    $Self->{Translation}->{'Hard Disk'} = 'ハードディスク';
    $Self->{Translation}->{'Capacity'} = '容量';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'ネットワークアダプタ';
    $Self->{Translation}->{'IP over DHCP'} = 'DHCP有効';
    $Self->{Translation}->{'IP Address'} = 'IP アドレス';
    $Self->{Translation}->{'Graphic Adapter'} = 'グラフィックアダプタ';
    $Self->{Translation}->{'Other Equipment'} = 'その他の機器';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'ワランティ終了日';
    $Self->{Translation}->{'Install Date'} = 'インストール日付';
    $Self->{Translation}->{'Phone 1'} = '電話 1';
    $Self->{Translation}->{'Phone 2'} = '電話 2';
    $Self->{Translation}->{'E-Mail'} = 'メールアドレス';
    $Self->{Translation}->{'Network Address'} = 'ネットワークアドレス';
    $Self->{Translation}->{'Subnet Mask'} = 'サブネットマスク';
    $Self->{Translation}->{'Gateway'} = 'ゲートウェイ';
    $Self->{Translation}->{'Licence Type'} = 'ライセンスタイプ';
    $Self->{Translation}->{'Licence Key'} = 'ライセンスキー';
    $Self->{Translation}->{'Quantity'} = '数量';
    $Self->{Translation}->{'Expiration Date'} = '有効期限';
    $Self->{Translation}->{'Media'} = 'メディア';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = 'コンピューター';
    $Self->{Translation}->{'Hardware'} = 'ハードウェア';
    $Self->{Translation}->{'Network'} = 'ネットワーク';
    $Self->{Translation}->{'Software'} = 'ソフトウェア';
    $Self->{Translation}->{'Expired'} = '期限切れ';
    $Self->{Translation}->{'Maintenance'} = 'メンテナンス';
    $Self->{Translation}->{'Pilot'} = 'パイロット';
    $Self->{Translation}->{'Planned'} = '計画';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '修理';
    $Self->{Translation}->{'Retired'} = '引退';
    $Self->{Translation}->{'Review'} = 'レビュー';
    $Self->{Translation}->{'Test/QA'} = 'テスト / QA';
    $Self->{Translation}->{'Laptop'} = 'ノートPC';
    $Self->{Translation}->{'Desktop'} = 'デスクトップ';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'サーバ';
    $Self->{Translation}->{'Other'} = '他';
    $Self->{Translation}->{'Monitor'} = 'モニタ';
    $Self->{Translation}->{'Printer'} = 'プリンタ';
    $Self->{Translation}->{'Switch'} = 'スイッチ';
    $Self->{Translation}->{'Router'} = 'ルータ';
    $Self->{Translation}->{'WLAN Access Point'} = 'WLAN アクセスポイント';
    $Self->{Translation}->{'Security Device'} = 'セキュリティデバイス';
    $Self->{Translation}->{'Backup Device'} = 'バックアップ・デバイス';
    $Self->{Translation}->{'Mouse'} = 'マウス';
    $Self->{Translation}->{'Keyboard'} = 'キーボード';
    $Self->{Translation}->{'Camera'} = 'カメラ';
    $Self->{Translation}->{'Beamer'} = 'プロジェクター';
    $Self->{Translation}->{'Modem'} = 'モデム';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA カード';
    $Self->{Translation}->{'USB Device'} = 'USBデバイス';
    $Self->{Translation}->{'Docking Station'} = 'ドッキングステーション';
    $Self->{Translation}->{'Scanner'} = 'スキャナ';
    $Self->{Translation}->{'Building'} = '構築';
    $Self->{Translation}->{'Office'} = '事務所';
    $Self->{Translation}->{'Floor'} = '切捨て';
    $Self->{Translation}->{'Room'} = '室内';
    $Self->{Translation}->{'Rack'} = 'ラック';
    $Self->{Translation}->{'Workplace'} = '職場';
    $Self->{Translation}->{'Outlet'} = 'アウトレット';
    $Self->{Translation}->{'IT Facility'} = 'ITファシリティ';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = '電話会社';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'クライアントソフト';
    $Self->{Translation}->{'Middleware'} = 'ミドルウェア';
    $Self->{Translation}->{'Server Application'} = 'サーバソフト';
    $Self->{Translation}->{'Client OS'} = 'クライアントOS';
    $Self->{Translation}->{'Server OS'} = 'サーバOS';
    $Self->{Translation}->{'Admin Tool'} = '管理ツール';
    $Self->{Translation}->{'User Tool'} = 'ユーザ・ツール';
    $Self->{Translation}->{'Embedded'} = '埋め込みオブジェクト';
    $Self->{Translation}->{'Single Licence'} = 'シングルライセンス';
    $Self->{Translation}->{'Per User'} = 'ユーザ毎';
    $Self->{Translation}->{'Per Processor'} = 'プロセッサ毎';
    $Self->{Translation}->{'Per Server'} = 'サーバ毎';
    $Self->{Translation}->{'Per Node'} = 'ノード毎';
    $Self->{Translation}->{'Volume Licence'} = 'ボリュームライセンス';
    $Self->{Translation}->{'Enterprise Licence'} = 'エンタープライズ・ライセンス';
    $Self->{Translation}->{'Developer Licence'} = '開発者ライセンス';
    $Self->{Translation}->{'Demo'} = 'デモ';
    $Self->{Translation}->{'Time Restricted'} = '時間制限';
    $Self->{Translation}->{'Freeware'} = 'フリーウェア';
    $Self->{Translation}->{'Open Source'} = 'オープンソース';
    $Self->{Translation}->{'Unlimited'} = '無制限';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = 'CMDBの設定';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '同じConfigItemクラス（ \'class\'）内またはグローバル（ \'global\'）内でのみ一意の名前を確認します。これは、重複を検索するときに既存のConfigItemがすべて考慮されることを意味します。';
    $Self->{Translation}->{'Config Items'} = '構成アイテム';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '構成アイテムを追加';
    $Self->{Translation}->{'Config item edit.'} = '構成アイテムを修正';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '担当者Webインタフェースの履歴を記録する構成アイテムのイベントモジュール';
    $Self->{Translation}->{'Config item history.'} = '構成アイテムの履歴';
    $Self->{Translation}->{'Config item print.'} = '構成アイテムを印刷';
    $Self->{Translation}->{'Config item zoom.'} = '構成アイテムのズーム';
    $Self->{Translation}->{'ConfigItemNumber'} = '構成アイテム番号';
    $Self->{Translation}->{'Configuration Item Limit'} = '構成アイテムの制限';
    $Self->{Translation}->{'Configuration Item limit per page.'} = 'ページあたりの構成アイテムの表示制限';
    $Self->{Translation}->{'Configuration Management Database.'} = '構成管理データベース';
    $Self->{Translation}->{'Configuration item bulk module.'} = '構成アイテムの一括モジュール';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '担当者インターフェースの構成アイテムの検索バックエンドルーター';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '構成アイテムの作成または変更';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        'リンクオブジェクトウィジェット（LinkObject :: ViewMode = "complex"）で設定ボタンを使用できるアクションを定義します。 これらのアクションは、次のJSおよびCSSファイルを登録している必要があります。Core.AllocationList.css、Core.UI.AllocationList.js、Core.UI.Table.Sort.js、Core.Agent.TableFilters.js、及びCore.Agent .LinkObject.js';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        'ジェネリックインターフェイスを活用して構成アイテムを作成する権限を定義する。';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        'ジェネリック・インターフェースを活用して構成アイテムを削除するために必要な権限を定義します。';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        'ジェネリックインターフェイスを活用して構成アイテムを取得する権限を定義する。';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        'ジェネリックインターフェイスを活用して構成アイテムを検索する権限を定義する。';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        'ジェネリックインターフェイスを活用して構成アイテムを変更する権限を定義する。';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '構成アイテムを定義して、構成アイテムのリストの小さなビューを表示します。';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '各ConfigItemクラスの正規表現を個別に定義して、ConfigItem名を確認し、対応するエラーメッセージを表示します。';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '構成アイテムのクラスのデフォルトのサブオブジェクトを定義する。';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '管理インターフェイス Cl 定義エディタの行数を定義する。';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        'インシデント状態の高（例: 危険）から低（例: 情報）へのインシデント状態の順序を定義する。';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        'リンクされたチケットがCIのステータスに影響を与える可能性がある、関連するデプロイ・ステータスを定義します。';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '構成アイテムの画面の検索制限を定義します。';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '構成アイテムの検索画面の検索制限を定義します。';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '構成アイテムの一覧に表示される列を定義します。 このオプションは列の位置には影響しません。 注：クラス列は、フィルター「全て」が選択されている場合は常に使用可能です。';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '構成アイテムの検索に表示される列を定義します。 このオプションは列の位置には影響しません。';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'CIクラスに応じて、表示されるCIの列を設定項目の概要で定義します。 各エントリの先頭には、クラス名と二重コロン（つまり、Computer::)を付ける必要があります。全てのCIに共通のCI属性がいくつかあります。（コンピュータクラスの例：Computer::Name、Computer::CurDeplState、Computer::CreateTime）
 CI定義で定義されている個々のCI属性を表示するには、以下のスキームを使用する必要があります（コンピュータクラスの例）。
Computer::HardDisk::1,
Computer::HardDisk::1::Capacity::1,
Computer::HardDisk::2,
Computer::HardDisk::2::Capacity::1
CIクラスのエントリがない場合は、ITSMConfigItem :: Frontend :: AgentITSMConfigItem ### ShowColumnsの設定で定義されているデフォルトの列が表示されます。';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        'リンクされたCIのステータスに影響するリンクのタイプ（チケットの観点から名前が付けられます）を定義します。';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        'リンクされたCIのステータスに影響するチケットのタイプを定義します。';
    $Self->{Translation}->{'Delete Configuration Item'} = '構成アイテムを削除';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = 'デプロイ・ステータスのカラー';
    $Self->{Translation}->{'Duplicate'} = 'コピー';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '担当者フロントエンドが一度に複数の設定項目で機能するように設定項目一括操作機能を有効にします。';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        'リストされたグループの構成アイテム一括処理機能のみを有効にします。';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        'チケット構成アイテムのリンクに構成アイテムのステータスを設定するイベントモジュール';
    $Self->{Translation}->{'ITSM config item overview.'} = '構成アイテムの一覧';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = 'クラスを担当するグループをチェックするモジュール';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '構成アイテムを担当するグループをチェックするモジュール';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '構成タイテムのレポートを生成するためのモジュール';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        'インポート/エクスポートモジュールのオブジェクトバックエンドモジュール登録';
    $Self->{Translation}->{'Overview.'} = '概要';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        'デプロイメントのパラメータは、担当者Webインタフェースの環境設定ビューで色を示します。';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '担当者Webインタフェースの環境設定ビューで展開状態のパラメータ';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'ジェネラルカタログ属性のパーミッショングループ　の設定値';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        'ページのパラメータ（構成アイテムが表示されます）';
    $Self->{Translation}->{'Permission Group'} = '権限グループ';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '担当者Webインターフェイスで構成アイテム画面を使用するために必要な権限';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '担当者Webインターフェイスの構成アイテムの検索画面を使用するために必要な権限';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '担当者Webインターフェイス構成アイテムのズーム画面を使用するために必要な権限';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '担当者Webインターフェイスで構成アイテムの追加画面を使用するために必要な権限';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '担当者Webインターフェイスで構成アイテムの編集画面を使用するために必要な権限';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '担当者Webインターフェイスで構成アイテムの履歴画面を使用するために必要な権限';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '担当者Webインタフェースで構成アイテムの印刷画面を使用するために必要な権限';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '構成アイテムを削除するために必要な特権';
    $Self->{Translation}->{'Search config items.'} = '構成アイテムを検索';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        'チケットがCIにリンクされている場合、CIのインシデントのステータスを自動的に設定します。';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        'メニューにリンクを表示して、担当者Webインターフェイスのズーム表示で設定項目を削除します。';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '担当者Webインターフェイスで構成アイテムの履歴（逆順）を表示します。';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '構成アイテムの識別子。 ConfigItem＃、MyConfigItem＃、 デフォルトはConfigItem＃です。';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = 'クラス';
    $Self->{Translation}->{'global'} = 'グローバル';
    $Self->{Translation}->{'postproductive'} = '次の生産';
    $Self->{Translation}->{'preproductive'} = '再生産';
    $Self->{Translation}->{'productive'} = '生産';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::ko_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = '';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = '';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::lt_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = 'Pritraukti apžvalgą';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Tarp';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'Biuras';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::lv_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Starp';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'Birojs';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::mk_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Конфиг Член Менаџмент';
    $Self->{Translation}->{'Change class definition'} = 'Промени дефиниција на класа';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = 'Дефиниција';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Состојба на Инцидент';
    $Self->{Translation}->{'Deployment State'} = 'Состојба на Распоред';
    $Self->{Translation}->{'Class'} = 'Класа';
    $Self->{Translation}->{'Deployment State Type'} = 'Тип на Состојба на Распоред';
    $Self->{Translation}->{'Current Incident State'} = 'Сегашна Состојба на Инцидент';
    $Self->{Translation}->{'Current Incident State Type'} = 'Сегашен Тип на Состојба на Инцидент';
    $Self->{Translation}->{'Last changed'} = 'Последно променето';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Конфиг Член';
    $Self->{Translation}->{'Filter for Classes'} = 'Филтер за Класи';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Одбери Класа од листата за да креирате нов Конфиг Член.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'ITSM КонфигЧлен Bulk Action';
    $Self->{Translation}->{'Deployment state'} = 'Распоредувачка состојба';
    $Self->{Translation}->{'Incident state'} = 'Сосотјба на Инцидент';
    $Self->{Translation}->{'Link to another'} = 'Поврзи со друг';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = 'Невалиден Кофигурациски број на Член!';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'Бројот на друг Конфигурациски Член со линк ширина.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'Името на овој конфиг член';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'Името е веќе во употреба од страна на КонфигЧленовите со следнов/те Број(еви): %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = 'Зумирај';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = 'Конфиг Членови по страница';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Преварувај';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Исто така барај во претходните верзи?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Конфигурациски Член';
    $Self->{Translation}->{'Configuration Item Information'} = 'Конфигурациски Член Информација';
    $Self->{Translation}->{'Current Deployment State'} = 'Сегашна Состојба на Распоред';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = 'Прикажи една верзија';
    $Self->{Translation}->{'Show all versions'} = 'Прикажи ги сите верзии';
    $Self->{Translation}->{'Version Incident State'} = 'Верзија на Инцидент Состојба';
    $Self->{Translation}->{'Version Deployment State'} = 'Верзија на Состојба на Распорадување';
    $Self->{Translation}->{'Version Number'} = 'Број на Верзија';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Конфигурациски Член Детали за Верзија';
    $Self->{Translation}->{'Property'} = 'Својства';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = 'КонфигЧлен';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = 'Распоредувачката состојба на овој конфиг член';
    $Self->{Translation}->{'The incident state of this config item'} = 'Инидент состојба на овој конфиг член';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Измеѓу';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Максимален број на еден елемент';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Празните полиња индицираат дека сегашните вредности се задржани';
    $Self->{Translation}->{'Skipped'} = 'Скокнато';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Модел';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Сериски Број';
    $Self->{Translation}->{'CPU'} = 'Процесор';
    $Self->{Translation}->{'Ram'} = 'Ram меморија';
    $Self->{Translation}->{'Hard Disk'} = 'Тврд Диск';
    $Self->{Translation}->{'Capacity'} = 'Капацитет';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Мрежен Адаптер';
    $Self->{Translation}->{'IP over DHCP'} = 'IP преку DHCP';
    $Self->{Translation}->{'IP Address'} = 'IP Адреса';
    $Self->{Translation}->{'Graphic Adapter'} = 'Графички Адаптер';
    $Self->{Translation}->{'Other Equipment'} = 'Друга Опрема';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Датум на Истекување на Гаранција';
    $Self->{Translation}->{'Install Date'} = 'Датум на инсталирање';
    $Self->{Translation}->{'Phone 1'} = 'Телефон 1';
    $Self->{Translation}->{'Phone 2'} = 'Телефон 2';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'Мрежна Адреса';
    $Self->{Translation}->{'Subnet Mask'} = 'Маска на Подмрежа';
    $Self->{Translation}->{'Gateway'} = 'Портал';
    $Self->{Translation}->{'Licence Type'} = 'Тип на Лиценца';
    $Self->{Translation}->{'Licence Key'} = 'Клуч на Лиценца';
    $Self->{Translation}->{'Quantity'} = 'Квантитет';
    $Self->{Translation}->{'Expiration Date'} = 'Дата на истекување';
    $Self->{Translation}->{'Media'} = 'Медиа';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'Истечено';
    $Self->{Translation}->{'Maintenance'} = 'Одржување';
    $Self->{Translation}->{'Pilot'} = 'Пилот';
    $Self->{Translation}->{'Planned'} = 'Планирано';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Поправи';
    $Self->{Translation}->{'Retired'} = 'Прекинати';
    $Self->{Translation}->{'Review'} = 'Преглед';
    $Self->{Translation}->{'Test/QA'} = 'Тест/QA';
    $Self->{Translation}->{'Laptop'} = 'Лаптоп';
    $Self->{Translation}->{'Desktop'} = 'Работна површина';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Сервер';
    $Self->{Translation}->{'Other'} = 'Друго';
    $Self->{Translation}->{'Monitor'} = 'Монитор';
    $Self->{Translation}->{'Printer'} = 'Печатач';
    $Self->{Translation}->{'Switch'} = 'Преклопник(Switch)';
    $Self->{Translation}->{'Router'} = 'Рутер';
    $Self->{Translation}->{'WLAN Access Point'} = 'Бежична Локална Мрежа Место за Пристап';
    $Self->{Translation}->{'Security Device'} = 'Безбедносен Уред';
    $Self->{Translation}->{'Backup Device'} = 'Бекап Уред';
    $Self->{Translation}->{'Mouse'} = 'Глушец';
    $Self->{Translation}->{'Keyboard'} = 'Тастатура';
    $Self->{Translation}->{'Camera'} = 'Камера';
    $Self->{Translation}->{'Beamer'} = 'Бимер';
    $Self->{Translation}->{'Modem'} = 'Модем';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA Картичка';
    $Self->{Translation}->{'USB Device'} = 'USB Уред';
    $Self->{Translation}->{'Docking Station'} = 'Приклучна Станица';
    $Self->{Translation}->{'Scanner'} = 'Скенер';
    $Self->{Translation}->{'Building'} = 'Градејќи';
    $Self->{Translation}->{'Office'} = 'Канцеларија';
    $Self->{Translation}->{'Floor'} = 'Под';
    $Self->{Translation}->{'Room'} = 'Соба';
    $Self->{Translation}->{'Rack'} = 'Рамка';
    $Self->{Translation}->{'Workplace'} = 'Работноместо';
    $Self->{Translation}->{'Outlet'} = 'Излезни';
    $Self->{Translation}->{'IT Facility'} = 'IT Објект';
    $Self->{Translation}->{'LAN'} = 'Локална Мрежа';
    $Self->{Translation}->{'WLAN'} = 'Бежична Локална Мрежа';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Клиент Апликација';
    $Self->{Translation}->{'Middleware'} = 'Middleware';
    $Self->{Translation}->{'Server Application'} = 'Сервисна Апликација';
    $Self->{Translation}->{'Client OS'} = 'Клиент ОС';
    $Self->{Translation}->{'Server OS'} = 'Сервер ОС';
    $Self->{Translation}->{'Admin Tool'} = 'Админ Алатки';
    $Self->{Translation}->{'User Tool'} = 'Корисник Алатка';
    $Self->{Translation}->{'Embedded'} = 'Вградени';
    $Self->{Translation}->{'Single Licence'} = 'Единечна Лиценца';
    $Self->{Translation}->{'Per User'} = 'По Корисник';
    $Self->{Translation}->{'Per Processor'} = 'По Процесор';
    $Self->{Translation}->{'Per Server'} = 'По Сервер';
    $Self->{Translation}->{'Per Node'} = 'По Јазол';
    $Self->{Translation}->{'Volume Licence'} = 'Волумент на Лиценца';
    $Self->{Translation}->{'Enterprise Licence'} = 'Претпријатие Лиценца';
    $Self->{Translation}->{'Developer Licence'} = 'Развивач Лиценца';
    $Self->{Translation}->{'Demo'} = 'Демо';
    $Self->{Translation}->{'Time Restricted'} = 'Време на Забрана';
    $Self->{Translation}->{'Freeware'} = 'Бесплатен софтвер';
    $Self->{Translation}->{'Open Source'} = 'Отворен Извор';
    $Self->{Translation}->{'Unlimited'} = 'Неограничено';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        'Провери за уникатно име само во рамките на иста КонфигЧлен класа (\'class\') или глобално (\'global\'), што значи дека секој постојачки КонфигЧлен е земен во профилот кога се бараат дупликати.';
    $Self->{Translation}->{'Config Items'} = 'Конфиг Членови';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        'Конфиг член настан модули кои овозможуваат запишуваат логови во историја на агент интефејсот.';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = 'Конфигурациски Член Лимит';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Креирај и менаџирај дефиниции за Конфигурациски Членови.';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        'Дефинира Барани пермисии да креира ITSM конфигурациски членови користејќи Генерички Интерфејси.';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        'Дефинира Барани пермисии за  да земе ITSM конфигурациски членови користејќи Генерички Интерфејс.';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        'Дефинира Барани пермисии за ITSM конфигурациски членови користејќи Генерички Интерфејси.';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        'Дефинира Барани пермисии за надградба на ITSM конфигурациски членови на Генерички Интерфејс.';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        'Дефинира преглед модул за приказ на мал преглед на кофигурациска лчен листа.';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        'Дефинира регуларни изрази индивидуално за секоја КонфигЧлен класа за да провери КонфигЧлен име и да ги прикаже грешните кореспондирачките пораки.';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        'Дефинира стандарден субобјект за класата \'ITSMConfigItem\'.';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        'Дефинира број на редици за CI дефиницискиот уредник во админ интерфејсот.';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        'Дефинира лимит за барања за AgentITSMConfigItem екранот.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        'Дефинира лимит за барања за AgentITSMConfigItemSearch екранот.';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        'Дефинира приказ на колони во конфиг членот преглед. Оваа опција нема ефект на позицијата на колони. Забелешка: Класни колони се секогаш достапни ако филтерот \'All\' е одбран.';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        'Дефинира приказ на колони во конфиг член барање. Оваа опција нема ефект на позицијата на колоните.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Дефинира приказ на колони од CIs во конфиг членот прегледот во зависност од CI класата. Секој влез мора да има префикс со класно име и двојни колони (i.e. Computer::). Има неколку CI-Атрибути кои се чести со сите CIs (пример за класта Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). За приказ на идивидуални CI-Attributes како што се дефинирани во  CI-Дефиницијата, следнава шема мора да биде искористена (пример за класата  Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Ако нема влез за CI класа, тогаш стандардните колони се прикажани како што се дефинирани во поставувањето ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Дефинира приказ на колони од CIs во конфиг членот прегледот во зависност од CI класата. Секој влез мора да има префикс со класно име и двојни колони (i.e. Computer::). Има неколку CI-Атрибути кои се чести со сите CIs (пример за класта Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). За приказ на идивидуални CI-Attributes како што се дефинирани во  CI-Дефиницијата, следнава шема мора да биде искористена (пример за класата  Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Ако нема влез за CI класа, тогаш стандардните колони се прикажани како што се дефинирани во поставувањето ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        'Дефинира приказ на колони од CIs во конфиг членот прегледот во зависност од CI класата. Секој влез мора да има префикс со класно име и двојни колони (i.e. Computer::). Има неколку CI-Атрибути кои се чести со сите CIs (пример за класта Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). За приказ на идивидуални CI-Attributes како што се дефинирани во  CI-Дефиницијата, следнава шема мора да биде искористена (пример за класата  Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Ако нема влез за CI класа, тогаш стандардните колони се прикажани.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = 'Дупликат';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        'Овозможува конфигурациски член bulk action карактеристики за агент предендел за да работи со повење од ефен конфигурациски член во исто врме.';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        'Овозможува конфигурациски член bulk action карактеристики само за групните листи.';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        'Овозможува/оневозможува функционалности за проверка на КонфигЧленови за уникатни имиња. Пред да ја овозможите оваа опција треба да го проверите вашиот систем дали веќе постојат конфиг членови со исто име. Вие можете да го направите со скрипата  bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = 'Модул за проверка на одговорност на група за класа.';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        'Модул за проверка на одоговрност на група за конфигурациски член.';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = 'Модул за генерирање ITSM конфиг член статистика.';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        'Објект позадински модул регирстациа за увоз/извоз модулот.';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        'Параметри за боја на распоред состојбите во прегледот со опциите  на агент интерфејсот.';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        'Параметри за распоред состојбите во прегледот со опциите  на агент интерфејсот.';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Параметри за пример дозволите за групи од генерални каталог атрибути.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        'Параметри за страниви(во кои конфигурациските членови се прикажани).';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        'Барани премисии за употреба на ITSM  конфигурациски член екран во агент интерфејсот.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        'Барани премисии за употреба на ITSM  конфигурациски член екран пребарување во агент интерфејсот.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        'Барани премисии за употреба на ITSM  конфигурациски член зголемувачки екран во агент интерфејсот.';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        'Барани премисии за употреба на додавање на ITSM  конфигурациски член екран во агент интерфејсот.';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        'Барани премисии за употреба на уреди ITSM  конфигурациски член екран во агент интерфејсот.';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        'Барани премисии за употреба на историја на ITSM  конфигурациски член екран во агент интерфејсот.';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        'Барани премисии за употреба на принт на ITSM  конфигурациски член екран во агент интерфејсот.';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        'Одбира конфигурациски член број генерирачки модул.  "AutoIncrement"  инкрементира конфигурависки чен број, SystemID,ConfigItemClassID  и бројачот е искористен. Форматот е  "SystemID.ConfigItemClassID.Counter", п.р. 1205000004, 1205000005.';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        'Сетира распоред состојба на конфигурациски член bulk екранот од агент интерфејсот.';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        'Сетира инцидент состојба на конфигурациски член bulk екранот од агент интерфејсот.';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        'Прикажува линк во менито кое овозможува поврзување со конфигурациски член со друг објект во конфиг член  зголемен преглед во агент интерфејсот.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        'Прикажува линк во менито кое пристапува во историјата на  конфигурациски член во конфиг член преглед во агент интерфејсот.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        'Прикажува линк во менито кое овозможува пристап до историјата на конфигурациски член во конфиг член  зголемен преглед во агент интерфејсот.';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        'Прикажува линк во менито со дупликат конфигурациски член во конфиг член преглед во агент интерфејсот.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        'Прикажува линк во менито со дупликат конфигурациски член во зголемен преглед во агент интерфејсот.';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        'Прикажува линк во менито за уредување на конфигурациски член во зголемен преглед во агент интерфејсот.';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        'Прикажува линк во менито за печатење во конфигурациски член зголемен преглед во агент интерфејсот.';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        'Прикажува линк во менито со зголемување на кнфигурациски член во  кнфигурациски член преглед во агент интерфејсот.';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        'Прикажува  конфиг член историја (во обратен редослед) во агент интерфејсот.';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        'Иднетификувачот за конфигурациски член, п.р.  ConfigItem#, MyConfigItem#. Стандардот е ConfigItem#.';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::ms_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Pengurusan Config Item';
    $Self->{Translation}->{'Change class definition'} = 'Ubah definisi kelas';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = 'Definisi';
    $Self->{Translation}->{'Change'} = 'Ubah';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'insiden keadaan';
    $Self->{Translation}->{'Deployment State'} = 'Keadaan Pertukaran';
    $Self->{Translation}->{'Class'} = 'Kelas';
    $Self->{Translation}->{'Deployment State Type'} = 'Jenis Pertukaran Keadaan';
    $Self->{Translation}->{'Current Incident State'} = 'Insiden keadaan semasa';
    $Self->{Translation}->{'Current Incident State Type'} = 'Nyatakan Jenis Insiden Semasa';
    $Self->{Translation}->{'Last changed'} = 'Terkini berubah';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Config Item';
    $Self->{Translation}->{'Filter for Classes'} = 'Menapis Kelas';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Pilih Kelas dari senarai untuk mewujudkan Perkara Config baru.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'ItemKonfig ITSM Tindakan Pukal';
    $Self->{Translation}->{'Deployment state'} = 'Keadaan pertukaran';
    $Self->{Translation}->{'Incident state'} = 'Keadaan insiden';
    $Self->{Translation}->{'Link to another'} = 'Paut ke lain';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = 'Nombor Barangan Konfigurasi Tidak Sah!';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'Nombor Barangan Konfigurasi lain untuk dipautkan dengan.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = 'Anda benar ingin membuang item config ini?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'Nama config item ini';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'Nama sudah digunakan oleh ConfigItems dengan Nombor(nombor-nombor) berikut: %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = 'Kandungan sejarah';
    $Self->{Translation}->{'Createtime'} = 'Cipta masa';
    $Self->{Translation}->{'Zoom view'} = 'Pandangan dibesarkan';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'konteks Tetapan';
    $Self->{Translation}->{'Config Items per page'} = 'Item config per halaman';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Jalankan Carian';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Juga mencari dalam versi sebelumnya?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Perkara konfigurasi';
    $Self->{Translation}->{'Configuration Item Information'} = 'Maklumat Barangan Konfigurasi';
    $Self->{Translation}->{'Current Deployment State'} = 'Pertukaran keadaan semasa';
    $Self->{Translation}->{'Last changed by'} = 'Terakhir diubah oleh';
    $Self->{Translation}->{'Show one version'} = 'Tunjukkan satu versi';
    $Self->{Translation}->{'Show all versions'} = 'Tunjukkan semua versi';
    $Self->{Translation}->{'Version Incident State'} = 'Versi Insiden Negeri';
    $Self->{Translation}->{'Version Deployment State'} = 'Versi Keadaan Pertukaran';
    $Self->{Translation}->{'Version Number'} = 'Bilangan versi';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Konfigurasi Versi Item Keperinchian';
    $Self->{Translation}->{'Property'} = 'harta';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = 'ConfigItem';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = 'Tiada Keputusan!';
    $Self->{Translation}->{'Config Item Search Results'} = 'Hasil carian ConfigItem';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = 'Keadaan penempatan item ini config';
    $Self->{Translation}->{'The incident state of this config item'} = 'Keadaan insiden bagi config item ini';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Between';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Bilangan maksimum satu elemen';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Medan kosong menunjukkan bahawa nilai semasa disimpan';
    $Self->{Translation}->{'Skipped'} = 'Melangkaui';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Model';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Nombor Series';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = 'memori utama';
    $Self->{Translation}->{'Hard Disk'} = 'Cakera Keras';
    $Self->{Translation}->{'Capacity'} = 'Muatan';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Network Adapter';
    $Self->{Translation}->{'IP over DHCP'} = 'IP lebih DHCP';
    $Self->{Translation}->{'IP Address'} = 'IP Addresse';
    $Self->{Translation}->{'Graphic Adapter'} = 'Penyesuai grafik';
    $Self->{Translation}->{'Other Equipment'} = 'Peralatan lain-lain';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Tarikh Waranti Peluputan';
    $Self->{Translation}->{'Install Date'} = 'Pasang Tarikh';
    $Self->{Translation}->{'Phone 1'} = 'Telefon 1';
    $Self->{Translation}->{'Phone 2'} = 'Telefon 2';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'Alamat Rangkaian';
    $Self->{Translation}->{'Subnet Mask'} = 'Subnet Mask';
    $Self->{Translation}->{'Gateway'} = 'Gerbang';
    $Self->{Translation}->{'Licence Type'} = 'Jenis Lesen';
    $Self->{Translation}->{'Licence Key'} = 'Kunci Lesen';
    $Self->{Translation}->{'Quantity'} = 'Kuantiti';
    $Self->{Translation}->{'Expiration Date'} = 'Tarikh Tamat Tempoh';
    $Self->{Translation}->{'Media'} = 'Media';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'Tamat Tempoh';
    $Self->{Translation}->{'Maintenance'} = 'penyelenggaraan';
    $Self->{Translation}->{'Pilot'} = 'Pilot';
    $Self->{Translation}->{'Planned'} = 'Dirancang';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Membaiki';
    $Self->{Translation}->{'Retired'} = 'bersara';
    $Self->{Translation}->{'Review'} = 'mengkaji';
    $Self->{Translation}->{'Test/QA'} = 'Ujian/QA';
    $Self->{Translation}->{'Laptop'} = 'Laptop';
    $Self->{Translation}->{'Desktop'} = 'Desktop';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Pelayan';
    $Self->{Translation}->{'Other'} = 'lain-lain';
    $Self->{Translation}->{'Monitor'} = 'Monitor';
    $Self->{Translation}->{'Printer'} = 'pencetak';
    $Self->{Translation}->{'Switch'} = 'Tukar';
    $Self->{Translation}->{'Router'} = 'Router';
    $Self->{Translation}->{'WLAN Access Point'} = 'pusat akses WLAN';
    $Self->{Translation}->{'Security Device'} = 'Peranti Keselamatan';
    $Self->{Translation}->{'Backup Device'} = 'Peralatan Backup';
    $Self->{Translation}->{'Mouse'} = 'tetikus';
    $Self->{Translation}->{'Keyboard'} = 'Juruteknik';
    $Self->{Translation}->{'Camera'} = 'Kamera';
    $Self->{Translation}->{'Beamer'} = 'Beamer';
    $Self->{Translation}->{'Modem'} = 'Modem';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA Kad';
    $Self->{Translation}->{'USB Device'} = 'Peranti USB';
    $Self->{Translation}->{'Docking Station'} = 'Dok Stesen';
    $Self->{Translation}->{'Scanner'} = 'Pengimbas';
    $Self->{Translation}->{'Building'} = 'Bangunan';
    $Self->{Translation}->{'Office'} = 'Pejabat';
    $Self->{Translation}->{'Floor'} = 'tingkat';
    $Self->{Translation}->{'Room'} = 'Bilik';
    $Self->{Translation}->{'Rack'} = 'Rak';
    $Self->{Translation}->{'Workplace'} = 'Tempat kerja';
    $Self->{Translation}->{'Outlet'} = 'Outlet';
    $Self->{Translation}->{'IT Facility'} = 'IT Kemudahan';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telko';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Aplikasi pelanggan';
    $Self->{Translation}->{'Middleware'} = 'Middleware';
    $Self->{Translation}->{'Server Application'} = 'Permohonan Pelayan';
    $Self->{Translation}->{'Client OS'} = 'OS Pelanggan';
    $Self->{Translation}->{'Server OS'} = 'Server OS';
    $Self->{Translation}->{'Admin Tool'} = 'Admin Tool';
    $Self->{Translation}->{'User Tool'} = 'Alat pengguna';
    $Self->{Translation}->{'Embedded'} = 'terbenam';
    $Self->{Translation}->{'Single Licence'} = 'Lesen Sahaja';
    $Self->{Translation}->{'Per User'} = 'setiap Pengguna';
    $Self->{Translation}->{'Per Processor'} = 'untuk Pemproses';
    $Self->{Translation}->{'Per Server'} = 'Setiap pelayan';
    $Self->{Translation}->{'Per Node'} = 'per Nod';
    $Self->{Translation}->{'Volume Licence'} = 'Lesen Jilid';
    $Self->{Translation}->{'Enterprise Licence'} = 'Lesen Enterprise';
    $Self->{Translation}->{'Developer Licence'} = 'Lesen Pemaju';
    $Self->{Translation}->{'Demo'} = 'Demo';
    $Self->{Translation}->{'Time Restricted'} = 'masa Terhad';
    $Self->{Translation}->{'Freeware'} = 'Freeware';
    $Self->{Translation}->{'Open Source'} = 'Sumber Terbuka';
    $Self->{Translation}->{'Unlimited'} = 'Tiada batasan';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        'Semak nama yang unik sahaja di dalam kelas ConfigItem yang sama (\'kelas\') atau di peringkat global (\'global\'), yang bermaksud setiap ConfigItem sedia ada diambil kira ketika mencari pendua.';
    $Self->{Translation}->{'Config Items'} = 'Barangan Konfigurasi';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = 'Tambah item config.';
    $Self->{Translation}->{'Config item edit.'} = 'Edit item config.';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        'Modul acara barangan konfigurasi yang membenarkan untuk log ke sejarah dalam antara muka agen.';
    $Self->{Translation}->{'Config item history.'} = 'Sejarah item config.';
    $Self->{Translation}->{'Config item print.'} = 'Cetak item config.';
    $Self->{Translation}->{'Config item zoom.'} = 'Zum item config.';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = 'Had Barangan Konfigurasi';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = 'Konfigurasi Pengurusan Pangkalan Data.';
    $Self->{Translation}->{'Configuration item bulk module.'} = 'Konfigurasi item modul pukal.';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        ' Router carian belakang barangan konfigurasi antara muka ejen.';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Cipta dan urus takrifan untuk Barangan Konfigurasi.';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        'Takrifkan Tindakan dimana butang tetapan itu ada dalam widget objek bersambung (LinkObject::ViewMode = "complex"). Sila pastikan yang Tindakan ini perlu didaftarkan yang berikut fail-fail JS dan CSS: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        'Mentakrifkan kebenaran yang diperlukan untuk mencipta barangan konfigurasi ITSM menggunakan Antara Muka Generik.';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        'Takrifkan keizinan yang diperlukan untuk membuang item konfigurasi ITSM menggunakan Antaramuka Umum.';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        'Mentakrifkan kebenaran yang diperlukan untuk mendapatkan barangan konfigurasi ITSM menggunakan Antara Muka Generik.';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        'Mentakrifkan kebenaran yang diperlukan untuk mencari barangan konfigurasi ITSM menggunakan Antara Muka Generik.';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        'Mentakrifkan kebenaran yang diperlukan untuk mengemaskini barangan konfigurasi ITSM menggunakan Antara Muka Generik.';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        'Mentakrifkan modul gambaran untuk menunjukkan pandangan yang kecil senarai item konfigurasi.';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        'Mentakrifkan ungkapan biasa secara individu untuk setiap kelas ItemKonfig untuk memeriksa nama ItemKonfig dan untuk menunjukkan mesej ralat sepadan.';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        'Definiert das Standard-Subobject der Klasse';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        'Menentukan bilangan baris untuk editor definisi CI dalam antara muka admin.';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        'Takrifkan susunan keadaan insiden daripada tinggi (cth. kritikal) ke rendah (cth. fungsian).';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        'Takrifkan keadaan penempatan yang releven dimana tiket bersambung boleh menjejaskan status bagi CI.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        'Mentakrifkan had carian untuk skrin ItemKonfigITSMAgen.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        'Mentakrifkan had carian untuk skrin CarianItemKonfigITSMAgen.';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        'Mentakrifkan ruangan ditunjukkan dalam gambaran keseluruhan item konfig. Pilihan ini tidak mempunyai kesan ke atas kedudukan ruangan. Nota: ruangan Kelas sentiasa ada jika penapis \'Semua\' dipilih.';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        'Mentakrifkan ruangan yang ditunjukkan dalam carian item konfig. Pilihan ini tidak mempunyai kesan ke atas kedudukan tiang.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Mentakrifkan ruangan ditunjukkan daripada CIs dalam gambaran keseluruhan item konfig bergantung kepada kelas CI . Setiap penyertaan hendaklah dimulakan dengan nama kelas dan titik bertindih dua (iaitu Komputer::). Terdapat beberapa sifat-CI yang ada pada kesemua CIs (contoh untuk kelas Komputer:Komputer::Nama, Komputer::CurDeplState, Komputer::MasaCipta). Untuk menunjukkan sifat individu CI sebagaimana yang ditakrifkan dalam Definisi-CI, skim berikut mesti digunakan (contoh untuk kelas Komputer): Komputer::harddisk::1, Komputer::harddisk::1::Kapasiti::1, komputer::harddisk::2, Komputer::harddisk::2::Kapasiti::1. Jika tiada penyertaan untuk kelas CI, maka ruangan default ditunjukkan sebagaimana yang ditakrifkan dalam ITSMKonfigItem::Bahagian Depan::AgentITSMKonfigItem###TunjukRuangan.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Mentakrifkan ruangan ditunjukkan daripada CIs dalam gambaran keseluruhan item konfig bergantung kepada kelas CI . Setiap penyertaan hendaklah dimulakan dengan nama kelas dan titik bertindih dua (iaitu Komputer::). Terdapat beberapa sifat-CI yang ada pada kesemua CIs (contoh untuk kelas Komputer:Komputer::Nama, Komputer::CurDeplState, Komputer::MasaCipta). Untuk menunjukkan sifat individu CI sebagaimana yang ditakrifkan dalam Definisi-CI, skim berikut mesti digunakan (contoh untuk kelas Komputer): Komputer::harddisk::1, Komputer::harddisk::1::Kapasiti::1, komputer::harddisk::2, Komputer::harddisk::2::Kapasiti::1. Jika tiada penyertaan untuk kelas CI, maka ruangan default ditunjukkan sebagaimana yang ditakrifkan dalam ITSMKonfigItem::Bahagian Depan::AgentITSMKonfigItem###TunjukRuangan.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        'Mentakrifkan ruangan ditunjukkan CIs dalam pandangan pautan jadual kompleks, bergantung kepada kelas CI . Setiap penyertaan hendaklah dimulakan dengan nama kelas dan titik bertindih dua (iaitu Komputer::). Terdapat beberapa Sifat-sifat-CI yang sama kepada semua CIs (contoh untuk kelas Komputer:Komputer::Nama, Komputer::CurDeplState, Komputer::CreateTime). Untuk menunjukkan sifat individu CI sebagaimana yang ditakrifkan dalam Definisi-CI, skim berikut mesti digunakan (contoh untuk kelas Komputer): Komputer::harddisk::1, Komputer::harddisk::1::Kapasiti::1, komputer::harddisk::2, Komputer::harddisk::2::Kapasiti::1. Jika tiada penyertaan untuk kelas CI, maka ruangan default dipaparkan.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        'Takrifkan jenis sambungan yang mana (dinamakan daripada perspektif tiket) boleh menjejaskan status bagi satu sambungan CI.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        'Takrifkan jenis tiket yang mana boleh menjejaskan status bagi satu sambungan CI.';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = 'Penempatan Keadaan Warna.';
    $Self->{Translation}->{'Duplicate'} = 'Gandakan';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        'Membolehkan ciri tindakan pukal item konfigurasi untuk frontend ejen untuk bekerja pada lebih daripada satu item konfigurasi pada satu masa.';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        'Membolehkan ciri item konfigurasi tindakan pukal hanya untuk kumpulan yang disenaraikan.';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        'Membolehkan/melumpuhkan fungsi untuk menyemak KonfigItem untuk nama-nama yang unik. Sebelum membolehkan pilihan ini anda perlu menyemak sistem anda untuk barangan konfig yang telah sedia ada dengan nama yang sama. Anda boleh melakukan ini dengan bin skrip/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        'Model acara untuk menyediakan status-itemconfig atas sambungan-itemconfig-tiket.';
    $Self->{Translation}->{'ITSM config item overview.'} = 'pandangan item config ITSM.';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = 'Modul untuk menyemak kumpulan bertanggungjawab untuk kelas.';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        'Modul untuk menyemak kumpulan bertanggungjawab untuk item konfigurasi.';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = 'Modul untuk menjana statistik konfig item ITSM.';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        'Objekt-Backend Modul Registration des Import/Export Moduls.';
    $Self->{Translation}->{'Overview.'} = 'Pandangan keseluruhan.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        'Parameter untuk warna keadaan mengatur kedudukan dalam paparan pilihan dari paparan ejen.';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        'Parameter untuk keadaan mengatur kedudukan dalam paparan pilihan dari paparan ejen.';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parameter untuk contoh kumpulan yang dibenarkan pada ciri-ciri katalog.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        'Parameter untuk muka surat (di mana item konfigurasi ditunjukkan).';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        'Memerlukan kebenaran untuk menggunakan skrin item konfigurasi ITSM dalam paparan ejen.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        'Memerlukan kebenaran untuk menggunakan skrin carian item konfigurasi ITSM dalam paparan ejen.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        'Memerlukan kebenaran untuk menggunakan skrin zum item konfigurasi ITSM dalam paparan ejen.';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        'Memerlukan kebenaran untuk menggunakan skrin tambah item konfigurasi ITSM dalam paparan ejen.';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        'Memerlukan kebenaran untuk menggunakan skrin edit item konfigurasi ITSM dalam paparan ejen.';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        'Memerlukan kebenaran untuk menggunakan skrin sejarah item konfigurasi ITSM dalam paparan ejen.';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        'Memerlukan kebenaran untuk menggunakan skrin cetak item konfigurasi ITSM dalam paparan ejen.';
    $Self->{Translation}->{'Required privileges to delete config items.'} = 'Keistimewaan-keistimewaan dikehendaki untuk memadamkan barang-barang config.';
    $Self->{Translation}->{'Search config items.'} = 'Cari barang-barang config.';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        'Memilih item konfigurasi modul penjana nombor. "PenambahanAuto" menambah bilangan item tatarajah, SistemID, KonfigItemClassID dan kaunter digunakan. Format ini adalah "SystemID.ConfigItemClassID.Counter ", contohnya 1205000004, 1205000005.';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        'Menentukan keadaan insiden bagi CI secara automatik apabila suatu Tiket disambungkan kepada suatu CI.';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        'Set keadaan penempatan dalam skrin pukal item konfigurasi antara muka ejen.';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        'Set keadaan insiden dalam skrin pukal item konfigurasi antara muka ejen.';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        'Menunjukkan pautan dalam menu yang membolehkan menghubungkan item konfigurasi dengan objek lain dalam pandangan zum item konfig bagi antara muka ejen.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        'Tunjuk pautan dalam menu untuk mengakses sejarah item konfigurasi di dalam gambaran keseluruhan item konfigurasi antara muka agen.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        'Menunjukkan pautan dalam menu untuk mengakses sejarah item konfigurasi dalam pandangan zum antara muka ejen.';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        'Menunjukkan satu sambungan kepada menu untuk memadam satu barang konfigurasi  dalam pandangan zumnya daripada antaramuka ejen.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        'Tunjuk pautan dalam menu untuk menyalin item konfigurasi di dalam gambaran keseluruhan item konfigurasi antara muka agen.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        'Menunjukkan pautan dalam menu untuk menyalin barangan konfigurasi dalam pandangan zum antara muka ejen.';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        'Menunjukkan pautan dalam menu untuk edit barangan konfigurasi dalam pandangan zum antara muka ejen.';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        'menunjukkan suatu sambungan pada menu untuk kembali dalam zoom item konfigurasi daripada antara muka agen';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        'Menunjukkan pautan dalam menu untuk mencetak barangan konfigurasi dalam pandangan zum antara muka ejen.';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        'Tunjuk pautan dalam menu untuk zum ke dalam barangan konfigurasi di dalam gambaran keseluruhan barangan konfigurasi antara muka agen.';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        'Tunjuk sejarah barangan konfig (urutan terbalik) dalam antara muka agen.';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        'Pengecam untuk item konfigurasi, contoh: ConfigItem#, MyConfigItem#. Default ialah ConfigItem#.';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = 'kelas';
    $Self->{Translation}->{'global'} = 'global';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::nb_NO_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Administrasjon av konfigurasjonsenheter';
    $Self->{Translation}->{'Change class definition'} = 'Endre klassedefinisjon';
    $Self->{Translation}->{'Config Item Class'} = 'Konfigurasjonsenhetsklasse';
    $Self->{Translation}->{'Definition'} = 'Definisjon';
    $Self->{Translation}->{'Change'} = 'Endre';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Hendelsestilstand';
    $Self->{Translation}->{'Deployment State'} = 'Produksjonssettingstilstand';
    $Self->{Translation}->{'Class'} = 'Klasse';
    $Self->{Translation}->{'Deployment State Type'} = 'Type produksjonssettingstilstand';
    $Self->{Translation}->{'Current Incident State'} = 'Nåværende hendelsestilstand';
    $Self->{Translation}->{'Current Incident State Type'} = 'Nåværende type hendelsestilstand';
    $Self->{Translation}->{'Last changed'} = 'Sist endret';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Konfigurasjonsenhet';
    $Self->{Translation}->{'Filter for Classes'} = 'Filter for klasser';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Velg en klasse fra listen for å opprette en ny konfigurasjonsenhet.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'Masseredigering av konfigurasjonsenheter';
    $Self->{Translation}->{'Deployment state'} = 'Produksjonssettingstilstand';
    $Self->{Translation}->{'Incident state'} = 'Hendelsestilstand';
    $Self->{Translation}->{'Link to another'} = 'Koble til en annen';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = 'Ugyldig konfigurasjonsenhetsnummer!';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'Nummeret til en annen konfigurasjonsenhet du vil koble til.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = 'Vil du virkelig fjerne denne konfigurasjonsenheten?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'Navnet til denne konfigurasjonsenheten';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'Navnet er allerede i bruk for konfigurasjonsenheter med følgende nummer: %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = 'Historien til konfigurasjonsenheten: %s';
    $Self->{Translation}->{'History Content'} = 'Historikk';
    $Self->{Translation}->{'Createtime'} = 'Opprettet';
    $Self->{Translation}->{'Zoom view'} = 'Zoom visning';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Kontekstvalg';
    $Self->{Translation}->{'Config Items per page'} = 'Konfigurasjonsenheter per side';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = 'En generisk ITSM konfigurasjonsenhetstabell';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Utfør søket';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Også søke i tidligere versjoner?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Konfigurasjonsenhet';
    $Self->{Translation}->{'Configuration Item Information'} = 'Opplysninger om konfigurasjonsenheten';
    $Self->{Translation}->{'Current Deployment State'} = 'Nåværende produksjonssettingstilstand';
    $Self->{Translation}->{'Last changed by'} = 'Sist endret av';
    $Self->{Translation}->{'Show one version'} = 'Vis én versjon';
    $Self->{Translation}->{'Show all versions'} = 'Vis alle versjoner';
    $Self->{Translation}->{'Version Incident State'} = 'Hendelsestilstand for versjonen';
    $Self->{Translation}->{'Version Deployment State'} = 'Produksjonssettingstilstand for versjonen';
    $Self->{Translation}->{'Version Number'} = 'Versjon';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Versjonsdetaljer for enheten';
    $Self->{Translation}->{'Property'} = 'Egenskap';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = 'Ingen tilgang er gitt til klassen!';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = 'Oversikt: ITSM konfigurasjonsenhet';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = 'Ingen konfigurasjonsenhetsID er gitt!';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = 'Du må velge minst en konfigurasjonsenhet!';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        'Du har ikke skrivetilgang til denne konfigurasjonsenhet: %s.';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = 'Fant ikke konfigurasjonsenheten "%s" i databasen!';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = 'Kunne ikke slette konfigurasjonsenheten med ID %s!';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = 'Ingen versjon funnet for konfigurasjonsenheten med ID %s!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = 'Konfigurasjonsenhet';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = 'Ingen resultater!';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = 'operativ';
    $Self->{Translation}->{'warning'} = 'advarsel';
    $Self->{Translation}->{'incident'} = 'hendelse';
    $Self->{Translation}->{'The deployment state of this config item'} = 'Produksjonssettingstilstand for denne konfigurasjonsenheten';
    $Self->{Translation}->{'The incident state of this config item'} = 'Hendelsestilstanden for denne konfigurasjonsenheten';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Mellom';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Maks. antall av ett element';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Tomme felter indikerer at de nåværende verdiene beholdes';
    $Self->{Translation}->{'Skipped'} = 'Hoppet over';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Modell';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Serienummer';
    $Self->{Translation}->{'CPU'} = 'Prosessor';
    $Self->{Translation}->{'Ram'} = 'Internminne';
    $Self->{Translation}->{'Hard Disk'} = 'Harddisk';
    $Self->{Translation}->{'Capacity'} = 'Kapasitet';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Nettverkskort';
    $Self->{Translation}->{'IP over DHCP'} = 'IP fra DHCP';
    $Self->{Translation}->{'IP Address'} = 'IP-adresse';
    $Self->{Translation}->{'Graphic Adapter'} = 'Grafikkort';
    $Self->{Translation}->{'Other Equipment'} = 'Annet utstyr';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Utløpsdato for garanti';
    $Self->{Translation}->{'Install Date'} = 'Installasjonsdato';
    $Self->{Translation}->{'Phone 1'} = 'Telefon 1';
    $Self->{Translation}->{'Phone 2'} = 'Telefon 2';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'Nettverksadresse';
    $Self->{Translation}->{'Subnet Mask'} = 'Subnettmaske';
    $Self->{Translation}->{'Gateway'} = 'Ruter';
    $Self->{Translation}->{'Licence Type'} = 'Lisenstype';
    $Self->{Translation}->{'Licence Key'} = 'Lisensnøkkel';
    $Self->{Translation}->{'Quantity'} = 'Antall';
    $Self->{Translation}->{'Expiration Date'} = 'Utgår dato';
    $Self->{Translation}->{'Media'} = 'Media';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = 'Nettverk';
    $Self->{Translation}->{'Software'} = 'Programvare';
    $Self->{Translation}->{'Expired'} = 'Utgått';
    $Self->{Translation}->{'Maintenance'} = 'Vedlikehold';
    $Self->{Translation}->{'Pilot'} = 'Pilot';
    $Self->{Translation}->{'Planned'} = 'Planlagt';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Reparasjon';
    $Self->{Translation}->{'Retired'} = 'Utfaset';
    $Self->{Translation}->{'Review'} = 'Evaluering';
    $Self->{Translation}->{'Test/QA'} = 'Test/Kvalitetskontroll';
    $Self->{Translation}->{'Laptop'} = 'Bærbar';
    $Self->{Translation}->{'Desktop'} = 'Stasjonær';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Tjener';
    $Self->{Translation}->{'Other'} = 'Annet';
    $Self->{Translation}->{'Monitor'} = 'Skjerm';
    $Self->{Translation}->{'Printer'} = 'Skriver';
    $Self->{Translation}->{'Switch'} = 'Svitsj';
    $Self->{Translation}->{'Router'} = 'Ruter';
    $Self->{Translation}->{'WLAN Access Point'} = 'WLAN-basestasjon';
    $Self->{Translation}->{'Security Device'} = 'Sikkerhetsenhet';
    $Self->{Translation}->{'Backup Device'} = 'Backup-enhet';
    $Self->{Translation}->{'Mouse'} = 'Mus';
    $Self->{Translation}->{'Keyboard'} = 'Tastatur';
    $Self->{Translation}->{'Camera'} = 'Kamera';
    $Self->{Translation}->{'Beamer'} = 'Prosjektør';
    $Self->{Translation}->{'Modem'} = 'Modem';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA-kort';
    $Self->{Translation}->{'USB Device'} = 'USB-enhet';
    $Self->{Translation}->{'Docking Station'} = 'Dokkingstasjon';
    $Self->{Translation}->{'Scanner'} = 'Skanner';
    $Self->{Translation}->{'Building'} = 'Bygning';
    $Self->{Translation}->{'Office'} = 'Kontor';
    $Self->{Translation}->{'Floor'} = 'Etasje';
    $Self->{Translation}->{'Room'} = 'Rom';
    $Self->{Translation}->{'Rack'} = 'Rack';
    $Self->{Translation}->{'Workplace'} = 'Arbeidssted';
    $Self->{Translation}->{'Outlet'} = 'Kontakt';
    $Self->{Translation}->{'IT Facility'} = 'IT-fasilitet';
    $Self->{Translation}->{'LAN'} = 'Lokalnett';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Teleutstyr';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Klient-applikasjon';
    $Self->{Translation}->{'Middleware'} = 'Mellomvare';
    $Self->{Translation}->{'Server Application'} = 'Tjener-applikasjon';
    $Self->{Translation}->{'Client OS'} = 'Klient-OS';
    $Self->{Translation}->{'Server OS'} = 'Tjener-OS';
    $Self->{Translation}->{'Admin Tool'} = 'Administratorverktøy';
    $Self->{Translation}->{'User Tool'} = 'Brukerverktøy';
    $Self->{Translation}->{'Embedded'} = 'Innebygd';
    $Self->{Translation}->{'Single Licence'} = 'Enkeltlisens';
    $Self->{Translation}->{'Per User'} = 'Per Bruker';
    $Self->{Translation}->{'Per Processor'} = 'Per Prosessor';
    $Self->{Translation}->{'Per Server'} = 'Per Tjener';
    $Self->{Translation}->{'Per Node'} = 'Per Node';
    $Self->{Translation}->{'Volume Licence'} = 'Volumlisens';
    $Self->{Translation}->{'Enterprise Licence'} = 'Enterprise-lisens';
    $Self->{Translation}->{'Developer Licence'} = 'Utviklerlisens';
    $Self->{Translation}->{'Demo'} = 'Demo';
    $Self->{Translation}->{'Time Restricted'} = 'Tidsbegrenset';
    $Self->{Translation}->{'Freeware'} = 'Gratis programvare';
    $Self->{Translation}->{'Open Source'} = 'Åpen kildekode';
    $Self->{Translation}->{'Unlimited'} = 'Ubegrenset';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = 'Konfigurasjonsenheter';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = 'Farge for produksjonssettingstilstand';
    $Self->{Translation}->{'Duplicate'} = 'Duplisér';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = 'Oversikt over ITSM konfigurasjonsenheter.';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = 'Oversikt.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parametere for tilgangsgruppe-eksempel i attributtene for generell katalog.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = 'Tilgangsgruppe';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = 'Søk etter konfigurasjonsenheter.';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = 'klasse';
    $Self->{Translation}->{'global'} = 'global';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::nl_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Configuratie-item beheer';
    $Self->{Translation}->{'Change class definition'} = 'Wijzig definitie';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = 'Definitie';
    $Self->{Translation}->{'Change'} = 'Wijzigen';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Incidentstatus';
    $Self->{Translation}->{'Deployment State'} = 'Status';
    $Self->{Translation}->{'Class'} = 'Klasse';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = 'Huidige Incidentstatus';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = 'Laatst gewijzigd';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Configuratie-item';
    $Self->{Translation}->{'Filter for Classes'} = 'Filter op klassen';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Selecteer een klasse uit de lijst om een nieuw configuratie-item aan te maken.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'De naam van dit configuratie-item';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = 'Geschiedenisinhoud';
    $Self->{Translation}->{'Createtime'} = 'Maak tijd';
    $Self->{Translation}->{'Zoom view'} = 'Detailoverzicht';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Contextinstellingen';
    $Self->{Translation}->{'Config Items per page'} = 'Aantal configuratie-items per pagina';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Ook in eerdere versies zoeken?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Configuratie-item';
    $Self->{Translation}->{'Configuration Item Information'} = 'Configuratie-item informatie';
    $Self->{Translation}->{'Current Deployment State'} = 'Actuele status';
    $Self->{Translation}->{'Last changed by'} = 'Laatst gewijzigd door';
    $Self->{Translation}->{'Show one version'} = 'Toon één versie';
    $Self->{Translation}->{'Show all versions'} = 'Toon alle versies';
    $Self->{Translation}->{'Version Incident State'} = 'Versie Incident-status';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = 'Versienummer';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Configuratie-item versie details';
    $Self->{Translation}->{'Property'} = 'Attribuut';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = 'Geen resultaat!';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = 'operationeel';
    $Self->{Translation}->{'warning'} = 'waarschuwing';
    $Self->{Translation}->{'incident'} = 'incident';
    $Self->{Translation}->{'The deployment state of this config item'} = 'De status van dit configuratie-item';
    $Self->{Translation}->{'The incident state of this config item'} = 'De incidentstatus van dit configuratie-item';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Tussen';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Maximale hoeveelheid';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = 'Overgeslagen';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Model';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Serienummer';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = 'RAM';
    $Self->{Translation}->{'Hard Disk'} = 'Harde schijf';
    $Self->{Translation}->{'Capacity'} = 'Capaciteit';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Netwerk adapter';
    $Self->{Translation}->{'IP over DHCP'} = 'IP via DHCP';
    $Self->{Translation}->{'IP Address'} = 'IP adres';
    $Self->{Translation}->{'Graphic Adapter'} = 'Videokaart';
    $Self->{Translation}->{'Other Equipment'} = 'Overige uitrusting';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Einddatum garantie';
    $Self->{Translation}->{'Install Date'} = 'Installatiedatum';
    $Self->{Translation}->{'Phone 1'} = 'Telefoon 1';
    $Self->{Translation}->{'Phone 2'} = 'Telefoon 2';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'Netwerkadres';
    $Self->{Translation}->{'Subnet Mask'} = 'Subnet Mask';
    $Self->{Translation}->{'Gateway'} = 'Gateway';
    $Self->{Translation}->{'Licence Type'} = 'Licentietype';
    $Self->{Translation}->{'Licence Key'} = 'Licentiesleutel';
    $Self->{Translation}->{'Quantity'} = 'Hoeveelheid';
    $Self->{Translation}->{'Expiration Date'} = 'Expiratiedatum';
    $Self->{Translation}->{'Media'} = 'Media';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'Verlopen';
    $Self->{Translation}->{'Maintenance'} = 'In onderhoud';
    $Self->{Translation}->{'Pilot'} = 'Pilot';
    $Self->{Translation}->{'Planned'} = 'Geplanned';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'In reparatie';
    $Self->{Translation}->{'Retired'} = 'Uitgefaseerd';
    $Self->{Translation}->{'Review'} = 'Review';
    $Self->{Translation}->{'Test/QA'} = 'Test/QA';
    $Self->{Translation}->{'Laptop'} = 'Laptop';
    $Self->{Translation}->{'Desktop'} = 'Desktop';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Server';
    $Self->{Translation}->{'Other'} = 'Anders';
    $Self->{Translation}->{'Monitor'} = 'Monitor';
    $Self->{Translation}->{'Printer'} = 'Printer';
    $Self->{Translation}->{'Switch'} = 'Switch';
    $Self->{Translation}->{'Router'} = 'Router';
    $Self->{Translation}->{'WLAN Access Point'} = 'WLAN Access Point';
    $Self->{Translation}->{'Security Device'} = 'Security device';
    $Self->{Translation}->{'Backup Device'} = 'Backupapparaat';
    $Self->{Translation}->{'Mouse'} = 'Muis';
    $Self->{Translation}->{'Keyboard'} = 'Toetsenbord';
    $Self->{Translation}->{'Camera'} = 'Camera';
    $Self->{Translation}->{'Beamer'} = 'Beamer';
    $Self->{Translation}->{'Modem'} = 'Modem';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA kaart';
    $Self->{Translation}->{'USB Device'} = 'USB apparaat';
    $Self->{Translation}->{'Docking Station'} = 'Docking station';
    $Self->{Translation}->{'Scanner'} = 'Scanner';
    $Self->{Translation}->{'Building'} = 'Gebouw';
    $Self->{Translation}->{'Office'} = 'Kantoor';
    $Self->{Translation}->{'Floor'} = 'Etage';
    $Self->{Translation}->{'Room'} = 'Kamer';
    $Self->{Translation}->{'Rack'} = 'Rack';
    $Self->{Translation}->{'Workplace'} = 'Werkplek';
    $Self->{Translation}->{'Outlet'} = 'Aansluiting';
    $Self->{Translation}->{'IT Facility'} = 'Serverruimte';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Clienttoepassing';
    $Self->{Translation}->{'Middleware'} = 'Middleware';
    $Self->{Translation}->{'Server Application'} = 'Servertoepassing';
    $Self->{Translation}->{'Client OS'} = 'Client besturingssysteem';
    $Self->{Translation}->{'Server OS'} = 'Server besturingssysteem';
    $Self->{Translation}->{'Admin Tool'} = 'Admin Tool';
    $Self->{Translation}->{'User Tool'} = 'User Tool';
    $Self->{Translation}->{'Embedded'} = 'Embedded';
    $Self->{Translation}->{'Single Licence'} = 'Single Licence';
    $Self->{Translation}->{'Per User'} = 'Per gebruiker';
    $Self->{Translation}->{'Per Processor'} = 'Per CPU';
    $Self->{Translation}->{'Per Server'} = 'Per server';
    $Self->{Translation}->{'Per Node'} = 'Per node';
    $Self->{Translation}->{'Volume Licence'} = 'Volume Licentie';
    $Self->{Translation}->{'Enterprise Licence'} = 'Enterpriselicentie';
    $Self->{Translation}->{'Developer Licence'} = 'Ontwikkelaarslicentie';
    $Self->{Translation}->{'Demo'} = 'Demo';
    $Self->{Translation}->{'Time Restricted'} = 'Time restricted';
    $Self->{Translation}->{'Freeware'} = 'Freeware';
    $Self->{Translation}->{'Open Source'} = 'Open Source';
    $Self->{Translation}->{'Unlimited'} = 'Ongelimiteerd';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = 'Configuratie-items';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Aanmaken en beheren van definities voor configuratie-items.';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        'Definieer acties waarbij een instellingenknop beschikbaar is in de widget voor gekoppelde objecten (LinkObject::ViewMode = "complex"). Houd er rekening mee dat deze acties de volgende JS- en CSS-bestanden moeten hebben geregistreerd: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js en Core.Agent .LinkObject.js.';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = 'Dupliceren';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parameters voor de voorbeeldmachtigingsgroepen van de algemene cataloguskenmerken.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = 'Toestemmingsgroep';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::pl_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Zarządzanie Elementem Konfiguracji';
    $Self->{Translation}->{'Change class definition'} = 'Zmień definicję klasy';
    $Self->{Translation}->{'Config Item Class'} = 'Klasa Elementu Konfiguracji';
    $Self->{Translation}->{'Definition'} = 'Definicja';
    $Self->{Translation}->{'Change'} = 'Zmiana';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Stan zdarzenia';
    $Self->{Translation}->{'Deployment State'} = 'Stan wdrożenia';
    $Self->{Translation}->{'Class'} = 'Klasa';
    $Self->{Translation}->{'Deployment State Type'} = 'Typ stanu wdrożenia';
    $Self->{Translation}->{'Current Incident State'} = 'Aktualny stan incydentu';
    $Self->{Translation}->{'Current Incident State Type'} = 'Aktualny typ stanu';
    $Self->{Translation}->{'Last changed'} = 'Ostatnio zmienione';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Element Konfiguracji';
    $Self->{Translation}->{'Filter for Classes'} = 'Filtr dla klas';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Zaznacz klasę na liście aby utworzyć nowy element konfiguracji.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'Akcja grupowa na Elementach Konfiguracji ITSM';
    $Self->{Translation}->{'Deployment state'} = 'Stan wdrożenia';
    $Self->{Translation}->{'Incident state'} = 'Stan zdarzenia';
    $Self->{Translation}->{'Link to another'} = 'Połącz z innym';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = 'Nieprawidłowy numer elementu konfiguracji!';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'Numer innego elementu konfiguracji do połączenia.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = 'Czy na pewno chcesz usunąć ten element konfiguracji?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'Nazwa tego elementu konfiguracji';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'Nazwa jest już używana przez CI o następujących numerach: %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = 'Historia Elementu Konfiguracji: %s';
    $Self->{Translation}->{'History Content'} = 'Zawartość historii';
    $Self->{Translation}->{'Createtime'} = 'Utworzone';
    $Self->{Translation}->{'Zoom view'} = 'Widok szczegółowy';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Ustawienia kontekstu';
    $Self->{Translation}->{'Config Items per page'} = 'Liczba Elementów Konfiguracji na stronę';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = 'Ogólna tabela elementów konfiguracji ITSM';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Uruchom wyszukiwanie';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Uwzględnić poprzednie wersje?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Element Konfiguracji';
    $Self->{Translation}->{'Configuration Item Information'} = 'Informacje o Elemencie Konfiguracji';
    $Self->{Translation}->{'Current Deployment State'} = 'Aktualny stan wdrożenia';
    $Self->{Translation}->{'Last changed by'} = 'Ostatnio zmienione przez';
    $Self->{Translation}->{'Show one version'} = 'Pokaż jedną wersję';
    $Self->{Translation}->{'Show all versions'} = 'Pokaż wszystkie wersje';
    $Self->{Translation}->{'Version Incident State'} = 'Stan';
    $Self->{Translation}->{'Version Deployment State'} = 'Stan wdrożenia wersji';
    $Self->{Translation}->{'Version Number'} = 'Wersja';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Szczegóły wersji Elementu Konfiguracji';
    $Self->{Translation}->{'Property'} = 'Właściwość';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = 'Przegląd: Elementy Konfiguracji ITSM';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        'Nie masz prawa zapisu do tego elementu konfiguracji: %s.';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = 'Nowa wersja (ID=%s)';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = 'Łącze do %s (typ=%s) dodane';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = 'Łącze do %s (typ=%s) usunięte';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = 'Atrybut %s zmieniony z "%s" na "%s"';
    $Self->{Translation}->{'Version %s deleted'} = 'Wersja %s usunięta';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = 'Element Konfiguracji';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = 'Brak Wyników!';
    $Self->{Translation}->{'Config Item Search Results'} = 'Wyniki wyszukiwania Elementów Konfiguracji';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = 'operacyjny';
    $Self->{Translation}->{'warning'} = 'ostrzeżenie';
    $Self->{Translation}->{'incident'} = 'incydent';
    $Self->{Translation}->{'The deployment state of this config item'} = 'Stan wdrożenia tego CI';
    $Self->{Translation}->{'The incident state of this config item'} = 'Stan zdarzenia tego CI';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = 'Pokazane elementy konfiguracji';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Pomiędzy';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Maksymalna liczba wystąpień pojednyczego elemetntu';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Puste pola znaczają, że zatrzymane będą bieżące wartości';
    $Self->{Translation}->{'Skipped'} = 'Pominięte';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Model';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Numer seryjny';
    $Self->{Translation}->{'CPU'} = 'Procesor';
    $Self->{Translation}->{'Ram'} = 'RAM';
    $Self->{Translation}->{'Hard Disk'} = 'Dysk twardy';
    $Self->{Translation}->{'Capacity'} = 'Pojemność';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Karta sieciowa';
    $Self->{Translation}->{'IP over DHCP'} = 'Adres IP z DHCP';
    $Self->{Translation}->{'IP Address'} = 'Adres IP';
    $Self->{Translation}->{'Graphic Adapter'} = 'Karta graficzna';
    $Self->{Translation}->{'Other Equipment'} = 'Pozostałe wyposażenie';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Data wygaśnięcia gwarancji';
    $Self->{Translation}->{'Install Date'} = 'Data instalacji';
    $Self->{Translation}->{'Phone 1'} = 'Telefon 1';
    $Self->{Translation}->{'Phone 2'} = 'Telefon 2';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'Adres sieciowy';
    $Self->{Translation}->{'Subnet Mask'} = 'Maska sieci';
    $Self->{Translation}->{'Gateway'} = 'Brama';
    $Self->{Translation}->{'Licence Type'} = 'Typ licencji';
    $Self->{Translation}->{'Licence Key'} = 'Klucz licencyjny';
    $Self->{Translation}->{'Quantity'} = 'Ilość';
    $Self->{Translation}->{'Expiration Date'} = 'Data wygaśnięcia';
    $Self->{Translation}->{'Media'} = 'Nośnik';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'Wygasło';
    $Self->{Translation}->{'Maintenance'} = 'Utrzymanie';
    $Self->{Translation}->{'Pilot'} = 'Pilotaż';
    $Self->{Translation}->{'Planned'} = 'Planowany';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Naprawa';
    $Self->{Translation}->{'Retired'} = 'Wycofany';
    $Self->{Translation}->{'Review'} = 'Przegląd';
    $Self->{Translation}->{'Test/QA'} = 'Test/QA';
    $Self->{Translation}->{'Laptop'} = 'Laptop';
    $Self->{Translation}->{'Desktop'} = 'Desktop';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Serwer';
    $Self->{Translation}->{'Other'} = 'Inne';
    $Self->{Translation}->{'Monitor'} = 'Monitor';
    $Self->{Translation}->{'Printer'} = 'Drukarka';
    $Self->{Translation}->{'Switch'} = 'Switch';
    $Self->{Translation}->{'Router'} = 'Ruter';
    $Self->{Translation}->{'WLAN Access Point'} = 'Access Point';
    $Self->{Translation}->{'Security Device'} = 'Urządzenie zabezpieczające';
    $Self->{Translation}->{'Backup Device'} = 'Urządzenie do tworzenia kopii zapasowej';
    $Self->{Translation}->{'Mouse'} = 'Myszka';
    $Self->{Translation}->{'Keyboard'} = 'Klawiatura';
    $Self->{Translation}->{'Camera'} = 'Kamera';
    $Self->{Translation}->{'Beamer'} = 'Projektor';
    $Self->{Translation}->{'Modem'} = 'Modem';
    $Self->{Translation}->{'PCMCIA Card'} = 'Karta PCMCIA';
    $Self->{Translation}->{'USB Device'} = 'Urządzenie USB';
    $Self->{Translation}->{'Docking Station'} = 'Stacja dokująca';
    $Self->{Translation}->{'Scanner'} = 'Skaner';
    $Self->{Translation}->{'Building'} = 'Budynek';
    $Self->{Translation}->{'Office'} = 'Biuro';
    $Self->{Translation}->{'Floor'} = 'Piętro';
    $Self->{Translation}->{'Room'} = 'Pokój';
    $Self->{Translation}->{'Rack'} = 'Rack';
    $Self->{Translation}->{'Workplace'} = 'Miejsce pracy';
    $Self->{Translation}->{'Outlet'} = 'Punkt sprzedaży';
    $Self->{Translation}->{'IT Facility'} = 'Dział IT';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Aplikacja kliencka';
    $Self->{Translation}->{'Middleware'} = 'Middleware';
    $Self->{Translation}->{'Server Application'} = 'Aplikacja serwerowa';
    $Self->{Translation}->{'Client OS'} = 'System kliencki';
    $Self->{Translation}->{'Server OS'} = 'System serwerowy';
    $Self->{Translation}->{'Admin Tool'} = 'Narzędzie administratora';
    $Self->{Translation}->{'User Tool'} = 'Narzędzie Użytkownika';
    $Self->{Translation}->{'Embedded'} = 'Osadzony';
    $Self->{Translation}->{'Single Licence'} = 'Pojedyńcza licencja';
    $Self->{Translation}->{'Per User'} = 'Na Użytkownika';
    $Self->{Translation}->{'Per Processor'} = 'Na procesor';
    $Self->{Translation}->{'Per Server'} = 'Na serwer';
    $Self->{Translation}->{'Per Node'} = 'Na urządzenie';
    $Self->{Translation}->{'Volume Licence'} = 'Zbiorcza licencja';
    $Self->{Translation}->{'Enterprise Licence'} = 'Licencja Enterprise';
    $Self->{Translation}->{'Developer Licence'} = 'Licencja developerska';
    $Self->{Translation}->{'Demo'} = 'Demo';
    $Self->{Translation}->{'Time Restricted'} = 'Czasowo ograniczona';
    $Self->{Translation}->{'Freeware'} = 'Freeware';
    $Self->{Translation}->{'Open Source'} = 'Open source';
    $Self->{Translation}->{'Unlimited'} = 'Bez limitu';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = 'Limit liczby CI';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Tworzenie i zarządzanie definicjami elementów konfiguracji.';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        'Określa akcje dla których jest dostępny przycisk ustawień w widżecie połączonych obiektów (LinkObject::ViewMode = "complex"). Zwróć uwagę, że te akcje muszą mieć zarejestrowane następujące pliki JS i CSS: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = 'Duplikat';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = 'Przegląd.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parametry dla przykładowych grup uprawnień atrybutów katalogu głównego.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = 'Uprawnienia grup';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::pt_BR_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Gerenciamento de Itens de Configuração';
    $Self->{Translation}->{'Change class definition'} = 'Mudar definição de classe';
    $Self->{Translation}->{'Config Item Class'} = 'Classe do Item de Configuração';
    $Self->{Translation}->{'Definition'} = 'Definição';
    $Self->{Translation}->{'Change'} = 'Alterar';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Estado de Incidente';
    $Self->{Translation}->{'Deployment State'} = 'Estado de Implantação';
    $Self->{Translation}->{'Class'} = 'Classe';
    $Self->{Translation}->{'Deployment State Type'} = 'Tipo de Estado de Implantação';
    $Self->{Translation}->{'Current Incident State'} = 'Estado do Incidente Atual';
    $Self->{Translation}->{'Current Incident State Type'} = 'Estado Atual de Incidente por tipo';
    $Self->{Translation}->{'Last changed'} = 'Última modificação';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Item de Configuração';
    $Self->{Translation}->{'Filter for Classes'} = 'Filtro para Classes';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Selecione uma classe a partir da lista para criar um novo item de configuração.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'Ação em Massa em ICs ITSM';
    $Self->{Translation}->{'Deployment state'} = 'Estado de implantação';
    $Self->{Translation}->{'Incident state'} = 'Estado de incidente';
    $Self->{Translation}->{'Link to another'} = 'Associar a outro';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = 'Número de Item de Configuração inválido!';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'O número de outro Item de Configuração para associar.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = 'Deseja realmente excluir este item de configuração?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'O nome deste item de configuração';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'Nome já em uso pelos ICs com os seguintes Número(s): %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = 'Histórico do Item de Configuração: %s';
    $Self->{Translation}->{'History Content'} = 'Conteúdo do Histórico';
    $Self->{Translation}->{'Createtime'} = 'Hora de criação';
    $Self->{Translation}->{'Zoom view'} = 'Visão de detalhe';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Configurações de Contexto';
    $Self->{Translation}->{'Config Items per page'} = 'Itens de configuração por página';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = 'Uma tabela genérica de Itens de Configuração GSTI';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Executar Pesquisa';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Procurar nas verões anteriores também?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Item de configuração';
    $Self->{Translation}->{'Configuration Item Information'} = 'Informação do Item de Configuração';
    $Self->{Translation}->{'Current Deployment State'} = 'Estado de Implantação Atual';
    $Self->{Translation}->{'Last changed by'} = 'Última modificação por';
    $Self->{Translation}->{'Show one version'} = 'Visualizar uma versão';
    $Self->{Translation}->{'Show all versions'} = 'Visualizar todas as versões';
    $Self->{Translation}->{'Version Incident State'} = 'Estado de Incidente da Versão';
    $Self->{Translation}->{'Version Deployment State'} = 'Estado de Implantação da Versão';
    $Self->{Translation}->{'Version Number'} = 'Número da versão';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Detalhes da Versão do Item de Configuração';
    $Self->{Translation}->{'Property'} = 'Propriedade';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = 'Nenhum acesso para a classe é dado!';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = 'Visão geral: ITSM ConfigItem';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = 'Nenhum ConfigItemID é dado!';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = 'Você precisa selecionar pelo menos um item de configuração!';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        'Você não tem acesso de escrita para este item de configuração: %s.';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = 'Item de configuração "%s" não encontrado na base de dados!';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = 'Não foi possível excluir o ID de item de configuração %s!';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = 'Nenhuma versão encontrada para o ConfigItemID %s!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = 'Nenhum ConfigItemID, DuplicateID ou ClassID é dado!';
    $Self->{Translation}->{'No access is given!'} = 'Nenhum acesso é dado!';
    $Self->{Translation}->{'No definition was defined for class %s!'} = 'Nenhuma definição para a classe %s!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = 'Não é possível exibir histórico. Nenhum ConfigItemID é dado!';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = 'Não é possível exibir histórico. Nenhum direito de acesso é dado!';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = 'Novo Item de Configuração (ID=%s)';
    $Self->{Translation}->{'New version (ID=%s)'} = 'Nova versão (ID = %s)';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = 'Estado de implementação atualizado (novo=%s , antigo=%s)';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = 'Estado de incidente atualizado (novo=%s, antigo=%s)';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = 'Item de Configuração (ID=%s) excluído';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = 'Vínculo para %s (tipo=%s) adicionado';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = 'Vínculo para %s (tipo=%s) deletado';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = 'Definição de Item de Configuração (ID=%s) atualizada';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = 'Nome atualizado (novo=%s, antigo=%s)';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = 'Atributo %s atualizado de "%s" para "%s"';
    $Self->{Translation}->{'Version %s deleted'} = 'Versão %s excluída';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = 'Nenhum ConfigItemID ou VersionID é dado!';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = 'Não foi possível exibir item de configuração. Nenhum direito de acesso é dado!';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = 'ConfigItemID %s não encontrado na base de dados!';
    $Self->{Translation}->{'VersionID %s not found in database!'} = 'VersionID %s não encontrado na base de dados!';
    $Self->{Translation}->{'ConfigItem'} = 'Item de configuração';
    $Self->{Translation}->{'printed by %s at %s'} = 'Impresso por %s em %s';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = 'ClassID inválido!';
    $Self->{Translation}->{'No ClassID is given!'} = 'Nenhum ClassID é dado!';
    $Self->{Translation}->{'No access rights for this class given!'} = 'Nenhum direito de acesso para esta classe é dado!';
    $Self->{Translation}->{'No Result!'} = 'Nenhum resultado!';
    $Self->{Translation}->{'Config Item Search Results'} = 'Resultados da pesquisa de Itens de configuração';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        'Não foi possível exibir item. Nenhum direito de acesso para ConfigItem é dado!';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = 'O estado de implantação deste item de configuração';
    $Self->{Translation}->{'The incident state of this config item'} = 'O estado de incidente deste item de configuração';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = 'Exibir itens de configuração';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Entre';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Número máximo de um elemento';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Os campos vazios indicam que os valores atuais são mantidos';
    $Self->{Translation}->{'Skipped'} = 'Desconsiderados';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Modelo';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Número de Série';
    $Self->{Translation}->{'CPU'} = 'Processador';
    $Self->{Translation}->{'Ram'} = 'Memória RAM';
    $Self->{Translation}->{'Hard Disk'} = 'Disco Rígido';
    $Self->{Translation}->{'Capacity'} = 'Capacidade';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Adaptador de Rede';
    $Self->{Translation}->{'IP over DHCP'} = 'DHCP';
    $Self->{Translation}->{'IP Address'} = 'Endereço IP';
    $Self->{Translation}->{'Graphic Adapter'} = 'Adaptador Gráfico';
    $Self->{Translation}->{'Other Equipment'} = 'Outro Equipamento';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Data de Expiração da Garantia';
    $Self->{Translation}->{'Install Date'} = 'Data Instalação';
    $Self->{Translation}->{'Phone 1'} = 'Telefone 1';
    $Self->{Translation}->{'Phone 2'} = 'Telefone 2';
    $Self->{Translation}->{'E-Mail'} = 'E-Mail';
    $Self->{Translation}->{'Network Address'} = 'Endereço de Rede';
    $Self->{Translation}->{'Subnet Mask'} = 'Máscara de Subrede';
    $Self->{Translation}->{'Gateway'} = 'Roteador Padrão';
    $Self->{Translation}->{'Licence Type'} = 'Tipo de Licença';
    $Self->{Translation}->{'Licence Key'} = 'Chave de Licença';
    $Self->{Translation}->{'Quantity'} = 'Quantidade';
    $Self->{Translation}->{'Expiration Date'} = 'Data de Expiração';
    $Self->{Translation}->{'Media'} = 'Mídia';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = 'Computador';
    $Self->{Translation}->{'Hardware'} = 'Hardware';
    $Self->{Translation}->{'Network'} = 'Rede';
    $Self->{Translation}->{'Software'} = 'Software';
    $Self->{Translation}->{'Expired'} = 'Expirado';
    $Self->{Translation}->{'Maintenance'} = 'Manutenção';
    $Self->{Translation}->{'Pilot'} = 'Piloto';
    $Self->{Translation}->{'Planned'} = 'Planejado';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Reparo';
    $Self->{Translation}->{'Retired'} = 'Retirado';
    $Self->{Translation}->{'Review'} = 'Revisão';
    $Self->{Translation}->{'Test/QA'} = 'Teste de Qualidade';
    $Self->{Translation}->{'Laptop'} = 'Notebook';
    $Self->{Translation}->{'Desktop'} = 'Desktop';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Servidor';
    $Self->{Translation}->{'Other'} = 'Outro';
    $Self->{Translation}->{'Monitor'} = 'Monitor';
    $Self->{Translation}->{'Printer'} = 'Impressora';
    $Self->{Translation}->{'Switch'} = 'Switch';
    $Self->{Translation}->{'Router'} = 'Roteador';
    $Self->{Translation}->{'WLAN Access Point'} = 'WLAN Access Point';
    $Self->{Translation}->{'Security Device'} = 'Dispositivo de Segurança';
    $Self->{Translation}->{'Backup Device'} = 'Dispositivo de Backup';
    $Self->{Translation}->{'Mouse'} = 'Mouse';
    $Self->{Translation}->{'Keyboard'} = 'Teclado';
    $Self->{Translation}->{'Camera'} = 'Câmera';
    $Self->{Translation}->{'Beamer'} = 'Projetor';
    $Self->{Translation}->{'Modem'} = 'Modem';
    $Self->{Translation}->{'PCMCIA Card'} = 'Cartão PCMCIA';
    $Self->{Translation}->{'USB Device'} = 'Dispositivo USB';
    $Self->{Translation}->{'Docking Station'} = 'Docking Station';
    $Self->{Translation}->{'Scanner'} = 'Scanner';
    $Self->{Translation}->{'Building'} = 'Versão';
    $Self->{Translation}->{'Office'} = 'Escritório';
    $Self->{Translation}->{'Floor'} = 'Andar';
    $Self->{Translation}->{'Room'} = 'Sala';
    $Self->{Translation}->{'Rack'} = 'Rack';
    $Self->{Translation}->{'Workplace'} = 'Local de Trabalho';
    $Self->{Translation}->{'Outlet'} = 'Tomada';
    $Self->{Translation}->{'IT Facility'} = 'Departamento de TI';
    $Self->{Translation}->{'LAN'} = 'REDE';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telecomunicação';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Aplicação Cliente';
    $Self->{Translation}->{'Middleware'} = 'Middleware';
    $Self->{Translation}->{'Server Application'} = 'Aplicação Servidor';
    $Self->{Translation}->{'Client OS'} = 'SO Cliente';
    $Self->{Translation}->{'Server OS'} = 'SO Servidor';
    $Self->{Translation}->{'Admin Tool'} = 'Ferramenta de Administração';
    $Self->{Translation}->{'User Tool'} = 'Ferramenta do Usuário';
    $Self->{Translation}->{'Embedded'} = 'Integrado';
    $Self->{Translation}->{'Single Licence'} = 'Licença Única';
    $Self->{Translation}->{'Per User'} = 'Por Usuário';
    $Self->{Translation}->{'Per Processor'} = 'Por Processador';
    $Self->{Translation}->{'Per Server'} = 'Por Servidor';
    $Self->{Translation}->{'Per Node'} = 'Por Nó';
    $Self->{Translation}->{'Volume Licence'} = 'Licença por Volume';
    $Self->{Translation}->{'Enterprise Licence'} = 'Licença Corporativa';
    $Self->{Translation}->{'Developer Licence'} = 'Licença de Desenvolvedor';
    $Self->{Translation}->{'Demo'} = 'Demonstração';
    $Self->{Translation}->{'Time Restricted'} = 'Tempo Restrito';
    $Self->{Translation}->{'Freeware'} = 'Livre';
    $Self->{Translation}->{'Open Source'} = 'Código aberto';
    $Self->{Translation}->{'Unlimited'} = 'Ilimitada';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = 'IC atribuido à empresa cliente';
    $Self->{Translation}->{'CIs assigned to customer user'} = 'IC atribuido ao usuário cliente';
    $Self->{Translation}->{'CMDB Settings'} = 'Configurações do CMDB';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        'Verifique se o nome é único apenas dentro da mesma classe de IC (\'classe) ou globalmente (\'global\'), o que significa que cada IC existente é levado em conta ao procurar por duplicações.';
    $Self->{Translation}->{'Config Items'} = 'Itens de Configuração';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = 'Adicionar item de configuração.';
    $Self->{Translation}->{'Config item edit.'} = 'Editar item de configuração.';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        'Módulo de evento de item de configuração que permite registrar o histórico da interface do atendente.';
    $Self->{Translation}->{'Config item history.'} = 'Histórivo do item de configuração.';
    $Self->{Translation}->{'Config item print.'} = 'Impressão do item de configuração.';
    $Self->{Translation}->{'Config item zoom.'} = 'Zoom de item de configuração.';
    $Self->{Translation}->{'ConfigItemNumber'} = 'ConfigItemNumber';
    $Self->{Translation}->{'Configuration Item Limit'} = 'Limite de Item de Configuração';
    $Self->{Translation}->{'Configuration Item limit per page.'} = 'Limite do item de configuração por página.';
    $Self->{Translation}->{'Configuration Management Database.'} = 'Configuração do Gerenciamento do Banco de Dados.';
    $Self->{Translation}->{'Configuration item bulk module.'} = 'Módulo de ação em massa em Itens de Configuração.';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        'Roteador de pesquisa de item de configuração da interface do atendente.';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Criar e gerenciar as definições de Itens de Configuração.';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        'Defina Ações onde um botão de configurações está disponível no widget de objetos vinculados (LinkObject::ViewMode="complex"). Observe que essas ações devem ter registrado os seguintes arquivos JS e CSS: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js e Core.Agent .LinkObject.js.';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        'Define as permissões requeridas para criar itens de configuração ITSM usando a Interface Genérica.';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        'Define a permissão necessário para remover Itens de Configuração usando a Interface Genérica.';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        'Define as permissões requeridas para obter itens de configuração ITSM usando a Interface Genérica.';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        'Define as permissões requeridas para localizar itens de configuração ITSM usando a Interface Genérica.';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        'Define as permissões requeridas para atualizar itens de configuração ITSM usando a Interface Genérica.';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        'Define um módulo de visão geral para mostrar a visão pequena da lista de itens de configuração.';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        'Definie expressões regulares individualmente para cada classe de IC para verificar o nome do IC e para exibir as mensagens de erro correspondentes.';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        'Define o subobjeto padrão da classe \'ITSMConfigItem\'.';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        'Define o número de linhas do editor de definição do IC na interface de administração.';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        'Define a ordem dos estados de incidente de alto (ex: crítico) para baixo (ex: funcional)';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        'Define os estados de implantação onde chamados associados podem afetar o estados de um IC.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        'Define o limite de pesquisa para a tela AgentITSMConfigItem.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        'Define o limite de pesquisa para a tela AgentITSMConfigItemSearch.';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        'Define as colunas exibidas na visão geral de itens de configuração. Esta opção não tem efeito na posição da coluna. Nota: A coluna classe está sempre disponível se o filtro "Todos" estiver selecionado.';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        'Define as colunas mostradas na busca de item de configuração. Esta opção não tem efeito sobre a posição da coluna.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Define as colunas de ICs exibidas na visão geral de itens de configuração, dependendo da classe do IC. Cada entrada deve ser prefixada com o nome da classe e duplo dois pontos (ex. Computador::). Há alguns atributos de IC que são comuns a todos os ICs (tal como a classe Computador: Computador::Nome, Computador::EstadoImplAtual, Computador::HoraCriacao). Para exibir atributos de IC individualmente como definido na definição do IC, o seguinte esquema deve ser usado (classe Computador, por exemplo): Computador::DiscoRigido::1, Computador::DiscoRigido::1::Capacidade::1, Computador::DiscoRigido::2, Computador::DiscoRigido::2::Capacidade::1. Se não há entrada para a classe IC, então as colunas padrão são exibidas como definido na configuração ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Define as colunas de ICs exibidas na tela de pesquisa de itens de configuração, dependendo da classe do IC. Cada entrada deve ser prefixada com o nome da classe e duplo dois pontos (ex. Computador::). Há alguns atributos de IC que são comuns a todos os ICs (tal como a classe Computador: Computador::Nome, Computador::EstadoImplAtual, Computador::HoraCriacao). Para exibir atributos de IC individualmente como definido na definição do IC, o seguinte esquema deve ser usado (classe Computador, por exemplo): Computador::DiscoRigido::1, Computador::DiscoRigido::1::Capacidade::1, Computador::DiscoRigido::2, Computador::DiscoRigido::2::Capacidade::1. Se não há entrada para a classe IC, então as colunas padrão são exibidas como definido na configuração ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        'Define as colunas de ICs exibidas na tela de tabela complexa de links, dependendo da classe do IC. Cada entrada deve ser prefixada com o nome da classe e duplo dois pontos (ex. Computador::). Há alguns atributos de IC que são comuns a todos os ICs (tal como a classe Computador: Computador::Nome, Computador::EstadoImplAtual, Computador::HoraCriacao). Para exibir atributos de IC individualmente como definido na definição do IC, o seguinte esquema deve ser usado (classe Computador, por exemplo): Computador::DiscoRigido::1, Computador::DiscoRigido::1::Capacidade::1, Computador::DiscoRigido::2, Computador::DiscoRigido::2::Capacidade::1. Se não há entrada para a classe IC, então as colunas padrão são exibidas.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        'Define qual tipo de associação (nomeado a partir da perspectiva do chamado) pode afetar o estado de um IC associado.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        'Define qual tipo de chamado pode afetar o estado de um IC associado.';
    $Self->{Translation}->{'Delete Configuration Item'} = 'Excluir item de configuração';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = 'Cor do Estado de Implantação';
    $Self->{Translation}->{'Duplicate'} = 'Duplicar';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        'Habilita a função de ação em massa em itens de configuração na interface de atendente para trabalhar em mais de um IC por vez.';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        'Habilita a função de ação em massa em itens de configuração apenas para grupos listados.';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        'Habilita/desabilita a funcionalidade de verificação de nomes únicos de ICs. Antes de habilitar esta opção, você deve verificar se há ICs com nomes duplicados em seu sistema. Você pode fazer isso por meio do script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        'Módulo de Evento para configurar o estado de item de configuração quando um chamado for associado a um IC.';
    $Self->{Translation}->{'ITSM config item overview.'} = 'Visão Geral  do Item de Configuração.';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = 'Módulo para verificar o grupo responsável por uma classe.';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        'Módulo para verificar o grupo responsável por um item de configuração.';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = 'Módulo para gerar estatísticas do item de configuração ITSM.';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        'Modulo de registro de objeto para o módulo de importação/exportação.';
    $Self->{Translation}->{'Overview.'} = 'Visão Geral.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        'Parâmetros para a cor dos estados de implantação na tela de preferências da interface de atendente.';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        'Parâmetros de estado de implantação nas preferências de exibição da interface de atendente.';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parâmetros do grupos de permissão de exemplo dos atributos do catálogo geral.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        'Parâmetros das páginas (nas quais os itens de configuração são mostrados).';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        'Permissões necessárias para usar a tela de item de configuração ITSM na interface de atendente.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        'Permissões necessárias para usar a tela de pesquisa de item de configuração ITSM na interface de atendente.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        'Permissões necessárias para usar a tela de detalhes do item de configuração ITSM na interface de atendente.';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        'Permissões necessárias para usar a tela de adição de item de configuração ITSM na interface de atendente.';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        'Permissões necessárias para usar a tela de edição de item de configuração ITSM na interface de atendente.';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        'Permissões necessárias para usar a tela de histórico de item de configuração ITSM na interface de atendente.';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        'Permissões necessárias para usar a tela de impressão de item de configuração ITSM na interface do atendente.';
    $Self->{Translation}->{'Required privileges to delete config items.'} = 'Privilégios necessários para apagar itens de configuração.';
    $Self->{Translation}->{'Search config items.'} = 'Procurar itens de configuração.';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        'Seleciona o módulo de geração de número de item de configuração. "Autoincremento" incrementa o número do item de configuração, o SystemID, o ConfigItemClassID e o Counter são usados. O formato é "SystemID.ConfigItemClassID.Counter", por exemplo, 1205000004, 1205000005.';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        'Configurar o estado de um IC automaticamente quando um Chamado for associado.';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        'Configura o estado de implantação na tela de ação em massa de ICs na interface de atendente.';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        'Configura o estado de incidente na tela de ação em massa de ICs na interface de atendente.';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        'Mostra um link no menu que permite associar um item de configuração com um outro objeto na visão de detalhes do item de configuração da interface de atendente.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        'Mostra um link no menu para acessar o histórico de um IC na visão geral de itens de configuração na interface de atendente.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        'Mostra um link no menu para acessar o histórico de item de configuração na visão de detalhes do mesmo na interface de atendente.';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        'Mostra um link no menu para apagar o item de configuração na Visão Detalhada na interface de atendente.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        'Mostra um link no menu para duplicar um IC na visão geral de itens de configuração na interface de atendente.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        'Mostra um link no menu para duplicar um item de configuração na visão de detalhes do mesmo na interface de atendente.';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        'Mostra um link no menu para editar um item de configuração na visão de detalhes do mesmo na interface de atendente.';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        'Mostra um link no menu para retornar à tela de visão de detalhes de item de configuração da interface de atendente.';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        'Mostra um link no menu para imprimir um item de configuração na visão de detalhes do mesmo na interface de atendente.';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        'Mostra um link no menu para detalhar um IC na visão geral de itens de configuração na interface de atendente.';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        'Mostra o histórico do item configuração (ordem reversa) na interface de atendente.';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        'O identificador para um item de configuração, ex. ItemConfig#, MeuItemConfig#. O padrão é ItemConfig#.';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = 'class';
    $Self->{Translation}->{'global'} = 'global';
    $Self->{Translation}->{'postproductive'} = 'pós-produtivo';
    $Self->{Translation}->{'preproductive'} = 'improdutivo';
    $Self->{Translation}->{'productive'} = 'produtivo';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::pt_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Gestão de Itens de Configuração';
    $Self->{Translation}->{'Change class definition'} = 'Mudar definição de classe';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = 'Definição';
    $Self->{Translation}->{'Change'} = 'Alterar';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Estado de Incidente';
    $Self->{Translation}->{'Deployment State'} = 'Estado de Implementação';
    $Self->{Translation}->{'Class'} = 'Classe';
    $Self->{Translation}->{'Deployment State Type'} = 'Tipo de Estado de Implementação';
    $Self->{Translation}->{'Current Incident State'} = 'Estado do Incidente Atual';
    $Self->{Translation}->{'Current Incident State Type'} = 'Estado Atual de Incidente por tipo';
    $Self->{Translation}->{'Last changed'} = 'Última modificação';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Item de Configuração';
    $Self->{Translation}->{'Filter for Classes'} = 'Filtro para Classes';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Selecione uma classe a partir da lista para criar um novo item de configuração.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = 'Estado de Implementação';
    $Self->{Translation}->{'Incident state'} = 'Estado de incidente';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'O nome deste item de configuração';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = 'Histórico do conteúdo';
    $Self->{Translation}->{'Createtime'} = 'Hora de criação';
    $Self->{Translation}->{'Zoom view'} = 'Vista de detalhe';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Configurações de Contexto';
    $Self->{Translation}->{'Config Items per page'} = 'Itens de configuração por página';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Executar Pesquisa';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Procurar nas versões anteriores também?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Item de configuração';
    $Self->{Translation}->{'Configuration Item Information'} = 'Informação do Item de Configuração';
    $Self->{Translation}->{'Current Deployment State'} = 'Estado de Implementação Atual';
    $Self->{Translation}->{'Last changed by'} = 'Última alteração por';
    $Self->{Translation}->{'Show one version'} = 'Visualizar uma versão';
    $Self->{Translation}->{'Show all versions'} = 'Visualizar todas as versões';
    $Self->{Translation}->{'Version Incident State'} = 'Estado da Versão de Incidentes';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = 'Número da versão';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Detalhes da Versão do Item de Configuração';
    $Self->{Translation}->{'Property'} = 'Propriedade';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = 'Item de configuração';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = 'Sem resultado!';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = 'operativo';
    $Self->{Translation}->{'warning'} = 'aviso';
    $Self->{Translation}->{'incident'} = 'incidente';
    $Self->{Translation}->{'The deployment state of this config item'} = 'O estado de implementação do item de configuração';
    $Self->{Translation}->{'The incident state of this config item'} = 'O estado de incidente deste item de configuração';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Entre';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Número máximo de um elemento';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Os campos vazios indicam que os valores atuais são mantidos';
    $Self->{Translation}->{'Skipped'} = 'Desconsiderados';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Modelo';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Número de Série';
    $Self->{Translation}->{'CPU'} = 'Processador';
    $Self->{Translation}->{'Ram'} = 'Memória RAM';
    $Self->{Translation}->{'Hard Disk'} = 'Disco Rígido';
    $Self->{Translation}->{'Capacity'} = 'Capacidade';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Adaptador de Rede';
    $Self->{Translation}->{'IP over DHCP'} = 'DHCP';
    $Self->{Translation}->{'IP Address'} = 'Endereço IP';
    $Self->{Translation}->{'Graphic Adapter'} = 'Adaptador Gráfico';
    $Self->{Translation}->{'Other Equipment'} = 'Outro Equipamento';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Data de Expiração da Garantia';
    $Self->{Translation}->{'Install Date'} = 'Data Instalação';
    $Self->{Translation}->{'Phone 1'} = 'Telefone 1';
    $Self->{Translation}->{'Phone 2'} = 'Telefone 2';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'Endereço de Rede';
    $Self->{Translation}->{'Subnet Mask'} = 'Máscara de rede';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = 'Tipo de Licença';
    $Self->{Translation}->{'Licence Key'} = 'Chave de Licença';
    $Self->{Translation}->{'Quantity'} = 'QuantAntiguidade';
    $Self->{Translation}->{'Expiration Date'} = 'Data de Expiração';
    $Self->{Translation}->{'Media'} = 'Media';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'Expirado';
    $Self->{Translation}->{'Maintenance'} = 'Manutenção';
    $Self->{Translation}->{'Pilot'} = 'Piloto';
    $Self->{Translation}->{'Planned'} = 'Planeado';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Reparação';
    $Self->{Translation}->{'Retired'} = 'Retirado';
    $Self->{Translation}->{'Review'} = 'Revisão';
    $Self->{Translation}->{'Test/QA'} = 'Teste de Qualidade';
    $Self->{Translation}->{'Laptop'} = 'portátil';
    $Self->{Translation}->{'Desktop'} = 'Área ';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = 'Servidor';
    $Self->{Translation}->{'Other'} = 'Outro';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = 'Impressora';
    $Self->{Translation}->{'Switch'} = 'Switch';
    $Self->{Translation}->{'Router'} = 'Router';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = 'Dispositivo de Segurança';
    $Self->{Translation}->{'Backup Device'} = 'Dispositivo de Backup';
    $Self->{Translation}->{'Mouse'} = 'rato';
    $Self->{Translation}->{'Keyboard'} = 'Teclado';
    $Self->{Translation}->{'Camera'} = 'Câmara';
    $Self->{Translation}->{'Beamer'} = 'Projetor';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = 'Placa PCMCIA';
    $Self->{Translation}->{'USB Device'} = 'Dispositivo USB';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = 'Scanner';
    $Self->{Translation}->{'Building'} = 'Prédio';
    $Self->{Translation}->{'Office'} = 'Gabinete';
    $Self->{Translation}->{'Floor'} = 'Piso';
    $Self->{Translation}->{'Room'} = 'Sala';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = 'Local de Trabalho';
    $Self->{Translation}->{'Outlet'} = 'Tomada';
    $Self->{Translation}->{'IT Facility'} = 'Departamento de TI';
    $Self->{Translation}->{'LAN'} = 'REDE';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = 'Telecomunicações';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = 'Aplicação Cliente';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = 'Servidor Aplicacional ';
    $Self->{Translation}->{'Client OS'} = 'SO Cliente';
    $Self->{Translation}->{'Server OS'} = 'SO Servidor';
    $Self->{Translation}->{'Admin Tool'} = 'Ferramenta de Administração';
    $Self->{Translation}->{'User Tool'} = 'Ferramenta de Utilizador';
    $Self->{Translation}->{'Embedded'} = 'Integrado';
    $Self->{Translation}->{'Single Licence'} = 'Licença Única';
    $Self->{Translation}->{'Per User'} = 'Por Utilizador';
    $Self->{Translation}->{'Per Processor'} = 'Por Processador';
    $Self->{Translation}->{'Per Server'} = 'Por Servidor';
    $Self->{Translation}->{'Per Node'} = 'Por Posto';
    $Self->{Translation}->{'Volume Licence'} = 'Licença por Volume';
    $Self->{Translation}->{'Enterprise Licence'} = 'Licença Corporativa';
    $Self->{Translation}->{'Developer Licence'} = 'Licença de desenvolvimento';
    $Self->{Translation}->{'Demo'} = 'Demonstração';
    $Self->{Translation}->{'Time Restricted'} = 'Tempo Restrito';
    $Self->{Translation}->{'Freeware'} = 'Livre';
    $Self->{Translation}->{'Open Source'} = 'Código aberto';
    $Self->{Translation}->{'Unlimited'} = 'Ilimitada';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = 'Itens de Configuração';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        'Módulo de evento de item de configuração que permite registar o histórico da interface do agente.';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = 'Limite de Items de Configuração';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        'Router de pesquisa de item de configuração da interface do agente.';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Criar e gerir as definições de Itens de Configuração.';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        'Definir Acções onde um botão de configurações está disponível na widget the objectos ligados (LinkObject::ViewMode = "complex"). Estas Acções devem estar registadas nos seguintes ficheiros JS e CSS: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        'Define um módulo de vista geral para mostrar a vista pequena da lista de itens de configuração.';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        'Define o subobjeto por omissão da classe \'ITSMConfigItem\'.';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        'Define as colunas mostradas na busca de item de configuração. Esta opção não tem efeito sobre a posição da coluna.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = 'Duplicar';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = 'Módulo para verificar o grupo responsável por uma classe.';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        'Módulo para verificar o grupo responsável por um item de configuração.';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = 'Módulo para gerar estatísticas do item de configuração ITSM.';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        'Modulo de registo de objeto para o módulo de importação/exportação.';
    $Self->{Translation}->{'Overview.'} = 'Visão Geral.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        'Parâmetros de estado de implementação nas preferências de visualização da interface de agente.';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parâmetros do grupos de permissão de exemplo dos atributos do catálogo geral.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        'Parâmetros das páginas (nas quais os itens de configuração são visualizados).';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        'Permissões necessárias para usar o formulário de item de configuração ITSM de agente.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        'Permissões necessárias para usar o formulário de pesquisa de item de configuração ITSM de agente.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        'Permissões necessárias para usar o formulário de detalhes do item de configuração ITSM de agente.';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        'Permissões necessárias para usar o formulário de adição de item de configuração ITSM de agente.';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        'Permissões necessárias para usar o formulário de edição de item de configuração ITSM de agente.';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        'Permissões necessárias para usar o formulário de histórico de item de configuração ITSM de agente.';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        'Permissões necessárias para usar o formulário de impressão de item de configuração ITSM na interface do agente.';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        'Seleciona o módulo de geração de número de item de configuração. "Autoincremento" incrementa o número do item de configuração, o SystemID, o ConfigItemClassID e o Counter são utilizados. O formato é "SystemID.ConfigItemClassID.Counter", por exemplo, 1205000004, 1205000005.';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        'Mostra um link no menu que permite associar um item de configuração a outro objeto na vista de detalhes do item de configuração de agente.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        'Mostra um link no menu para aceder o histórico de item de configuração na vista de agente.';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        'Mostra um link no menu para duplicar um item de configuração na vista de agente.';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        'Mostra um link no menu para editar um item de configuração na vista de agente.';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        'Mostra um link no menu para imprimir um item de configuração na vista de detalhes de agente.';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        'Mostra o histórico do item configuração (ordem inversa) no interface de agente.';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::ro_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = 'Vedere de detaliu';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = 'avertizare';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = '';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = 'Sărit';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = 'Altele';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'Birou';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::ru_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Управление Конфигурационными единицами';
    $Self->{Translation}->{'Change class definition'} = 'Изменить описание класса';
    $Self->{Translation}->{'Config Item Class'} = 'Класс конфигурационной единицы';
    $Self->{Translation}->{'Definition'} = 'Описание';
    $Self->{Translation}->{'Change'} = 'Изменение';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Состояние инцидента';
    $Self->{Translation}->{'Deployment State'} = 'Состояние использования';
    $Self->{Translation}->{'Class'} = 'Класс';
    $Self->{Translation}->{'Deployment State Type'} = 'Тип Состояния использования';
    $Self->{Translation}->{'Current Incident State'} = 'Текущее состояние инцидента';
    $Self->{Translation}->{'Current Incident State Type'} = 'Тип Текущего состояния инцидента';
    $Self->{Translation}->{'Last changed'} = 'Время последнего изменения';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Основные средства';
    $Self->{Translation}->{'Filter for Classes'} = 'Фильтр для классов';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Выберите класс из списка для создания новой КЕ';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'ITSM ConfigItem массовое действие';
    $Self->{Translation}->{'Deployment state'} = 'Состояние использования';
    $Self->{Translation}->{'Incident state'} = 'Состояние инцидента';
    $Self->{Translation}->{'Link to another'} = 'Связать с другим';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = 'Неверный номер конфигурационной единицы!';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'Количество других КЕ для связывания.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = 'Вы действительно хотите удалить эту конфигурационную единицу?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'Имя этого учетного элемента';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'Имя уже используется другой КЕ со следующим номером (ами): %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = 'История конфигурационной единицы: %s';
    $Self->{Translation}->{'History Content'} = 'Содержимое истории';
    $Self->{Translation}->{'Createtime'} = 'Время создания';
    $Self->{Translation}->{'Zoom view'} = 'Подробный показ';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Параметры контекста';
    $Self->{Translation}->{'Config Items per page'} = 'Конфигурационных единиц на страницу';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = 'Общая таблица элементов конфигурации ITSM';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Выполнить поиск';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Искать и в предыдущих версиях?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Конфигурационная единица';
    $Self->{Translation}->{'Configuration Item Information'} = ' Информация о Конф. ед.';
    $Self->{Translation}->{'Current Deployment State'} = 'Текущее состояние использования';
    $Self->{Translation}->{'Last changed by'} = 'Автор последнего изменения';
    $Self->{Translation}->{'Show one version'} = 'Показать одну версию';
    $Self->{Translation}->{'Show all versions'} = 'Показать все версии';
    $Self->{Translation}->{'Version Incident State'} = 'Состояние инцидента версии';
    $Self->{Translation}->{'Version Deployment State'} = 'Состояние использования версии объекта';
    $Self->{Translation}->{'Version Number'} = 'Номер версии';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Полная информация о КЕ выбранной версии';
    $Self->{Translation}->{'Property'} = 'Свойство';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = 'Нет доступа к классу!';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = 'Обзор: ITSM ConfigItem';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = 'Нет ConfigItemID!';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = 'Вам нужна хотя бы одна выбранная конфигурационная единица!';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        'У вас нет доступа на запись этой конфигурационной единицы: %s.';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = 'Конфигурационная единица "%s" не найдена в базе!';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = 'Не удалось удалить конфигурационную единицу с ID %s!';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = 'Не найдена версия конфигурационной единицы %s!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = 'Не присвоен ConfigItemID, DuplicateID или ClassID!';
    $Self->{Translation}->{'No access is given!'} = 'Нет доступа!';
    $Self->{Translation}->{'No definition was defined for class %s!'} = 'Класс %s не был определен!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = 'Невозможно показать историю, нет ConfigItemID!';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = 'Невозможно показать историю, недостаточно прав!';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = 'Новая КЕ (ID=%s)';
    $Self->{Translation}->{'New version (ID=%s)'} = 'Новая версия (ID=%s)';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = 'Состояние использования обновлено (новое=%s, старое=%s)';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = 'Состояние инцидента обновлено (новое=%s, старое=%s)';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = 'КЕ (ID=%s) удалена';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = 'Связь с %s (тип=%s) добавлена';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = 'Связь с %s (тип=%s) удалена';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = 'Описание КЕ обновлено (ID=%s)';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = 'Имя обновлено (новое=%s, старое=%s)';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = 'Атрибут %s обновлен с "%s" до "%s"';
    $Self->{Translation}->{'Version %s deleted'} = 'Версия %s удалена';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = 'Не присвоен ConfigItemID или VersionID!';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = 'Невозможно отобразить конфигурационную единицу, нет доступа!';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = 'Конфигурационная единица %s не найдена в базе!';
    $Self->{Translation}->{'VersionID %s not found in database!'} = 'Версия %s не найдена в базе!';
    $Self->{Translation}->{'ConfigItem'} = 'Конфигурационная единица';
    $Self->{Translation}->{'printed by %s at %s'} = 'напечатано %s в %s';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = 'Неверный ClassID!';
    $Self->{Translation}->{'No ClassID is given!'} = 'ClassID не присвоен!';
    $Self->{Translation}->{'No access rights for this class given!'} = 'Нет доступа для этого класса!';
    $Self->{Translation}->{'No Result!'} = 'Нет результата/Решения!';
    $Self->{Translation}->{'Config Item Search Results'} = 'Результаты поиска конфигурационных единиц';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        'Невозможно отобразить элемент, недостаточно прав для КЕ!';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = 'Состояние использования этой КЕ';
    $Self->{Translation}->{'The incident state of this config item'} = 'Состояние инцидента для этой КЕ';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Между';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Максимальное количество одного элемента';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Пустые поля показывают, что текущие значения сохранятся';
    $Self->{Translation}->{'Skipped'} = 'Пропущено';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Модель';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Серийный номер';
    $Self->{Translation}->{'CPU'} = 'ЦПУ';
    $Self->{Translation}->{'Ram'} = 'ОЗУ';
    $Self->{Translation}->{'Hard Disk'} = 'Жесткий диск';
    $Self->{Translation}->{'Capacity'} = 'Объем';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Сетевой адаптер';
    $Self->{Translation}->{'IP over DHCP'} = 'IP по DHCP';
    $Self->{Translation}->{'IP Address'} = 'IP адрес';
    $Self->{Translation}->{'Graphic Adapter'} = 'Видео адаптер';
    $Self->{Translation}->{'Other Equipment'} = 'Другое оборудование';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Срок истечения гарантии';
    $Self->{Translation}->{'Install Date'} = 'Дата установки';
    $Self->{Translation}->{'Phone 1'} = 'Телефон 1';
    $Self->{Translation}->{'Phone 2'} = 'Телефон 2';
    $Self->{Translation}->{'E-Mail'} = 'E-Mail';
    $Self->{Translation}->{'Network Address'} = 'Сетевой адрес';
    $Self->{Translation}->{'Subnet Mask'} = 'Маска подсети';
    $Self->{Translation}->{'Gateway'} = 'Шлюз';
    $Self->{Translation}->{'Licence Type'} = 'Тип лицензии';
    $Self->{Translation}->{'Licence Key'} = 'Лицензионный ключ';
    $Self->{Translation}->{'Quantity'} = 'Количество';
    $Self->{Translation}->{'Expiration Date'} = 'Срок лицензии';
    $Self->{Translation}->{'Media'} = 'Мультимедийные средства';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = 'Компьютер';
    $Self->{Translation}->{'Hardware'} = 'Аппаратное обеспечение';
    $Self->{Translation}->{'Network'} = 'Сеть';
    $Self->{Translation}->{'Software'} = 'Программное обеспечение';
    $Self->{Translation}->{'Expired'} = 'Устарело';
    $Self->{Translation}->{'Maintenance'} = 'Техническое обслуживание';
    $Self->{Translation}->{'Pilot'} = 'Эксперимент';
    $Self->{Translation}->{'Planned'} = 'Запланированно';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Ремонт';
    $Self->{Translation}->{'Retired'} = 'Списано';
    $Self->{Translation}->{'Review'} = 'Экспертиза';
    $Self->{Translation}->{'Test/QA'} = 'Тестирование';
    $Self->{Translation}->{'Laptop'} = 'Ноутбук';
    $Self->{Translation}->{'Desktop'} = 'Настольный ПК';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Сервер';
    $Self->{Translation}->{'Other'} = 'Другое';
    $Self->{Translation}->{'Monitor'} = 'Монитор';
    $Self->{Translation}->{'Printer'} = 'Принтер';
    $Self->{Translation}->{'Switch'} = 'Коммутатор';
    $Self->{Translation}->{'Router'} = 'Маршрутизатор';
    $Self->{Translation}->{'WLAN Access Point'} = 'WLAN точка доступа';
    $Self->{Translation}->{'Security Device'} = 'Смарт-карта';
    $Self->{Translation}->{'Backup Device'} = 'Устройство бэкапа';
    $Self->{Translation}->{'Mouse'} = 'Мышка';
    $Self->{Translation}->{'Keyboard'} = 'Клавиатура';
    $Self->{Translation}->{'Camera'} = 'Камера';
    $Self->{Translation}->{'Beamer'} = 'Проектор';
    $Self->{Translation}->{'Modem'} = 'Модем';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA карта';
    $Self->{Translation}->{'USB Device'} = 'USB устройство';
    $Self->{Translation}->{'Docking Station'} = 'Док-станция';
    $Self->{Translation}->{'Scanner'} = 'Сканер';
    $Self->{Translation}->{'Building'} = 'Здание';
    $Self->{Translation}->{'Office'} = 'Офис';
    $Self->{Translation}->{'Floor'} = 'Этаж';
    $Self->{Translation}->{'Room'} = 'Кабинет';
    $Self->{Translation}->{'Rack'} = 'Стойка';
    $Self->{Translation}->{'Workplace'} = 'Рабочее место';
    $Self->{Translation}->{'Outlet'} = 'Розетка';
    $Self->{Translation}->{'IT Facility'} = 'ИТ объект';
    $Self->{Translation}->{'LAN'} = 'Сетевая карта';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Клиентское приложение';
    $Self->{Translation}->{'Middleware'} = 'Промежуточное ПО';
    $Self->{Translation}->{'Server Application'} = 'Серверное приложение';
    $Self->{Translation}->{'Client OS'} = 'Клиентская ОС';
    $Self->{Translation}->{'Server OS'} = 'Серверная ОС';
    $Self->{Translation}->{'Admin Tool'} = 'Системные утилиты';
    $Self->{Translation}->{'User Tool'} = 'Пользовательские утилиты';
    $Self->{Translation}->{'Embedded'} = 'Встроенный';
    $Self->{Translation}->{'Single Licence'} = 'Однократная лицензия';
    $Self->{Translation}->{'Per User'} = 'На пользователя';
    $Self->{Translation}->{'Per Processor'} = 'На ЦПУ';
    $Self->{Translation}->{'Per Server'} = 'На сервер';
    $Self->{Translation}->{'Per Node'} = 'На ноду(узел)';
    $Self->{Translation}->{'Volume Licence'} = 'Корпоративная лицензия';
    $Self->{Translation}->{'Enterprise Licence'} = 'Корпоративная лицензия';
    $Self->{Translation}->{'Developer Licence'} = 'Лицензия для разработчика';
    $Self->{Translation}->{'Demo'} = 'Демо-версия';
    $Self->{Translation}->{'Time Restricted'} = 'Ограниченная по времени';
    $Self->{Translation}->{'Freeware'} = 'Бесплатное ПО';
    $Self->{Translation}->{'Open Source'} = 'ПО с открытым кодом';
    $Self->{Translation}->{'Unlimited'} = 'Без ограничений';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        'Проверка уникальности Имени внутри текущего класса КЕ(\'class\') или глобально (\'global\'),что означает, что любая существующая КЕ учитывается при поиске дубликатов ';
    $Self->{Translation}->{'Config Items'} = 'Конфигурационные единицы';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = 'Создать конфигурационную единицу';
    $Self->{Translation}->{'Config item edit.'} = 'Редактировать конфигурационную единицу';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        'Config item (КЕ) event module ведущий запись в историю в агентском интерфейсе.';
    $Self->{Translation}->{'Config item history.'} = 'История конфигурационной единицы';
    $Self->{Translation}->{'Config item print.'} = 'Печать конфигурационной единицы';
    $Self->{Translation}->{'Config item zoom.'} = 'Просмотр конфигурационной единицы';
    $Self->{Translation}->{'ConfigItemNumber'} = 'Номер конфигурационной единицы';
    $Self->{Translation}->{'Configuration Item Limit'} = ' Лимит КЕ';
    $Self->{Translation}->{'Configuration Item limit per page.'} = 'Лимит КЕ на страницу';
    $Self->{Translation}->{'Configuration Management Database.'} = 'База управления конфигурацией';
    $Self->{Translation}->{'Configuration item bulk module.'} = 'Объемный модуль КЕ';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        'Модуль поиска КЕ в агентском интерфейсе';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Создание и управление описаниями конфигурационных единиц.';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        'Задает Действия/Actions когда кнопка настройки доступна в связанном виджете (LinkObject::ViewMode = "complex"). Обратите внимание, что эти Действия/Actions должны иметь зарегистрированные JS или CSS файлы: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js и Core.Agent.LinkObject.js.';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        'Задать требуемые права для создания ITSM КЕ с использованием Generic Interface.';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        'Задать требуемые права для удаления ITSM КЕ с использованием Generic Interface.';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        'Задает требуемые права для получения ITSM КЕ с использованием Generic Interface.';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        'Задает требуемые права для поиска ITSM КЕ с использованием Generic Interface';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        'Задает требуемые права для одновления ITSM КЕ с использованием Generic Interface';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        'Задает модуль просмотра КЕ в кратком виде.';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        'Задает регулярные выражения для каждого отдельного класса для проверки имен КЕ и выдачи сообщений об ошибках.';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        'Задает умалчивамый подобъект для класса \'ITSMConfigItem\'.';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        'Задает количество строк для редактора описаний КЕ (CI) в интерфейсе админа.';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        'Задает порядок состояний инцидентов от высоких (например, критических) до низких (например, функциональных).';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        'Задает соответствующие состояния развертывания, когда связанные заявки могут влиять на статус КЕ.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        'Задает лимит поиска объектов для экрана AgentITSMConfigItem';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        'Задает лимит поиска объектов для экрана AgentITSMConfigItemSearch';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        'Задает список показываемых столбцов в обзоре КЕ. Этот параметр не влияет на их порядок. Помните: Столбец Класс доступен только при условии выбора в фильтре "Все"(All)';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        'Задает список показываемых столбцов в результатах поиска КЕ. Этот параметр не влияет на порядок их отображения.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Задает список отображаемых столбцов при просмотре списков КЕ в зависимости от класса. Каждое имя атрибута должно иметь в качестве префикса - имя класса и два двоеточия (i.e. Computer::). Есть ряд атрибутов, общих для всех классов (например, для класса Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). Для отображения индивидуальных атрибутов, специфичных для конкретного класса из описания КЕ (CI-Definition), используется следующий синтаксис (например, для класса Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Если нет атрибутов для конкретного класса, отображаются столбцы заданные параметром ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Задает список отображаемых столбцов в результатах поиска КЕ в зависимости от выбранного класса. Каждое имя атрибута должно иметь в качестве префикса - имя класса и два двоеточия (i.e. Computer::). Есть ряд атрибутов, общих для всех классов (например, для класса Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). Для отображения индивидуальных атрибутов, специфичных для конкретного класса из описания КЕ (CI-Definition), используется следующий синтаксис (например, для класса Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Если нет атрибутов для конкретного класса, отображаются столбцы заданные параметром ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        'Задает перечень отображаемых столбцов отображаемых КЕ в таблице при создании связи, в зависимости от класса. Каждое имя атрибута должно иметь в качестве префикса - имя класса и два двоеточия (i.e. Computer::). Есть ряд атрибутов, общих для всех классов (например, для класса Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). Для отображения индивидуальных атрибутов, специфичных для конкретного класса из описания КЕ (CI-Definition), используется следующий синтаксис (например, для класса Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Если нет атрибутов для конкретного класса, отображаются столбцы заданные параметром ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        'Определяет, какой тип связи (названный с точки зрения заявки) может повлиять на статус связанной КЕ.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        'Определяет, какой тип заявки может повлиять на статус связанной КЕ.';
    $Self->{Translation}->{'Delete Configuration Item'} = 'Удалить конфигурационную единицу';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = 'Цвет состояния использования';
    $Self->{Translation}->{'Duplicate'} = 'Резерв';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        'Включает возможность массовых действий с КЕ для интерфейса агента для работы с несколькими КЕ одновременно.';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        'Включает возможность массовых действий с КЕ только для перечисленных групп.';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        'Включает/выключает проверку уникальности имен КЕ. До включения этого параметра, проверьте вашу БД на наличие дубликатов имен КЕ. Вы можете сделать это с помощью скрипта bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        'Модуль установки статуса КЕ для связанной с ней заявки.';
    $Self->{Translation}->{'ITSM config item overview.'} = 'Обзор конфигурационных единиц ITSM.';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = 'Модуль для проверки группы ответственной за класс';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        'Модуль для проверки группы ответственной за КЕ';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = 'Модуль формирования статистики по КЕ.';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        'Object backend module registration для модуля Import/Export.';
    $Self->{Translation}->{'Overview.'} = 'Обзор';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        'Параметры для цветов обозначающих состояния использования КЕ в настройках обзоров в интерфейсе агента.';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        'Параметры для состояний использования КЕ в настройках обзоров в интерфейсе агента.';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Параметры для примерных групповых прав для атрибутов Общего каталога';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        'Параметры страниц (на которых отображаются КЕ)';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        'Требуемые права для использования обзора КЕ в CMDB агентом';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        'Требуемые права для использования поиска в CMDB агентом';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        'Требуемые права для использования полного просмотра КЕ (zoom view) в CMDB агентом';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        'Требуемые права для добавления КЕ в CMDB агентом';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        'Требуемые права для редактирования содержимого КЕ в CMDB агентом';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        'Требуемые права для просмотра истории содержимого КЕ в CMDB агентом';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        'Требуемые права для печати характеристик КЕ в CMDB агентом';
    $Self->{Translation}->{'Required privileges to delete config items.'} = 'Необходимы права для удаления конфигурационных единиц.';
    $Self->{Translation}->{'Search config items.'} = 'Поиск конфигурационных единиц.';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        'Выбор модуля генерации номера КЕ. "AutoIncrement" последовательно увеличивает номер на 1.Для формирования номера используются SystemID, ConfigItemClassID и счетчик. Формат - "SystemID.ConfigItemClassID.Counter", т.е. 1205000004, 1205000005.';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        'Установить состояние инцидента КЕ автоматически, когда заявка связана с КЕ.';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        'Задает состояние использования на экране массовых действий с КЕ в интерфейсе агента.';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        'Задает состояние инцидента на экране массовых действий с КЕ в интерфейсе агента.';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        'Показывает пункт меню Связать (Link) для организации связи КЕ с другим объектом на экране полного (zoom view) просмотра';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        'Показывает пункт меню для доступа к истории КЕ при просмотре КЕ в интерфейсе агента.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        'Показывает пункт меню История для просмотра истории КЕ';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        'Показывает пункт меню для удаления конфигурационной единицы при подробном просмотре в интерфейсе агента.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        'Показывает пункт меню Дублировать для создания новой КЕ путем копирования характеристик текущей КЕ';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        'Показывает пункт меню Дублировать для создания новой КЕ путем копирования характеристик текущей КЕ';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        'Показывает пункт меню Редактировать для изменения характеристик КЕ';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        'Показывает пункт меню Назад при подробном просмотре конфигурационных единиц в интерфейсе агента.';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        'Показывает пункт меню Печать для печати характеристик КЕ';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        'Показывает пункт меню для подробного просмотра КЕ при обзоре КЕ в интерфейсе агента';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        'Показ истории содержимого КЕ (порядок показа)';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        'Идентификатор (префикс) к номеру КЕ, т.е КЕ_№, КЕ#, Учетный_элемент_№. По умолчанию - ConfigItem# (или его перевод).';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = 'класс';
    $Self->{Translation}->{'global'} = 'глобальный';
    $Self->{Translation}->{'postproductive'} = 'постпродуктивный';
    $Self->{Translation}->{'preproductive'} = 'предпродуктивный';
    $Self->{Translation}->{'productive'} = 'продуктивный';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::sk_SK_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = 'Zväčšiť pohľad';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = '';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'Kancelária';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::sl_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = 'Povečan pogled';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Med';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'Pisarna';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::sr_Cyrl_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Управљање конфигурационим ставкама';
    $Self->{Translation}->{'Change class definition'} = 'Дефиниција класе промене';
    $Self->{Translation}->{'Config Item Class'} = 'Класа конфигурационе ставке';
    $Self->{Translation}->{'Definition'} = 'Дефиниција';
    $Self->{Translation}->{'Change'} = 'Промени';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Стање инцидента';
    $Self->{Translation}->{'Deployment State'} = 'Статус распоређивања';
    $Self->{Translation}->{'Class'} = 'Класа';
    $Self->{Translation}->{'Deployment State Type'} = 'Тип статуса распоређивања';
    $Self->{Translation}->{'Current Incident State'} = 'Тренутно стање инцидента';
    $Self->{Translation}->{'Current Incident State Type'} = 'Тип тренутног статуса инцидента';
    $Self->{Translation}->{'Last changed'} = 'Задњи пут промењено';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Конфигурациона ставка';
    $Self->{Translation}->{'Filter for Classes'} = 'Филтер за класе';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Изаберите класу из листе ради креирања нове конфигурационе ставке.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'ITSM масовна акција на тикету';
    $Self->{Translation}->{'Deployment state'} = 'Статус распоређивања';
    $Self->{Translation}->{'Incident state'} = 'Стање инцидента';
    $Self->{Translation}->{'Link to another'} = 'Повежи са другим';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = 'Неисправан број конфигурационе ставке!';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'Број друге конфигурационе ставке за повезивање.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = 'Да ли стварно желите да обришете ову конфигурациону ставку?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'Назив ове конфигурационе ставке';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'Назив је већ у употреби на конфигурационим ставкама број: %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = 'Историјат конфигурационе ставке: %s';
    $Self->{Translation}->{'History Content'} = 'Садржај историје';
    $Self->{Translation}->{'Createtime'} = 'Време креирања';
    $Self->{Translation}->{'Zoom view'} = 'Увећани преглед';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Подешавање контекста';
    $Self->{Translation}->{'Config Items per page'} = 'Конфигурационе ставке по страни';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = 'Генеричка табела ITSM конфигурационе ставке';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Покрени претрагу';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Претражите и у претхотним верзијама?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Конфигурациона ставка';
    $Self->{Translation}->{'Configuration Item Information'} = 'Информација о конфигурационој ставки';
    $Self->{Translation}->{'Current Deployment State'} = 'Тренутни статус распоређивања';
    $Self->{Translation}->{'Last changed by'} = 'Променио';
    $Self->{Translation}->{'Show one version'} = 'Прикажи једну верзију';
    $Self->{Translation}->{'Show all versions'} = 'Покажи све верзије';
    $Self->{Translation}->{'Version Incident State'} = 'Верзија - статус инцидента';
    $Self->{Translation}->{'Version Deployment State'} = 'Верзија - статус примене';
    $Self->{Translation}->{'Version Number'} = 'Број верзије';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Детаљи верзије конфигурационе ставке';
    $Self->{Translation}->{'Property'} = 'Својство';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = 'Није дат приступ класи!';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = 'Преглед: ITSM конфигурационе ставке';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = 'Није дат ConfigItemID!';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = 'Потребна је бар једна изабрана конфигурациона ставка!';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        'Немате право уписа за ову конфигурациону ставку: %s.';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = 'Конфигурациона ставка "%s" није нађена у бази података!';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = 'Није било могуће обрисати ConfigItemID %s!';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = 'Није пронађена верзија за ConfigItemID %s!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = 'Нису дати ConfigItemID, DuplicateID или ClassID!';
    $Self->{Translation}->{'No access is given!'} = 'Није дат приступ!';
    $Self->{Translation}->{'No definition was defined for class %s!'} = 'Није пронађена дефиниција за класу %s!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = 'Не може се приказати историјат, јер није дат ConfigItemID!';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = 'Не може се приказати историјат, није дозвољен приступ!';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = 'Нова конфигурациона ставка (ID=%s)';
    $Self->{Translation}->{'New version (ID=%s)'} = 'Нова верзија (ID=%s)';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = 'Ажурирано стање распореда (ново=%s, старо=%s) ';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = 'Ажурирано стање инцидента (ново=%s, старо=%s)';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = 'Обрисана конфигурациона ставка (ID=%s)';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = 'Повезано са %s (тип=%s)';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = 'Обрисана веза са %s (тип=%s)';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = 'Ажурирана дефиниција конфигурационе ставке (ID=%s)';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = 'Ажуриран назив (нови=%s, стари=%s)';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = 'Ажуриран атрибут%s са "%s" на "%s"';
    $Self->{Translation}->{'Version %s deleted'} = 'Верзија %s је обрисана';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = 'Нису дати ConfigItemID или VersionID!';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = 'Не може се приказати конфигурациона ставка, није дат приступ!';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = 'ИД конфигурационе ставке %s није нађен у бази података!';
    $Self->{Translation}->{'VersionID %s not found in database!'} = 'Ид верзије %s није нађен у бази података!';
    $Self->{Translation}->{'ConfigItem'} = 'Конфигурациона ставка';
    $Self->{Translation}->{'printed by %s at %s'} = 'штампану од стране %s у %s';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = 'Неисправан ИД класе!';
    $Self->{Translation}->{'No ClassID is given!'} = 'Није дат ИД класе!';
    $Self->{Translation}->{'No access rights for this class given!'} = 'Није дат приступ овој класи!';
    $Self->{Translation}->{'No Result!'} = 'Нема резултата!';
    $Self->{Translation}->{'Config Item Search Results'} = 'Резултат претраге конфигурационих ставки';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        'Не може се приказати ставка, није дат приступ за ову конфигурациону ставку!';
    $Self->{Translation}->{'operational'} = 'оперативни';
    $Self->{Translation}->{'warning'} = 'упозорење';
    $Self->{Translation}->{'incident'} = 'инцидент';
    $Self->{Translation}->{'The deployment state of this config item'} = 'Статус примене ове конфигурационе ставке';
    $Self->{Translation}->{'The incident state of this config item'} = 'Статус инцидента ове конфигурационе ставке';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = 'Приказане конфигурационе ставке';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Између';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Максимални број једног елемента';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Празна поља указују да ће актуелне вредности бити задржане';
    $Self->{Translation}->{'Skipped'} = 'Прескочено';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Модел';
    $Self->{Translation}->{'Customer Company'} = 'Фирма клијента';
    $Self->{Translation}->{'Serial Number'} = 'Серијски број';
    $Self->{Translation}->{'CPU'} = 'Процесор';
    $Self->{Translation}->{'Ram'} = 'RAM';
    $Self->{Translation}->{'Hard Disk'} = 'Хард диск';
    $Self->{Translation}->{'Capacity'} = 'Капацитет';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Мрежни адаптер';
    $Self->{Translation}->{'IP over DHCP'} = 'IP преко DHCP';
    $Self->{Translation}->{'IP Address'} = 'IP адреса';
    $Self->{Translation}->{'Graphic Adapter'} = 'Графички адаптер';
    $Self->{Translation}->{'Other Equipment'} = 'Друга опрема';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Датум истицања гаранције';
    $Self->{Translation}->{'Install Date'} = 'Датум инсталације';
    $Self->{Translation}->{'Phone 1'} = 'Телефон 1';
    $Self->{Translation}->{'Phone 2'} = 'Телефон 2';
    $Self->{Translation}->{'E-Mail'} = 'Имејл';
    $Self->{Translation}->{'Network Address'} = 'Мрежна адреса';
    $Self->{Translation}->{'Subnet Mask'} = 'Подмрежна маска';
    $Self->{Translation}->{'Gateway'} = 'Мрежни пролаз';
    $Self->{Translation}->{'Licence Type'} = 'Тип лиценце';
    $Self->{Translation}->{'Licence Key'} = 'Лиценцни кључ';
    $Self->{Translation}->{'Quantity'} = 'Количина';
    $Self->{Translation}->{'Expiration Date'} = 'Датум истицања';
    $Self->{Translation}->{'Media'} = 'Медија';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = 'Рачунар';
    $Self->{Translation}->{'Hardware'} = 'Хардвер';
    $Self->{Translation}->{'Network'} = 'Мрежа';
    $Self->{Translation}->{'Software'} = 'Софтвер';
    $Self->{Translation}->{'Expired'} = 'Истекло';
    $Self->{Translation}->{'Maintenance'} = 'Одржавање';
    $Self->{Translation}->{'Pilot'} = 'Пилот';
    $Self->{Translation}->{'Planned'} = 'Планирано';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'На поправци';
    $Self->{Translation}->{'Retired'} = 'Расходовано';
    $Self->{Translation}->{'Review'} = 'Рецензија';
    $Self->{Translation}->{'Test/QA'} = 'Тест/QA';
    $Self->{Translation}->{'Laptop'} = 'Лаптоп';
    $Self->{Translation}->{'Desktop'} = 'Десктоп';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Сервер';
    $Self->{Translation}->{'Other'} = 'Друго';
    $Self->{Translation}->{'Monitor'} = 'Монитор';
    $Self->{Translation}->{'Printer'} = 'Штампач';
    $Self->{Translation}->{'Switch'} = 'Свич';
    $Self->{Translation}->{'Router'} = 'Рутер';
    $Self->{Translation}->{'WLAN Access Point'} = 'WLAN приступна тачка';
    $Self->{Translation}->{'Security Device'} = 'Сигурносни уређај';
    $Self->{Translation}->{'Backup Device'} = 'Уређај за резервне копије';
    $Self->{Translation}->{'Mouse'} = 'Миш';
    $Self->{Translation}->{'Keyboard'} = 'Тастатура';
    $Self->{Translation}->{'Camera'} = 'Камера';
    $Self->{Translation}->{'Beamer'} = 'Пројектор';
    $Self->{Translation}->{'Modem'} = 'Модем';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA картица';
    $Self->{Translation}->{'USB Device'} = 'USB уређај';
    $Self->{Translation}->{'Docking Station'} = 'Прикључна станица';
    $Self->{Translation}->{'Scanner'} = 'Скенер';
    $Self->{Translation}->{'Building'} = 'Зграда';
    $Self->{Translation}->{'Office'} = 'Канцеларија';
    $Self->{Translation}->{'Floor'} = 'Спрат';
    $Self->{Translation}->{'Room'} = 'Соба';
    $Self->{Translation}->{'Rack'} = 'Рек';
    $Self->{Translation}->{'Workplace'} = 'Радно место';
    $Self->{Translation}->{'Outlet'} = 'Утичница';
    $Self->{Translation}->{'IT Facility'} = 'ИТ објекат';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Клијентска апликација';
    $Self->{Translation}->{'Middleware'} = 'Посреднички програм';
    $Self->{Translation}->{'Server Application'} = 'Серверска апликација';
    $Self->{Translation}->{'Client OS'} = 'Клијентски OS';
    $Self->{Translation}->{'Server OS'} = 'Серверски OS';
    $Self->{Translation}->{'Admin Tool'} = 'Административни алат';
    $Self->{Translation}->{'User Tool'} = 'Кориснички алат';
    $Self->{Translation}->{'Embedded'} = 'Уграђен';
    $Self->{Translation}->{'Single Licence'} = 'Појединачна лиценца';
    $Self->{Translation}->{'Per User'} = 'По кориснику';
    $Self->{Translation}->{'Per Processor'} = 'По процесору';
    $Self->{Translation}->{'Per Server'} = 'По серверу';
    $Self->{Translation}->{'Per Node'} = 'По чвору';
    $Self->{Translation}->{'Volume Licence'} = 'Количинске лиценце';
    $Self->{Translation}->{'Enterprise Licence'} = 'Компанијска лиценца';
    $Self->{Translation}->{'Developer Licence'} = 'Развојна лиценца';
    $Self->{Translation}->{'Demo'} = 'Демонстрациони';
    $Self->{Translation}->{'Time Restricted'} = 'Временски ограничено';
    $Self->{Translation}->{'Freeware'} = 'Бесплатни **';
    $Self->{Translation}->{'Open Source'} = 'Отворени код';
    $Self->{Translation}->{'Unlimited'} = 'Неограничено';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'У реду';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = 'Додељене конфигурационе ставке';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = 'Конфигурационе ставке додељене фирми клијента';
    $Self->{Translation}->{'CIs assigned to customer user'} = 'Конфигурационе ставке додељене клијенту кориснику';
    $Self->{Translation}->{'CMDB Settings'} = 'CMDB подешавања';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        'Провера јединственог имена само у оквиру исте класе конфигурационих ставки (\'класа\') или глобално (\'глобално\'), што значи да је свака постојећа Конфигурациона ставка узета у обзир у провери дупликата.';
    $Self->{Translation}->{'Config Items'} = 'Конфигурационе ставке';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = 'Додавање конфигурационе ставке.';
    $Self->{Translation}->{'Config item edit.'} = 'Уређивање конфигурационе ставке.';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        'Модул догађаја конфигурационих ставки који омогућава бележење у историјат у интерфејсу оператера.';
    $Self->{Translation}->{'Config item history.'} = 'Историјат конфигурационе ставке.';
    $Self->{Translation}->{'Config item print.'} = 'Штампа конфигурационе ставке';
    $Self->{Translation}->{'Config item zoom.'} = 'Детаљни приказ конфигурацине ставке.';
    $Self->{Translation}->{'ConfigItemNumber'} = 'Број конфигурационе ставке';
    $Self->{Translation}->{'Configuration Item Limit'} = 'Ограничење конфигурационе ставке';
    $Self->{Translation}->{'Configuration Item limit per page.'} = 'Ограничење конфигурационих ставки по страни.';
    $Self->{Translation}->{'Configuration Management Database.'} = 'База података управљања конфигурацијом.';
    $Self->{Translation}->{'Configuration item bulk module.'} = 'Модул масовне акције на конфигурационим ставкама.';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        'Конфигурациона ставка претражује позадински рутер у интерфејсу оператера.';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Креирање и управљање дефиниција за конфигурационе ставке.';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        'Дефинише акције где је дугме поставки доступно у повезаном графичком елементу објекта (LinkObject::ViewMode = "complex"). Молимо да имате на уму да ове Акције морају да буду регистроване у следећим JS и CSS датотекама: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js и Core.Agent.LinkObject.js.';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        'Дефинише неопходне дозволе за креирање ITSM конфигурационих ставки коришћењем генеричког интерфејса.';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        'Одређује потребне дозволе за брисање ITSM конфигурационих ставки кроз генерички интерфејс.';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        'Дефинише неопходне дозволе за добијање ITSM конфигурационих ставки коришћењем генеричког интерфејса.';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        'Дефинише неопходне дозволе за претрагу ITSM конфигурационих ставки коришћењем генеричког интерфејса.';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        'Дефинише неопходне дозволе за ажурирање ITSM конфигурационих ставки коришћењем генеричког интерфејса.';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        'Дефинише модул прегледа за мали приказ листе конфигурационих ставки. ';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        'Одређује регуларне изразе за сваку класу конфигурационе ставке ради провере назива конфигурационе ставке и приказа припадајућих порука о грешкама.';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        'Одређује подразумевани подобјекат класе ITSM конфигурациона ставка.';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        'Одређује број редова за едитор дефиниције CI у административном интерфејсу.';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        'Одређује редослед инцидентних стања од високог (нпр критично) до ниског (нпр функционално).';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        'Одређује релевантна стања распоређивања где повезани тикети могу да утичу на статус CI.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        'Дефинише границу претраге за екран AgentITSMConfigItem.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        'Дефинише границу претраге за екран AgentITSMConfigItemSearch.';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        'Одређује приказане колоне у прегледу конфигурационих ставки. Ова опције нема утицај на позиције колона. Напомена: уколико је изабран филтер \'Све\' класа колоне је увек доступна.';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        'Дефинише приказане колоне у претрази конфигурационих ставки. Ова опције нема утицај на позиције колона.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Дефинише приказане колоне конфигурационих ставки у прегледу у зависности од класе. Сваки унос мора садржати префикс са називом класе и двотачком (нпр. Computer::). Постоји пар атрибута који су заједнички за све класе (нпр. за класу Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). За приказ појединачних атрибута из дефиниције, морате користити следећу шему (пример за класу Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Уколико не постоји унос за појединачну класу, биће приказане подразумеване колоне дефинисане у ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Дефинише приказане колоне конфигурационих ставки у претрази у зависности од класе. Сваки унос мора садржати префикс са називом класе и двотачком (нпр. Computer::). Постоји пар атрибута који су заједнички за све класе (нпр. за класу Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). За приказ појединачних атрибута из дефиниције, морате користити следећу шему (пример за класу Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Уколико не постоји унос за појединачну класу, биће приказане подразумеване колоне дефинисане у ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        'Дефинише приказане колоне конфигурационих ставки у додатку командне табле у зависности од класе. Сваки унос мора садржати префикс са називом класе и двотачком (нпр. Computer::). Постоји пар атрибута који су заједнички за све класе (нпр. за класу Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). За приказ појединачних атрибута из дефиниције, морате користити следећу шему (пример за класу Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Уколико не постоји унос за појединачну класу, биће приказане подразумеване колоне дефинисане у AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (кључ DefaultColumns).';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        'Дефинише приказане колоне конфигурационих ставки у табели веза у зависности од класе. Уколико не постоји унос за појединачну класу, биће приказане подразумеване колоне.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        'Дефинише приказане колоне конфигурационих ставки у табели веза у зависности од класе. Сваки унос мора садржати префикс са називом класе и двотачком (нпр. Computer::). Постоји пар атрибута који су заједнички за све класе (нпр. за класу Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). За приказ појединачних атрибута из дефиниције, морате користити следећу шему (пример за класу Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Уколико не постоји унос за појединачну класу, биће приказане подразумеване колоне.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        'Одређује који тип везе (гледано из перспективе тикета) може да утиче на повезану CI.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        'Одређује који тип тикета може да утиче на статус повезане CI.';
    $Self->{Translation}->{'Delete Configuration Item'} = 'Брисање конфигурационе ставке';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = 'Боја статуса распоређивања';
    $Self->{Translation}->{'Duplicate'} = 'Дупликат';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        'Омогућава масовну акцију на конфигурационим ставкама у приступном систему оператера на више од једне ставке истовремено.';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        'Активира својство масовне акције на конфигирационим ставкама само за излистане групе.';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        'Активира/деактивира функцију за проверу јединствености имана конфигурационих ставки. Пре активирања ове опције требате проверити у вашем систему постојање дупликата на постојећим ставкама. То можете урадити помоћу скрипта bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates. ';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        'Модул догађаја за подешавање статуса конфигурационих ставки приликом постављања веза.';
    $Self->{Translation}->{'ITSM config item overview.'} = 'Преглед ITSM конфигурационе ставке.';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = 'Модул за проверу групе одговорне за класу.';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        'Модул за проверу групе одговорне за конфигурациону ставку.';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = 'Модул за генерисање статистике ITSM конфигурационих ставки.';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        'Регистрација модула објекта за додатак увоз/извоз.';
    $Self->{Translation}->{'Overview.'} = 'Преглед.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Параметри за позадински приказ контролне табле листе конфигурационих ставки фирме клијента у интерфејсу оператера. "Limit" дефинише подразумевани број приказаних ставки. "Group" се користи да ограничи приступ додатку (нпр. Group: admin;group1;group2;). "Default" одређује да ли је додатак подразумевано активиран или да је неопходно да га корисник мануелно активира. "CacheTTLLocal" је време у минутима за кеширање додатка.';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        'Параметри за боје статуса примене на приказу поставки у интерфејсу оператера.';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        'Параметри за статусе примене на приказу поставки у интерфејсу оператера.';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Параметри за атрибуте узорка групе за дозволе општег каталога';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        'Параметри за странице (на којима су конфигурационе ставке приказане).';
    $Self->{Translation}->{'Permission Group'} = 'Група приступа';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        'Потребне дозволе за употребу екрана ITSM конфигурационе ставке у интерфејсу оператера.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        'Потребне дозволе за употребу екрана претраге ITSM конфигурационе ставке у интерфејсу оператера.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        'Потребне дозволе за употребу екрана деталног приказа ITSM конфигурационе ставке у интерфејсу оператера.';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        'Потребне дозволе за употребу екрана додавања ITSM конфигурационе ставке у интерфејсу оператера.';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        'Потребне дозволе за употребу екрана измене ITSM конфигурационе ставке у интерфејсу оператера.';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        'Потребне дозволе за употребу екрана историјата ITSM конфигурационе ставке у интерфејсу оператера.';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        'Потребне дозволе за употребу екрана штампе ITSM конфигурационе ставке у интерфејсу оператера.';
    $Self->{Translation}->{'Required privileges to delete config items.'} = 'Потребна права за брисање конфигурационе ставке.';
    $Self->{Translation}->{'Search config items.'} = 'Претрага конфигурационих ставки.';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        'Бира модул оператора аутоматског увећања конфигурационе ставке. Оператор аутоматског увећања повећава број ставке, SystemID, ConfigItemClassID и коришћени бројач. Формат је "SystemID.ConfigItemClassID.бројач", нпр 1205000004, 1205000005.';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        'Поставља стање инцидента конфигурационе ставке када је тикет повезан са ставком.';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        'Одређује статус примене на екрану масовне акције конфигурационе ставке у интерфејсу оператера.';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        'Одређује статус инцидента на екрану масовне акције конфигурационе ставке у интерфејсу оператера.';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        'У менију приказује везу која омогућава повезивање конфигурационе ставке са другим објектом на детаљном приказу конфигурације у интерфејсу оператера.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        'У менију приказује везу за приступ историјату конфигурационе ставке на прегледу конфигурационе ставке у интерфејсу оператера.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        'У менију приказује везу за приступ историјату конфигурационе ставке на детаљном приказу у интерфејсу оператера.';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        'У менију приказује везу за брисање конфигурационе ставке на њеном детаљном приказу у интерфејсу оператера.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        'У менију приказује везу за умножавање конфигурационе ставке на прегледу конфигурационе ставке у интерфејсу оператера.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        'У менију приказује везу за умножавање конфигурационе ставке на њеном детаљном приказу у интерфејсу оператера.';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        'У менију приказује везу за измену конфигурационе ставке на њеном детаљном приказу у интерфејсу оператера.';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        'У менију приказује везу за повратак на детаљни приказ конфигурационе ставке  у интерфејсу оператера.';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        'У менију приказује везу за штампу конфигурационе ставке на њеном детаљном приказу у интерфејсу оператера.';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        'У менију приказује везу за детаљни приказ конфигурационе ставке на прегледу конфигурационе ставке у интерфејсу оператера.';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        'Приказује историјат конфигурационе ставке (обрнут редослед) у интерфејсу оператера.';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        'Идентификатор за конфигурациону ставку, нпр. ConfigItem#, MyConfigItem#. Подразумевано је ConfigItem#.';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = 'класа';
    $Self->{Translation}->{'global'} = 'глобално';
    $Self->{Translation}->{'postproductive'} = 'пост-продукција';
    $Self->{Translation}->{'preproductive'} = 'пре-продукција';
    $Self->{Translation}->{'productive'} = 'продукција';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::sr_Latn_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Upravljanje konfiguracionim stavkama';
    $Self->{Translation}->{'Change class definition'} = 'Definicija klase promene';
    $Self->{Translation}->{'Config Item Class'} = 'Klasa konfiguracione stavke';
    $Self->{Translation}->{'Definition'} = 'Definicija';
    $Self->{Translation}->{'Change'} = 'Promeni';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Stanje incidenta';
    $Self->{Translation}->{'Deployment State'} = 'Status raspoređivanja';
    $Self->{Translation}->{'Class'} = 'Klasa';
    $Self->{Translation}->{'Deployment State Type'} = 'Tip statusa raspoređivanja';
    $Self->{Translation}->{'Current Incident State'} = 'Trenutno stanje incidenta';
    $Self->{Translation}->{'Current Incident State Type'} = 'Tip trenutnog statusa incidenta';
    $Self->{Translation}->{'Last changed'} = 'Zadnji put promenjeno';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Konfiguraciona stavka';
    $Self->{Translation}->{'Filter for Classes'} = 'Filter za klase';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Izaberite klasu iz liste radi kreiranja nove konfiguracione stavke.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'ITSM masovna akcija na tiketu';
    $Self->{Translation}->{'Deployment state'} = 'Status raspoređivanja';
    $Self->{Translation}->{'Incident state'} = 'Stanje incidenta';
    $Self->{Translation}->{'Link to another'} = 'Poveži sa drugim';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = 'Neispravan broj konfiguracione stavke!';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'Broj druge konfiguracione stavke za povezivanje.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = 'Da li stvarno želite da obrišete ovu konfiguracionu stavku?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'Naziv ove konfiguracione stavke';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'Naziv je već u upotrebi na konfiguracionim stavkama broj: %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = 'Istorijat konfiguracione stavke: %s';
    $Self->{Translation}->{'History Content'} = 'Sadržaj istorije';
    $Self->{Translation}->{'Createtime'} = 'Vreme kreiranja';
    $Self->{Translation}->{'Zoom view'} = 'Uvećani pregled';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Podešavanje konteksta';
    $Self->{Translation}->{'Config Items per page'} = 'Konfiguracione stavke po strani';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = 'Generička tabela ITSM konfiguracione stavke';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Pokreni pretragu';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Pretražite i u prethotnim verzijama?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Konfiguraciona stavka';
    $Self->{Translation}->{'Configuration Item Information'} = 'Informacija o konfiguracionoj stavki';
    $Self->{Translation}->{'Current Deployment State'} = 'Trenutni status raspoređivanja';
    $Self->{Translation}->{'Last changed by'} = 'Promenio';
    $Self->{Translation}->{'Show one version'} = 'Prikaži jednu verziju';
    $Self->{Translation}->{'Show all versions'} = 'Pokaži sve verzije';
    $Self->{Translation}->{'Version Incident State'} = 'Verzija - status incidenta';
    $Self->{Translation}->{'Version Deployment State'} = 'Verzija - status primene';
    $Self->{Translation}->{'Version Number'} = 'Broj verzije';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Detalji verzije konfiguracione stavke';
    $Self->{Translation}->{'Property'} = 'Svojstvo';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = 'Nije dat pristup klasi!';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = 'Pregled: ITSM konfiguracione stavke';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = 'Nije dat ConfigItemID!';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = 'Potrebna je bar jedna izabrana konfiguraciona stavka!';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        'Nemate pravo upisa za ovu konfiguracionu stavku: %s.';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = 'Konfiguraciona stavka "%s" nije nađena u bazi podataka!';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = 'Nije bilo moguće obrisati ConfigItemID %s!';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = 'Nije pronađena verzija za ConfigItemID %s!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = 'Nisu dati ConfigItemID, DuplicateID ili ClassID!';
    $Self->{Translation}->{'No access is given!'} = 'Nije dat pristup!';
    $Self->{Translation}->{'No definition was defined for class %s!'} = 'Nije pronađena definicija za klasu %s!';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = 'Ne može se prikazati istorijat, jer nije dat ConfigItemID!';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = 'Ne može se prikazati istorijat, nije dozvoljen pristup!';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = 'Nova konfiguraciona stavka (ID=%s)';
    $Self->{Translation}->{'New version (ID=%s)'} = 'Nova verzija (ID=%s)';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = 'Ažurirano stanje rasporeda (novo=%s, staro=%s) ';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = 'Ažurirano stanje incidenta (novo=%s, staro=%s)';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = 'Obrisana konfiguraciona stavka (ID=%s)';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = 'Povezano sa %s (tip=%s)';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = 'Obrisana veza sa %s (tip=%s)';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = 'Ažurirana definicija konfiguracione stavke (ID=%s)';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = 'Ažuriran naziv (novi=%s, stari=%s)';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = 'Ažuriran atribut%s sa "%s" na "%s"';
    $Self->{Translation}->{'Version %s deleted'} = 'Verzija %s je obrisana';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = 'Nisu dati ConfigItemID ili VersionID!';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = 'Ne može se prikazati konfiguraciona stavka, nije dat pristup!';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = 'ID konfiguracione stavke %s nije nađen u bazi podataka!';
    $Self->{Translation}->{'VersionID %s not found in database!'} = 'Id verzije %s nije nađen u bazi podataka!';
    $Self->{Translation}->{'ConfigItem'} = 'Konfiguraciona stavka';
    $Self->{Translation}->{'printed by %s at %s'} = 'štampanu od strane %s u %s';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = 'Neispravan ID klase!';
    $Self->{Translation}->{'No ClassID is given!'} = 'Nije dat ID klase!';
    $Self->{Translation}->{'No access rights for this class given!'} = 'Nije dat pristup ovoj klasi!';
    $Self->{Translation}->{'No Result!'} = 'Nema rezultata!';
    $Self->{Translation}->{'Config Item Search Results'} = 'Rezultat pretrage konfiguracionih stavki';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        'Ne može se prikazati stavka, nije dat pristup za ovu konfiguracionu stavku!';
    $Self->{Translation}->{'operational'} = 'operativni';
    $Self->{Translation}->{'warning'} = 'upozorenje';
    $Self->{Translation}->{'incident'} = 'incident';
    $Self->{Translation}->{'The deployment state of this config item'} = 'Status primene ove konfiguracione stavke';
    $Self->{Translation}->{'The incident state of this config item'} = 'Status incidenta ove konfiguracione stavke';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = 'Prikazane konfiguracione stavke';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Između';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Maksimalni broj jednog elementa';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Prazna polja ukazuju da će aktuelne vrednosti biti zadržane';
    $Self->{Translation}->{'Skipped'} = 'Preskočeno';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Model';
    $Self->{Translation}->{'Customer Company'} = 'Firma klijenta';
    $Self->{Translation}->{'Serial Number'} = 'Serijski broj';
    $Self->{Translation}->{'CPU'} = 'Procesor';
    $Self->{Translation}->{'Ram'} = 'RAM';
    $Self->{Translation}->{'Hard Disk'} = 'Hard disk';
    $Self->{Translation}->{'Capacity'} = 'Kapacitet';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Mrežni adapter';
    $Self->{Translation}->{'IP over DHCP'} = 'IP preko DHCP';
    $Self->{Translation}->{'IP Address'} = 'IP adresa';
    $Self->{Translation}->{'Graphic Adapter'} = 'Grafički adapter';
    $Self->{Translation}->{'Other Equipment'} = 'Druga oprema';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Datum isticanja garancije';
    $Self->{Translation}->{'Install Date'} = 'Datum instalacije';
    $Self->{Translation}->{'Phone 1'} = 'Telefon 1';
    $Self->{Translation}->{'Phone 2'} = 'Telefon 2';
    $Self->{Translation}->{'E-Mail'} = 'Imejl';
    $Self->{Translation}->{'Network Address'} = 'Mrežna adresa';
    $Self->{Translation}->{'Subnet Mask'} = 'Podmrežna maska';
    $Self->{Translation}->{'Gateway'} = 'Mrežni prolaz';
    $Self->{Translation}->{'Licence Type'} = 'Tip licence';
    $Self->{Translation}->{'Licence Key'} = 'Licencni ključ';
    $Self->{Translation}->{'Quantity'} = 'Količina';
    $Self->{Translation}->{'Expiration Date'} = 'Datum isticanja';
    $Self->{Translation}->{'Media'} = 'Medija';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = 'Računar';
    $Self->{Translation}->{'Hardware'} = 'Hardver';
    $Self->{Translation}->{'Network'} = 'Mreža';
    $Self->{Translation}->{'Software'} = 'Softver';
    $Self->{Translation}->{'Expired'} = 'Isteklo';
    $Self->{Translation}->{'Maintenance'} = 'Održavanje';
    $Self->{Translation}->{'Pilot'} = 'Pilot';
    $Self->{Translation}->{'Planned'} = 'Planirano';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Na popravci';
    $Self->{Translation}->{'Retired'} = 'Rashodovano';
    $Self->{Translation}->{'Review'} = 'Recenzija';
    $Self->{Translation}->{'Test/QA'} = 'Test/QA';
    $Self->{Translation}->{'Laptop'} = 'Laptop';
    $Self->{Translation}->{'Desktop'} = 'Desktop';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Server';
    $Self->{Translation}->{'Other'} = 'Drugo';
    $Self->{Translation}->{'Monitor'} = 'Monitor';
    $Self->{Translation}->{'Printer'} = 'Štampač';
    $Self->{Translation}->{'Switch'} = 'Svič';
    $Self->{Translation}->{'Router'} = 'Ruter';
    $Self->{Translation}->{'WLAN Access Point'} = 'WLAN pristupna tačka';
    $Self->{Translation}->{'Security Device'} = 'Sigurnosni uređaj';
    $Self->{Translation}->{'Backup Device'} = 'Uređaj za rezervne kopije';
    $Self->{Translation}->{'Mouse'} = 'Miš';
    $Self->{Translation}->{'Keyboard'} = 'Tastatura';
    $Self->{Translation}->{'Camera'} = 'Kamera';
    $Self->{Translation}->{'Beamer'} = 'Projektor';
    $Self->{Translation}->{'Modem'} = 'Modem';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA kartica';
    $Self->{Translation}->{'USB Device'} = 'USB uređaj';
    $Self->{Translation}->{'Docking Station'} = 'Priključna stanica';
    $Self->{Translation}->{'Scanner'} = 'Skener';
    $Self->{Translation}->{'Building'} = 'Zgrada';
    $Self->{Translation}->{'Office'} = 'Kancelarija';
    $Self->{Translation}->{'Floor'} = 'Sprat';
    $Self->{Translation}->{'Room'} = 'Soba';
    $Self->{Translation}->{'Rack'} = 'Rek';
    $Self->{Translation}->{'Workplace'} = 'Radno mesto';
    $Self->{Translation}->{'Outlet'} = 'Utičnica';
    $Self->{Translation}->{'IT Facility'} = 'IT objekat';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Klijentska aplikacija';
    $Self->{Translation}->{'Middleware'} = 'Posrednički program';
    $Self->{Translation}->{'Server Application'} = 'Serverska aplikacija';
    $Self->{Translation}->{'Client OS'} = 'Klijentski OS';
    $Self->{Translation}->{'Server OS'} = 'Serverski OS';
    $Self->{Translation}->{'Admin Tool'} = 'Administrativni alat';
    $Self->{Translation}->{'User Tool'} = 'Korisnički alat';
    $Self->{Translation}->{'Embedded'} = 'Ugrađen';
    $Self->{Translation}->{'Single Licence'} = 'Pojedinačna licenca';
    $Self->{Translation}->{'Per User'} = 'Po korisniku';
    $Self->{Translation}->{'Per Processor'} = 'Po procesoru';
    $Self->{Translation}->{'Per Server'} = 'Po serveru';
    $Self->{Translation}->{'Per Node'} = 'Po čvoru';
    $Self->{Translation}->{'Volume Licence'} = 'Količinske licence';
    $Self->{Translation}->{'Enterprise Licence'} = 'Kompanijska licenca';
    $Self->{Translation}->{'Developer Licence'} = 'Razvojna licenca';
    $Self->{Translation}->{'Demo'} = 'Demonstracioni';
    $Self->{Translation}->{'Time Restricted'} = 'Vremenski ograničeno';
    $Self->{Translation}->{'Freeware'} = 'Besplatni **';
    $Self->{Translation}->{'Open Source'} = 'Otvoreni kod';
    $Self->{Translation}->{'Unlimited'} = 'Neograničeno';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'U redu';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = 'Dodeljene konfiguracione stavke';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = 'Konfiguracione stavke dodeljene firmi klijenta';
    $Self->{Translation}->{'CIs assigned to customer user'} = 'Konfiguracione stavke dodeljene klijentu korisniku';
    $Self->{Translation}->{'CMDB Settings'} = 'CMDB podešavanja';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        'Provera jedinstvenog imena samo u okviru iste klase konfiguracionih stavki (\'klasa\') ili globalno (\'globalno\'), što znači da je svaka postojeća Konfiguraciona stavka uzeta u obzir u proveri duplikata.';
    $Self->{Translation}->{'Config Items'} = 'Konfiguracione stavke';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = 'Dodavanje konfiguracione stavke.';
    $Self->{Translation}->{'Config item edit.'} = 'Uređivanje konfiguracione stavke.';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        'Modul događaja konfiguracionih stavki koji omogućava beleženje u istorijat u interfejsu operatera.';
    $Self->{Translation}->{'Config item history.'} = 'Istorijat konfiguracione stavke.';
    $Self->{Translation}->{'Config item print.'} = 'Štampa konfiguracione stavke';
    $Self->{Translation}->{'Config item zoom.'} = 'Detaljni prikaz konfiguracine stavke.';
    $Self->{Translation}->{'ConfigItemNumber'} = 'Broj konfiguracione stavke';
    $Self->{Translation}->{'Configuration Item Limit'} = 'Ograničenje konfiguracione stavke';
    $Self->{Translation}->{'Configuration Item limit per page.'} = 'Ograničenje konfiguracionih stavki po strani.';
    $Self->{Translation}->{'Configuration Management Database.'} = 'Baza podataka upravljanja konfiguracijom.';
    $Self->{Translation}->{'Configuration item bulk module.'} = 'Modul masovne akcije na konfiguracionim stavkama.';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        'Konfiguraciona stavka pretražuje pozadinski ruter u interfejsu operatera.';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Kreiranje i upravljanje definicija za konfiguracione stavke.';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        'Definiše akcije gde je dugme postavki dostupno u povezanom grafičkom elementu objekta (LinkObject::ViewMode = "complex"). Molimo da imate na umu da ove Akcije moraju da budu registrovane u sledećim JS i CSS datotekama: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js i Core.Agent.LinkObject.js.';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        'Definiše neophodne dozvole za kreiranje ITSM konfiguracionih stavki korišćenjem generičkog interfejsa.';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        'Određuje potrebne dozvole za brisanje ITSM konfiguracionih stavki kroz generički interfejs.';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        'Definiše neophodne dozvole za dobijanje ITSM konfiguracionih stavki korišćenjem generičkog interfejsa.';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        'Definiše neophodne dozvole za pretragu ITSM konfiguracionih stavki korišćenjem generičkog interfejsa.';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        'Definiše neophodne dozvole za ažuriranje ITSM konfiguracionih stavki korišćenjem generičkog interfejsa.';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        'Definiše modul pregleda za mali prikaz liste konfiguracionih stavki. ';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        'Određuje regularne izraze za svaku klasu konfiguracione stavke radi provere naziva konfiguracione stavke i prikaza pripadajućih poruka o greškama.';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        'Određuje podrazumevani podobjekat klase ITSM konfiguraciona stavka.';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        'Određuje broj redova za editor definicije CI u administrativnom interfejsu.';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        'Određuje redosled incidentnih stanja od visokog (npr kritično) do niskog (npr funkcionalno).';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        'Određuje relevantna stanja raspoređivanja gde povezani tiketi mogu da utiču na status CI.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        'Definiše granicu pretrage za ekran AgentITSMConfigItem.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        'Definiše granicu pretrage za ekran AgentITSMConfigItemSearch.';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        'Određuje prikazane kolone u pregledu konfiguracionih stavki. Ova opcije nema uticaj na pozicije kolona. Napomena: ukoliko je izabran filter \'Sve\' klasa kolone je uvek dostupna.';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        'Definiše prikazane kolone u pretrazi konfiguracionih stavki. Ova opcije nema uticaj na pozicije kolona.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Definiše prikazane kolone konfiguracionih stavki u pregledu u zavisnosti od klase. Svaki unos mora sadržati prefiks sa nazivom klase i dvotačkom (npr. Computer::). Postoji par atributa koji su zajednički za sve klase (npr. za klasu Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). Za prikaz pojedinačnih atributa iz definicije, morate koristiti sledeću šemu (primer za klasu Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Ukoliko ne postoji unos za pojedinačnu klasu, biće prikazane podrazumevane kolone definisane u ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Definiše prikazane kolone konfiguracionih stavki u pretrazi u zavisnosti od klase. Svaki unos mora sadržati prefiks sa nazivom klase i dvotačkom (npr. Computer::). Postoji par atributa koji su zajednički za sve klase (npr. za klasu Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). Za prikaz pojedinačnih atributa iz definicije, morate koristiti sledeću šemu (primer za klasu Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Ukoliko ne postoji unos za pojedinačnu klasu, biće prikazane podrazumevane kolone definisane u ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        'Definiše prikazane kolone konfiguracionih stavki u dodatku komandne table u zavisnosti od klase. Svaki unos mora sadržati prefiks sa nazivom klase i dvotačkom (npr. Computer::). Postoji par atributa koji su zajednički za sve klase (npr. za klasu Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). Za prikaz pojedinačnih atributa iz definicije, morate koristiti sledeću šemu (primer za klasu Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Ukoliko ne postoji unos za pojedinačnu klasu, biće prikazane podrazumevane kolone definisane u AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (ključ DefaultColumns).';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        'Definiše prikazane kolone konfiguracionih stavki u tabeli veza u zavisnosti od klase. Ukoliko ne postoji unos za pojedinačnu klasu, biće prikazane podrazumevane kolone.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        'Definiše prikazane kolone konfiguracionih stavki u tabeli veza u zavisnosti od klase. Svaki unos mora sadržati prefiks sa nazivom klase i dvotačkom (npr. Computer::). Postoji par atributa koji su zajednički za sve klase (npr. za klasu Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). Za prikaz pojedinačnih atributa iz definicije, morate koristiti sledeću šemu (primer za klasu Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. Ukoliko ne postoji unos za pojedinačnu klasu, biće prikazane podrazumevane kolone.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        'Određuje koji tip veze (gledano iz perspektive tiketa) može da utiče na povezanu CI.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        'Određuje koji tip tiketa može da utiče na status povezane CI.';
    $Self->{Translation}->{'Delete Configuration Item'} = 'Brisanje konfiguracione stavke';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = 'Boja statusa raspoređivanja';
    $Self->{Translation}->{'Duplicate'} = 'Duplikat';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        'Omogućava masovnu akciju na konfiguracionim stavkama u pristupnom sistemu operatera na više od jedne stavke istovremeno.';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        'Aktivira svojstvo masovne akcije na konfigiracionim stavkama samo za izlistane grupe.';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        'Aktivira/deaktivira funkciju za proveru jedinstvenosti imana konfiguracionih stavki. Pre aktiviranja ove opcije trebate proveriti u vašem sistemu postojanje duplikata na postojećim stavkama. To možete uraditi pomoću skripta bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates. ';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        'Modul događaja za podešavanje statusa konfiguracionih stavki prilikom postavljanja veza.';
    $Self->{Translation}->{'ITSM config item overview.'} = 'Pregled ITSM konfiguracione stavke.';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = 'Modul za proveru grupe odgovorne za klasu.';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        'Modul za proveru grupe odgovorne za konfiguracionu stavku.';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = 'Modul za generisanje statistike ITSM konfiguracionih stavki.';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        'Registracija modula objekta za dodatak uvoz/izvoz.';
    $Self->{Translation}->{'Overview.'} = 'Pregled.';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        'Parametri za pozadinski prikaz kontrolne table liste konfiguracionih stavki firme klijenta u interfejsu operatera. "Limit" definiše podrazumevani broj prikazanih stavki. "Group" se koristi da ograniči pristup dodatku (npr. Group: admin;group1;group2;). "Default" određuje da li je dodatak podrazumevano aktiviran ili da je neophodno da ga korisnik manuelno aktivira. "CacheTTLLocal" je vreme u minutima za keširanje dodatka.';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        'Parametri za boje statusa primene na prikazu postavki u interfejsu operatera.';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        'Parametri za statuse primene na prikazu postavki u interfejsu operatera.';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parametri za atribute uzorka grupe za dozvole opšteg kataloga';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        'Parametri za stranice (na kojima su konfiguracione stavke prikazane).';
    $Self->{Translation}->{'Permission Group'} = 'Grupa pristupa';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        'Potrebne dozvole za upotrebu ekrana ITSM konfiguracione stavke u interfejsu operatera.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        'Potrebne dozvole za upotrebu ekrana pretrage ITSM konfiguracione stavke u interfejsu operatera.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        'Potrebne dozvole za upotrebu ekrana detalnog prikaza ITSM konfiguracione stavke u interfejsu operatera.';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        'Potrebne dozvole za upotrebu ekrana dodavanja ITSM konfiguracione stavke u interfejsu operatera.';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        'Potrebne dozvole za upotrebu ekrana izmene ITSM konfiguracione stavke u interfejsu operatera.';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        'Potrebne dozvole za upotrebu ekrana istorijata ITSM konfiguracione stavke u interfejsu operatera.';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        'Potrebne dozvole za upotrebu ekrana štampe ITSM konfiguracione stavke u interfejsu operatera.';
    $Self->{Translation}->{'Required privileges to delete config items.'} = 'Potrebna prava za brisanje konfiguracione stavke.';
    $Self->{Translation}->{'Search config items.'} = 'Pretraga konfiguracionih stavki.';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        'Bira modul operatora automatskog uvećanja konfiguracione stavke. Operator automatskog uvećanja povećava broj stavke, SystemID, ConfigItemClassID i korišćeni brojač. Format je "SystemID.ConfigItemClassID.brojač", npr 1205000004, 1205000005.';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        'Postavlja stanje incidenta konfiguracione stavke kada je tiket povezan sa stavkom.';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        'Određuje status primene na ekranu masovne akcije konfiguracione stavke u interfejsu operatera.';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        'Određuje status incidenta na ekranu masovne akcije konfiguracione stavke u interfejsu operatera.';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        'U meniju prikazuje vezu koja omogućava povezivanje konfiguracione stavke sa drugim objektom na detaljnom prikazu konfiguracije u interfejsu operatera.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        'U meniju prikazuje vezu za pristup istorijatu konfiguracione stavke na pregledu konfiguracione stavke u interfejsu operatera.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        'U meniju prikazuje vezu za pristup istorijatu konfiguracione stavke na detaljnom prikazu u interfejsu operatera.';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        'U meniju prikazuje vezu za brisanje konfiguracione stavke na njenom detaljnom prikazu u interfejsu operatera.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        'U meniju prikazuje vezu za umnožavanje konfiguracione stavke na pregledu konfiguracione stavke u interfejsu operatera.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        'U meniju prikazuje vezu za umnožavanje konfiguracione stavke na njenom detaljnom prikazu u interfejsu operatera.';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        'U meniju prikazuje vezu za izmenu konfiguracione stavke na njenom detaljnom prikazu u interfejsu operatera.';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        'U meniju prikazuje vezu za povratak na detaljni prikaz konfiguracione stavke  u interfejsu operatera.';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        'U meniju prikazuje vezu za štampu konfiguracione stavke na njenom detaljnom prikazu u interfejsu operatera.';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        'U meniju prikazuje vezu za detaljni prikaz konfiguracione stavke na pregledu konfiguracione stavke u interfejsu operatera.';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        'Prikazuje istorijat konfiguracione stavke (obrnut redosled) u interfejsu operatera.';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        'Identifikator za konfiguracionu stavku, npr. ConfigItem#, MyConfigItem#. Podrazumevano je ConfigItem#.';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = 'klasa';
    $Self->{Translation}->{'global'} = 'globalno';
    $Self->{Translation}->{'postproductive'} = 'post-produkcija';
    $Self->{Translation}->{'preproductive'} = 'pre-produkcija';
    $Self->{Translation}->{'productive'} = 'produkcija';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::sv_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Hantering av konfigurationsenheter';
    $Self->{Translation}->{'Change class definition'} = 'Ändra klassdefinition';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = 'Definition';
    $Self->{Translation}->{'Change'} = 'Ändra';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Incidentläge';
    $Self->{Translation}->{'Deployment State'} = 'Driftläge';
    $Self->{Translation}->{'Class'} = 'Klass';
    $Self->{Translation}->{'Deployment State Type'} = 'Driftlägestyper';
    $Self->{Translation}->{'Current Incident State'} = 'Nuvarande incidentläge';
    $Self->{Translation}->{'Current Incident State Type'} = 'Nuvarande typ av incidentläge';
    $Self->{Translation}->{'Last changed'} = 'Senast ändrad';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Konfigurationsenhet';
    $Self->{Translation}->{'Filter for Classes'} = 'Filtrera klasser';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Välj en klass från listan för att skapa en ny konfigurationsenhet';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = 'Driftläge';
    $Self->{Translation}->{'Incident state'} = 'Incidentläge';
    $Self->{Translation}->{'Link to another'} = 'Länka till annan';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'Namnet på denna konfigurationsenhet';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'Namnet används redan i konfigurationsenheter med förljande nummer: %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = 'Historikinnehåll';
    $Self->{Translation}->{'Createtime'} = 'Tidpunkt för skapande';
    $Self->{Translation}->{'Zoom view'} = 'Fokusvy';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Anpassa vy';
    $Self->{Translation}->{'Config Items per page'} = 'Konfigurationsenheter per sida';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Utför sökning';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Sök även i tidigare versioner?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Konfigurationsenhet';
    $Self->{Translation}->{'Configuration Item Information'} = 'Information om konfigurationsenhet';
    $Self->{Translation}->{'Current Deployment State'} = 'Nuvarande driftläge';
    $Self->{Translation}->{'Last changed by'} = 'Senast ändrad av';
    $Self->{Translation}->{'Show one version'} = 'Visa en version';
    $Self->{Translation}->{'Show all versions'} = 'Visa alla versioner';
    $Self->{Translation}->{'Version Incident State'} = 'Incidentläge för version';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = 'Versionsnummer';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Versionsinformation för konfigurationsenhet';
    $Self->{Translation}->{'Property'} = 'Egenskap';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = 'Konfigurationsenhet';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = 'Inget resultat!';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = 'Driftläge för konfigurationsenhet';
    $Self->{Translation}->{'The incident state of this config item'} = 'Incidentläge för konfigurationsenhet';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Mellan';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Max antal av ett element';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Tomma fält anger att tidigare värden ska användas.';
    $Self->{Translation}->{'Skipped'} = 'Hoppade över';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Modell';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Serienummer';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = 'RAM';
    $Self->{Translation}->{'Hard Disk'} = 'Hårddisk';
    $Self->{Translation}->{'Capacity'} = 'Kapacitet';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Nätverkskort';
    $Self->{Translation}->{'IP over DHCP'} = 'IP över DHCP';
    $Self->{Translation}->{'IP Address'} = 'IP-adress';
    $Self->{Translation}->{'Graphic Adapter'} = 'Grafikkort';
    $Self->{Translation}->{'Other Equipment'} = 'Annan utrustning';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Sista dag för garanti';
    $Self->{Translation}->{'Install Date'} = 'Installationsdatum';
    $Self->{Translation}->{'Phone 1'} = 'Telefon 1';
    $Self->{Translation}->{'Phone 2'} = 'Telefon 2';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'Nätverksadress';
    $Self->{Translation}->{'Subnet Mask'} = 'Subnätmask';
    $Self->{Translation}->{'Gateway'} = 'Gateway';
    $Self->{Translation}->{'Licence Type'} = 'Licenstyper';
    $Self->{Translation}->{'Licence Key'} = 'Licensnyckel';
    $Self->{Translation}->{'Quantity'} = 'Mängd';
    $Self->{Translation}->{'Expiration Date'} = 'Löper ut datum';
    $Self->{Translation}->{'Media'} = 'Media';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'Löpt ut';
    $Self->{Translation}->{'Maintenance'} = 'Underhåll';
    $Self->{Translation}->{'Pilot'} = 'Pilot';
    $Self->{Translation}->{'Planned'} = 'Planlagd';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Reperation';
    $Self->{Translation}->{'Retired'} = 'Pensionerad';
    $Self->{Translation}->{'Review'} = 'Översyn';
    $Self->{Translation}->{'Test/QA'} = 'Test/QA';
    $Self->{Translation}->{'Laptop'} = 'Laptop';
    $Self->{Translation}->{'Desktop'} = 'Desktop';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Server';
    $Self->{Translation}->{'Other'} = 'Annan';
    $Self->{Translation}->{'Monitor'} = 'Monitor';
    $Self->{Translation}->{'Printer'} = 'Skrivare';
    $Self->{Translation}->{'Switch'} = 'Switch';
    $Self->{Translation}->{'Router'} = 'Router';
    $Self->{Translation}->{'WLAN Access Point'} = 'WLAN Accesspunkt';
    $Self->{Translation}->{'Security Device'} = 'Säkerhetsutrustning';
    $Self->{Translation}->{'Backup Device'} = 'Backupenhet';
    $Self->{Translation}->{'Mouse'} = 'Mus';
    $Self->{Translation}->{'Keyboard'} = 'Tangentbord';
    $Self->{Translation}->{'Camera'} = 'Kamera';
    $Self->{Translation}->{'Beamer'} = 'Beamer';
    $Self->{Translation}->{'Modem'} = 'Modem';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA-kort';
    $Self->{Translation}->{'USB Device'} = 'USB-enhet';
    $Self->{Translation}->{'Docking Station'} = 'Dockningsstation';
    $Self->{Translation}->{'Scanner'} = 'Scanner';
    $Self->{Translation}->{'Building'} = 'Byggnad';
    $Self->{Translation}->{'Office'} = 'Kontor';
    $Self->{Translation}->{'Floor'} = 'Våning';
    $Self->{Translation}->{'Room'} = 'Rum';
    $Self->{Translation}->{'Rack'} = 'Rack';
    $Self->{Translation}->{'Workplace'} = 'Arbetsplats';
    $Self->{Translation}->{'Outlet'} = 'Väggkontakt';
    $Self->{Translation}->{'IT Facility'} = 'IT-byggnad';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Klientapplikation';
    $Self->{Translation}->{'Middleware'} = 'Middleware';
    $Self->{Translation}->{'Server Application'} = 'Serverapplikation';
    $Self->{Translation}->{'Client OS'} = 'Klient-OS';
    $Self->{Translation}->{'Server OS'} = 'Server-OS';
    $Self->{Translation}->{'Admin Tool'} = 'Administrationsverktyg';
    $Self->{Translation}->{'User Tool'} = 'Användarverktyg';
    $Self->{Translation}->{'Embedded'} = 'Embedded';
    $Self->{Translation}->{'Single Licence'} = 'Enstaka licens';
    $Self->{Translation}->{'Per User'} = 'Per användare';
    $Self->{Translation}->{'Per Processor'} = 'Per processor';
    $Self->{Translation}->{'Per Server'} = 'Per server';
    $Self->{Translation}->{'Per Node'} = 'Per nod';
    $Self->{Translation}->{'Volume Licence'} = 'Volymlicens';
    $Self->{Translation}->{'Enterprise Licence'} = 'Enterpriselicens';
    $Self->{Translation}->{'Developer Licence'} = 'Utvecklarlicens';
    $Self->{Translation}->{'Demo'} = 'Demo';
    $Self->{Translation}->{'Time Restricted'} = 'Tidsbegränsad';
    $Self->{Translation}->{'Freeware'} = 'Freeware';
    $Self->{Translation}->{'Open Source'} = 'Öppen källkod';
    $Self->{Translation}->{'Unlimited'} = 'Obegränsad';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = 'Konfigurationsenheter';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = 'Begränsning konfigurationsenheter';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Skapa och hantera definitioner för konfigurationsenheter';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        'Anger antal rader för KE-editorn i admingränssnittet.';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = 'Duplicera';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Parametrar för exempel på behörighetsegrupper i grundkatalogens attribut.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::sw_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'Usanidi wa usimamizi wa kipengele ';
    $Self->{Translation}->{'Change class definition'} = 'Badili ufafanuzi wa tabaka';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = 'Ufafanuzi';
    $Self->{Translation}->{'Change'} = 'Badilisha';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Hali ya tukio';
    $Self->{Translation}->{'Deployment State'} = 'Hali ya kutumia';
    $Self->{Translation}->{'Class'} = 'Tabaka';
    $Self->{Translation}->{'Deployment State Type'} = 'Aina ya hali ya kutumia';
    $Self->{Translation}->{'Current Incident State'} = 'Hali ya tukio la sasa';
    $Self->{Translation}->{'Current Incident State Type'} = 'Aina ya hali ya tukio la sasa';
    $Self->{Translation}->{'Last changed'} = 'Mwisho kubadilishwa';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Usanidi wa Kipengele ';
    $Self->{Translation}->{'Filter for Classes'} = 'Chuja kwa ajili ya matabaka';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'Chagua tabaka kutoka kwenye orodha ya kutengeneza kipengele cha usanidi kipya. ';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'Vitendo vya wingi vya kipengele cha usanidi cha ITSM';
    $Self->{Translation}->{'Deployment state'} = 'Hali ya kutumia';
    $Self->{Translation}->{'Incident state'} = 'Hali ya tukio';
    $Self->{Translation}->{'Link to another'} = 'Unganisha na nyingine';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = 'Namba ya kipengele cha usanidi n batili!';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'Namba ya kipengele kingine cha usanidi cha kuunganisha nacho.';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = 'Jina la hiki kipengele cha usanidi';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'Jina hili tayari limekwisha tumikana kipengele cha usanidi chenye namba i(z)fuatazo: %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = 'Maudhui ya historia';
    $Self->{Translation}->{'Createtime'} = 'Muda wa kutengeneza';
    $Self->{Translation}->{'Zoom view'} = 'Mandhari iliyokuzwa';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'Mipangilio ya muktadha';
    $Self->{Translation}->{'Config Items per page'} = 'Vipengele vya usanidi kwa kila ukurusa';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'Tafuta';
    $Self->{Translation}->{'Also search in previous versions?'} = 'Pia tafuta katika toleo lililopita?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'Kipengele cha Usanidi';
    $Self->{Translation}->{'Configuration Item Information'} = 'Taarifa za kipengele cha usanidi';
    $Self->{Translation}->{'Current Deployment State'} = 'Hali ya kutumia ya sasa';
    $Self->{Translation}->{'Last changed by'} = 'Mwisho kubadilishwa na';
    $Self->{Translation}->{'Show one version'} = 'Onyesha toleo moja';
    $Self->{Translation}->{'Show all versions'} = 'Onyesha matoleo yote';
    $Self->{Translation}->{'Version Incident State'} = 'Toleo la hali ya tukio';
    $Self->{Translation}->{'Version Deployment State'} = 'Hali ya toleo la kutumia';
    $Self->{Translation}->{'Version Number'} = 'Namba ya toleo';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'Toleo la kipengele cha usanidi kwa undani.';
    $Self->{Translation}->{'Property'} = 'Mali';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = 'Kipengele cha usanidi';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = 'Hali ya matumizi ya hii kipengele cha usanidi ';
    $Self->{Translation}->{'The incident state of this config item'} = 'Hali ya tukio ya hiki kipengele cha usanidi';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Katikati';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'Kima cha juu cha namba ya elementi mojawapo';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Uga zilizowazi zinaonyesha kwamba thamani za sasa zimewekwa.';
    $Self->{Translation}->{'Skipped'} = 'Rukwa';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Mfano';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Namba tambulishi';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = 'Ram';
    $Self->{Translation}->{'Hard Disk'} = 'Diski kuu';
    $Self->{Translation}->{'Capacity'} = 'Uwezo';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Adapta ya mtandao';
    $Self->{Translation}->{'IP over DHCP'} = 'IP kwa DHCP';
    $Self->{Translation}->{'IP Address'} = 'Anwani ya IP';
    $Self->{Translation}->{'Graphic Adapter'} = 'Adapta ya Mchoro';
    $Self->{Translation}->{'Other Equipment'} = 'Vifaa vingine';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Tarehe ya kuisha kw audhamini.';
    $Self->{Translation}->{'Install Date'} = 'Tarehe ya kusakinisha';
    $Self->{Translation}->{'Phone 1'} = 'Simu ya 1';
    $Self->{Translation}->{'Phone 2'} = 'Simu ya 2';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'Anwani ya mtandao';
    $Self->{Translation}->{'Subnet Mask'} = 'Subnet mask';
    $Self->{Translation}->{'Gateway'} = 'Kichanganishi mtandao';
    $Self->{Translation}->{'Licence Type'} = 'Aina ya leseni';
    $Self->{Translation}->{'Licence Key'} = 'Neno kuu la leseni';
    $Self->{Translation}->{'Quantity'} = 'Wingi';
    $Self->{Translation}->{'Expiration Date'} = 'Tarehe ya mwisho';
    $Self->{Translation}->{'Media'} = 'Midia';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = 'Mtandao';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'Imeisha';
    $Self->{Translation}->{'Maintenance'} = 'Matengenezo';
    $Self->{Translation}->{'Pilot'} = 'Kwanza';
    $Self->{Translation}->{'Planned'} = 'Iliyopangwa';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Rekebisha';
    $Self->{Translation}->{'Retired'} = 'Staafu';
    $Self->{Translation}->{'Review'} = 'Mapitio';
    $Self->{Translation}->{'Test/QA'} = 'Jaribio/ Maswali na Majibu';
    $Self->{Translation}->{'Laptop'} = 'Kompyuta ya mkononi';
    $Self->{Translation}->{'Desktop'} = 'Eneo kazi';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'Seva';
    $Self->{Translation}->{'Other'} = 'Engine';
    $Self->{Translation}->{'Monitor'} = 'Monita';
    $Self->{Translation}->{'Printer'} = 'Kichapishi';
    $Self->{Translation}->{'Switch'} = 'Swichi';
    $Self->{Translation}->{'Router'} = 'Kipanga njia';
    $Self->{Translation}->{'WLAN Access Point'} = 'Kiungo ufikivu cha WLAN';
    $Self->{Translation}->{'Security Device'} = 'Kifaa cha ulinzi';
    $Self->{Translation}->{'Backup Device'} = 'Kifaa cha chelezo';
    $Self->{Translation}->{'Mouse'} = 'Kipanya';
    $Self->{Translation}->{'Keyboard'} = 'Boabonye';
    $Self->{Translation}->{'Camera'} = 'Kamera';
    $Self->{Translation}->{'Beamer'} = 'Projector';
    $Self->{Translation}->{'Modem'} = 'Modemu';
    $Self->{Translation}->{'PCMCIA Card'} = 'Kadi ya PCMCIA';
    $Self->{Translation}->{'USB Device'} = 'Kifaa cha USB';
    $Self->{Translation}->{'Docking Station'} = 'Kituo egeshi';
    $Self->{Translation}->{'Scanner'} = 'Kitambazo';
    $Self->{Translation}->{'Building'} = 'Jengo';
    $Self->{Translation}->{'Office'} = 'Ofisi';
    $Self->{Translation}->{'Floor'} = 'Sakafu';
    $Self->{Translation}->{'Room'} = 'Chumba';
    $Self->{Translation}->{'Rack'} = 'Rack';
    $Self->{Translation}->{'Workplace'} = 'Eneo la kazi';
    $Self->{Translation}->{'Outlet'} = 'Sehemu ya kutokea';
    $Self->{Translation}->{'IT Facility'} = 'P';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Programu-tumizi a mteja';
    $Self->{Translation}->{'Middleware'} = 'Middleware';
    $Self->{Translation}->{'Server Application'} = 'Seva ya programu-tumizi';
    $Self->{Translation}->{'Client OS'} = 'OS ya mteja';
    $Self->{Translation}->{'Server OS'} = 'OS ya seva';
    $Self->{Translation}->{'Admin Tool'} = 'Kifaa cha kiongozi';
    $Self->{Translation}->{'User Tool'} = 'Kifaa cha mtumiaji';
    $Self->{Translation}->{'Embedded'} = 'Pakichwa';
    $Self->{Translation}->{'Single Licence'} = 'Leseni moja';
    $Self->{Translation}->{'Per User'} = 'Kwa mtumiaji';
    $Self->{Translation}->{'Per Processor'} = 'Kwa kichakato';
    $Self->{Translation}->{'Per Server'} = 'Kwa seva';
    $Self->{Translation}->{'Per Node'} = 'Kwa fundo';
    $Self->{Translation}->{'Volume Licence'} = 'Leseni ya sauti';
    $Self->{Translation}->{'Enterprise Licence'} = 'Leseni ya Enterprie';
    $Self->{Translation}->{'Developer Licence'} = 'Leseni ya mtengenezaji';
    $Self->{Translation}->{'Demo'} = 'Onyesho';
    $Self->{Translation}->{'Time Restricted'} = 'Muda umezuiliwa';
    $Self->{Translation}->{'Freeware'} = 'Programu ya bure';
    $Self->{Translation}->{'Open Source'} = 'Chanzo cha wazi';
    $Self->{Translation}->{'Unlimited'} = 'Haina kikomo';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Sawa';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        'Angalia majina ya kipekee tu katika tabaka sawa la kipengele cha usanidi (\'Tabaka\') au zote (\'Zote\'), ambayo inamaanisha kila kipengele cha usanidi kilichopo kimejumuishwa wakati nakala pacha zinaangaliwa.';
    $Self->{Translation}->{'Config Items'} = 'Vipengele vya usanidi';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        'Sanidi moduli ya tukio ya kipengele ambacho kinawezesha kuingia katika historia katika kiolesura cha wakala.';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = 'Kikomo cha kipengele cha usanidi';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        'Kipanga njia cha mazingira ya nyuma cha utafutaji wa kipengele cha usanidi cha kiolesura cha wakala. ';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'Tengeneza na simamia ufafanuzi wa vipengele vya usanidi.';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        'Inafafanua ruhusa zinazohitajika kutengeneza vipengele vya usanidi vya ITSM kwa kutumia kiolesura cha jeneriki.';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        'Inafafanua ruhusa zinazohitajika kupatavipengele vya usanidi vya ITSM kwa kutumia kiolesura cha jeneriki.';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        'Inafafanua ruhusa zinazohitajika kutafuta vipengele vya usanidi vya ITSM kwa kutumia kiolesura cha jeneriki.';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        'Inafafanua ruhusa zinazohitajika kusasisha vipengele vya usanidi vya ITSM kwa kutumia kiolesura cha jeneriki.';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        'Inafafanua moduli ya mapitio kuonyesha mandhari ndogo ya orodha ya vipengele vya usanidi.';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        'Inafafanua semi za mara kwa mara mojamoja kwa kila tabaka la kipengele ch usanidi kuangalia jina la kipengele cha usanidi na kuonyesha ujumbe wa makosa yanayohusiana.';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        'Inafafanua kipengele cha chaguo msingi cha tabaka la \'Kipengelecha Usanidi cha ITSM\'.';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        'Inafafanua namba ya safu mlalo kwa mhariri wa ufafanuzi wa CI katika kiolesura cha kiongozi';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        'Inawezesha upeo wa kutafuta kwa skerini ya Kipengele cha usanidi cha ITSM cha wakala.';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        'Inafafanua kikomo cha kutafuta kwa skrini ya Kutafuta Kipengele cha usanidi cha ITSM  cha Wakala.';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        'Inaelezea safu wima zote zilizoonyeshwa katika mapitio ya kipengele cha usanidi. Hili chaguo halina madhara katika nafasi ya safuwima. Kidokezo: Tabaka la safuwima linapatikana mara zote kama kichujio \'Vyote\' itachaguliwa.';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        'Inafafanua safu wima zilizoonyeshwa katika utafutaji wa kipengele cha usanidi. Chaguo hili halina madhara katika nafasi ya safuwima.';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Inafafanua safu wima za CI zilizoonyeshwa katika mapitio ya kipengele cha usanidi kutegemeana na tabaka la CI. Kila ingizo lazima liwe na kiambishi awali jina la tabaka na :: ( mfano Tarakilishi::)Kuna viumbi vichache vya CI ambavyo vipo kwa CI zote (mfano kwenye tabaka la tarakilishi: tarakilishi::jina, Tarakilishi::Hali yaCurDep, Tarakilishi::Muda wa kutengeneza). Kuonyesha kiumbi cha CI kimoja kimoja kama vilivyo fafanuliwa kwenye ufafanuzi wa CI, mpangilio ufuatao lazima utumike (mfano kwenye tabaka tarakilishi): Tarakilishi::diski kuu::1, Tarakilishi::Diski kuu ngumu::1Uwezo::1, Tarakilishi::Diski kuu ngumu::2, Tarakilishi::Diski kuu ngumu::2::Uwezo::1. Kama hakuna ingizo kwa tabaka la CI, safuwima chaguo msingi zinaonyeshwa kama zilivyofafanuliwa katika mpangilio
ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'Inafafanua safu wima za CI zilizoonyeshwa katika kutafuta kipengele cha usanidi kutegemeana na tabaka la CI. Kila ingizo lazima liwe na kiambishi awali jina la tabaka na :: ( mfano Tarakilishi::)Kuna viumbi vichache vya CI ambavyo vipo kwa CI zote (mfano kwenye tabaka la tarakilishi: tarakilishi::jina, Tarakilishi::Hali yaCurDep, Tarakilishi::Muda wa kutengeneza). Kuonyesha kiumbi cha CI kimoja kimoja kama vilivyo fafanuliwa kwenye ufafanuzi wa CI, mpangilio ufuatao lazima utumike (mfano kwenye tabaka tarakilishi): Tarakilishi::diski kuu::1, Tarakilishi::Diski kuu ngumu::1Uwezo::1, Tarakilishi::Diski kuu ngumu::2, Tarakilishi::Diski kuu ngumu::2::Uwezo::1. Kama hakuna ingizo kwa tabaka la CI, safuwima chaguo msingi zinaonyeshwa kama zilivyofafanuliwa katika mpangilio
ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        'Inafafanua safu wima za CI zilizoonyeshwa katika mandhari ya jedwali tata la kiungo,kutegemeana na tabaka la CI. Kila ingizo lazima liwe na kiambishi awali jina la tabaka na :: ( mfano Tarakilishi::).Kuna viumbi vichache vya CI ambavyo vipo kwa CI zote (mfano kwenye tabaka la tarakilishi: tarakilishi::jina, Tarakilishi::Hali yaCurDep, Tarakilishi::Muda wa kutengeneza). Kuonyesha kiumbi cha CI kimoja kimoja kama vilivyo fafanuliwa kwenye ufafanuzi wa CI, mpangilio ufuatao lazima utumike (mfano kwenye tabaka tarakilishi): Tarakilishi::diski kuu::1, Tarakilishi::Diski kuu ngumu::1Uwezo::1, Tarakilishi::Diski kuu ngumu::2, Tarakilishi::Diski kuu ngumu::2::Uwezo::1. Kama hakuna ingizo kwa tabaka la CI, safuwima chaguo msingi zinaonyeshwa.';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = 'Nakala pacha';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        'Inawezesha kipengele cha kitendo ch wingi cha kipengele cha usanidi kwa maingira ya mbele ya wakala kufanya kazi katika kipengele cha usanidi zaidi ya kimoja kwa muda.';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        'Inawezesha kipengele cha kitendo cha wingi cha kipengele cha usanidi kwa makundi yaliyoorodheshwa tu.';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        'Wezesha/Kutowezesha kitendeakazi cha kuangalia vipengele vya usanidi kwa majina ya pekee. Kabla ya kuwezesha chaguo hili unatakiwa kuangalia mfumo wako kwa vipengele vya usanidi kama vipo tayari kwa majina nakala pacha. Unaweza kufanya hivi na maandiko bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = 'Moduli ya kuangalia makundi yanayohusika na tabaka.';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        'Module ya kuangalia makundi yanayohusika na kipengele cha usanidi.';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = 'Moduli ya kutengeneza takwimu za kipengele cha usanidi cha ITSM>';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        'Usajili wa moduli ya mazingira ya nyuma ya kipengee kwa moduli ya Kuingiza/kuhamisha.';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        'Vigezo kwa ajili ya rangi ya hali za matumizi kama mandhari ya mapendeleo ya kiolesura cha wakala.';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        'Vigezo kwa ajili ya hali za matumizi kama mandhari ya mapendeleo ya kiolesura cha wakala.';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Vigezo kwa mfano ruhusa za vikundi kwa sifa za katalogi za ujumla.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        'Vigezo kwa ajili ya ukurasa (ambavyo vipengele vya usanidi vimeonyeshwa).';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        'Inahitaji ruhusa kutumia skrini ya kipengele cha usanidi cha ITSM katika kiolesura cha wakala.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        'Inahitaji ruhusa kutumia skrini ya kutafuta kipengele cha usanidi cha ITSM katika kiolesura cha wakala.';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        'Inahitaji ruhusa kutumia skrini iliyokuzwa ya kipengele cha usanidi cha ITSM katika kiolesura cha wakala.';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        'Inahitaji ruhusa kutumia skrini ya kipengele cha usanidi cha ITSM kilichoongeza katika kiolesura cha wakala.';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        'Inahitaji ruhusa kutumia skrini ya kipengele cha usanidi cha ITSM kilichohaririwa katika kiolesura cha wakala.';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        'Inahitaji ruhusa kutumia skrini ya kipengele cha usanidi cha ITSM cha historia katika kiolesura cha wakala.';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        'Inahitaji ruhusa kutumia skrini ya kipengele cha usanidi cha ITSM cha kuchapisha katika kiolesura cha wakala.';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        'Inachagua moduli ya kuzalisha namba ya kipengele cha usanidi. "OngezaOtomatiki" inaongeza namba ya kipengele ya usanidi, kitambulisho cha mfumo, kitambulisho cha tabaka la kipengele sanidi na kihesabuji vinatumika. Umbizo ni "Kitambulisho cha mfumo.kitambulisho cha tabaka la kipengele sanidi.Kihesabuji". mfano 1205000004, 1205000005.';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        'Inaweka hali ya kutumia katika skrini ya wingi wa kipengele cha usanidi ya kiolesura cha wakala.';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        'Inaweka hali ya tukio katika skrini ya kipengele cha usanidi ya kiolesura cha wakala.';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        'Inaonyesha kiongo katika menyu ambacho kinaunganisha kipengele cha usanidi na kipengee kingine katika mandhari iliyokuzwa ya kipengele cha usanidi ya kioleusura cha wakala.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        'Inaonyesha kiungo kwenye  menyu kuweza kufikia historia ya kipengele cha usanidi katika mapitio ya kipengele ya usanidi ya kiolesura cha wakala.';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        'Inaonyesha kiungo kwenye  menyu kuweza kufikia historia ya kipengele cha usanidi katika mandhari yake iliyokuzwa ya kiolesura cha wakala.';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        'Onyesha kiungo katika menyu kutoa nakala pacha ya kipengele cha usanidi katika mapitio ya kipengele cha usanidi ya kiolesura cha wakala.';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        'Onyesha kiungo katika menyu kutoa nakala pacha ya kipengele cha usanidi katika mandhari yake ya kipengele cha usanidi cha kiolesura cha wakala.';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        'Onyesha kiungo katika menyu kuhariri  kipengele cha usanidi katika mandhari yake iliyokuzwa ya kiolesura cha wakala.';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        'Onyesha kiungo katika menyu kuchapisha kipengele cha usanidi katika mandhari yake iliyokuzwa ya kiolesura cha wakala.';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        'Onyesha kiungo katika menyu kukuza katika kipengele cha usanidi katika mapitio ya kipengele cha usanidi ya kiolesura cha wakala.';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        'Inaonyesha historia ya kipengele cha usanidi (kwa mpangilio wa kurudi nyuma) katika kiolesura cha wakala.';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        'Kitambulishi cha kipengele cha usanidi, mfano.ConfigItem#, MyConfigItem#. Chagu-msingi ni ConfigItem#.';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::th_TH_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = 'การจัดการ Config Item';
    $Self->{Translation}->{'Change class definition'} = 'นิยามของคลาสการเปลี่ยนแปลง';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = 'นิยาม';
    $Self->{Translation}->{'Change'} = 'เปลี่ยน';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'สถานภาพของเหต์การณ์';
    $Self->{Translation}->{'Deployment State'} = 'สถานภาพการใช้งาน';
    $Self->{Translation}->{'Class'} = 'คลาส';
    $Self->{Translation}->{'Deployment State Type'} = 'ประเภทสถานภาพการใช้งาน';
    $Self->{Translation}->{'Current Incident State'} = 'สถานภาพของเหต์การณ์ปัจจุบัน';
    $Self->{Translation}->{'Current Incident State Type'} = 'ประเภทสถานภาพของเหต์การณ์ปัจจุบัน';
    $Self->{Translation}->{'Last changed'} = 'การเปลี่ยนแปลงล่าสุด';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = 'Config Item';
    $Self->{Translation}->{'Filter for Classes'} = 'ตัวกรองสำหรับคลาส';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = 'เลือกคลาสจากรายการที่จะสร้าง Config Item ใหม่';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'การทำงานเป็นกลุ่มของ ITSM ConfigItem';
    $Self->{Translation}->{'Deployment state'} = 'สถานภาพการใช้งาน';
    $Self->{Translation}->{'Incident state'} = 'สถานภาพของเหตุการณ์';
    $Self->{Translation}->{'Link to another'} = 'เชื่อมโยงไปที่อื่น';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = 'จำนวนรายการการกำหนดค่าไม่ถูกต้อง!';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = 'จำนวนของรายการการกำหนดค่าอื่นๆที่ต้องการเชื่อมโยง';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = ' ชื่อของรายการการตั้งค่านี้\'';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        'ชื่อที่ถูกการใช้งานโดย ConfigItems มีจำนวนดังต่อไปนี้ (s): %s';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = 'เนื้อหาประวัติ';
    $Self->{Translation}->{'Createtime'} = 'เวลาที่สร้าง';
    $Self->{Translation}->{'Zoom view'} = 'มุมมองการซูม';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = 'การตั้งค่าข้อความ';
    $Self->{Translation}->{'Config Items per page'} = 'Config Items ในแต่ละหน้า';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = 'ดำเนินการค้นหา';
    $Self->{Translation}->{'Also search in previous versions?'} = 'ค้นหาเวอร์ชั่นที่แล้วด้วยหรือไม่?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = 'รายการการกำหนดค่า';
    $Self->{Translation}->{'Configuration Item Information'} = 'ข้อมูลรายการการกำหนดค่า';
    $Self->{Translation}->{'Current Deployment State'} = 'สถานภาพการใช้งานปัจจุบัน';
    $Self->{Translation}->{'Last changed by'} = 'การเปลี่ยนแปลงล่าสุดโดย';
    $Self->{Translation}->{'Show one version'} = 'แสดงหนึ่งเวอร์ชั่น';
    $Self->{Translation}->{'Show all versions'} = 'แสดงเวอร์ชั่นทั้งหมด';
    $Self->{Translation}->{'Version Incident State'} = 'เวอร์ชั่นของสถานภาพของเหตุการณ์';
    $Self->{Translation}->{'Version Deployment State'} = 'เว่อร์ชั่นสถานภาพการใช้งาน';
    $Self->{Translation}->{'Version Number'} = 'หมายเลขเวอร์ชั้น';
    $Self->{Translation}->{'Configuration Item Version Details'} = 'เนื้อหาเวอร์ชั่น Configuration Item';
    $Self->{Translation}->{'Property'} = 'คุณสมบัติ';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = 'ConfigItem';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = 'ไม่มีผลลัพธ์!';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = 'สถานภาพการพัฒนาการของ config item นี้ ';
    $Self->{Translation}->{'The incident state of this config item'} = 'สถานภาพเหตุการณ์ของ config item นี';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'ระหว่าง';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = 'จำนวนสูงสุดของหนึ่งอุปกรณ์';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'เขตข้อมูลว่างเปล่าบ่งชี้ว่าค่าปัจจุบันจะถูกเก็บไว้';
    $Self->{Translation}->{'Skipped'} = 'ข้ามแล้ว';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'โมเดล';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'หมายเลขซีเรียล';
    $Self->{Translation}->{'CPU'} = 'ซีพียู';
    $Self->{Translation}->{'Ram'} = 'แรม';
    $Self->{Translation}->{'Hard Disk'} = 'ฮาร์ดดิสก์';
    $Self->{Translation}->{'Capacity'} = 'ความจุ';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'การ์ดเชื่อมต่อเครือข่าย';
    $Self->{Translation}->{'IP over DHCP'} = 'IP ผ่าน DHCP';
    $Self->{Translation}->{'IP Address'} = 'ที่อยู่ไอพี';
    $Self->{Translation}->{'Graphic Adapter'} = 'ตัวปรับต่อกราฟฟิค';
    $Self->{Translation}->{'Other Equipment'} = 'อุปกรณ์อื่น ๆ';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'การรับประกันวันหมดอายุ';
    $Self->{Translation}->{'Install Date'} = 'วันติดตั้ง';
    $Self->{Translation}->{'Phone 1'} = 'โทรศัพท์ 1';
    $Self->{Translation}->{'Phone 2'} = 'โทรศัพท์ 2';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'ที่อยู่เครือข่าย';
    $Self->{Translation}->{'Subnet Mask'} = 'เครือข่ายย่อยมาส์ก';
    $Self->{Translation}->{'Gateway'} = 'ประตูสัญญาณ';
    $Self->{Translation}->{'Licence Type'} = 'ประเภทลายเซนต์';
    $Self->{Translation}->{'Licence Key'} = 'กุญแจลายเซนต์';
    $Self->{Translation}->{'Quantity'} = 'จำนวน';
    $Self->{Translation}->{'Expiration Date'} = 'วันหมดอายุ';
    $Self->{Translation}->{'Media'} = 'มีเดีย';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'หมดอายุ';
    $Self->{Translation}->{'Maintenance'} = 'การดูแลรักษา';
    $Self->{Translation}->{'Pilot'} = 'การนำร่อง';
    $Self->{Translation}->{'Planned'} = 'การวางแผน';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'ซ่อม';
    $Self->{Translation}->{'Retired'} = 'เกษียณ';
    $Self->{Translation}->{'Review'} = 'ตัวอย่าง';
    $Self->{Translation}->{'Test/QA'} = 'ทดสอบ/QA';
    $Self->{Translation}->{'Laptop'} = 'แล็ปท็อป';
    $Self->{Translation}->{'Desktop'} = 'เดสทอป';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = 'เซิร์ฟเวอร์';
    $Self->{Translation}->{'Other'} = 'อื่นๆ';
    $Self->{Translation}->{'Monitor'} = 'จอภาพ';
    $Self->{Translation}->{'Printer'} = 'เครื่องพิมพ์';
    $Self->{Translation}->{'Switch'} = 'สับเปลี่ยน';
    $Self->{Translation}->{'Router'} = 'เราเตอร์';
    $Self->{Translation}->{'WLAN Access Point'} = 'จุดเชื่อมต่อWLAN';
    $Self->{Translation}->{'Security Device'} = 'อุปกรณ์รักษาความปลอดภัย';
    $Self->{Translation}->{'Backup Device'} = 'อุปกรณ์สำรองข้อมูล';
    $Self->{Translation}->{'Mouse'} = 'เม้าส์';
    $Self->{Translation}->{'Keyboard'} = 'แป้นพิมพ์';
    $Self->{Translation}->{'Camera'} = 'กล้อง';
    $Self->{Translation}->{'Beamer'} = 'ผู้คาน';
    $Self->{Translation}->{'Modem'} = 'โมเดม';
    $Self->{Translation}->{'PCMCIA Card'} = 'บัตร PCMCIA';
    $Self->{Translation}->{'USB Device'} = 'อุปกรณ์ USB';
    $Self->{Translation}->{'Docking Station'} = 'สถานี Docking';
    $Self->{Translation}->{'Scanner'} = 'เครื่องสแกน';
    $Self->{Translation}->{'Building'} = 'การสร้าง';
    $Self->{Translation}->{'Office'} = 'ออฟฟิศ';
    $Self->{Translation}->{'Floor'} = 'ชั้น';
    $Self->{Translation}->{'Room'} = 'ห้อง';
    $Self->{Translation}->{'Rack'} = 'จำพวก';
    $Self->{Translation}->{'Workplace'} = 'ที่ทำงาน';
    $Self->{Translation}->{'Outlet'} = 'เอาท์เล็ท';
    $Self->{Translation}->{'IT Facility'} = 'สิ่งอำนวยความสะดวกด้านไอที';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'WLAN';
    $Self->{Translation}->{'Telco'} = 'Telco';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'แอพลิเคชันของลูกค้า';
    $Self->{Translation}->{'Middleware'} = 'มิดเดิ้ลแวย์';
    $Self->{Translation}->{'Server Application'} = 'แอพลิเคชันเซิร์ฟเวอร์';
    $Self->{Translation}->{'Client OS'} = 'OS ของลูกค้า';
    $Self->{Translation}->{'Server OS'} = 'OS เซิร์ฟเวอร์';
    $Self->{Translation}->{'Admin Tool'} = 'เครื่องมือแอดมิน';
    $Self->{Translation}->{'User Tool'} = 'เครื่องมือของผู้ใช้';
    $Self->{Translation}->{'Embedded'} = 'การฝังตัว';
    $Self->{Translation}->{'Single Licence'} = 'ลายเซนต์เดียว';
    $Self->{Translation}->{'Per User'} = 'ต่อผู้ใช้';
    $Self->{Translation}->{'Per Processor'} = 'สำหรับตัวประมวลผล';
    $Self->{Translation}->{'Per Server'} = 'ต่อเซิร์ฟเวอร์';
    $Self->{Translation}->{'Per Node'} = 'ต่อหนึ่งโหนด';
    $Self->{Translation}->{'Volume Licence'} = 'ลายเซนต์อุปกรณ์เสียง';
    $Self->{Translation}->{'Enterprise Licence'} = 'ลายเวนต์เอ็นเตอร์ไพรส์';
    $Self->{Translation}->{'Developer Licence'} = 'ลายเซนต์ผู้พัฒนา';
    $Self->{Translation}->{'Demo'} = 'การสาธิต';
    $Self->{Translation}->{'Time Restricted'} = 'เวลา จำกัด';
    $Self->{Translation}->{'Freeware'} = 'ฟรีแวร์';
    $Self->{Translation}->{'Open Source'} = 'โอเพนซอร์ส';
    $Self->{Translation}->{'Unlimited'} = 'ไม่จำกัด';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'โอเค';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        'ตรวจสอบชื่อเฉพาะเท่านั้นที่อยู่ใน ConfigItem ชั้นเดียวกัน (ชั้น) หรือทั้งหมด (ทั้งหมด) ซึ่งหมายความว่าConfigItem ที่มีอยู่ทั้งหมดจะถูกนำเข้าบัญชีเพื่อค้นหารายการที่ซ้ำกัน';
    $Self->{Translation}->{'Config Items'} = 'Config Items';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        'โมดูลกิจกรรมของConfig item ที่ช่วยให้เข้าสู่ระบบประวัติในอินเตอร์เฟสของเอเย่นต์';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = 'การจำกัด Configuration Item';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        'Configuration Item ค้นหา backend router ของอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = 'สร้างและจัดการคำนิยามสำหรับ Configuration Items';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        'กำหนดสิทธิ์ที่จำเป็นในการสร้างรายการการตั้งค่า ITSMโดยใช้อินเตอร์เฟซทั่วไป';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        'กำหนดสิทธิ์ที่จำเป็นในการเอารายการการตั้งค่า ITSMโดยใช้อินเตอร์เฟซทั่วไป';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        'กำหนดสิทธิ์ที่จำเป็นในการค้นหารายการการตั้งค่า ITSMโดยใช้อินเตอร์เฟซทั่วไป';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        'กำหนดสิทธิ์ที่จำเป็นในการอัพเดตรายการการตั้งค่า ITSMโดยใช้อินเตอร์เฟซทั่วไป';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        'กำหนดโมดูลภาพรวมในการแสดงมุมมองขนาดเล็กของรายการ  Configuration Items';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        'กำหนดนิพจน์ทั่วไปเป็นรายบุคคลในConfigItem แต่ละชั้น เพื่อตรวจสอบชื่อ ConfigItem และเพื่อแสดงข้อความผิดพลาดที่สอดคล้องกัน';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        'กำหนดอบเจกต์ย่อยเริ่มต้นของคลาส \'ITSMConfigItem\'';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        'กำหนดจำนวนแถวสำหรับตัวแก้ไขการนิยาม CI ในอินเตอร์เฟสของผู้ดูแลระบบ';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        'กำหนดขีดจำกัดของการค้นหาสำหรับหน้าจอ AgentITSMConfigItem';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        'กำหนดขีดจำกัดของการค้นหาสำหรับหน้าจอ AgentITSMConfigItemSearch';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        'กำหนดคอลัมน์ที่จะแสดงในภาพรวมของconfig item ตัวเลือกนี้จะไม่มีผลต่อตำแหน่งของคอลัมน์ หมายเหตุ: ระดับของคอลัมน์สามารถใช้ได้เสมอถ้าตัวกรอง \'ทั้งหมด\' ถูกเลือก';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        'กำหนดคอลัมน์ที่จะแสดงในการค้นหา config item ตัวเลือกนี้จะไม่มีผลต่อตำแหน่งของคอลัมน์ ';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'กำหนดคอลัมน์ที่แสดง CIs ในภาพรวมของรายการการตั้งค่าซึ่งขึ้นอยู่กับคลาส CI รายการการป้อนแต่ละรายการจะต้องนำหน้าด้วยชื่อคลาสและเครื่องหมายทวิภาคคู่ (เช่นคอมพิวเตอร์: :) มีแอตทริบิวต์ CI เพียงไม่กี่แอตทริบิวต์ ที่ใช้กันทั่วไปใน CIs ทั้งหมด (ตัวอย่างสำหรับคลาสคอมพิวเตอร์: คอมพิวเตอร์::ชื่อ, คอมพิวเตอร์::CurDeplState, คอมพิวเตอร์::CreateTime) ในการแสดงแอตทริบิวต์ CI ตามที่กำหนดไว้ในคำนิยาม CI จะต้องใช้รูปแบบต่อไปนี้(ตัวอย่างสำหรับคลาสคอมพิวเตอร์): คอมพิวเตอร์::ฮาร์ดดิสก์::1, คอมพิวเตอร์::ฮาร์ดดิสก์::1::ความจุ::1, คอมพิวเตอร์::ฮาร์ดดิสก์::2, คอมพิวเตอร์::ฮาร์ดดิสก์::2::ความจุ::1  หากไม่มีการป้อนคลาสที่ CI แล้วคอลัมน์เริ่มต้นจะแสดงตามที่กำหนดไว้ในการตั้งค่า ITSMConfigItem ::Frontend :: AgentITSMConfigItem ### ShowColumns';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        'กำหนดคอลัมน์ที่แสดง CIs ในการค้นหาของรายการการตั้งค่าซึ่งขึ้นอยู่กับคลาส CI รายการการป้อนแต่ละรายการจะต้องนำหน้าด้วยชื่อคลาสและเครื่องหมายทวิภาคคู่ (เช่นคอมพิวเตอร์: :) มีแอตทริบิวต์ CI เพียงไม่กี่แอตทริบิวต์ ที่ใช้กันทั่วไปใน CIs ทั้งหมด (ตัวอย่างสำหรับคลาสคอมพิวเตอร์: คอมพิวเตอร์::ชื่อ, คอมพิวเตอร์::CurDeplState, คอมพิวเตอร์::CreateTime) ในการแสดงแอตทริบิวต์ CI ตามที่กำหนดไว้ในคำนิยาม CI จะต้องใช้รูปแบบต่อไปนี้(ตัวอย่างสำหรับคลาสคอมพิวเตอร์): คอมพิวเตอร์::ฮาร์ดดิสก์::1, คอมพิวเตอร์::ฮาร์ดดิสก์::1::ความจุ::1, คอมพิวเตอร์::ฮาร์ดดิสก์::2, คอมพิวเตอร์::ฮาร์ดดิสก์::2::ความจุ::1 หากไม่มีการป้อนคลาสที่ CI แล้วคอลัมน์เริ่มต้นจะแสดงตามที่กำหนดไว้ในการตั้งค่า ITSMConfigItem ::Frontend :: AgentITSMConfigItem ### ShowColumns';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        'กำหนดคอลัมน์ที่แสดง CIs ในตารางการเชื่อมโยงมุมมองที่ซับซ้อนซึ่งขึ้นอยู่กับคลาส CI รายการการป้อนแต่ละรายการจะต้องนำหน้าด้วยชื่อคลาสและเครื่องหมายทวิภาคคู่ (เช่นคอมพิวเตอร์: :) มีแอตทริบิวต์ CI เพียงไม่กี่แอตทริบิวต์ ที่ใช้กันทั่วไปใน CIs ทั้งหมด (ตัวอย่างสำหรับคลาสคอมพิวเตอร์: คอมพิวเตอร์::ชื่อ, คอมพิวเตอร์::CurDeplState, คอมพิวเตอร์::CreateTime) ในการแสดงแอตทริบิวต์ CI ตามที่กำหนดไว้ในคำนิยาม CI จะต้องใช้รูปแบบต่อไปนี้(ตัวอย่างสำหรับคลาสคอมพิวเตอร์): คอมพิวเตอร์::ฮาร์ดดิสก์::1, คอมพิวเตอร์::ฮาร์ดดิสก์::1::ความจุ::1, คอมพิวเตอร์::ฮาร์ดดิสก์::2, คอมพิวเตอร์::ฮาร์ดดิสก์::2::ความจุ::1 หากไม่มีการป้อนคลาสที่ CI แล้วคอลัมน์เริ่มต้นจะแสดงตามที่กำหนดไว้ในการตั้งค่า ITSMConfigItem ::Frontend :: AgentITSMConfigItem ### ShowColumns';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = 'ซ้ำ';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        'เปิดใช้งานการตั้งค่ารายการคุณลักษณะของการทำงานเป็นกลุ่ม สำหรับฟรอนต์เอนของเอเย่นต์เพื่อทำงานกับรายการการกำหนดค่าที่มีมากกว่าหนึ่งในหนึ่งครั้ง';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        'เปิดใช้งานการตั้งค่ารายการคุณลักษณะของการทำงานเป็นกลุ่มเฉพาะสำหรับกลุ่มที่ระบุไว้';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        'เปิด / ปิดการใช้งานฟังก์ชั่นเพื่อตรวจสอบ ConfigItems สำหรับชื่อที่ไม่ซ้ำกัน ก่อนเปิดใช้งานตัวเลือกนี้คุณควรตรวจสอบระบบของคุณสำหรับรายการการตั้งค่าที่มีอยู่แล้วด้วยชื่อที่ซ้ำกัน คุณสามารถทำเช่นนี้ด้วยสคริปต์ดังต่อไปนี้ bin / otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = 'โมดูลในการตรวจสอบผู้รับผิดชอบกลุ่มสำหรับคลาส';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        'โมดูลในการตรวจสอบผู้รับผิดชอบกลุ่มสำหรับ configuration item';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = 'โมดูลสร้างสถิติ ITSM config item';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        'การลงทะเบียนโมดูล Object backend สำหรับโมดูลนำเข้าหรือส่งออก';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        'พารามิเตอร์สำหรับการพัฒนาสถานภาพของสีในมุมมองการตั้งค่าของอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        'พารามิเตอร์สำหรับการพัฒนาสถานภาพในมุมมองการตั้งค่าของอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'พารามิเตอร์สำหรับตัวอย่างกลุ่มที่ได้รับอนุญาตของแอตทริบิวต์แค็ตตาล็อกทั่วไป';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        'พารามิเตอร์สำหรับแต่ละหน้า(ซึ่ง configuration items แสดง)';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        'จำเป็นต้องมีการอนุญาติในการใช้งานในหน้าจอรายการการตั้งค่า ITSM ในอินเตอร์เฟซของเอเย่นต์';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        'จำเป็นต้องมีการอนุญาติในการใช้งานในหน้าจอการค้นหารายการการตั้งค่า ITSM ในอินเตอร์เฟซของเอเย่นต์';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        'จำเป็นต้องมีการอนุญาติในการใช้งานในหน้าจอการซูมรายการการตั้งค่า ITSM ในอินเตอร์เฟซของเอเย่นต์';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        'จำเป็นต้องมีการอนุญาติในการใช้งานการเพิ่มในหน้าจอรายการการตั้งค่า ITSM ในอินเตอร์เฟซของเอเย่นต์';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        'จำเป็นต้องมีการอนุญาติในการใช้งานการแก้ไขในหน้าจอรายการการตั้งค่า ITSM ในอินเตอร์เฟซของเอเย่นต์';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        'จำเป็นต้องมีการอนุญาติในการใช้งานประวัติในหน้าจอรายการการตั้งค่า ITSM ในอินเตอร์เฟซของเอเย่นต์';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        'จำเป็นต้องมีการอนุญาติในการใช้งานการพิมพ์ในหน้าจอรายการการตั้งค่า ITSM ในอินเตอร์เฟซของเอเย่นต์';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        'เลือกหมายเลขรายการตั้งค่าตัวกำเนิดโมดูล "AutoIncrement" จะเพิ่มหมายเลขรายการกำหนดค่า SystemID ConfigItemClassID และตัวนับที่มีการใช้ รูปแบบคือ "SystemID.ConfigItemClassID.Counter" เช่น 1205000004, 1205000005';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        'ตั้งค่าสถานะการใช้งานในหน้าจอการตั้งค่ารายการเป็นกลุ่มของอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        'ตั้งค่าสถานะของเหตุการณ์ในหน้าจอการตั้งค่ารายการเป็นกลุ่มของอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        'แสดงลิงค์ในเมนูเพื่อลิงค์รายการการตั้งค่ากับออบเจกค์อื่นในการซูมในอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        'แสดงลิงค์ในเมนูในการที่จะเข้าชมประวัติของ configuration item ในมุมมองทั่วไปของconfiguration itemในอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        'แสดงลิงค์ในเมนูในการที่จะเข้าชมประวัติของ configuration item ในมุมมองการซูมของอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        'แสดงลิงค์ในเมนูในการที่จะจำลอง configuration item ในมุมมองทั่วไปของconfiguration item ในอินเตอร์เฟซเอเย่นต์ ';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        'แสดงลิงค์ในเมนูในการที่จะจำลองconfiguration item ในมุมมองการซูมของอินเตอร์เฟซเอเย่นต์ ';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        'แสดงลิงค์ในเมนูในการที่จะแก้ไข configuration item ในมุมมองการซูมของอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        'แสดงลิงค์ในเมนูในการที่จะพิมพ์ configuration item ในมุมมองการซูมของอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        'แสดงลิงค์ในเมนูเพื่อซูมรายการการตั้งค่าในมุมมองภาพรวมของรายการการตั้งค่าในอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        'แสดงประวัติ config item (ลำดับย้อนกลับ) ในอินเตอร์เฟซเอเย่นต์';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        'ตัวบ่งชี้สำหรับ  Configuration Item เช่น ConfigItem#, MyConfigItem# ค่าเริ่มต้นคือ ConfigItem#';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::tr_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = 'Yakınlaştırma görünümü';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Arasında';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = '';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = '';
    $Self->{Translation}->{'Office'} = 'Ofis';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = '';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Genel katalog özniteliklerinin örnek izin gruplarının parametreleri.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::uk_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = '';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = 'Збільшений вид';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = ' Між';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '';
    $Self->{Translation}->{'Skipped'} = '';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '';
    $Self->{Translation}->{'CPU'} = '';
    $Self->{Translation}->{'Ram'} = '';
    $Self->{Translation}->{'Hard Disk'} = '';
    $Self->{Translation}->{'Capacity'} = '';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '';
    $Self->{Translation}->{'IP over DHCP'} = '';
    $Self->{Translation}->{'IP Address'} = '';
    $Self->{Translation}->{'Graphic Adapter'} = '';
    $Self->{Translation}->{'Other Equipment'} = '';
    $Self->{Translation}->{'Warranty Expiration Date'} = '';
    $Self->{Translation}->{'Install Date'} = '';
    $Self->{Translation}->{'Phone 1'} = '';
    $Self->{Translation}->{'Phone 2'} = '';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '';
    $Self->{Translation}->{'Subnet Mask'} = '';
    $Self->{Translation}->{'Gateway'} = '';
    $Self->{Translation}->{'Licence Type'} = '';
    $Self->{Translation}->{'Licence Key'} = '';
    $Self->{Translation}->{'Quantity'} = '';
    $Self->{Translation}->{'Expiration Date'} = '';
    $Self->{Translation}->{'Media'} = '';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '';
    $Self->{Translation}->{'Maintenance'} = '';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = '';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '';
    $Self->{Translation}->{'Retired'} = '';
    $Self->{Translation}->{'Review'} = '';
    $Self->{Translation}->{'Test/QA'} = '';
    $Self->{Translation}->{'Laptop'} = '';
    $Self->{Translation}->{'Desktop'} = '';
    $Self->{Translation}->{'PDA'} = '';
    $Self->{Translation}->{'Server'} = '';
    $Self->{Translation}->{'Other'} = '';
    $Self->{Translation}->{'Monitor'} = '';
    $Self->{Translation}->{'Printer'} = '';
    $Self->{Translation}->{'Switch'} = '';
    $Self->{Translation}->{'Router'} = '';
    $Self->{Translation}->{'WLAN Access Point'} = '';
    $Self->{Translation}->{'Security Device'} = '';
    $Self->{Translation}->{'Backup Device'} = 'Пристрій резервного копіювання';
    $Self->{Translation}->{'Mouse'} = '';
    $Self->{Translation}->{'Keyboard'} = '';
    $Self->{Translation}->{'Camera'} = '';
    $Self->{Translation}->{'Beamer'} = 'Beamer';
    $Self->{Translation}->{'Modem'} = '';
    $Self->{Translation}->{'PCMCIA Card'} = '';
    $Self->{Translation}->{'USB Device'} = '';
    $Self->{Translation}->{'Docking Station'} = '';
    $Self->{Translation}->{'Scanner'} = '';
    $Self->{Translation}->{'Building'} = 'будинок';
    $Self->{Translation}->{'Office'} = 'Офіс';
    $Self->{Translation}->{'Floor'} = '';
    $Self->{Translation}->{'Room'} = '';
    $Self->{Translation}->{'Rack'} = '';
    $Self->{Translation}->{'Workplace'} = '';
    $Self->{Translation}->{'Outlet'} = '';
    $Self->{Translation}->{'IT Facility'} = '';
    $Self->{Translation}->{'LAN'} = '';
    $Self->{Translation}->{'WLAN'} = '';
    $Self->{Translation}->{'Telco'} = '';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '';
    $Self->{Translation}->{'Middleware'} = '';
    $Self->{Translation}->{'Server Application'} = '';
    $Self->{Translation}->{'Client OS'} = '';
    $Self->{Translation}->{'Server OS'} = '';
    $Self->{Translation}->{'Admin Tool'} = 'Адмін-інструмент';
    $Self->{Translation}->{'User Tool'} = '';
    $Self->{Translation}->{'Embedded'} = '';
    $Self->{Translation}->{'Single Licence'} = '';
    $Self->{Translation}->{'Per User'} = '';
    $Self->{Translation}->{'Per Processor'} = '';
    $Self->{Translation}->{'Per Server'} = '';
    $Self->{Translation}->{'Per Node'} = '';
    $Self->{Translation}->{'Volume Licence'} = '';
    $Self->{Translation}->{'Enterprise Licence'} = '';
    $Self->{Translation}->{'Developer Licence'} = '';
    $Self->{Translation}->{'Demo'} = '';
    $Self->{Translation}->{'Time Restricted'} = '';
    $Self->{Translation}->{'Freeware'} = '';
    $Self->{Translation}->{'Open Source'} = '';
    $Self->{Translation}->{'Unlimited'} = '';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Параметри для прикладу дозволу груп загальних атрибутів каталогу.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::vi_VN_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '';
    $Self->{Translation}->{'Change class definition'} = '';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '';
    $Self->{Translation}->{'Change'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = 'Trạng thái sự cố';
    $Self->{Translation}->{'Deployment State'} = '';
    $Self->{Translation}->{'Class'} = '';
    $Self->{Translation}->{'Deployment State Type'} = '';
    $Self->{Translation}->{'Current Incident State'} = 'Trạng thái sự cố hiện thời';
    $Self->{Translation}->{'Current Incident State Type'} = '';
    $Self->{Translation}->{'Last changed'} = 'Thay đổi lần cuối';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '';
    $Self->{Translation}->{'Filter for Classes'} = '';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '';
    $Self->{Translation}->{'Createtime'} = '';
    $Self->{Translation}->{'Zoom view'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '';
    $Self->{Translation}->{'Config Items per page'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '';
    $Self->{Translation}->{'Also search in previous versions?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '';
    $Self->{Translation}->{'Configuration Item Information'} = '';
    $Self->{Translation}->{'Current Deployment State'} = '';
    $Self->{Translation}->{'Last changed by'} = '';
    $Self->{Translation}->{'Show one version'} = '';
    $Self->{Translation}->{'Show all versions'} = '';
    $Self->{Translation}->{'Version Incident State'} = '';
    $Self->{Translation}->{'Version Deployment State'} = '';
    $Self->{Translation}->{'Version Number'} = '';
    $Self->{Translation}->{'Configuration Item Version Details'} = '';
    $Self->{Translation}->{'Property'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '';
    $Self->{Translation}->{'Config Item Search Results'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = 'Trạng thái triển khai của đối tượng này';
    $Self->{Translation}->{'The incident state of this config item'} = '';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = 'Giữa';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = 'Trường rỗng nghĩa là giá trị hiện tại không đổi';
    $Self->{Translation}->{'Skipped'} = 'Bị bỏ qua';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = 'Chủng loại';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = 'Số seri';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = 'Bộ nhớ';
    $Self->{Translation}->{'Hard Disk'} = 'Ổ cứng';
    $Self->{Translation}->{'Capacity'} = 'Dung lượng';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = 'Card mạng';
    $Self->{Translation}->{'IP over DHCP'} = 'Cấp IP qua DHCP';
    $Self->{Translation}->{'IP Address'} = 'Địa chỉ IP';
    $Self->{Translation}->{'Graphic Adapter'} = 'Card đồ hoạ';
    $Self->{Translation}->{'Other Equipment'} = 'Thiết bị khác';
    $Self->{Translation}->{'Warranty Expiration Date'} = 'Ngày hết hạn bảo hành';
    $Self->{Translation}->{'Install Date'} = 'Ngày cài đặt';
    $Self->{Translation}->{'Phone 1'} = 'Điện thoại 1';
    $Self->{Translation}->{'Phone 2'} = 'Điện thoại 2';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = 'Địa chỉ mạng';
    $Self->{Translation}->{'Subnet Mask'} = 'Mặt nạ lớp mạng';
    $Self->{Translation}->{'Gateway'} = 'Cổng truy cập';
    $Self->{Translation}->{'Licence Type'} = 'Loại giấy phép';
    $Self->{Translation}->{'Licence Key'} = 'Khoá bản quyền';
    $Self->{Translation}->{'Quantity'} = 'Số lượng';
    $Self->{Translation}->{'Expiration Date'} = 'Ngày hết hạn';
    $Self->{Translation}->{'Media'} = 'Phương tiện';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = 'Đã hết hạn';
    $Self->{Translation}->{'Maintenance'} = 'Bảo trì';
    $Self->{Translation}->{'Pilot'} = '';
    $Self->{Translation}->{'Planned'} = 'Theo kế hoạch';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = 'Sửa chữa';
    $Self->{Translation}->{'Retired'} = 'Không sử dụng';
    $Self->{Translation}->{'Review'} = 'Thẩm định';
    $Self->{Translation}->{'Test/QA'} = 'Kiểm tra/Hỏi đáp';
    $Self->{Translation}->{'Laptop'} = 'Máy xách tay';
    $Self->{Translation}->{'Desktop'} = 'Máy để bàn';
    $Self->{Translation}->{'PDA'} = 'Thiết bị cầm tay';
    $Self->{Translation}->{'Server'} = 'Máy chủ';
    $Self->{Translation}->{'Other'} = 'Khác';
    $Self->{Translation}->{'Monitor'} = 'Màn hình';
    $Self->{Translation}->{'Printer'} = 'Máy in';
    $Self->{Translation}->{'Switch'} = 'Bộ chuyển mạch';
    $Self->{Translation}->{'Router'} = 'Bộ định tuyến';
    $Self->{Translation}->{'WLAN Access Point'} = 'Điểm truy cập không dây';
    $Self->{Translation}->{'Security Device'} = 'Thiết bị bảo mật';
    $Self->{Translation}->{'Backup Device'} = 'Thiết bị lưu trữ';
    $Self->{Translation}->{'Mouse'} = 'Chuột';
    $Self->{Translation}->{'Keyboard'} = 'Bàn phím';
    $Self->{Translation}->{'Camera'} = 'Camera';
    $Self->{Translation}->{'Beamer'} = '';
    $Self->{Translation}->{'Modem'} = 'Bộ điều biến';
    $Self->{Translation}->{'PCMCIA Card'} = 'Card PCMCIA';
    $Self->{Translation}->{'USB Device'} = 'Thiết bị USB';
    $Self->{Translation}->{'Docking Station'} = 'Máy trạm cố định';
    $Self->{Translation}->{'Scanner'} = 'Máy quets';
    $Self->{Translation}->{'Building'} = 'Toà nhà';
    $Self->{Translation}->{'Office'} = 'Văn phòng';
    $Self->{Translation}->{'Floor'} = 'Tầng';
    $Self->{Translation}->{'Room'} = 'Phòng';
    $Self->{Translation}->{'Rack'} = 'Tủ Rack';
    $Self->{Translation}->{'Workplace'} = 'Nơi làm việc';
    $Self->{Translation}->{'Outlet'} = 'Ổ cắm';
    $Self->{Translation}->{'IT Facility'} = 'Cơ sở CNTT';
    $Self->{Translation}->{'LAN'} = 'LAN';
    $Self->{Translation}->{'WLAN'} = 'Mạng không dây';
    $Self->{Translation}->{'Telco'} = 'Viễn thông';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = 'Ứng dụng khách';
    $Self->{Translation}->{'Middleware'} = 'Phần mềm trung gian';
    $Self->{Translation}->{'Server Application'} = 'Ứng dụng máy chủ';
    $Self->{Translation}->{'Client OS'} = 'HĐH khách';
    $Self->{Translation}->{'Server OS'} = 'HĐH máy chủ';
    $Self->{Translation}->{'Admin Tool'} = 'Công cụ quản lý';
    $Self->{Translation}->{'User Tool'} = 'Công cụ người dùng';
    $Self->{Translation}->{'Embedded'} = 'Nhúng';
    $Self->{Translation}->{'Single Licence'} = 'Giấy phép đơn';
    $Self->{Translation}->{'Per User'} = 'Trên người dùng';
    $Self->{Translation}->{'Per Processor'} = 'Trên bộ xử lý';
    $Self->{Translation}->{'Per Server'} = 'Trên máy chủ';
    $Self->{Translation}->{'Per Node'} = 'Trên node';
    $Self->{Translation}->{'Volume Licence'} = 'Cấp phép số lượng lớn';
    $Self->{Translation}->{'Enterprise Licence'} = 'Giấy phép doanh nghiệp';
    $Self->{Translation}->{'Developer Licence'} = 'Giấy phép phát triển';
    $Self->{Translation}->{'Demo'} = 'Chạy thử';
    $Self->{Translation}->{'Time Restricted'} = 'Giới hạn thời gian';
    $Self->{Translation}->{'Freeware'} = 'Phần mềm miễn phí';
    $Self->{Translation}->{'Open Source'} = 'Mã nguồn mở';
    $Self->{Translation}->{'Unlimited'} = 'Không giới hạn';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = 'Nhân đôi';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        'Tham số cho nhóm cấp phép của thuộc tính danh mục chung.';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::zh_CN_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '管理配置项';
    $Self->{Translation}->{'Change class definition'} = '修改类定义';
    $Self->{Translation}->{'Config Item Class'} = '配置项类';
    $Self->{Translation}->{'Definition'} = '定义';
    $Self->{Translation}->{'Change'} = '变更';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '故障状态';
    $Self->{Translation}->{'Deployment State'} = '部署状态';
    $Self->{Translation}->{'Class'} = '类';
    $Self->{Translation}->{'Deployment State Type'} = '部署状态类型';
    $Self->{Translation}->{'Current Incident State'} = '当前故障状态';
    $Self->{Translation}->{'Current Incident State Type'} = '当前的故障状态类型';
    $Self->{Translation}->{'Last changed'} = '最后修改';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '配置项';
    $Self->{Translation}->{'Filter for Classes'} = '类过滤器';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '先从列表中选择类，然后创建新的配置项。';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = 'ITSM 配置项 批量操作';
    $Self->{Translation}->{'Deployment state'} = '部署状态';
    $Self->{Translation}->{'Incident state'} = '事件状态';
    $Self->{Translation}->{'Link to another'} = '链接到其他';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '无效的配置项编号！';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '将要链接的另外一个配置项编号。';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '你真的想要删除这个配置项吗?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '为这个配置项命名';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '编号为(s):%s的配置项已经使用了此名字';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '配置项：%s 的历史信息';
    $Self->{Translation}->{'History Content'} = '历史值';
    $Self->{Translation}->{'Createtime'} = '创建时间';
    $Self->{Translation}->{'Zoom view'} = '详情视图';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '上下文设置';
    $Self->{Translation}->{'Config Items per page'} = '每页配置项个数';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '一个通用的ITSM配置项表格';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '搜索';
    $Self->{Translation}->{'Also search in previous versions?'} = '同时搜索以前的版本吗?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '配置项';
    $Self->{Translation}->{'Configuration Item Information'} = '配置项信息';
    $Self->{Translation}->{'Current Deployment State'} = '当前的部署状态';
    $Self->{Translation}->{'Last changed by'} = '上次修改人';
    $Self->{Translation}->{'Show one version'} = '显示一个版本';
    $Self->{Translation}->{'Show all versions'} = '显示所有版本';
    $Self->{Translation}->{'Version Incident State'} = '版本故障状态';
    $Self->{Translation}->{'Version Deployment State'} = '版本部署状态';
    $Self->{Translation}->{'Version Number'} = '版本号';
    $Self->{Translation}->{'Configuration Item Version Details'} = '配置项版本详情';
    $Self->{Translation}->{'Property'} = '属性';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '没有类的访问权限！';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = 'ITSM配置项概览';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '没有指定配置项ID！';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '您至少需要一个选定的配置项！';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '你没有此配置项的写权限：%s。';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '在数据库中找不到配置项“%s”！';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '无法删除配置项ID %s！';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '找不到配置项ID%s的版本！';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '没有指定配置项ID、DuplicateID或ClassID！';
    $Self->{Translation}->{'No access is given!'} = '没有访问权限！';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '没有类 %s的定义！';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '不能显示历史信息，因为没有指定配置项ID！';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '无法显示历史记录，因为没有访问权限！';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '新建配置项（ID=%s）';
    $Self->{Translation}->{'New version (ID=%s)'} = '新建版本（ID=%s）';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '部署状态已更新（新=%s，旧=%s）';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '故障状态已更新（新=%s，旧=%s）';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '配置项（ID=%s）已删除';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '到%s链接的（类型=%s）已添加';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '到%s链接的（类型=%s）已删除';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '配置项定义已更新（ID=%s）';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '名称已更新（新=%s，旧=%s）';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '属性%s已从“%s”更新为“%s”';
    $Self->{Translation}->{'Version %s deleted'} = '版本%s 已删除';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '没有指定配置项ID或版本ID！';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '无法显示配置项，因为没有访问权限！';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '在数据库中找不到配置项ID %s！';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '在数据库中找不到版本ID %s！';
    $Self->{Translation}->{'ConfigItem'} = '配置项';
    $Self->{Translation}->{'printed by %s at %s'} = '打印人：%s ，打印日期：%s';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '无效的类ID！';
    $Self->{Translation}->{'No ClassID is given!'} = '没有指定类ID！';
    $Self->{Translation}->{'No access rights for this class given!'} = '没有这个指定类的访问权限！';
    $Self->{Translation}->{'No Result!'} = '无结果！';
    $Self->{Translation}->{'Config Item Search Results'} = '配置项搜索结果';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '无法显示配置项，因为没有给定配置项的访问权限！';
    $Self->{Translation}->{'operational'} = '正常';
    $Self->{Translation}->{'warning'} = '警告';
    $Self->{Translation}->{'incident'} = '故障';
    $Self->{Translation}->{'The deployment state of this config item'} = '配置项部署状态';
    $Self->{Translation}->{'The incident state of this config item'} = '配置项故障状态';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '已显示的配置项';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = '时间区间';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '单个元素的最大数量';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '空字段表示保持当前值';
    $Self->{Translation}->{'Skipped'} = '跳过的';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '型号';
    $Self->{Translation}->{'Customer Company'} = '客户单位';
    $Self->{Translation}->{'Serial Number'} = '序列号';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = '内存';
    $Self->{Translation}->{'Hard Disk'} = '硬盘';
    $Self->{Translation}->{'Capacity'} = '容量';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '网卡';
    $Self->{Translation}->{'IP over DHCP'} = '从DHCP获取IP';
    $Self->{Translation}->{'IP Address'} = 'IP地址';
    $Self->{Translation}->{'Graphic Adapter'} = '显卡';
    $Self->{Translation}->{'Other Equipment'} = '其它设备';
    $Self->{Translation}->{'Warranty Expiration Date'} = '保修过期日期';
    $Self->{Translation}->{'Install Date'} = '安装日期';
    $Self->{Translation}->{'Phone 1'} = '电话1';
    $Self->{Translation}->{'Phone 2'} = '电话2';
    $Self->{Translation}->{'E-Mail'} = 'E-Mail';
    $Self->{Translation}->{'Network Address'} = '网络地址';
    $Self->{Translation}->{'Subnet Mask'} = '子网掩码';
    $Self->{Translation}->{'Gateway'} = '网关';
    $Self->{Translation}->{'Licence Type'} = '许可类别';
    $Self->{Translation}->{'Licence Key'} = '许可密钥';
    $Self->{Translation}->{'Quantity'} = '数量';
    $Self->{Translation}->{'Expiration Date'} = '过期日期';
    $Self->{Translation}->{'Media'} = '介质';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '计算机';
    $Self->{Translation}->{'Hardware'} = '硬件';
    $Self->{Translation}->{'Network'} = '网络';
    $Self->{Translation}->{'Software'} = '软件';
    $Self->{Translation}->{'Expired'} = '已过期';
    $Self->{Translation}->{'Maintenance'} = '维护';
    $Self->{Translation}->{'Pilot'} = '试验';
    $Self->{Translation}->{'Planned'} = '已计划';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '修复';
    $Self->{Translation}->{'Retired'} = '报废';
    $Self->{Translation}->{'Review'} = '复审';
    $Self->{Translation}->{'Test/QA'} = '测试/品质保证';
    $Self->{Translation}->{'Laptop'} = '笔记本';
    $Self->{Translation}->{'Desktop'} = '台式电脑';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = '服务器';
    $Self->{Translation}->{'Other'} = '其它';
    $Self->{Translation}->{'Monitor'} = '显示器';
    $Self->{Translation}->{'Printer'} = '打印机';
    $Self->{Translation}->{'Switch'} = '交换机';
    $Self->{Translation}->{'Router'} = '路由器';
    $Self->{Translation}->{'WLAN Access Point'} = 'WLAN 无线访问点';
    $Self->{Translation}->{'Security Device'} = '安全设备';
    $Self->{Translation}->{'Backup Device'} = '备份设备';
    $Self->{Translation}->{'Mouse'} = '鼠标';
    $Self->{Translation}->{'Keyboard'} = '键盘';
    $Self->{Translation}->{'Camera'} = '照相机';
    $Self->{Translation}->{'Beamer'} = '投影仪';
    $Self->{Translation}->{'Modem'} = '调解调器';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA卡';
    $Self->{Translation}->{'USB Device'} = 'USB 设备';
    $Self->{Translation}->{'Docking Station'} = '坞站';
    $Self->{Translation}->{'Scanner'} = '扫描仪';
    $Self->{Translation}->{'Building'} = '大厦';
    $Self->{Translation}->{'Office'} = '办公室';
    $Self->{Translation}->{'Floor'} = '楼层';
    $Self->{Translation}->{'Room'} = '房间';
    $Self->{Translation}->{'Rack'} = '机架';
    $Self->{Translation}->{'Workplace'} = '工作间';
    $Self->{Translation}->{'Outlet'} = '插座';
    $Self->{Translation}->{'IT Facility'} = 'IT设施';
    $Self->{Translation}->{'LAN'} = '局域网';
    $Self->{Translation}->{'WLAN'} = '无线网络';
    $Self->{Translation}->{'Telco'} = '电话公司';
    $Self->{Translation}->{'GSM'} = 'GSM';
    $Self->{Translation}->{'Client Application'} = '客户端应用程序';
    $Self->{Translation}->{'Middleware'} = '中间件';
    $Self->{Translation}->{'Server Application'} = '服务器应用程序';
    $Self->{Translation}->{'Client OS'} = '客户操作系统';
    $Self->{Translation}->{'Server OS'} = '服务器操作系统';
    $Self->{Translation}->{'Admin Tool'} = '管理工具';
    $Self->{Translation}->{'User Tool'} = '用户工具';
    $Self->{Translation}->{'Embedded'} = '嵌入式';
    $Self->{Translation}->{'Single Licence'} = '单一许可';
    $Self->{Translation}->{'Per User'} = '每个用户';
    $Self->{Translation}->{'Per Processor'} = '每个处理器';
    $Self->{Translation}->{'Per Server'} = '每个服务器';
    $Self->{Translation}->{'Per Node'} = '每个节点';
    $Self->{Translation}->{'Volume Licence'} = '批量授权';
    $Self->{Translation}->{'Enterprise Licence'} = '企业许可';
    $Self->{Translation}->{'Developer Licence'} = '开发许可';
    $Self->{Translation}->{'Demo'} = '演示';
    $Self->{Translation}->{'Time Restricted'} = '时间限制';
    $Self->{Translation}->{'Freeware'} = '免费软件';
    $Self->{Translation}->{'Open Source'} = '开源';
    $Self->{Translation}->{'Unlimited'} = '无限制的';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = '确定';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '已分配的配置项';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '已分配给客户单位的配置项';
    $Self->{Translation}->{'CIs assigned to customer user'} = '已分配给客户用户的配置项';
    $Self->{Translation}->{'CMDB Settings'} = 'CMDB设置';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '检查是否为唯一名称的范围是仅在\'class\'（配置项类）内还是\'global\'(全局)，就是在查找重复配置项名称时的范围。';
    $Self->{Translation}->{'Config Items'} = '配置项';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '添加配置项。';
    $Self->{Translation}->{'Config item edit.'} = '编辑配置项。';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '服务人员界面用于记录配置项事件到历史的模块。';
    $Self->{Translation}->{'Config item history.'} = '配置项历史。';
    $Self->{Translation}->{'Config item print.'} = '打印配置项。';
    $Self->{Translation}->{'Config item zoom.'} = '配置项详情。';
    $Self->{Translation}->{'ConfigItemNumber'} = 'ConfigItemNumber（配置项编号）';
    $Self->{Translation}->{'Configuration Item Limit'} = '配置项限制';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '每个页面上的配置项限制。';
    $Self->{Translation}->{'Configuration Management Database.'} = '配置管理数据库。';
    $Self->{Translation}->{'Configuration item bulk module.'} = '配置项批处理模块。';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '服务人员界面的配置项搜索后端路由。';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '创建和管理配置项定义';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '定义链接对象小部件(LinkObject::ViewMode = \"complex\")设置按钮中的操作。请注意，这些操作必须已经在以下JS和CSS文件中注册：Core.AllocationList.css、Core.UI.AllocationList.js、 Core.UI.Table.Sort.js、Core.Agent.TableFilters.js和Core.Agent.LinkObject.js。';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '定义使用通用接口创建ITSM配置项所需的权限。';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '定义使用通用接口删除ITSM配置项所需的权限。';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '定义使用通用接口获取ITSM配置项所需的权限。';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '定义使用通用接口搜索ITSM配置项所需的权限。';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '定义使用通用接口更新ITSM配置项所需的权限。';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '定义显示配置项列表简洁视图的概览模块。';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '为每个配置项类定义独立的正则表达式，以检查配置项名称并显示相应的错误消息。';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '定义\'ITSMConfigItem\'类的默认子对象。';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '定义在系统管理面板中修改配置项定义的编辑器的行数。';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '定义故障状态从高（如紧急）到低（如正常）的顺序。';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '定义关联的部署状态，即链接的工单可以影响配置项状态的情况。';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '定义AgentITSMConfigItem（服务人员配置项）窗口的搜索限制';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '定义AgentITSMConfigItemSearch（服务人员配置项搜索）窗口的搜索限制';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '定义配置项概览显示的字段。这个选项不会影响字段的位置。注意：如果选择“全部”，则“Class（类）”字段总是会显示。';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '定义配置项搜索中显示的字段。这个选项不会影响字段的位置。';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '定义配置项概览视图中按配置项类分类显示的字段。每个条目均需以类名为前缀再加上双冒号（如计算机::）。有一些配置项属性是所有配置项都有的（例如，对于计算机类：计算机::名称、计算机::CurDeplState当前部署状态、计算机::CreateTime创建时间）。要显示在配置项定义中定义的独特的配置项属性，必须使用下面的方案（例如：对于计算机类：计算机::硬盘::1、计算机::硬盘::容量::1、计算机::硬盘::2、计算机::硬盘::2计算机::硬盘::2）。如果一个配置项类没有条目存在，则使用在ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns中设置的默认字段。';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '定义按配置项类分类的配置项搜索结果中显示的字段。每个条目均需以类名为前缀再加上双冒号（如计算机::）。有一些配置项属性是所有配置项都有的（例如，对于计算机类：计算机::名称、计算机::CurDeplState当前部署状态、计算机::CreateTime创建时间）。要显示在配置项定义中定义的独特的配置项属性，必须使用下面的方案（例如：对于计算机类：计算机::硬盘::1、计算机::硬盘::容量::1、计算机::硬盘::2、计算机::硬盘::2计算机::硬盘::2）。如果一个配置项类没有条目存在，则使用在ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns中设置的默认字段。';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '定义配置项小部件中按配置项类分类显示的字段。每个条目均需以类名为前缀再加上双冒号（如计算机::）。有一些配置项属性是所有配置项都有的（例如，对于计算机类：Computer::Name、Computer::CurDeplState当前部署状态、Computer::CreateTime创建时间）。要显示在配置项定义中定义的独特的配置项属性，必须使用下面的方案（例如：对于计算机类：Computer::HardDisk::1【计算机::硬盘::1】、Computer::HardDisk::1::Capacity::1【计算机::硬盘::容量::1】、 Computer::HardDisk::2【计算机::硬盘::2】、Computer::HardDisk::2::Capacity::1【计算机::硬盘::2::容量::1】）。如果一个配置项类没有条目存在，则使用在AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany 中设置的默认字段(DefaultColumns键)。';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '在所有配置项类的链接表复杂视图中定义要显示的配置项字段。 如果没有条目，则显示默认字段。';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '定义链接表复合视图中按配置项类分类显示的字段。每个条目均需以类名为前缀再加上双冒号（如计算机::）。有一些配置项属性是所有配置项都有的（例如，对于计算机类：计算机::名称、计算机::CurDeplState当前部署状态、计算机::CreateTime创建时间）。要显示在配置项定义中定义的独特的配置项属性，必须使用下面的方案（例如：对于计算机类：计算机::硬盘::1、计算机::硬盘::容量::1、计算机::硬盘::2、计算机::硬盘::2计算机::硬盘::2）。如果一个配置项类没有条目存在，则使用在ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns中设置的默认字段。';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '定义哪一类(从工单角度命名的)链接可以影响一个链接的配置项的状态。';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '定义哪一类工单可以影响一个链接的配置项的状态。';
    $Self->{Translation}->{'Delete Configuration Item'} = '删除配置项';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '部署状态的颜色';
    $Self->{Translation}->{'Duplicate'} = '复制';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '在服务人员界面启用配置项批量操作功能，可以一次处理多个配置项。';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '只有列表中的组才有权限使用配置项批量操作功能。';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '启用/禁用检查配置项唯一名称功能。在启用这个选项前，您应该检查系统中是否已经存在重名的配置项，您可以使用这个脚本来检查重名情况：bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates。';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '在工单-配置项的链接中设置配置项-状态的事件模块。';
    $Self->{Translation}->{'ITSM config item overview.'} = 'ITSM配置项概览。';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '用于检查配置项类的组权限的模块。';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '用于检查配置项的组权限的模块。';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '用于生成ITSM配置项统计的模块';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '用于导入/导出模块的对象后端模块注册。';
    $Self->{Translation}->{'Overview.'} = '概览。';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '服务人员界面客户客户单位配置项概览的仪表板后端的参数。"Limit（限制）" 是默认的显示条目数，“Group”（组）用于到本插件的访问权限限制（如 Group:admin;group1;group2）。“Default（默认）”代表这个插件是默认启用还是需要用户手动启用。“CacheTTL”是本插件的缓存过期时间（单位：分钟）。';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '服务人员界面偏好设置视图中用于部署状态颜色的参数。';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '服务人员界面偏好设置视图中用于部署状态的参数。';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '目录属性样例-权限组的参数。';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '显示配置项的页面参数。';
    $Self->{Translation}->{'Permission Group'} = '权限组';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '服务人员界面使用ITSM配置项窗口需要的权限。';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '服务人员界面使用ITSM配置项搜索窗口需要的权限。';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '服务人员界面使用ITSM配置项详情窗口需要的权限。';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '服务人员界面使用添加ITSM配置项窗口需要的权限。';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '服务人员界面使用编辑ITSM配置项窗口需要的权限。';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '服务人员界面使用ITSM配置项历史窗口需要的权限。';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '服务人员界面使用ITSM配置项打印窗口需要的权限。';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '删除配置项所需的权限。';
    $Self->{Translation}->{'Search config items.'} = '搜索配置项。';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '选择配置项号码生成器模块。“AutoIncrement自动增量”使用系统ID、配置项类ID和计数器来增加配置项号，格式为：系统ID.配置项类ID.计数器，如1205000004、1205000005。';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '当一个工单链接到一个配置项时自动设置这个配置项的故障状态。';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '在服务人员界面配置项批量操作窗口设置部署状态。';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '在服务人员界面配置项批量操作窗口设置事件状态。';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '在服务人员界面配置项详情视图中，在菜单中显示一个允许链接配置项到另一对象的链接。';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '在服务人员界面配置项概览视图中，在菜单中显示一个允许访问配置项历史的链接。';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '在服务人员界面配置项详情视图中，在菜单中显示一个允许访问配置项历史的链接。';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '在服务人员界面配置项详情视图中，在菜单中显示一个删除配置项的链接。';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '在服务人员界面配置项概览视图中，在菜单中显示一个复制配置项的链接。';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '在服务人员界面配置项详情视图中，在菜单中显示一个复制配置项的链接。';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '在服务人员界面配置项详情视图中，在菜单中显示一个编辑配置项的链接。';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '在服务人员界面配置项详情视图中，在菜单中显示一个返回链接。';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '在服务人员界面配置项详情视图中，在菜单中显示一个打印配置项的链接。';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '在服务人员界面配置项概览视图中，在菜单中显示一个查看配置项详情的链接。';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '在服务人员界面显示配置项历史（倒序）。';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '配置项标识符，如ConfigItem#、MyConfigItem#，默认是ConfigItem#。';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '类';
    $Self->{Translation}->{'global'} = '全局';
    $Self->{Translation}->{'postproductive'} = '闲置';
    $Self->{Translation}->{'preproductive'} = '准备投产';
    $Self->{Translation}->{'productive'} = '投产';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Language::zh_TW_ITSMConfigurationManagement;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem.tt
    $Self->{Translation}->{'This field is required and the value must be alphanumeric.'} = '';
    $Self->{Translation}->{'Must be unique and only accepts alphanumeric characters.'} = '';
    $Self->{Translation}->{'This is the order in which this field will be shown on the screens where it is active.'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminDynamicFieldConfigItem/Config.tt
    $Self->{Translation}->{'Config item class'} = '';
    $Self->{Translation}->{'The selected config item class is invalid or does not exist.'} =
        '';
    $Self->{Translation}->{'Config item deployment states'} = '';
    $Self->{Translation}->{'Config item link type'} = '';
    $Self->{Translation}->{'Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.'} =
        '';
    $Self->{Translation}->{'Config item link source'} = '';
    $Self->{Translation}->{'Config item link removal'} = '';
    $Self->{Translation}->{'Activate this option to remove links between the ticket and config items that are removed from this field.'} =
        '';
    $Self->{Translation}->{'Config item key'} = '';
    $Self->{Translation}->{'These dynamic fields will be filled with values of the same selected config item(s).'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AdminITSMConfigItem.tt
    $Self->{Translation}->{'Config Item Management'} = '管理配置項';
    $Self->{Translation}->{'Change class definition'} = '修改類定義';
    $Self->{Translation}->{'Config Item Class'} = '';
    $Self->{Translation}->{'Definition'} = '定義';
    $Self->{Translation}->{'Change'} = '修改';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentDashboardITSMConfigItemGeneric.tt
    $Self->{Translation}->{'Incident State'} = '故障狀態';
    $Self->{Translation}->{'Deployment State'} = '部署狀態';
    $Self->{Translation}->{'Class'} = '類別';
    $Self->{Translation}->{'Deployment State Type'} = '部署狀態類型';
    $Self->{Translation}->{'Current Incident State'} = '當前的故障狀態';
    $Self->{Translation}->{'Current Incident State Type'} = '當前的故障狀態類型';
    $Self->{Translation}->{'Last changed'} = '最後修改';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemAdd.tt
    $Self->{Translation}->{'Config Item'} = '配置項';
    $Self->{Translation}->{'Filter for Classes'} = '過濾類';
    $Self->{Translation}->{'Select a Class from the list to create a new Config Item.'} = '先從列表中選擇類別，然後建立新的配置項。';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemBulk.tt
    $Self->{Translation}->{'ITSM ConfigItem Bulk Action'} = '';
    $Self->{Translation}->{'Deployment state'} = '';
    $Self->{Translation}->{'Incident state'} = '';
    $Self->{Translation}->{'Link to another'} = '';
    $Self->{Translation}->{'Invalid Configuration Item number!'} = '';
    $Self->{Translation}->{'The number of another Configuration Item to link with.'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemCustomerCIsWidget.tt
    $Self->{Translation}->{'Customer related config items'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemDelete.tt
    $Self->{Translation}->{'Do you really want to delete this config item?'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemEdit.tt
    $Self->{Translation}->{'The name of this config item'} = '為這個配置項命名';
    $Self->{Translation}->{'Name is already in use by the ConfigItems with the following Number(s): %s'} =
        '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemHistory.tt
    $Self->{Translation}->{'History of Config Item: %s'} = '';
    $Self->{Translation}->{'History Content'} = '歷史内容';
    $Self->{Translation}->{'Createtime'} = '創建時間';
    $Self->{Translation}->{'Zoom view'} = '缩放視圖';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewNavBar.tt
    $Self->{Translation}->{'Context Settings'} = '上下文設置';
    $Self->{Translation}->{'Config Items per page'} = '每頁配置項個數';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemOverviewSmall.tt
    $Self->{Translation}->{'A generic ITSM Configuration Item table'} = '';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemSearch.tt
    $Self->{Translation}->{'Run Search'} = '搜索';
    $Self->{Translation}->{'Also search in previous versions?'} = '同時搜索以前的版本?';

    # TT Template: Kernel/Output/HTML/Templates/Standard/AgentITSMConfigItemZoom.tt
    $Self->{Translation}->{'Configuration Item'} = '配置項';
    $Self->{Translation}->{'Configuration Item Information'} = '配置項信息';
    $Self->{Translation}->{'Current Deployment State'} = '當前的部署狀態';
    $Self->{Translation}->{'Last changed by'} = '最後修改於';
    $Self->{Translation}->{'Show one version'} = '顯示一個版本';
    $Self->{Translation}->{'Show all versions'} = '顯示所有版本';
    $Self->{Translation}->{'Version Incident State'} = '版本故障狀態';
    $Self->{Translation}->{'Version Deployment State'} = '版本部屬狀態';
    $Self->{Translation}->{'Version Number'} = '版本號碼';
    $Self->{Translation}->{'Configuration Item Version Details'} = '配置項版本詳情';
    $Self->{Translation}->{'Property'} = '屬性';

    # Perl Module: Kernel/Modules/AgentITSMConfigItem.pm
    $Self->{Translation}->{'No access to Class is given!'} = '';
    $Self->{Translation}->{'Overview: ITSM ConfigItem'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemBulk.pm
    $Self->{Translation}->{'No ConfigItemID is given!'} = '';
    $Self->{Translation}->{'You need at least one selected Configuration Item!'} = '';
    $Self->{Translation}->{'You don\'t have write access to this configuration item: %s.'} =
        '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemDelete.pm
    $Self->{Translation}->{'Config item "%s" not found in database!'} = '';
    $Self->{Translation}->{'Was not able to delete the configitem ID %s!'} = '';
    $Self->{Translation}->{'No version found for ConfigItemID %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemEdit.pm
    $Self->{Translation}->{'No ConfigItemID, DuplicateID or ClassID is given!'} = '';
    $Self->{Translation}->{'No access is given!'} = '';
    $Self->{Translation}->{'No definition was defined for class %s!'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemHistory.pm
    $Self->{Translation}->{'Can\'t show history, no ConfigItemID is given!'} = '';
    $Self->{Translation}->{'Can\'t show history, no access rights given!'} = '';
    $Self->{Translation}->{'New ConfigItem (ID=%s)'} = '';
    $Self->{Translation}->{'New version (ID=%s)'} = '';
    $Self->{Translation}->{'Deployment state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Incident state updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'ConfigItem (ID=%s) deleted'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) added'} = '';
    $Self->{Translation}->{'Link to %s (type=%s) deleted'} = '';
    $Self->{Translation}->{'ConfigItem definition updated (ID=%s)'} = '';
    $Self->{Translation}->{'Name updated (new=%s, old=%s)'} = '';
    $Self->{Translation}->{'Attribute %s updated from "%s" to "%s"'} = '';
    $Self->{Translation}->{'Version %s deleted'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemPrint.pm
    $Self->{Translation}->{'No ConfigItemID or VersionID is given!'} = '';
    $Self->{Translation}->{'Can\'t show config item, no access rights given!'} = '';
    $Self->{Translation}->{'ConfigItemID %s not found in database!'} = '';
    $Self->{Translation}->{'VersionID %s not found in database!'} = '';
    $Self->{Translation}->{'ConfigItem'} = '配置項';
    $Self->{Translation}->{'printed by %s at %s'} = '';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemSearch.pm
    $Self->{Translation}->{'Invalid ClassID!'} = '';
    $Self->{Translation}->{'No ClassID is given!'} = '';
    $Self->{Translation}->{'No access rights for this class given!'} = '';
    $Self->{Translation}->{'No Result!'} = '沒有結果!';
    $Self->{Translation}->{'Config Item Search Results'} = '配置項目搜尋結果';

    # Perl Module: Kernel/Modules/AgentITSMConfigItemZoom.pm
    $Self->{Translation}->{'Can\'t show item, no access rights for ConfigItem are given!'} =
        '';
    $Self->{Translation}->{'operational'} = '';
    $Self->{Translation}->{'warning'} = '';
    $Self->{Translation}->{'incident'} = '';
    $Self->{Translation}->{'The deployment state of this config item'} = '配置項部署狀態';
    $Self->{Translation}->{'The incident state of this config item'} = '配置項故障狀態';

    # Perl Module: Kernel/Output/HTML/Dashboard/ITSMConfigItemGeneric.pm
    $Self->{Translation}->{'Shown config items'} = '';

    # Perl Module: Kernel/Output/HTML/ITSMConfigItem/LayoutDate.pm
    $Self->{Translation}->{'Between'} = '';

    # Perl Module: Kernel/System/ImportExport/ObjectBackend/ITSMConfigItem.pm
    $Self->{Translation}->{'Maximum number of one element'} = '此元素的最大數量';
    $Self->{Translation}->{'Empty fields indicate that the current values are kept'} = '空字段表示當前值被保存';
    $Self->{Translation}->{'Skipped'} = '跳過的';

    # Perl Module: var/packagesetup/ITSMConfigurationManagement.pm
    $Self->{Translation}->{'Model'} = '型號';
    $Self->{Translation}->{'Customer Company'} = '';
    $Self->{Translation}->{'Serial Number'} = '序列號';
    $Self->{Translation}->{'CPU'} = 'CPU';
    $Self->{Translation}->{'Ram'} = '内存';
    $Self->{Translation}->{'Hard Disk'} = '硬碟';
    $Self->{Translation}->{'Capacity'} = '容量';
    $Self->{Translation}->{'FQDN'} = '';
    $Self->{Translation}->{'Network Adapter'} = '網路卡';
    $Self->{Translation}->{'IP over DHCP'} = '從DHCP取得IP';
    $Self->{Translation}->{'IP Address'} = 'IP位址';
    $Self->{Translation}->{'Graphic Adapter'} = '顯示卡';
    $Self->{Translation}->{'Other Equipment'} = '其它設備';
    $Self->{Translation}->{'Warranty Expiration Date'} = '保養失效日期';
    $Self->{Translation}->{'Install Date'} = '安装日期';
    $Self->{Translation}->{'Phone 1'} = '電話1';
    $Self->{Translation}->{'Phone 2'} = '電話2';
    $Self->{Translation}->{'E-Mail'} = '';
    $Self->{Translation}->{'Network Address'} = '網路位址';
    $Self->{Translation}->{'Subnet Mask'} = '子網路遮照';
    $Self->{Translation}->{'Gateway'} = '閘道';
    $Self->{Translation}->{'Licence Type'} = '許可類别';
    $Self->{Translation}->{'Licence Key'} = '許可密鑰';
    $Self->{Translation}->{'Quantity'} = '數量';
    $Self->{Translation}->{'Expiration Date'} = '失效日期';
    $Self->{Translation}->{'Media'} = '介質';

    # XML Definition: ITSMConfigurationManagement.sopm
    $Self->{Translation}->{'Computer'} = '';
    $Self->{Translation}->{'Hardware'} = '';
    $Self->{Translation}->{'Network'} = '';
    $Self->{Translation}->{'Software'} = '';
    $Self->{Translation}->{'Expired'} = '到期';
    $Self->{Translation}->{'Maintenance'} = '維護';
    $Self->{Translation}->{'Pilot'} = '試驗';
    $Self->{Translation}->{'Planned'} = '已計劃';
    $Self->{Translation}->{'Production'} = '';
    $Self->{Translation}->{'Repair'} = '修復';
    $Self->{Translation}->{'Retired'} = '報廢';
    $Self->{Translation}->{'Review'} = '複審';
    $Self->{Translation}->{'Test/QA'} = '测試/QA';
    $Self->{Translation}->{'Laptop'} = '筆記本';
    $Self->{Translation}->{'Desktop'} = '桌上型電腦';
    $Self->{Translation}->{'PDA'} = 'PDA';
    $Self->{Translation}->{'Server'} = '服務器';
    $Self->{Translation}->{'Other'} = '其它';
    $Self->{Translation}->{'Monitor'} = '顯示器';
    $Self->{Translation}->{'Printer'} = '打印機';
    $Self->{Translation}->{'Switch'} = '交換器';
    $Self->{Translation}->{'Router'} = '路由器';
    $Self->{Translation}->{'WLAN Access Point'} = 'WLAN 無線基地台';
    $Self->{Translation}->{'Security Device'} = '安全設備';
    $Self->{Translation}->{'Backup Device'} = '備份設備';
    $Self->{Translation}->{'Mouse'} = '滑鼠';
    $Self->{Translation}->{'Keyboard'} = '鍵盤';
    $Self->{Translation}->{'Camera'} = '照相機';
    $Self->{Translation}->{'Beamer'} = '投影儀';
    $Self->{Translation}->{'Modem'} = '調解調器';
    $Self->{Translation}->{'PCMCIA Card'} = 'PCMCIA卡';
    $Self->{Translation}->{'USB Device'} = 'USB 設備';
    $Self->{Translation}->{'Docking Station'} = '塢站';
    $Self->{Translation}->{'Scanner'} = '掃描儀';
    $Self->{Translation}->{'Building'} = '大廈';
    $Self->{Translation}->{'Office'} = '辦公室';
    $Self->{Translation}->{'Floor'} = '樓層';
    $Self->{Translation}->{'Room'} = '房間';
    $Self->{Translation}->{'Rack'} = '機架';
    $Self->{Translation}->{'Workplace'} = '工作間';
    $Self->{Translation}->{'Outlet'} = '插座';
    $Self->{Translation}->{'IT Facility'} = 'IT設施';
    $Self->{Translation}->{'LAN'} = '區域網路';
    $Self->{Translation}->{'WLAN'} = '無線網路';
    $Self->{Translation}->{'Telco'} = '電話公司';
    $Self->{Translation}->{'GSM'} = '';
    $Self->{Translation}->{'Client Application'} = '客户端應用程序';
    $Self->{Translation}->{'Middleware'} = '中間件';
    $Self->{Translation}->{'Server Application'} = '服務器應用程序';
    $Self->{Translation}->{'Client OS'} = '客户操作系統';
    $Self->{Translation}->{'Server OS'} = '服務器操作系統';
    $Self->{Translation}->{'Admin Tool'} = '管理工具';
    $Self->{Translation}->{'User Tool'} = '用戶工具';
    $Self->{Translation}->{'Embedded'} = '嵌入式';
    $Self->{Translation}->{'Single Licence'} = '單一許可';
    $Self->{Translation}->{'Per User'} = '每個用戶';
    $Self->{Translation}->{'Per Processor'} = '每個處理器';
    $Self->{Translation}->{'Per Server'} = '每個服務器';
    $Self->{Translation}->{'Per Node'} = '每個節點';
    $Self->{Translation}->{'Volume Licence'} = '批量授權';
    $Self->{Translation}->{'Enterprise Licence'} = '企業執照';
    $Self->{Translation}->{'Developer Licence'} = '開發許可';
    $Self->{Translation}->{'Demo'} = '演示';
    $Self->{Translation}->{'Time Restricted'} = '時間限制';
    $Self->{Translation}->{'Freeware'} = '免費軟件';
    $Self->{Translation}->{'Open Source'} = '開源';
    $Self->{Translation}->{'Unlimited'} = '無限制的';

    # JS File: var/httpd/htdocs/js/ITSM.Agent.ConfigItem.Zoom.js
    $Self->{Translation}->{'Ok'} = 'Ok';

    # SysConfig
    $Self->{Translation}->{'AlternativeTo'} = '';
    $Self->{Translation}->{'Assigned CIs'} = '';
    $Self->{Translation}->{'Autoloading of ITSMConfigurationManagement extensions.'} = '';
    $Self->{Translation}->{'CIs assigned to customer company'} = '';
    $Self->{Translation}->{'CIs assigned to customer user'} = '';
    $Self->{Translation}->{'CMDB Settings'} = '';
    $Self->{Translation}->{'Check for a unique name only within the same ConfigItem class (\'class\') or globally (\'global\'), which means every existing ConfigItem is taken into account when looking for duplicates.'} =
        '';
    $Self->{Translation}->{'Config Items'} = '配置項';
    $Self->{Translation}->{'Config item (dropdown)'} = '';
    $Self->{Translation}->{'Config item (multiselect)'} = '';
    $Self->{Translation}->{'Config item add.'} = '';
    $Self->{Translation}->{'Config item edit.'} = '';
    $Self->{Translation}->{'Config item event module that enables logging to history in the agent interface.'} =
        '';
    $Self->{Translation}->{'Config item history.'} = '';
    $Self->{Translation}->{'Config item print.'} = '';
    $Self->{Translation}->{'Config item zoom.'} = '';
    $Self->{Translation}->{'ConfigItemNumber'} = '';
    $Self->{Translation}->{'Configuration Item Limit'} = '';
    $Self->{Translation}->{'Configuration Item limit per page.'} = '';
    $Self->{Translation}->{'Configuration Management Database.'} = '';
    $Self->{Translation}->{'Configuration item bulk module.'} = '';
    $Self->{Translation}->{'Configuration item search backend router of the agent interface.'} =
        '';
    $Self->{Translation}->{'Create and manage the definitions for Configuration Items.'} = '創建和管理配置項定義';
    $Self->{Translation}->{'Define Actions where a settings button is available in the linked objects widget (LinkObject::ViewMode = "complex"). Please note that these Actions must have registered the following JS and CSS files: Core.AllocationList.css, Core.UI.AllocationList.js, Core.UI.Table.Sort.js, Core.Agent.TableFilters.js and Core.Agent.LinkObject.js.'} =
        '';
    $Self->{Translation}->{'Define a mapping between CI classes and a Fontawesome icons which are shown in the agent interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to create ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to delete ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to get ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to search ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines Required permissions to update ITSM configuration items using the Generic Interface.'} =
        '';
    $Self->{Translation}->{'Defines an overview module to show the small view of a configuration item list.'} =
        '';
    $Self->{Translation}->{'Defines regular expressions individually for each ConfigItem class to check the ConfigItem name and to show corresponding error messages.'} =
        '';
    $Self->{Translation}->{'Defines the default subobject of the class \'ITSMConfigItem\'.'} =
        '';
    $Self->{Translation}->{'Defines the number of rows for the CI definition editor in the admin interface.'} =
        '';
    $Self->{Translation}->{'Defines the order of incident states from high (e.g. cricital) to low (e.g. functional).'} =
        '';
    $Self->{Translation}->{'Defines the relevant deployment states where linked tickets can affect the status of a CI.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItem screen.'} =
        '';
    $Self->{Translation}->{'Defines the search limit for the AgentITSMConfigItemSearch screen.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item overview. This option has no effect on the position of the column. Note: Class column is always available if filter \'All\' is selected.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns in the config item search. This option has no effect on the position of the column.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item overview depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item search depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumns.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the config item widget depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that are common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown as defined in the setting AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany (key DefaultColumns).'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view for all CI classes. If there is no entry, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines the shown columns of CIs in the link table complex view, depending on the CI class. Each entry must be prefixed with the class name and double colons (i.e. Computer::). There are a few CI-Attributes that common to all CIs (example for the class Computer: Computer::Name, Computer::CurDeplState, Computer::CreateTime). To show individual CI-Attributes as defined in the CI-Definition, the following scheme must be used (example for the class Computer): Computer::HardDisk::1, Computer::HardDisk::1::Capacity::1, Computer::HardDisk::2, Computer::HardDisk::2::Capacity::1. If there is no entry for a CI class, then the default columns are shown.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Defines which type of link (named from the ticket perspective) is used to link tickets and config items.'} =
        '';
    $Self->{Translation}->{'Defines which type of ticket can affect the status of a linked CI.'} =
        '';
    $Self->{Translation}->{'Delete Configuration Item'} = '';
    $Self->{Translation}->{'DependsOn'} = '';
    $Self->{Translation}->{'Deployment State Color'} = '';
    $Self->{Translation}->{'Duplicate'} = '複製';
    $Self->{Translation}->{'Dynamic fields administration'} = '';
    $Self->{Translation}->{'DynamicFieldConfigItem'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer id.'} = '';
    $Self->{Translation}->{'Enable search for config items based on the customer user id.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature for the agent frontend to work on more than one configuration item at a time.'} =
        '';
    $Self->{Translation}->{'Enables configuration item bulk action feature only for the listed groups.'} =
        '';
    $Self->{Translation}->{'Enables/disables the functionality to check ConfigItems for unique names. Before enabling this option you should check your system for already existing config items with duplicate names. You can do this with the script bin/otrs.Console.pl Admin::ITSM::Configitem::ListDuplicates.'} =
        '';
    $Self->{Translation}->{'Event Module that links a ConfigItem.'} = '';
    $Self->{Translation}->{'Event module to set configitem-status on ticket-configitem-link.'} =
        '';
    $Self->{Translation}->{'ITSM config item overview.'} = '';
    $Self->{Translation}->{'Limit for config item search.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a class.'} = '';
    $Self->{Translation}->{'Module to check the group responsible for a configuration item.'} =
        '';
    $Self->{Translation}->{'Module to generate ITSM config item statistics.'} = '';
    $Self->{Translation}->{'No longer in use. Do not change. Required due to compatibility with previous package versions.'} =
        '';
    $Self->{Translation}->{'Object backend module registration for the import/export module.'} =
        '';
    $Self->{Translation}->{'Overview.'} = '概況';
    $Self->{Translation}->{'Parameters for the dashboard backend of the customer company config item overview of the agent interface . "Limit" is the number of entries shown by default. "Group" is used to restrict the access to the plugin (e. g. Group: admin;group1;group2;). "Default" determines if the plugin is enabled by default or if the user needs to enable it manually. "CacheTTLLocal" is the cache time in minutes for the plugin.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states color in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the deployment states in the preferences view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Parameters for the example permission groups of the general catalog attributes.'} =
        '';
    $Self->{Translation}->{'Parameters for the pages (in which the configuration items are shown).'} =
        '';
    $Self->{Translation}->{'Permission Group'} = '';
    $Self->{Translation}->{'RelevantTo'} = '';
    $Self->{Translation}->{'Registers an output filter that shows a config item widget.'} = '';
    $Self->{Translation}->{'Required group permissions to use the customer config item widget in the agent interface.'} =
    $Self->{Translation}->{'Registers a ITSMConfigItem-EventModule that will save related attachments.'} = '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item search screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the ITSM configuration item zoom screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the add ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the config item dialog in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the edit ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the history ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required permissions to use the print ITSM configuration item screen in the agent interface.'} =
        '';
    $Self->{Translation}->{'Required privileges to delete config items.'} = '';
    $Self->{Translation}->{'Search config items.'} = '';
    $Self->{Translation}->{'Selects the configuration item number generator module. "AutoIncrement" increments the configuration item number, the SystemID, the ConfigItemClassID and the counter are used. The format is "SystemID.ConfigItemClassID.Counter", e.g. 1205000004, 1205000005.'} =
        '';
    $Self->{Translation}->{'Set the incident state of a CI automatically when a Ticket is Linked to a CI.'} =
        '';
    $Self->{Translation}->{'Sets the deployment state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Sets the incident state in the configuration item bulk screen of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu that allows linking a configuration item with another object in the config item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to access the history of a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to delete a configuration item in its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to duplicate a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to edit a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to go back in the configuration item zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to print a configuration item in the its zoom view of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows a link in the menu to zoom into a configuration item in the configuration item overview of the agent interface.'} =
        '';
    $Self->{Translation}->{'Shows the config item history (reverse ordered) in the agent interface.'} =
        '';
    $Self->{Translation}->{'The identifier for a configuration item, e.g. ConfigItem#, MyConfigItem#. The default is ConfigItem#.'} =
        '';
    $Self->{Translation}->{'Ticket event module that creates and removes links between tickets and config items.'} =
        '';
    $Self->{Translation}->{'Ticket event module that updates DynamicFields.'} = '';
    $Self->{Translation}->{'This configuration registers a frontend module for the agent interface that provides the AJAX interface for AgentITSMConfigItemCustomerCIsWidget.'} =
        '';
    $Self->{Translation}->{'class'} = '';
    $Self->{Translation}->{'global'} = '';
    $Self->{Translation}->{'postproductive'} = '';
    $Self->{Translation}->{'preproductive'} = '';
    $Self->{Translation}->{'productive'} = '';


    push @{ $Self->{JavaScriptStrings} // [] }, (
    'No',
    'Ok',
    'Please enter at least one search value or * to find anything.',
    'Settings',
    'Submit',
    'Yes',
    'none',
    );

}

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpNb2R1bGVzOjpBSkFYRHluYW1pY0ZpZWxkQ29uZmlnSXRlbTsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6VmFyaWFibGVDaGVjayBxdyg6YWxsKTsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAogICAgJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpMb2cnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpXZWI6OlJlcXVlc3QnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpEeW5hbWljRmllbGQ6OkNvbmZpZ0l0ZW0nLAopOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRTZWxmID0geyVQYXJhbX07CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJExvZ09iamVjdCAgICAgICAgICAgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyk7CiAgICBteSAkTGF5b3V0T2JqZWN0ICAgICAgICAgICAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnKTsKICAgIG15ICRQYXJhbU9iamVjdCAgICAgICAgICAgICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OldlYjo6UmVxdWVzdCcpOwogICAgbXkgJER5bmFtaWNGaWVsZENvbmZpZ0l0ZW1PYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6RHluYW1pY0ZpZWxkOjpDb25maWdJdGVtJyk7CgogICAgaWYgKCAhJFNlbGYtPntTdWJhY3Rpb259ICkgewogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ1BhcmFtZXRlciBTdWJhY3Rpb24gaXMgbWlzc2luZy4nLAogICAgICAgICk7CgogICAgICAgIHJldHVybjsKICAgIH0KCiAgICBteSAkRGF0YTsKCiAgICAjIFRPRE86IHdlIHNob3VsZCB1c2UgdGhlIHNhbWUgbmFtaW5nIGxpa2UgREYtTERBUCAoQXV0b0ZpbGwpIG9yIHN0YW5kYXJkaXplIHRoZXNlCiAgICBpZiAoICRTZWxmLT57U3ViYWN0aW9ufSBlcSAnR2V0QWRkaXRpb25hbERGU3RvcmFnZURhdGEnICkgewogICAgICAgIG15ICRTb3VyY2VEeW5hbWljRmllbGROYW1lID0gJFBhcmFtT2JqZWN0LT5HZXRQYXJhbSggUGFyYW0gPT4gJ1NvdXJjZUR5bmFtaWNGaWVsZE5hbWUnICk7CiAgICAgICAgaWYgKCAhZGVmaW5lZCAkU291cmNlRHluYW1pY0ZpZWxkTmFtZSB8fCAhbGVuZ3RoICRTb3VyY2VEeW5hbWljRmllbGROYW1lICkgewogICAgICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ1BhcmFtZXRlciBTb3VyY2VEeW5hbWljRmllbGROYW1lIGlzIG1pc3NpbmcuJywKICAgICAgICAgICAgKTsKCiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CgogICAgICAgIG15IEBTZWxlY3RlZENvbmZpZ0l0ZW1JRHMgPSAkUGFyYW1PYmplY3QtPkdldEFycmF5KAogICAgICAgICAgICBQYXJhbSA9PiAnU2VsZWN0ZWRDb25maWdJdGVtSURzW10nLAogICAgICAgICAgICBSYXcgICA9PiAxLAogICAgICAgICk7CgogICAgICAgICMgSWYgbm8gY29uZmlnIGl0ZW0gSURzIHdlcmUgZ2l2ZW4sIHRoZSBhcnJheSBjb250YWlucyBvbmUgZWxlbWVudCB3aXRoIGFuIGVtcHR5IHN0cmluZy4KICAgICAgICAjIFJlbW92ZSBpdC4KICAgICAgICBAU2VsZWN0ZWRDb25maWdJdGVtSURzID0gZ3JlcCB7IGRlZmluZWQgJF8gJiYgbGVuZ3RoICRfIH0gQFNlbGVjdGVkQ29uZmlnSXRlbUlEczsKCiAgICAgICAgJERhdGEgPSAkRHluYW1pY0ZpZWxkQ29uZmlnSXRlbU9iamVjdC0+R2V0QWRkaXRpb25hbERGU3RvcmFnZURhdGEoCiAgICAgICAgICAgIFNvdXJjZUR5bmFtaWNGaWVsZE5hbWUgPT4gJFNvdXJjZUR5bmFtaWNGaWVsZE5hbWUsCiAgICAgICAgICAgIFNlbGVjdGVkQ29uZmlnSXRlbUlEcyAgPT4gXEBTZWxlY3RlZENvbmZpZ0l0ZW1JRHMsCiAgICAgICAgICAgIFN0b3JhZ2VUeXBlICAgICAgICAgICAgPT4gJ0Zyb250ZW5kJywKICAgICAgICAgICAgVXNlcklEICAgICAgICAgICAgICAgICA9PiAkU2VsZi0+e1VzZXJJRH0sCiAgICAgICAgKTsKICAgIH0KCiAgICBteSAkSlNPTiA9ICRMYXlvdXRPYmplY3QtPkpTT05FbmNvZGUoCiAgICAgICAgRGF0YSA9PiAkRGF0YSwKICAgICk7CgogICAgcmV0dXJuICRMYXlvdXRPYmplY3QtPkF0dGFjaG1lbnQoCiAgICAgICAgQ29udGVudFR5cGUgPT4gJ2FwcGxpY2F0aW9uL2pzb247IGNoYXJzZXQ9JyAuICRMYXlvdXRPYmplY3QtPntDaGFyc2V0fSwKICAgICAgICBDb250ZW50ICAgICA9PiAkSlNPTiAvLyAnW10nLAogICAgICAgIFR5cGUgICAgICAgID0+ICdpbmxpbmUnLAogICAgICAgIE5vQ2FjaGUgICAgID0+IDEsCiAgICApOwp9CgoxOwo=
# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Modules::AdminDynamicFieldConfigItem;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::Output::HTML::Layout',
    'Kernel::System::DynamicField',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::LinkObject',
    'Kernel::System::Valid',
    'Kernel::System::Web::Request',
);

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

our $ObjectManagerDisabled = 1;

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

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

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

    $Self->{ObjectTypeConfig} = $ConfigObject->Get('DynamicFields::ObjectType');
    $Self->{FieldTypeConfig}  = $ConfigObject->Get('DynamicFields::Driver') || {};

    return $Self;
}

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

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

    $LayoutObject->AddJSOnDocumentCompleteIfNotExists(
        Key  => 'AdminDynamicFieldConfigItem',
        Code => 'Core.Agent.Admin.DynamicFieldConfigItem.Init();',
    );

    if ( $Self->{Subaction} eq 'Add' ) {
        return $Self->_Add(%Param);
    }
    elsif ( $Self->{Subaction} eq 'AddAction' ) {
        $LayoutObject->ChallengeTokenCheck();

        return $Self->_AddAction(%Param);
    }
    elsif ( $Self->{Subaction} eq 'Change' ) {
        return $Self->_Change(%Param);
    }
    elsif ( $Self->{Subaction} eq 'ChangeAction' ) {
        $LayoutObject->ChallengeTokenCheck();

        return $Self->_ChangeAction(%Param);
    }

    return $LayoutObject->ErrorScreen(
        Message => "Undefined subaction.",
    );
}

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

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

    my %GetParam;
    NEEDED:
    for my $Needed (qw(ObjectType FieldType FieldOrder)) {
        $GetParam{$Needed} = $ParamObject->GetParam( Param => $Needed );
        next NEEDED if defined $GetParam{$Needed} && length $GetParam{$Needed};

        return $LayoutObject->ErrorScreen(
            Message => "Need $Needed.",
        );
    }

    my $ObjectTypeName = $Self->{ObjectTypeConfig}->{ $GetParam{ObjectType} }->{DisplayName} || '';
    my $FieldTypeName  = $Self->{FieldTypeConfig}->{ $GetParam{FieldType} }->{DisplayName}   || '';

    return $Self->_ShowScreen(
        %Param,
        %GetParam,
        Mode           => 'Add',
        BreadcrumbText => $LayoutObject->{LanguageObject}
            ->Translate( 'Add %s field', $LayoutObject->{LanguageObject}->Translate($FieldTypeName) ),
        ObjectTypeName => $ObjectTypeName,
        FieldTypeName  => $FieldTypeName,
    );
}

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

    my $ConfigObject       = $Kernel::OM->Get('Kernel::Config');
    my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');
    my $LayoutObject       = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $ParamObject        = $Kernel::OM->Get('Kernel::System::Web::Request');

    my %Errors;
    my %GetParam;

    NEEDED:
    for my $Needed (qw(Name Label FieldOrder)) {
        $GetParam{$Needed} = $ParamObject->GetParam( Param => $Needed );
        next NEEDED if defined $GetParam{$Needed} && length $GetParam{$Needed};

        $Errors{ $Needed . 'ServerError' }        = 'ServerError';
        $Errors{ $Needed . 'ServerErrorMessage' } = 'This field is required.';
    }

    if ( $GetParam{Name} ) {

        # check if name is alphanumeric
        if ( $GetParam{Name} !~ m{\A (?: [a-zA-Z] | \d )+ \z}xms ) {
            $Errors{NameServerError} = 'ServerError';
            $Errors{NameServerErrorMessage} =
                'The field does not contain only ASCII letters and numbers.';
        }

        # check if name is duplicated
        my %DynamicFieldsList = %{
            $DynamicFieldObject->DynamicFieldList(
                Valid      => 0,
                ResultType => 'HASH',
            )
        };
        %DynamicFieldsList = reverse %DynamicFieldsList;

        if ( $DynamicFieldsList{ $GetParam{Name} } ) {
            $Errors{NameServerError}        = 'ServerError';
            $Errors{NameServerErrorMessage} = 'There is another field with the same name.';
        }
    }

    if ( $GetParam{FieldOrder} ) {

        # check if field order is numeric and positive
        if ( $GetParam{FieldOrder} !~ m{\A (?: \d )+ \z}xms ) {
            $Errors{FieldOrderServerError}        = 'ServerError';
            $Errors{FieldOrderServerErrorMessage} = 'The field must be numeric.';
        }
    }

    for my $ConfigParam (
        qw(ObjectType ObjectTypeName FieldType FieldTypeName ValidID ConfigItemClass ConfigItemLinkType ConfigItemLinkSource ConfigItemLinkRemoval)
        )
    {
        $GetParam{$ConfigParam} = $ParamObject->GetParam( Param => $ConfigParam );
    }

    my @DeplStates = $ParamObject->GetArray(
        Param => 'DeplStates',
        Raw   => 1,
    );
    if (@DeplStates) {
        $GetParam{DeplStates} = \@DeplStates;
    }

    # uncorrectable errors
    if ( !$GetParam{ValidID} ) {
        return $LayoutObject->ErrorScreen(
            Message => 'Need ValidID.',
        );
    }

    if ( !$GetParam{ConfigItemClass} ) {
        $Errors{ConfigItemClassServerError} = 'ServerError';
    }

    my @AdditionalDFStorage = $Self->_AdditionalDFStorageGet();
    $GetParam{AdditionalDFStorage} = \@AdditionalDFStorage;

    my %AdditionalDFStorageErrors = $Self->_AdditionalDFStorageValidate(%GetParam);
    %Errors = ( %Errors, %AdditionalDFStorageErrors );

    if (%Errors) {
        return $Self->_ShowScreen(
            %Param,
            %Errors,
            %GetParam,
            Mode => 'Add',
        );
    }

    my $DynamicFieldConfig;
    my $DynamicFieldDriverRegistration = $ConfigObject->Get('DynamicFields::Driver');

    if (
        IsHashRefWithData($DynamicFieldDriverRegistration)
        && IsHashRefWithData( $DynamicFieldDriverRegistration->{ $GetParam{FieldType} } )
        && IsHashRefWithData( $DynamicFieldDriverRegistration->{ $GetParam{FieldType} }->{Config} )
        )
    {
        $DynamicFieldConfig = $DynamicFieldDriverRegistration->{ $GetParam{FieldType} }->{Config};
    }

    # overwrite dynamic field config
    KEY:
    for my $Key (
        qw( ConfigItemClass DeplStates ConfigItemLinkType ConfigItemLinkSource ConfigItemLinkRemoval AdditionalDFStorage )
        )
    {
        next KEY if !defined $GetParam{$Key};
        $DynamicFieldConfig->{$Key} = $GetParam{$Key};
    }

    my $FieldID = $DynamicFieldObject->DynamicFieldAdd(
        Name       => $GetParam{Name},
        Label      => $GetParam{Label},
        FieldOrder => $GetParam{FieldOrder},
        FieldType  => $GetParam{FieldType},
        ObjectType => $GetParam{ObjectType},
        Config     => $DynamicFieldConfig || {},
        ValidID    => $GetParam{ValidID},
        UserID     => $Self->{UserID},
    );

    if ( !$FieldID ) {
        return $LayoutObject->ErrorScreen(
            Message => 'Could not create the new field.',
        );
    }

    return $LayoutObject->Redirect(
        OP => "Action=AdminDynamicField",
    );
}

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

    my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');
    my $LayoutObject       = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $ParamObject        = $Kernel::OM->Get('Kernel::System::Web::Request');

    my $FieldID = $ParamObject->GetParam( Param => 'ID' );
    if ( !$FieldID ) {
        return $LayoutObject->ErrorScreen(
            Message => 'Need ID.',
        );
    }

    my $DynamicFieldConfig = $DynamicFieldObject->DynamicFieldGet(
        ID => $FieldID,
    );
    if ( !IsHashRefWithData($DynamicFieldConfig) ) {
        return $LayoutObject->ErrorScreen(
            Message => "Could not get config for dynamic field $FieldID.",
        );
    }

    my %Config;
    if ( IsHashRefWithData( $DynamicFieldConfig->{Config} ) ) {
        %Config = %{ $DynamicFieldConfig->{Config} };
    }

    # Initialize GetParam with read-only values that cannot be changed anymore in change dialog.
    my %GetParam = (
        ObjectType      => $DynamicFieldConfig->{ObjectType},
        FieldType       => $DynamicFieldConfig->{FieldType},
        ConfigItemClass => $Config{ConfigItemClass},
    );

    my $ObjectTypeName = $Self->{ObjectTypeConfig}->{ $GetParam{ObjectType} }->{DisplayName} || '';
    my $FieldTypeName  = $Self->{FieldTypeConfig}->{ $GetParam{FieldType} }->{DisplayName}   || '';

    return $Self->_ShowScreen(
        %Param,
        %GetParam,
        %{$DynamicFieldConfig},
        %Config,
        ID             => $FieldID,
        Mode           => 'Change',
        BreadcrumbText => $LayoutObject->{LanguageObject}
            ->Translate( 'Change %s field', $LayoutObject->{LanguageObject}->Translate($FieldTypeName) ),
        ObjectTypeName => $ObjectTypeName,
        FieldTypeName  => $FieldTypeName,
    );
}

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

    my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');
    my $LayoutObject       = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $ParamObject        = $Kernel::OM->Get('Kernel::System::Web::Request');

    my %Errors;
    my %GetParam;

    NEEDED:
    for my $Needed (qw(Name Label FieldOrder)) {
        $GetParam{$Needed} = $ParamObject->GetParam( Param => $Needed );
        next NEEDED if defined $GetParam{$Needed} && length $GetParam{$Needed};

        $Errors{ $Needed . 'ServerError' }        = 'ServerError';
        $Errors{ $Needed . 'ServerErrorMessage' } = 'This field is required.';
    }

    my $FieldID = $ParamObject->GetParam( Param => 'ID' );
    if ( !$FieldID ) {
        return $LayoutObject->ErrorScreen(
            Message => 'Need ID.',
        );
    }
    my $DynamicFieldConfig = $DynamicFieldObject->DynamicFieldGet(
        ID => $FieldID,
    );
    if ( !IsHashRefWithData($DynamicFieldConfig) ) {
        return $LayoutObject->ErrorScreen(
            Message => "Could not get config for dynamic field $FieldID.",
        );
    }

    if ( $GetParam{Name} ) {

        # check if name is lowercase
        if ( $GetParam{Name} !~ m{\A (?: [a-zA-Z] | \d )+ \z}xms ) {
            $Errors{NameServerError} = 'ServerError';
            $Errors{NameServerErrorMessage} =
                'The field does not contain only ASCII letters and numbers.';
        }

        # check if name is duplicated
        my %DynamicFieldsList = %{
            $DynamicFieldObject->DynamicFieldList(
                Valid      => 0,
                ResultType => 'HASH',
            )
        };
        %DynamicFieldsList = reverse %DynamicFieldsList;

        if (
            $DynamicFieldsList{ $GetParam{Name} } &&
            $DynamicFieldsList{ $GetParam{Name} } ne $FieldID
            )
        {
            $Errors{NameServerError}        = 'ServerError';
            $Errors{NameServerErrorMessage} = 'There is another field with the same name.';
        }

        # if it's an internal field, its name should not change
        if (
            $DynamicFieldConfig->{InternalField} &&
            $DynamicFieldsList{ $GetParam{Name} } ne $FieldID
            )
        {
            $Errors{NameServerError}        = 'ServerError';
            $Errors{NameServerErrorMessage} = 'The name for this field should not change.';
            $Param{InternalField}           = $DynamicFieldConfig->{InternalField};
        }
    }

    if ( $GetParam{FieldOrder} ) {

        # check if field order is numeric and positive
        if ( $GetParam{FieldOrder} !~ m{\A (?: \d )+ \z}xms ) {
            $Errors{FieldOrderServerError}        = 'ServerError';
            $Errors{FieldOrderServerErrorMessage} = 'The field must be numeric.';
        }
    }

    for my $ConfigParam (
        qw(ObjectType ObjectTypeName FieldType FieldTypeName DefaultValue ValidID ConfigItemLinkType ConfigItemLinkSource ConfigItemLinkRemoval)
        )
    {
        $GetParam{$ConfigParam} = $ParamObject->GetParam( Param => $ConfigParam );
    }

    # Config item class is read-only in change, so set it to the stored value every time.
    $GetParam{ConfigItemClass} = $DynamicFieldConfig->{Config}->{ConfigItemClass};

    my @DeplStates = $ParamObject->GetArray(
        Param => 'DeplStates',
        Raw   => 1,
    );
    $GetParam{DeplStates} = \@DeplStates;
    $DynamicFieldConfig->{Config}->{DeplStates} = \@DeplStates;

    $DynamicFieldConfig->{Config}->{ConfigItemLinkType}    = $GetParam{ConfigItemLinkType};
    $DynamicFieldConfig->{Config}->{ConfigItemLinkSource}  = $GetParam{ConfigItemLinkSource};
    $DynamicFieldConfig->{Config}->{ConfigItemLinkRemoval} = $GetParam{ConfigItemLinkRemoval};

    if ( !$GetParam{ValidID} ) {
        return $LayoutObject->ErrorScreen(
            Message => 'Need ValidID.',
        );
    }

    my @AdditionalDFStorage = $Self->_AdditionalDFStorageGet();
    $GetParam{AdditionalDFStorage} = \@AdditionalDFStorage;

    my %AdditionalDFStorageErrors = $Self->_AdditionalDFStorageValidate(%GetParam);
    %Errors = ( %Errors, %AdditionalDFStorageErrors );

    # return to change screen if errors
    if (%Errors) {
        return $Self->_ShowScreen(
            %Param,
            %Errors,
            %GetParam,
            ID   => $FieldID,
            Mode => 'Change',
        );
    }

    $DynamicFieldConfig->{Config}->{AdditionalDFStorage} = $GetParam{AdditionalDFStorage};

    # update dynamic field (FieldType and ObjectType cannot be changed - use old values)
    my $UpdateSuccess = $DynamicFieldObject->DynamicFieldUpdate(
        ID         => $FieldID,
        Name       => $GetParam{Name},
        Label      => $GetParam{Label},
        FieldOrder => $GetParam{FieldOrder},
        FieldType  => $DynamicFieldConfig->{FieldType},
        ObjectType => $DynamicFieldConfig->{ObjectType},
        Config     => $DynamicFieldConfig->{Config} || {},
        ValidID    => $GetParam{ValidID},
        UserID     => $Self->{UserID},
    );

    if ( !$UpdateSuccess ) {
        return $LayoutObject->ErrorScreen(
            Message => "Could not update the field $GetParam{Name}.",
        );
    }

    if ( $ParamObject->GetParam( Param => 'ContinueAfterSave' ) ) {
        return $LayoutObject->Redirect(
            OP =>
                "Action=$Self->{Action};Subaction=Change;ObjectType=$GetParam{ObjectType};FieldType=$GetParam{FieldType};ID=$FieldID"
        );
    }

    return $LayoutObject->Redirect(
        OP => "Action=AdminDynamicField",
    );
}

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

    my $DynamicFieldObject   = $Kernel::OM->Get('Kernel::System::DynamicField');
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
    my $LayoutObject         = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $ValidObject          = $Kernel::OM->Get('Kernel::System::Valid');
    my $LinkObject           = $Kernel::OM->Get('Kernel::System::LinkObject');
    my $LanguageObject       = $Kernel::OM->Get('Kernel::Language');

    $Param{DisplayFieldName} = 'New';

    if ( $Param{Mode} eq 'Change' ) {
        $Param{ShowWarning}      = 'ShowWarning';
        $Param{DisplayFieldName} = $Param{Name};
    }

    my $Output = $LayoutObject->Header();
    $Output .= $LayoutObject->NavigationBar();

    my $DynamicFieldList = $DynamicFieldObject->DynamicFieldListGet(
        Valid => 0,
    );

    # get the list of order numbers (is already sorted).
    my @DynamicfieldOrderList;
    my %DynamicfieldNamesList;
    for my $Dynamicfield ( @{$DynamicFieldList} ) {
        push @DynamicfieldOrderList, $Dynamicfield->{FieldOrder};
        $DynamicfieldNamesList{ $Dynamicfield->{FieldOrder} } = $Dynamicfield->{Label};
    }

    # when adding we need to create an extra order number for the new field
    if ( $Param{Mode} eq 'Add' ) {

        # get the last element form the order list and add 1
        my $LastOrderNumber = $DynamicfieldOrderList[-1];
        $LastOrderNumber++;

        # add this new order number to the end of the list
        push @DynamicfieldOrderList, $LastOrderNumber;
    }

    # show the names of the other fields to ease ordering
    my %OrderNamesList;
    my $CurrentlyText = $LayoutObject->{LanguageObject}->Translate('Currently') . ': ';
    for my $OrderNumber ( sort @DynamicfieldOrderList ) {
        $OrderNamesList{$OrderNumber} = $OrderNumber;
        if ( $DynamicfieldNamesList{$OrderNumber} && $OrderNumber ne $Param{FieldOrder} ) {
            $OrderNamesList{$OrderNumber} = $OrderNumber . ' - '
                . $CurrentlyText
                . $DynamicfieldNamesList{$OrderNumber};
        }
    }

    my $DynamicFieldOrderStrg = $LayoutObject->BuildSelection(
        Data          => \%OrderNamesList,
        Name          => 'FieldOrder',
        SelectedValue => $Param{FieldOrder} || 1,
        PossibleNone  => 0,
        Translation   => 0,
        Sort          => 'NumericKey',
        Class         => 'W75pc Validate_Number',
    );

    my %ValidList    = $ValidObject->ValidList();
    my $ValidityStrg = $LayoutObject->BuildSelection(
        Data         => \%ValidList,
        Name         => 'ValidID',
        SelectedID   => $Param{ValidID} || 1,
        PossibleNone => 0,
        Translation  => 1,
        Class        => 'W50pc',
    );

    my %ShowParams = $Self->_AdditionalDFStorageShow(
        %Param
    );
    %Param = ( %Param, %ShowParams );

    my $ReadonlyInternalField = '';

    # Internal fields can not be deleted and name should not change.
    if ( $Param{InternalField} ) {
        $LayoutObject->Block(
            Name => 'InternalField',
            Data => {%Param},
        );
        $ReadonlyInternalField = 'readonly="readonly"';
    }

    my $ClassList = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    my $ConfigItemClasses = [ sort values %{ $ClassList || {} } ];
    my %ConfigItemClasses = map { $_ => $_ } @{$ConfigItemClasses};

    # Add 'invalid' note to config class name if the class is invalid
    # or does not exist (anymore).
    my $SelectedConfigItemClassIsInvalid;
    if (
        $Param{Mode} eq 'Change'
        && $Param{ConfigItemClass}
        && !$ConfigItemClasses{ $Param{ConfigItemClass} }
        )
    {
        $SelectedConfigItemClassIsInvalid = 1;
        $ConfigItemClasses{ $Param{ConfigItemClass} } = $LanguageObject->Translate( $Param{ConfigItemClass} )
            . ' ('
            . $LanguageObject->Translate('invalid')
            . ')';
    }

    # disable field in change dialogs
    my $Disabled = $Param{Mode} eq 'Change' ? 1 : 0;

    $Param{ConfigItemClassStrg} = $LayoutObject->BuildSelection(
        Data         => \%ConfigItemClasses,
        SelectedID   => $Param{ConfigItemClass},
        Class        => 'Modernize Validate_Required  ' . ( $Param{ConfigItemClassServerError} || ' ' ),
        Translation  => 1,
        Name         => 'ConfigItemClass',
        PossibleNone => 1,
        Disabled     => $Disabled,
        Sort         => 'AlphanumericKey',
    );

    # selection of deployment status
    my $DeplStateList = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    my $DeplStates = [ sort values %{ $DeplStateList || {} } ];
    my %DeplStates = map { $_ => $_ } @{$DeplStates};

    my $DeplStateSelectionHTML = $LayoutObject->BuildSelection(
        Data         => \%DeplStates,
        Name         => 'DeplStates',
        PossibleNone => 0,
        Translation  => 1,
        Class        => 'Modernize',
        Multiple     => 1,
        SelectedID   => $Param{DeplStates},
    );

    # selection of link type
    my %PossibleConfigItemLinkTypes = $LinkObject->PossibleTypesList(
        Object1 => $Param{ObjectType},
        Object2 => 'ITSMConfigItem',
    );

    my @PossibleConfigItemLinkTypes = keys %PossibleConfigItemLinkTypes;

    my $ConfigItemLinkTypeSelectionHTML = $LayoutObject->BuildSelection(
        Data          => \@PossibleConfigItemLinkTypes,
        Name          => 'ConfigItemLinkType',
        PossibleNone  => 1,
        Class         => 'Modernize',
        Multiple      => 0,
        SelectedValue => $Param{ConfigItemLinkType},
    );

    # selection of config item link source
    my %ConfigItemLinkSources = (
        ITSMConfigItem     => 'Config item',
        $Param{ObjectType} => $Param{ObjectType},
    );

    my $ConfigItemLinkSourceSelectionHTML = $LayoutObject->BuildSelection(
        Data         => \%ConfigItemLinkSources,
        Name         => 'ConfigItemLinkSource',
        PossibleNone => 0,
        Class        => 'Modernize',
        Multiple     => 0,
        SelectedID   => $Param{ConfigItemLinkSource},
    );

    # selection for config item link removal
    my $ConfigItemLinkRemovalSelectionHTML = $LayoutObject->BuildSelection(
        Data => {
            0 => Translatable('No'),
            1 => Translatable('Yes'),
        },
        Name       => 'ConfigItemLinkRemoval',
        SelectedID => $Param{ConfigItemLinkRemoval} // 1,
        Class      => 'Modernize',
    );

    $Output .= $LayoutObject->Output(
        TemplateFile => 'AdminDynamicFieldConfigItem',
        Data         => {
            %Param,
            SelectedConfigItemClassIsInvalid   => $SelectedConfigItemClassIsInvalid,
            DeplStateSelectionHTML             => $DeplStateSelectionHTML,
            ConfigItemLinkTypeSelectionHTML    => $ConfigItemLinkTypeSelectionHTML,
            ConfigItemLinkSourceSelectionHTML  => $ConfigItemLinkSourceSelectionHTML,
            ConfigItemLinkRemovalSelectionHTML => $ConfigItemLinkRemovalSelectionHTML,
            ValidityStrg                       => $ValidityStrg,
            DynamicFieldOrderStrg              => $DynamicFieldOrderStrg,
            ReadonlyInternalField              => $ReadonlyInternalField,
        },
    );

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

    return $Output;
}

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

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

    my %ShowParams;

    # Only ticket dynamic fields are supported for additional dynamic field storage.
    return %ShowParams if !$Param{ObjectType};
    return %ShowParams if $Param{ObjectType} ne 'Ticket';

    my @AdditionalDFStorage;
    if ( IsArrayRefWithData( $Param{AdditionalDFStorage} ) ) {
        @AdditionalDFStorage = @{ $Param{AdditionalDFStorage} };
    }

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

    # Assemble available dynamic fields to be additionally filled.
    my $AdditionalDynamicFieldConfigs = $DynamicFieldObject->DynamicFieldListGet(
        ObjectType => 'Ticket',
    );

    # Filter out dynamic field that is currently being configured. It's not available
    # to be selected as an additional dynamic field to be filled.
    my @AdditionalDynamicFieldConfigs = @{$AdditionalDynamicFieldConfigs};
    if ( defined $Param{Name} ) {
        @AdditionalDynamicFieldConfigs = grep { $_->{Name} ne $Param{Name} } @{$AdditionalDynamicFieldConfigs};
    }

    my %AdditionalDynamicFieldSelection
        = map { $_->{Name} => $_->{Name} . '(' . $_->{Label} . ')' } @AdditionalDynamicFieldConfigs;

    my $AdditionalDFStorageValueCounter = 0;
    for my $Storage (@AdditionalDFStorage) {
        my $DynamicField  = $Storage->{DynamicField};
        my $ConfigItemKey = $Storage->{ConfigItemKey};
        my $Type          = $Storage->{Type};

        my $DynamicFieldError        = '';
        my $DynamicFieldErrorMessage = Translatable('This field is required');    # default in template

        my $ConfigItemKeyError        = '';
        my $ConfigItemKeyErrorMessage = Translatable('This field is required');    # default in template;

        if ( $Param{AdditionalDFStorageErrors} ) {

            # Dynamic field error
            if ( defined $Param{AdditionalDFStorageErrors}->[$AdditionalDFStorageValueCounter]->{DynamicField} ) {
                $DynamicFieldError = 'ServerError';
                $DynamicFieldErrorMessage
                    = $Param{AdditionalDFStorageErrors}->[$AdditionalDFStorageValueCounter]->{DynamicField};
            }

            # Config item key error
            if ( defined $Param{AdditionalDFStorageErrors}->[$AdditionalDFStorageValueCounter]->{ConfigItemKey} ) {
                $ConfigItemKeyError = 'ServerError';
                $ConfigItemKeyErrorMessage
                    = $Param{AdditionalDFStorageErrors}->[$AdditionalDFStorageValueCounter]->{ConfigItemKey};
            }
        }

        $AdditionalDFStorageValueCounter++;

        my $DynamicFieldSelection = $LayoutObject->BuildSelection(
            Data         => \%AdditionalDynamicFieldSelection,
            Sort         => 'AlphanumericValue',
            Name         => 'DynamicField_' . $AdditionalDFStorageValueCounter,
            SelectedID   => $DynamicField,
            PossibleNone => 1,
            Translation  => 0,
            Class        => "Modernize VariableWidth DataTable Validate_Required $DynamicFieldError",
        );

        my $TypeOption = $LayoutObject->BuildSelection(
            Data => {
                Frontend        => 'Frontend',
                Backend         => 'Backend',
                FrontendBackend => 'Frontend and Backend',
            },
            Sort           => 'IndividualKey',
            SortIndividual => [ 'Backend', 'Frontend', 'FrontendBackend' ],
            Name           => 'Type_' . $AdditionalDFStorageValueCounter,
            SelectedID     => $Type || 'Backend',
            PossibleNone   => 0,
            Translation    => 1,
            Class          => 'Modernize',
        );

        # create a value map row
        $LayoutObject->Block(
            Name => 'AdditionalDFStorageRow',
            Data => {
                AdditionalDFStorageValueCounter => $AdditionalDFStorageValueCounter,
                DynamicFieldSelection           => $DynamicFieldSelection,
                DynamicFieldErrorMessage        => $DynamicFieldErrorMessage,
                ConfigItemKey                   => $ConfigItemKey,
                ConfigItemKeyError              => $ConfigItemKeyError,
                ConfigItemKeyErrorMessage       => $ConfigItemKeyErrorMessage,
                TypeOption                      => $TypeOption,
            },
        );
    }

    $Param{TypeOption} = $LayoutObject->BuildSelection(
        Data => {
            Frontend        => 'Frontend',
            Backend         => 'Backend',
            FrontendBackend => 'Frontend and Backend',
        },
        Sort           => 'IndividualKey',
        SortIndividual => [ 'Backend', 'Frontend', 'FrontendBackend' ],
        Name           => 'Type',
        SelectedID     => $Param{Type} // 'Backend',
        PossibleNone   => 0,
        Translation    => 1,
        Class          => 'Modernize',
    );

    # create AdditionalDFStorage template
    $Param{DynamicFieldSelectionTemplate} = $LayoutObject->BuildSelection(
        Data         => \%AdditionalDynamicFieldSelection,
        Sort         => 'AlphanumericValue',
        Name         => 'DynamicField',
        PossibleNone => 1,
        Translation  => 0,
        Class        => 'Modernize VariableWidth DataTable',
    );

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

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

    $ShowParams{AdditionalDFStorageValueCounter} = $AdditionalDFStorageValueCounter;

    return %ShowParams;
}

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

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

    my @AdditionalDFStorage;

    my $ValueCounter = $ParamObject->GetParam( Param => 'AdditionalDFStorageValueCounter' ) // 0;

    VALUECOUNTERINDEX:
    for my $ValueCounterIndex ( 1 .. $ValueCounter ) {
        my $DynamicField = $ParamObject->GetParam( Param => 'DynamicField_' . $ValueCounterIndex );
        next VALUECOUNTERINDEX if !defined $DynamicField;

        my $ConfigItemKey = $ParamObject->GetParam( Param => 'ConfigItemKey_' . $ValueCounterIndex );
        my $Type          = $ParamObject->GetParam( Param => 'Type_' . $ValueCounterIndex ) // 'Backend';

        push @AdditionalDFStorage, {
            DynamicField  => $DynamicField,
            ConfigItemKey => $ConfigItemKey,
            Type          => $Type,
        };
    }

    return @AdditionalDFStorage;
}

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

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

    my @AdditionalDFStorage = @{ $Param{AdditionalDFStorage} // [] };

    my %UsedDynamicFields;
    my @StorageErrorMessages;
    my $StorageErrorFound;

    STORAGE:
    for my $Storage (@AdditionalDFStorage) {
        my $DynamicField  = $Storage->{DynamicField};
        my $ConfigItemKey = $Storage->{ConfigItemKey};
        my $Type          = $Storage->{Type};

        my %StorageErrorMessages;

        # Check dynamic field.
        if ( !defined $DynamicField || !length $DynamicField ) {
            $StorageErrorMessages{DynamicField} = Translatable('This field is required.');
        }
        elsif ( $UsedDynamicFields{$DynamicField} ) {
            $StorageErrorMessages{DynamicField} = Translatable('Dynamic field is configured more than once.');
        }
        else {
            my $DynamicFieldConfig = $DynamicFieldObject->DynamicFieldGet(
                Name => $DynamicField,
            );
            if ( !IsHashRefWithData($DynamicFieldConfig) ) {
                $StorageErrorMessages{DynamicField} = Translatable('Dynamic field does not exist or is invalid.');
            }
            elsif ( $DynamicFieldConfig->{ObjectType} ne 'Ticket' ) {
                $StorageErrorMessages{DynamicField} = Translatable('Only dynamic fields for tickets are allowed.');
            }
        }

        if ( defined $DynamicField && length $DynamicField ) {
            $UsedDynamicFields{$DynamicField} = 1;
        }

        # Check config item key.
        if ( !defined $ConfigItemKey || !length $ConfigItemKey ) {
            $StorageErrorMessages{ConfigItemKey} = Translatable('This field is required.');
        }

        # Important: push even if %StorageErrorMessages is empty
        # because the index in @StorageErrorMessages must match the one of @AdditionalDFStorage.
        push @StorageErrorMessages, \%StorageErrorMessages;

        $StorageErrorFound = 1 if %StorageErrorMessages;
    }

    my %Errors;
    if ($StorageErrorFound) {
        $Errors{AdditionalDFStorageErrors} = \@StorageErrorMessages;
    }

    return %Errors;
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Modules::AdminITSMConfigItem;

use strict;
use warnings;

our $ObjectManagerDisabled = 1;

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

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

    return $Self;
}

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

    # get needed objecs
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
    my $LayoutObject         = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    # get class list
    my $ClassList = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );
    return $LayoutObject->ErrorScreen() if !$ClassList;
    return $LayoutObject->ErrorScreen() if ref $ClassList ne 'HASH';
    return $LayoutObject->ErrorScreen() if !%{$ClassList};

    # get needed objects
    my $ParamObject      = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $UserObject       = $Kernel::OM->Get('Kernel::System::User');

    # ------------------------------------------------------------ #
    # definition list
    # ------------------------------------------------------------ #
    if ( $Self->{Subaction} eq 'DefinitionList' ) {

        # get class id
        my $ClassID = $ParamObject->GetParam( Param => 'ClassID' );

        return $LayoutObject->Redirect( OP => "Action=$Self->{Action}" ) if !$ClassID;

        # generate ClassOptionStrg
        my $ClassOptionStrg = $LayoutObject->BuildSelection(
            Data         => $ClassList,
            Name         => 'ClassID',
            PossibleNone => 1,
            Translation  => 0,
            SelectedID   => $ClassID,
            Class        => 'Modernize',
        );

        # output overview
        $LayoutObject->Block(
            Name => 'Overview',
            Data => {
                ClassOptionStrg => $ClassOptionStrg,
                ClassSelected   => {
                    ID   => $ClassID,
                    Name => $ClassList->{$ClassID},
                },
            },
        );

        # output overview result
        $LayoutObject->Block(
            Name => 'DefinitionList',
            Data => {
                Name => $ClassList->{$ClassID},
            },
        );

        # get definition list
        my $DefinitionList = $ConfigItemObject->DefinitionList(
            ClassID => $ClassID,
        );

        for my $Definition ( reverse @{$DefinitionList} ) {

            # get user data
            my $FullName = $UserObject->UserName(
                UserID => $Definition->{CreateBy},
            );

            # output definition
            $LayoutObject->Block(
                Name => 'DefinitionListRow',
                Data => {
                    %{$Definition},
                    Class        => $ClassList->{$ClassID},
                    CreateByUser => $FullName,
                },
            );
        }

        # ActionOverview
        $LayoutObject->Block(
            Name => 'ActionOverview',
        );

        # output header
        my $Output = $LayoutObject->Header();
        $Output .= $LayoutObject->NavigationBar();

        # generate output
        $Output .= $LayoutObject->Output(
            TemplateFile => 'AdminITSMConfigItem',
            Data         => \%Param,
        );
        $Output .= $LayoutObject->Footer();

        return $Output;
    }

    # ------------------------------------------------------------ #
    # definition view
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'DefinitionView' ) {

        # get definition id
        my $DefinitionID = $ParamObject->GetParam( Param => 'DefinitionID' );

        return $LayoutObject->Redirect( OP => "Action=$Self->{Action}" ) if !$DefinitionID;

        # get definition
        my $Definition = $ConfigItemObject->DefinitionGet(
            DefinitionID => $DefinitionID,
        );

        $Definition->{DefinitionString} = $LayoutObject->Ascii2Html(
            Text => $Definition->{Definition},
        );

        # generate ClassOptionStrg
        my $ClassOptionStrg = $LayoutObject->BuildSelection(
            Data         => $ClassList,
            Name         => 'ClassID',
            PossibleNone => 1,
            Translation  => 0,
            SelectedID   => $Definition->{ClassID},
            Class        => 'Modernize',
        );

        # output overview
        $LayoutObject->Block(
            Name => 'Overview',
            Data => {
                ClassOptionStrg => $ClassOptionStrg,
                ClassSelected   => {
                    ID   => $Definition->{ClassID},
                    Name => $Definition->{Class},
                },
                VersionSelected => {
                    ID      => $DefinitionID,
                    Version => $Definition->{Version},
                },
            },
        );

        # get user data
        my $UserName = $UserObject->UserName(
            UserID => $Definition->{CreateBy},
        );

        # output overview result
        $LayoutObject->Block(
            Name => 'DefinitionView',
            Data => {
                %{$Definition},
                CreateByUser => $UserName,
            },
        );

        # ActionOverview
        $LayoutObject->Block(
            Name => 'ActionOverview',
        );

        # output header
        my $Output = $LayoutObject->Header();
        $Output .= $LayoutObject->NavigationBar();

        # generate output
        $Output .= $LayoutObject->Output(
            TemplateFile => 'AdminITSMConfigItem',
            Data         => \%Param,
        );
        $Output .= $LayoutObject->Footer();

        return $Output;
    }

    # ------------------------------------------------------------ #
    # definition change
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'DefinitionChange' ) {

        # get class id
        my $ClassID = $ParamObject->GetParam( Param => 'ClassID' );

        return $LayoutObject->Redirect( OP => "Action=$Self->{Action}" ) if !$ClassID;

        # get class list
        my $ClassList = $GeneralCatalogObject->ItemList(
            Class => 'ITSM::ConfigItem::Class',
        );

        # get definition
        my $Definition = $ConfigItemObject->DefinitionGet(
            ClassID => $ClassID,
        );

        # generate ClassOptionStrg
        my $ClassOptionStrg = $LayoutObject->BuildSelection(
            Data         => $ClassList,
            Name         => 'ClassID',
            PossibleNone => 1,
            Translation  => 0,
            SelectedID   => $ClassID,
            Class        => 'Modernize',
        );

        # output overview
        $LayoutObject->Block(
            Name => 'Overview',
            Data => {
                ClassOptionStrg => $ClassOptionStrg,
                ClassSelected   => {
                    ID   => $Definition->{ClassID},
                    Name => $Definition->{Class},
                },
                Edit => 1,
            },
        );

        # output overview result
        $LayoutObject->Block(
            Name => 'DefinitionChange',
            Data => {
                %{$Definition},
                ClassID => $ClassID,
                Class   => $ClassList->{$ClassID},
                Rows =>
                    $Kernel::OM->Get('Kernel::Config')->Get("ITSMConfigItem::Frontend::$Self->{Action}")->{EditorRows}
                    || 30,
            },
        );

        # ActionOverview
        $LayoutObject->Block(
            Name => 'ActionOverview',
        );

        # output header
        my $Output = $LayoutObject->Header();
        $Output .= $LayoutObject->NavigationBar();

        # generate output
        $Output .= $LayoutObject->Output(
            TemplateFile => 'AdminITSMConfigItem',
            Data         => \%Param,
        );
        $Output .= $LayoutObject->Footer();

        return $Output;
    }

    # ------------------------------------------------------------ #
    # definition save
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'DefinitionSave' ) {
        my %Definition;

        # get params
        for my $FormParam (qw(ClassID Definition)) {
            $Definition{$FormParam} = $ParamObject->GetParam( Param => $FormParam ) || '';
        }
        for my $FormParam (qw(ClassID Definition)) {
            if ( !$Definition{$FormParam} ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  => "Need $FormParam!"
                );
                return $LayoutObject->Redirect( OP => "Action=$Self->{Action}" );
            }
        }

        # Validate YAML code by converting it to Perl.
        my $DefinitionRef = $Kernel::OM->Get('Kernel::System::YAML')->Load(
            Data => $Definition{Definition},
        );
        return $LayoutObject->ErrorScreen() if ref $DefinitionRef ne 'ARRAY';

        # add to database
        my $DefinitionID = $ConfigItemObject->DefinitionAdd(
            %Definition,
            UserID => $Self->{UserID},
        );

        return $LayoutObject->ErrorScreen() if !$DefinitionID;

        my $ContinueAfterSave = $ParamObject->GetParam( Param => 'ContinueAfterSave' );
        if ($ContinueAfterSave) {
            return $LayoutObject->Redirect(
                OP => "Action=$Self->{Action};Subaction=DefinitionChange;ClassID=$Definition{ClassID}",
            );
        }

        return $LayoutObject->Redirect( OP => "Action=$Self->{Action}" );
    }

    # ------------------------------------------------------------ #
    # config item class overview
    # ------------------------------------------------------------ #
    else {

        # generate ClassOptionStrg
        my $ClassOptionStrg = $LayoutObject->BuildSelection(
            Data         => $ClassList,
            Name         => 'ClassID',
            PossibleNone => 1,
            Translation  => 0,
            Class        => 'Modernize',
        );

        # output overview
        $LayoutObject->Block(
            Name => 'Overview',
            Data => {
                ClassOptionStrg => $ClassOptionStrg,
            },
        );

        # output overview result
        $LayoutObject->Block(
            Name => 'OverviewList',
        );

        for my $ClassID ( sort { $ClassList->{$a} cmp $ClassList->{$b} } keys %{$ClassList} ) {

            $LayoutObject->Block(
                Name => 'OverviewListRow',
                Data => {
                    ClassID => $ClassID,
                    Name    => $ClassList->{$ClassID},
                },
            );
        }

        # output header
        my $Output = $LayoutObject->Header();
        $Output .= $LayoutObject->NavigationBar();

        # generate output
        $Output .= $LayoutObject->Output(
            TemplateFile => 'AdminITSMConfigItem',
            Data         => \%Param,
        );
        $Output .= $LayoutObject->Footer();

        return $Output;
    }
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Modules::AgentITSMConfigItem;

use strict;
use warnings;

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 );

    return $Self;
}

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

    # get config of frontend module
    $Self->{Config} = $Kernel::OM->Get('Kernel::Config')->Get("ITSMConfigItem::Frontend::$Self->{Action}");

    # get config data
    $Self->{SearchLimit} = $Self->{Config}->{SearchLimit} || 10000;

    my $SessionObject = $Kernel::OM->Get('Kernel::System::AuthSession');

    # store last screen, used for backlinks
    $SessionObject->UpdateSessionID(
        SessionID => $Self->{SessionID},
        Key       => 'LastScreenView',
        Value     => $Self->{RequestedURL},
    );

    # store last screen overview
    $SessionObject->UpdateSessionID(
        SessionID => $Self->{SessionID},
        Key       => 'LastScreenOverview',
        Value     => $Self->{RequestedURL},
    );

    # get param object
    my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');

    # get default parameters, try to get filter (ClassID) from session if not given as parameter
    $Self->{Filter} = $ParamObject->GetParam( Param => 'Filter' )
        || $Self->{AgentITSMConfigItemClassFilter}
        || '';
    $Self->{View} = $ParamObject->GetParam( Param => 'View' ) || '';

    # store filter (ClassID) in session
    $SessionObject->UpdateSessionID(
        SessionID => $Self->{SessionID},
        Key       => 'AgentITSMConfigItemClassFilter',
        Value     => $Self->{Filter},
    );

    # get sorting parameters
    my $SortBy = $ParamObject->GetParam( Param => 'SortBy' )
        || $Self->{Config}->{'SortBy::Default'}
        || 'Number';

    # get ordering parameters
    my $OrderBy = $ParamObject->GetParam( Param => 'OrderBy' )
        || $Self->{Config}->{'Order::Default'}
        || 'Down';

    # set Sort and Order by as Arrays
    my @SortByArray  = ($SortBy);
    my @OrderByArray = ($OrderBy);

    # get general catalog object
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

    # get class list
    my $ClassList = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # get possible deployment state list for config items to be shown
    my $StateList = $GeneralCatalogObject->ItemList(
        Class       => 'ITSM::ConfigItem::DeploymentState',
        Preferences => {
            Functionality => [ 'preproductive', 'productive' ],
        },
    );

    # set the deployment state IDs parameter for the search
    my $DeplStateIDs;
    for my $DeplStateKey ( sort keys %{$StateList} ) {
        push @{$DeplStateIDs}, $DeplStateKey;
    }

    # to store the default class
    my $ClassIDAuto = '';

    # to store the NavBar filters
    my %Filters;

    # define position of the filter in the frontend
    my $PrioCounter = 1000;

    # to store the total number of config items in all classes that the user has access
    my $TotalCount;

    # to store all the clases that the user has access, used in search for filter 'All'
    my $AccessClassList;

    # my config item object
    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    CLASSID:
    for my $ClassID ( sort { ${$ClassList}{$a} cmp ${$ClassList}{$b} } keys %{$ClassList} ) {

        # show menu link only if user has access rights
        my $HasAccess = $ConfigItemObject->Permission(
            Scope   => 'Class',
            ClassID => $ClassID,
            UserID  => $Self->{UserID},
            Type    => $Self->{Config}->{Permission},
        );

        next CLASSID if !$HasAccess;

        # insert this class to be passed as search parameter for filter 'All'
        push @{$AccessClassList}, $ClassID;

        # count all records of this class
        my $ClassCount = $ConfigItemObject->ConfigItemCount(
            ClassID => $ClassID,
        );

        # add the config items number in this class to the total
        $TotalCount += $ClassCount;

        # increase the PrioCounter
        $PrioCounter++;

        # add filter with params for the search method
        $Filters{$ClassID} = {
            Name   => $ClassList->{$ClassID},
            Prio   => $PrioCounter,
            Count  => $ClassCount,
            Search => {
                ClassIDs         => [$ClassID],
                DeplStateIDs     => $DeplStateIDs,
                OrderBy          => \@SortByArray,
                OrderByDirection => \@OrderByArray,
                Limit            => $Self->{SearchLimit},
            },
        };

        # remember the first class id to show this in the overview
        # if no class id was given
        if ( !$ClassIDAuto ) {
            $ClassIDAuto = $ClassID;
        }
    }

    # if only one filter exists
    if ( scalar keys %Filters == 1 ) {

        # get the name of the only filter
        my ($FilterName) = keys %Filters;

        # activate this filter
        $Self->{Filter} = $FilterName;
    }
    else {

        # add default filter, which shows all items
        $Filters{All} = {
            Name   => 'All',
            Prio   => 1000,
            Count  => $TotalCount,
            Search => {
                ClassIDs         => $AccessClassList,
                DeplStateIDs     => $DeplStateIDs,
                OrderBy          => \@SortByArray,
                OrderByDirection => \@OrderByArray,
                Limit            => $Self->{SearchLimit},
            },
        };

        # if no filter was selected activate the filter for the default class
        if ( !$Self->{Filter} ) {
            $Self->{Filter} = $ClassIDAuto;
        }
    }

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

    # check if filter is valid
    if ( !$Filters{ $Self->{Filter} } ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('No access to Class is given!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # investigate refresh
    my $Refresh = $Self->{UserRefreshTime} ? 60 * $Self->{UserRefreshTime} : undef;

    # output header
    my $Output = $LayoutObject->Header(
        Title   => Translatable('Overview'),
        Refresh => $Refresh,
    );
    $Output .= $LayoutObject->NavigationBar();
    $LayoutObject->Print( Output => \$Output );
    $Output = '';

    # display all navbar filters
    my %NavBarFilter;
    for my $Filter ( sort keys %Filters ) {

        # display the navbar filter
        $NavBarFilter{ $Filters{$Filter}->{Prio} } = {
            Filter => $Filter,
            %{ $Filters{$Filter} },
        };
    }

    # search config items which match the selected filter
    my $ConfigItemIDs = $ConfigItemObject->ConfigItemSearchExtended(
        %{ $Filters{ $Self->{Filter} }->{Search} },
    );

    # find out which columns should be shown
    my @ShowColumns;
    if ( $Self->{Config}->{ShowColumns} ) {

        # get all possible columns from config
        my %PossibleColumn = %{ $Self->{Config}->{ShowColumns} };

        # show column "Class" if filter 'All' is selected
        if ( $Self->{Filter} eq 'All' ) {
            $PossibleColumn{Class} = '1';
        }

        # get the column names that should be shown
        COLUMNNAME:
        for my $Name ( sort keys %PossibleColumn ) {
            next COLUMNNAME if !$PossibleColumn{$Name};
            push @ShowColumns, $Name;
        }
    }

    # get the configured columns and reorganize them by class name
    if (
        IsArrayRefWithData( $Self->{Config}->{ShowColumnsByClass} )
        && $Self->{Filter}
        && $Self->{Filter} ne 'All'
        )
    {

        my %ColumnByClass;
        NAME:
        for my $Name ( @{ $Self->{Config}->{ShowColumnsByClass} } ) {
            my ( $Class, $Column ) = split /::/, $Name, 2;

            next NAME if !$Column;

            push @{ $ColumnByClass{$Class} }, $Column;
        }

        # check if there is a specific column config for the selected class
        my $SelectedClass = $ClassList->{ $Self->{Filter} };
        if ( $ColumnByClass{$SelectedClass} ) {
            @ShowColumns = @{ $ColumnByClass{$SelectedClass} };
        }
    }

    # show the list
    my $LinkPage =
        'Filter=' . $LayoutObject->Ascii2Html( Text => $Self->{Filter} )
        . ';View=' . $LayoutObject->Ascii2Html( Text => $Self->{View} )
        . ';SortBy=' . $LayoutObject->Ascii2Html( Text => $SortBy )
        . ';OrderBy=' . $LayoutObject->Ascii2Html( Text => $OrderBy )
        . ';';
    my $LinkSort =
        'Filter=' . $LayoutObject->Ascii2Html( Text => $Self->{Filter} )
        . ';View=' . $LayoutObject->Ascii2Html( Text => $Self->{View} )
        . ';';
    my $LinkFilter =
        'SortBy=' . $LayoutObject->Ascii2Html( Text => $SortBy )
        . ';OrderBy=' . $LayoutObject->Ascii2Html( Text => $OrderBy )
        . ';View=' . $LayoutObject->Ascii2Html( Text => $Self->{View} )
        . ';';

    # show config item list
    $Output .= $LayoutObject->ITSMConfigItemListShow(
        ConfigItemIDs => $ConfigItemIDs,
        Total         => scalar @{$ConfigItemIDs},
        View          => $Self->{View},
        Filter        => $Self->{Filter},
        Filters       => \%NavBarFilter,
        FilterLink    => $LinkFilter,
        TitleName     => $LayoutObject->{LanguageObject}->Translate('Overview: ITSM ConfigItem'),
        TitleValue    => $Filters{ $Self->{Filter} }->{Name},
        Env           => $Self,
        LinkPage      => $LinkPage,
        LinkSort      => $LinkSort,
        ShowColumns   => \@ShowColumns,
        SortBy        => $LayoutObject->Ascii2Html( Text => $SortBy ),
        OrderBy       => $LayoutObject->Ascii2Html( Text => $OrderBy ),
        RequestedURL  => $Self->{RequestedURL},
    );

    # add footer
    $Output .= $LayoutObject->Footer();

    return $Output;
}

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok1vZHVsZXM6OkFnZW50SVRTTUNvbmZpZ0l0ZW1BZGQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgS2VybmVsOjpMYW5ndWFnZSBxdyhUcmFuc2xhdGFibGUpOwpvdXIgJE9iamVjdE1hbmFnZXJEaXNhYmxlZCA9IDE7CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHslUGFyYW19OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgIHJldHVybiAkU2VsZjsKfQoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgZ2V0IGdlbmVyYWwgY2F0YWxvZyBvYmplY3QKICAgIG15ICRHZW5lcmFsQ2F0YWxvZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpOwoKICAgICMgZ2V0IGNsYXNzIGxpc3QKICAgIG15ICRDbGFzc0xpc3QgPSAkR2VuZXJhbENhdGFsb2dPYmplY3QtPkl0ZW1MaXN0KAogICAgICAgIENsYXNzID0+ICdJVFNNOjpDb25maWdJdGVtOjpDbGFzcycsCiAgICApOwoKICAgICMgY2hlY2sgZm9yIGFjY2VzcyByaWdodHMKICAgIGZvciBteSAkQ2xhc3NJRCAoIHNvcnQga2V5cyAleyRDbGFzc0xpc3R9ICkgewogICAgICAgIG15ICRIYXNBY2Nlc3MgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nKS0+UGVybWlzc2lvbigKICAgICAgICAgICAgVHlwZSAgPT4gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpDb25maWcnKS0+R2V0KCJJVFNNQ29uZmlnSXRlbTo6RnJvbnRlbmQ6OiRTZWxmLT57QWN0aW9ufSIpLT57UGVybWlzc2lvbn0sCiAgICAgICAgICAgIFNjb3BlID0+ICdDbGFzcycsCiAgICAgICAgICAgIENsYXNzSUQgPT4gJENsYXNzSUQsCiAgICAgICAgICAgIFVzZXJJRCAgPT4gJFNlbGYtPntVc2VySUR9LAogICAgICAgICk7CgogICAgICAgIGRlbGV0ZSAkQ2xhc3NMaXN0LT57JENsYXNzSUR9IGlmICEkSGFzQWNjZXNzOwogICAgfQoKICAgICMgZ2V0IGxheW91dCBvYmplY3QKICAgIG15ICRMYXlvdXRPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0Jyk7CgogICAgIyBzaG93IHRoZSBsaXN0IG9mIENJIGNsYXNzZXMgc29ydGVkIGJ5IG5hbWUKICAgIGZvciBteSAkSXRlbUlEICggc29ydCB7ICR7JENsYXNzTGlzdH17JGF9IGNtcCAkeyRDbGFzc0xpc3R9eyRifSB9IGtleXMgJXskQ2xhc3NMaXN0fSApIHsKCiAgICAgICAgIyBnZXQgaXRlbSBkYXRhCiAgICAgICAgbXkgJEl0ZW1EYXRhID0gJEdlbmVyYWxDYXRhbG9nT2JqZWN0LT5JdGVtR2V0KAogICAgICAgICAgICBJdGVtSUQgPT4gJEl0ZW1JRCwKICAgICAgICApOwoKICAgICAgICAjIG91dHB1dCBvdmVydmlldyBpdGVtIGxpc3QKICAgICAgICAkTGF5b3V0T2JqZWN0LT5CbG9jaygKICAgICAgICAgICAgTmFtZSA9PiAnT3ZlcnZpZXdJdGVtTGlzdCcsCiAgICAgICAgICAgIERhdGEgPT4gewogICAgICAgICAgICAgICAgQ2xhc3NJRCA9PiAkSXRlbUlELAogICAgICAgICAgICAgICAgTmFtZSAgICA9PiAkQ2xhc3NMaXN0LT57JEl0ZW1JRH0sCiAgICAgICAgICAgIH0sCiAgICAgICAgKTsKICAgIH0KCiAgICAjIG91dHB1dCBoZWFkZXIKICAgIG15ICRPdXRwdXQgPSAkTGF5b3V0T2JqZWN0LT5IZWFkZXIoCiAgICAgICAgVGl0bGUgPT4gVHJhbnNsYXRhYmxlKCdBZGQnKSwKICAgICk7CiAgICAkT3V0cHV0IC49ICRMYXlvdXRPYmplY3QtPk5hdmlnYXRpb25CYXIoKTsKCiAgICAjIG91dHB1dCBvdmVydmlldwogICAgJE91dHB1dCAuPSAkTGF5b3V0T2JqZWN0LT5PdXRwdXQoCiAgICAgICAgVGVtcGxhdGVGaWxlID0+ICdBZ2VudElUU01Db25maWdJdGVtQWRkJywKICAgICAgICBEYXRhICAgICAgICAgPT4gewogICAgICAgICAgICAlUGFyYW0sCiAgICAgICAgfSwKICAgICk7CgogICAgIyBvdXRwdXQgZm9vdGVyCiAgICAkT3V0cHV0IC49ICRMYXlvdXRPYmplY3QtPkZvb3RlcigpOwoKICAgIHJldHVybiAkT3V0cHV0Owp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpNb2R1bGVzOjpBZ2VudElUU01Db25maWdJdGVtQXR0YWNobWVudDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6VmFyaWFibGVDaGVjayBxdyg6YWxsKTsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAogICAgJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6QXR0YWNobWVudCcsCiAgICAnS2VybmVsOjpTeXN0ZW06OkxvZycsCiAgICAnS2VybmVsOjpTeXN0ZW06OldlYjo6UmVxdWVzdCcsCik7CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHslUGFyYW19OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgIHJldHVybiAkU2VsZjsKfQoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRMYXlvdXRPYmplY3QgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0Jyk7CiAgICBteSAkTG9nT2JqZWN0ICAgICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpOwogICAgbXkgJFBhcmFtT2JqZWN0ICAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpXZWI6OlJlcXVlc3QnKTsKICAgIG15ICRDSUF0dGFjaG1lbnRPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OkF0dGFjaG1lbnQnKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgTkVFREVEOgogICAgZm9yIG15ICROZWVkZWQgKHF3KEF0dGFjaG1lbnRJRCkpIHsKICAgICAgICAkUGFyYW17JE5lZWRlZH0gPSAkUGFyYW1PYmplY3QtPkdldFBhcmFtKCBQYXJhbSA9PiAkTmVlZGVkICk7CgogICAgICAgIG5leHQgTkVFREVEIGlmIGRlZmluZWQgJFBhcmFteyROZWVkZWR9OwoKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJQYXJhbWV0ZXIgJyROZWVkZWQnIGlzIG5lZWRlZCEiLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIG15ICVBdHRhY2htZW50ID0gJENJQXR0YWNobWVudE9iamVjdC0+Q0lBdHRhY2htZW50R2V0KAogICAgICAgIEF0dGFjaG1lbnRJRCA9PiAkUGFyYW17QXR0YWNobWVudElEfSwKICAgICk7CgogICAgaWYgKCAhJUF0dGFjaG1lbnQgKSB7CiAgICAgICAgcmV0dXJuICRMYXlvdXRPYmplY3QtPlJlZGlyZWN0KAogICAgICAgICAgICBPUCA9PiAkU2VsZi0+e0xhc3RTY3JlZW5WaWV3fSwKICAgICAgICApOwogICAgfQogICAgZWxzZSB7CiAgICAgICAgcmV0dXJuICRMYXlvdXRPYmplY3QtPkF0dGFjaG1lbnQoCiAgICAgICAgICAgICVBdHRhY2htZW50CiAgICAgICAgKTsKICAgIH0KfQoKMTsK
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --
package Kernel::Modules::AgentITSMConfigItemBulk;

use strict;
use warnings;

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 );

    return $Self;
}

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

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

    # check if bulk feature is enabled
    if ( !$ConfigObject->Get('ITSMConfigItem::Frontend::BulkFeature') ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Bulk feature is not enabled!'),
        );
    }

    # get param object
    my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');

    # get involved config items, filtering empty ConfigItemIDs
    my @ConfigItemIDs = grep {$_}
        $ParamObject->GetArray( Param => 'ConfigItemID' );

    # check needed stuff
    if ( !@ConfigItemIDs ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('No ConfigItemID is given!'),
            Comment => Translatable('You need at least one selected Configuration Item!'),
        );
    }
    my $Output = $LayoutObject->Header(
        Type => 'Small',
    );

    # declare the variables for all the parameters
    my %Error;

    my %GetParam;

    # get config item object
    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    # get all parameters and check for errors
    if ( $Self->{Subaction} eq 'Do' ) {

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

        # get all parameters
        for my $Key (
            qw( LinkTogether LinkTogetherAnother LinkType LinkTogetherLinkType DeplStateID
            InciStateID )
            )
        {
            $GetParam{$Key} = $ParamObject->GetParam( Param => $Key ) || '';
        }

        if ( $GetParam{'LinkTogetherAnother'} ) {
            $Kernel::OM->Get('Kernel::System::CheckItem')->StringClean(
                StringRef => \$GetParam{'LinkTogetherAnother'},
                TrimLeft  => 1,
                TrimRight => 1,
            );
            my $ConfigItemID = $ConfigItemObject->ConfigItemLookup(
                ConfigItemNumber => $GetParam{'LinkTogetherAnother'},
            );
            if ( !$ConfigItemID ) {
                $Error{'LinkTogetherAnotherInvalid'} = 'ServerError';
            }
        }
    }

    # process config item
    my @ConfigItemIDSelected;
    my $ActionFlag = 0;
    my $Counter    = 1;

    # get link object
    my $LinkObject = $Kernel::OM->Get('Kernel::System::LinkObject');

    CONFIGITEM_ID:
    for my $ConfigItemID (@ConfigItemIDs) {
        my $ConfigItem = $ConfigItemObject->ConfigItemGet(
            ConfigItemID => $ConfigItemID,
        );

        my $Config = $ConfigObject->Get("ITSMConfigItem::Frontend::AgentITSMConfigItemEdit");

        # check permissions
        my $Access = $ConfigItemObject->Permission(
            Scope  => 'Item',
            ItemID => $ConfigItemID,
            UserID => $Self->{UserID},
            Type   => $Config->{Permission},
        );

        if ( !$Access ) {

            # error screen, don't show config item
            $Output .= $LayoutObject->Notify(
                Data => $LayoutObject->{LanguageObject}->Translate(
                    'You don\'t have write access to this configuration item: %s.',
                    $ConfigItem->{Number},
                ),
            );
            next CONFIGITEM_ID;
        }

        # remember selected config item ids
        push @ConfigItemIDSelected, $ConfigItemID;

        # do some actions on CIs
        if ( ( $Self->{Subaction} eq 'Do' ) && ( !%Error ) ) {

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

            # bulk action version add
            if ( $GetParam{DeplStateID} || $GetParam{InciStateID} ) {

                # get current version of the config item
                my $CurrentVersion = $ConfigItemObject->VersionGet(
                    ConfigItemID => $ConfigItemID,
                    XMLDataGet   => 1,
                );

                my $NewDeplStateID = $CurrentVersion->{DeplStateID};
                my $NewInciStateID = $CurrentVersion->{InciStateID};

                if ( IsNumber( $GetParam{DeplStateID} ) ) {
                    $NewDeplStateID = $GetParam{DeplStateID};
                }
                if ( IsNumber( $GetParam{InciStateID} ) ) {
                    $NewInciStateID = $GetParam{InciStateID};
                }

                my $VersionID = $ConfigItemObject->VersionAdd(
                    ConfigItemID => $ConfigItemID,
                    Name         => $CurrentVersion->{Name},
                    DefinitionID => $CurrentVersion->{DefinitionID},
                    DeplStateID  => $NewDeplStateID,
                    InciStateID  => $NewInciStateID,
                    XMLData      => $CurrentVersion->{XMLData},
                    UserID       => $Self->{UserID},
                );
            }

            # bulk action links
            # link all config items to another config item
            if ( $GetParam{'LinkTogetherAnother'} ) {
                my $MainConfigItemID = $ConfigItemObject->ConfigItemLookup(
                    ConfigItemNumber => $GetParam{'LinkTogetherAnother'},
                );

                # split the type identifier
                my @Type = split q{::}, $GetParam{LinkType};

                if ( $Type[0] && $Type[1] && ( $Type[1] eq 'Source' || $Type[1] eq 'Target' ) ) {

                    my $SourceKey = $ConfigItemID;
                    my $TargetKey = $MainConfigItemID;

                    if ( $Type[1] eq 'Target' ) {
                        $SourceKey = $MainConfigItemID;
                        $TargetKey = $ConfigItemID;
                    }

                    for my $ConfigItemIDPartner (@ConfigItemIDs) {
                        if ( $MainConfigItemID ne $ConfigItemIDPartner ) {
                            $LinkObject->LinkAdd(
                                SourceObject => 'ITSMConfigItem',
                                SourceKey    => $SourceKey,
                                TargetObject => 'ITSMConfigItem',
                                TargetKey    => $TargetKey,
                                Type         => $Type[0],
                                State        => 'Valid',
                                UserID       => $Self->{UserID},
                            );
                        }
                    }
                }
            }

            # link together
            if ( $GetParam{'LinkTogether'} ) {

                # split the type identifier
                my @Type = split q{::}, $GetParam{LinkTogetherLinkType};

                if ( $Type[0] && $Type[1] && ( $Type[1] eq 'Source' || $Type[1] eq 'Target' ) ) {
                    for my $ConfigItemIDPartner (@ConfigItemIDs) {

                        my $SourceKey = $ConfigItemID;
                        my $TargetKey = $ConfigItemIDPartner;

                        if ( $Type[1] eq 'Target' ) {
                            $SourceKey = $ConfigItemIDPartner;
                            $TargetKey = $ConfigItemID;
                        }

                        if ( $ConfigItemID ne $ConfigItemIDPartner ) {
                            $LinkObject->LinkAdd(
                                SourceObject => 'ITSMConfigItem',
                                SourceKey    => $SourceKey,
                                TargetObject => 'ITSMConfigItem',
                                TargetKey    => $TargetKey,
                                Type         => $Type[0],
                                State        => 'Valid',
                                UserID       => $Self->{UserID},
                            );
                        }
                    }
                }
            }
            $ActionFlag = 1;
        }
        $Counter++;
    }

    # redirect
    if ($ActionFlag) {
        return $LayoutObject->PopupClose(
            URL => ( $Self->{LastScreenOverview} || 'Action=AgentDashboard' ),
        );
    }

    $Output .= $Self->_Mask(
        %Param,
        %GetParam,
        ConfigItemIDs => \@ConfigItemIDSelected,
        Errors        => \%Error,
    );
    $Output .= $LayoutObject->Footer(
        Type => 'Small',
    );
    return $Output;
}

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

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

    # prepare errors!
    if ( $Param{Errors} ) {
        for my $KeyError ( sort keys %{ $Param{Errors} } ) {
            $Param{$KeyError} = $LayoutObject->Ascii2Html( Text => $Param{Errors}->{$KeyError} );
        }
    }

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

    # remember config item ids
    if ( $Param{ConfigItemIDs} ) {
        for my $ConfigItemID ( @{ $Param{ConfigItemIDs} } ) {
            $LayoutObject->Block(
                Name => 'UsedConfigItemID',
                Data => {
                    ConfigItemID => $ConfigItemID,
                },
            );
        }
    }

    # get needed objects
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
    my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');

    $Self->{Config} = $ConfigObject->Get("ITSMConfigItem::Frontend::$Self->{Action}");

    # deployment state
    if ( $Self->{Config}->{DeplState} ) {
        my $DeplStateList = $GeneralCatalogObject->ItemList(
            Class => 'ITSM::ConfigItem::DeploymentState',
        );

        # generate DeplStateStrg
        $Param{DeplStateStrg} = $LayoutObject->BuildSelection(
            Data         => $DeplStateList,
            Name         => 'DeplStateID',
            PossibleNone => 1,
            SelectedID   => $Param{DeplStateID},
            Sort         => 'AlphanumericValue',
        );
        $LayoutObject->Block(
            Name => 'DeplState',
            Data => {%Param},
        );
    }

    # incident state
    if ( $Self->{Config}->{InciState} ) {

        # get incident state list
        my $InciStateList = $GeneralCatalogObject->ItemList(
            Class       => 'ITSM::Core::IncidentState',
            Preferences => {
                Functionality => [ 'operational', 'incident' ],
            },
        );

        # generate InciStateStrg
        $Param{InciStateStrg} = $LayoutObject->BuildSelection(
            Data         => $InciStateList,
            Name         => 'InciStateID',
            PossibleNone => 1,
            SelectedID   => $Param{InciStateID},
            Sort         => 'AlphanumericValue',
        );
        $LayoutObject->Block(
            Name => 'InciState',
            Data => {%Param},
        );
    }

    # get link object
    my $LinkObject = $Kernel::OM->Get('Kernel::System::LinkObject');

    # link types list
    # get possible types list
    my %PossibleTypesList = $LinkObject->PossibleTypesList(
        Object1 => 'ITSMConfigItem',
        Object2 => 'ITSMConfigItem',
        UserID  => $Self->{UserID},
    );

    # define blank line entry
    my %BlankLine = (
        Key      => '-',
        Value    => '-------------------------',
        Disabled => 1,
    );

    # create the selectable type list
    my $Counter = 0;
    my @SelectableTypesList;
    my @LinkTogetherTypeList;
    POSSIBLETYPE:
    for my $PossibleType ( sort { lc $a cmp lc $b } keys %PossibleTypesList ) {

        # lookup type id
        my $TypeID = $LinkObject->TypeLookup(
            Name   => $PossibleType,
            UserID => $Self->{UserID},
        );

        # get type
        my %Type = $LinkObject->TypeGet(
            TypeID => $TypeID,
            UserID => $Self->{UserID},
        );

        # type list for link together can contain only
        # link types which are not directed (not pointed)
        if ( !$Type{Pointed} ) {

            # create the source name
            my %SourceName;
            $SourceName{Key}   = $PossibleType . '::Source';
            $SourceName{Value} = $Type{SourceName};

            push @LinkTogetherTypeList, \%SourceName;
        }

        # create the source name
        my %SourceName;
        $SourceName{Key}   = $PossibleType . '::Source';
        $SourceName{Value} = $Type{SourceName};

        push @SelectableTypesList, \%SourceName;

        next POSSIBLETYPE if !$Type{Pointed};

        # create the target name
        my %TargetName;
        $TargetName{Key}   = $PossibleType . '::Target';
        $TargetName{Value} = $Type{TargetName};

        push @SelectableTypesList, \%TargetName;
    }
    continue {

        # add blank line
        push @SelectableTypesList, \%BlankLine;

        $Counter++;
    }

    # removed last (empty) entry
    pop @SelectableTypesList;

    # add blank lines on top and bottom of the list if more then two linktypes
    if ( $Counter > 2 ) {
        unshift @SelectableTypesList, \%BlankLine;
        push @SelectableTypesList, \%BlankLine;
    }

    # generate LinkTypeStrg
    $Param{LinkTypeStrg} = $LayoutObject->BuildSelection(
        Data         => \@SelectableTypesList,
        Name         => 'LinkType',
        PossibleNone => 0,
        SelectedID   => $Param{TypeIdentifier} || 'AlternativeTo::Source',
        Sort         => 'AlphanumericValue',
    );
    $Param{LinkTogetherLinkTypeStrg} = $LayoutObject->BuildSelection(
        Data         => \@LinkTogetherTypeList,
        Name         => 'LinkTogetherLinkType',
        PossibleNone => 0,
        SelectedID   => $Param{TypeIdentifier} || 'AlternativeTo::Source',
        Sort         => 'AlphanumericValue',
    );

    $Param{LinkTogetherYesNoOption} = $LayoutObject->BuildSelection(
        Data       => $ConfigObject->Get('YesNoOptions'),
        Name       => 'LinkTogether',
        SelectedID => $Param{LinkTogether} || 0,
    );

    # get output back
    return $LayoutObject->Output(
        TemplateFile => 'AgentITSMConfigItemBulk',
        Data         => \%Param
    );
}

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpNb2R1bGVzOjpBZ2VudElUU01Db25maWdJdGVtQ3VzdG9tZXJDSXNXaWRnZXQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0JywKICAgICdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW1DdXN0b21lckNJcycsCiAgICAnS2VybmVsOjpTeXN0ZW06OkxpbmtPYmplY3QnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpXZWI6OlJlcXVlc3QnLAopOwoKdXNlIEtlcm5lbDo6U3lzdGVtOjpWYXJpYWJsZUNoZWNrIHF3KDphbGwpOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRTZWxmID0geyVQYXJhbX07CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJFBhcmFtT2JqZWN0ICAgICAgICAgICAgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6V2ViOjpSZXF1ZXN0Jyk7CiAgICBteSAkTGF5b3V0T2JqZWN0ICAgICAgICAgICAgICAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnKTsKICAgIG15ICRMaW5rT2JqZWN0ICAgICAgICAgICAgICAgICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxpbmtPYmplY3QnKTsKICAgIG15ICRJVFNNQ29uZmlnSXRlbUN1c3RvbWVyQ0lzT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtQ3VzdG9tZXJDSXMnKTsKCiAgICBmb3IgbXkgJFBhcmFtIChxdyhUaWNrZXRJRCBDb25maWdJdGVtSUQgQ3VzdG9tZXJVc2VySUQgQ3VzdG9tZXJJRCkpIHsKICAgICAgICAkUGFyYW17JFBhcmFtfSA9ICRQYXJhbU9iamVjdC0+R2V0UGFyYW0oIFBhcmFtID0+ICRQYXJhbSApIC8vICcnOwogICAgfQoKICAgIG15ICRKU09OOwoKICAgIGlmICggJFNlbGYtPntTdWJhY3Rpb259IGVxICdMaW5rQWRkJyApIHsKICAgICAgICAkTGlua09iamVjdC0+TGlua0FkZCgKICAgICAgICAgICAgU291cmNlT2JqZWN0ID0+ICdUaWNrZXQnLAogICAgICAgICAgICBTb3VyY2VLZXkgICAgPT4gJFBhcmFte1RpY2tldElEfSwKICAgICAgICAgICAgVGFyZ2V0T2JqZWN0ID0+ICdJVFNNQ29uZmlnSXRlbScsCiAgICAgICAgICAgIFRhcmdldEtleSAgICA9PiAkUGFyYW17Q29uZmlnSXRlbUlEfSwKICAgICAgICAgICAgVHlwZSAgICAgICAgID0+ICdSZWxldmFudFRvJywKICAgICAgICAgICAgU3RhdGUgICAgICAgID0+ICdWYWxpZCcsCiAgICAgICAgICAgIFVzZXJJRCAgICAgICA9PiAkU2VsZi0+e1VzZXJJRH0sCiAgICAgICAgKTsKICAgIH0KICAgIGVsc2lmICggJFNlbGYtPntTdWJhY3Rpb259IGVxICdMaW5rRGVsZXRlJyApIHsKICAgICAgICAkTGlua09iamVjdC0+TGlua0RlbGV0ZSgKICAgICAgICAgICAgT2JqZWN0MSA9PiAnVGlja2V0JywKICAgICAgICAgICAgS2V5MSAgICA9PiAkUGFyYW17VGlja2V0SUR9LAogICAgICAgICAgICBPYmplY3QyID0+ICdJVFNNQ29uZmlnSXRlbScsCiAgICAgICAgICAgIEtleTIgICAgPT4gJFBhcmFte0NvbmZpZ0l0ZW1JRH0sCiAgICAgICAgICAgIFR5cGUgICAgPT4gJ1JlbGV2YW50VG8nLAogICAgICAgICAgICBVc2VySUQgID0+ICRTZWxmLT57VXNlcklEfSwKICAgICAgICApOwogICAgfQogICAgZWxzaWYgKCAkU2VsZi0+e1N1YmFjdGlvbn0gZXEgJ0N1c3RvbWVyVXBkYXRlJyAmJiAkUGFyYW17Q3VzdG9tZXJVc2VySUR9ICkgewoKICAgICAgICBteSBAQ29uZmlnSXRlbXMgPSAkSVRTTUNvbmZpZ0l0ZW1DdXN0b21lckNJc09iamVjdC0+R2V0UG9zc2libGVDdXN0b21lckNJcygKICAgICAgICAgICAgQ3VzdG9tZXJVc2VySUQgPT4gJFBhcmFte0N1c3RvbWVyVXNlcklEfSwKICAgICAgICAgICAgQ3VzdG9tZXJJRCAgICAgPT4gJFBhcmFte0N1c3RvbWVySUR9LAogICAgICAgICk7CgogICAgICAgIGlmIChAQ29uZmlnSXRlbXMpIHsKCiAgICAgICAgICAgIGZvciBteSAkQ29uZmlnSXRlbSAoQENvbmZpZ0l0ZW1zKSB7CiAgICAgICAgICAgICAgICAkTGF5b3V0T2JqZWN0LT5CbG9jaygKICAgICAgICAgICAgICAgICAgICBOYW1lID0+ICdDb25maWdJdGVtcycsCiAgICAgICAgICAgICAgICAgICAgRGF0YSA9PiB7CiAgICAgICAgICAgICAgICAgICAgICAgICV7JENvbmZpZ0l0ZW19LAogICAgICAgICAgICAgICAgICAgIH0sCiAgICAgICAgICAgICAgICApOwogICAgICAgICAgICB9CgogICAgICAgICAgICBteSAkQ3VzdG9tZXJDSVdpZGdldEhUTUwgPSAkTGF5b3V0T2JqZWN0LT5PdXRwdXQoCiAgICAgICAgICAgICAgICBUZW1wbGF0ZUZpbGUgPT4gJ0FnZW50SVRTTUNvbmZpZ0l0ZW1DdXN0b21lckNJc1dpZGdldCcsCiAgICAgICAgICAgICAgICBEYXRhICAgICAgICAgPT4ge30sCiAgICAgICAgICAgICk7CgogICAgICAgICAgICAkSlNPTiA9ICRMYXlvdXRPYmplY3QtPkpTT05FbmNvZGUoCiAgICAgICAgICAgICAgICBOb1F1b3RlcyA9PiAxLAogICAgICAgICAgICAgICAgRGF0YSAgICAgPT4gewogICAgICAgICAgICAgICAgICAgIENvbmZpZ0l0ZW1zID0+ICRDdXN0b21lckNJV2lkZ2V0SFRNTCwKICAgICAgICAgICAgICAgIH0sCiAgICAgICAgICAgICk7CiAgICAgICAgfQogICAgfQoKICAgIHJldHVybiAkTGF5b3V0T2JqZWN0LT5BdHRhY2htZW50KAogICAgICAgIENvbnRlbnRUeXBlID0+ICdhcHBsaWNhdGlvbi9qc29uOyBjaGFyc2V0PScgLiAkTGF5b3V0T2JqZWN0LT57Q2hhcnNldH0sCiAgICAgICAgQ29udGVudCAgICAgPT4gJEpTT04gLy8gJ3t9JywKICAgICAgICBUeXBlICAgICAgICA9PiAnaW5saW5lJywKICAgICAgICBOb0NhY2hlICAgICA9PiAxLAogICAgKTsKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQpwYWNrYWdlIEtlcm5lbDo6TW9kdWxlczo6QWdlbnRJVFNNQ29uZmlnSXRlbURlbGV0ZTsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSBLZXJuZWw6Okxhbmd1YWdlIHF3KFRyYW5zbGF0YWJsZSk7CgpvdXIgJE9iamVjdE1hbmFnZXJEaXNhYmxlZCA9IDE7CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHslUGFyYW19OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgIHJldHVybiAkU2VsZjsKfQoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgZ2V0IG5lZWRlZCBDb25maWdJdGVtSUQKICAgIG15ICRDb25maWdJdGVtSUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6V2ViOjpSZXF1ZXN0JyktPkdldFBhcmFtKCBQYXJhbSA9PiAnQ29uZmlnSXRlbUlEJyApOwoKICAgICMgZ2V0IGxheW91dCBvYmplY3QKICAgIG15ICRMYXlvdXRPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0Jyk7CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRDb25maWdJdGVtSUQgKSB7CiAgICAgICAgcmV0dXJuICRMYXlvdXRPYmplY3QtPkVycm9yU2NyZWVuKAogICAgICAgICAgICBNZXNzYWdlID0+IFRyYW5zbGF0YWJsZSgnTm8gQ29uZmlnSXRlbUlEIGlzIGdpdmVuIScpLAogICAgICAgICAgICBDb21tZW50ID0+IFRyYW5zbGF0YWJsZSgnUGxlYXNlIGNvbnRhY3QgdGhlIGFkbWluaXN0cmF0b3IuJyksCiAgICAgICAgKTsKICAgIH0KCiAgICAjIGdldCBjb25maWcgaXRlbSBvYmplY3QKICAgIG15ICRDb25maWdJdGVtT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtJyk7CgogICAgIyBnZXQgY29uZmlnIG9mIGZyb250ZW5kIG1vZHVsZQogICAgJFNlbGYtPntDb25maWd9ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpDb25maWcnKS0+R2V0KCJJVFNNQ29uZmlnSXRlbTo6RnJvbnRlbmQ6OiRTZWxmLT57QWN0aW9ufSIpOwoKICAgICMgY2hlY2sgcGVybWlzc2lvbnMKICAgIG15ICRBY2Nlc3MgPSAkQ29uZmlnSXRlbU9iamVjdC0+UGVybWlzc2lvbigKICAgICAgICBUeXBlICAgPT4gJFNlbGYtPntDb25maWd9LT57UGVybWlzc2lvbn0sCiAgICAgICAgU2NvcGUgID0+ICdJdGVtJywKICAgICAgICBBY3Rpb24gPT4gJFNlbGYtPntBY3Rpb259LAogICAgICAgIEl0ZW1JRCA9PiAkQ29uZmlnSXRlbUlELAogICAgICAgIFVzZXJJRCA9PiAkU2VsZi0+e1VzZXJJRH0sCiAgICApOwoKICAgICMgZXJyb3Igc2NyZWVuCiAgICBpZiAoICEkQWNjZXNzICkgewogICAgICAgIHJldHVybiAkTGF5b3V0T2JqZWN0LT5Ob1Blcm1pc3Npb24oCiAgICAgICAgICAgIE1lc3NhZ2UgPT4KICAgICAgICAgICAgICAgICRMYXlvdXRPYmplY3QtPntMYW5ndWFnZU9iamVjdH0tPlRyYW5zbGF0ZSggJ1lvdSBuZWVkICVzIHBlcm1pc3Npb25zIScsICRTZWxmLT57Q29uZmlnfS0+e1Blcm1pc3Npb259ICksCiAgICAgICAgICAgIFdpdGhIZWFkZXIgPT4gJ3llcycsCiAgICAgICAgKTsKICAgIH0KCiAgICAjIGdldCBjb25maWcgaXRlbSBkYXRhCiAgICBteSAkQ29uZmlnSXRlbSA9ICRDb25maWdJdGVtT2JqZWN0LT5Db25maWdJdGVtR2V0KAogICAgICAgIENvbmZpZ0l0ZW1JRCA9PiAkQ29uZmlnSXRlbUlELAogICAgICAgIFVzZXJJRCAgICAgICA9PiAkU2VsZi0+e1VzZXJJRH0sCiAgICApOwoKICAgICMgY2hlY2sgaWYgY29uZmlnIGl0ZW0gaXMgZm91bmQKICAgIGlmICggISRDb25maWdJdGVtICkgewogICAgICAgIHJldHVybiAkTGF5b3V0T2JqZWN0LT5FcnJvclNjcmVlbigKICAgICAgICAgICAgTWVzc2FnZSA9PgogICAgICAgICAgICAgICAgJExheW91dE9iamVjdC0+e0xhbmd1YWdlT2JqZWN0fS0+VHJhbnNsYXRlKCAnQ29uZmlnIGl0ZW0gIiVzIiBub3QgZm91bmQgaW4gZGF0YWJhc2UhJywgJENvbmZpZ0l0ZW1JRCApLAogICAgICAgICAgICBDb21tZW50ID0+IFRyYW5zbGF0YWJsZSgnUGxlYXNlIGNvbnRhY3QgdGhlIGFkbWluaXN0cmF0b3IuJyksCiAgICAgICAgKTsKICAgIH0KCiAgICAjIGRlbGV0ZSB0aGUgY29uZmlnIGl0ZW0KICAgIGlmICggJFNlbGYtPntTdWJhY3Rpb259IGVxICdDb25maWdJdGVtRGVsZXRlJyApIHsKCiAgICAgICAgIyBkZWxldGUgdGhlIGNvbmZpZyBpdGVtCiAgICAgICAgbXkgJENvdWxkRGVsZXRlQ29uZmlnSXRlbSA9ICRDb25maWdJdGVtT2JqZWN0LT5Db25maWdJdGVtRGVsZXRlKAogICAgICAgICAgICBDb25maWdJdGVtSUQgPT4gJENvbmZpZ0l0ZW1JRCwKICAgICAgICAgICAgVXNlcklEICAgICAgID0+ICRTZWxmLT57VXNlcklEfSwKICAgICAgICApOwoKICAgICAgICBpZiAoJENvdWxkRGVsZXRlQ29uZmlnSXRlbSkgewoKICAgICAgICAgICAgIyByZWRpcmVjdCB0byBjb25maWcgaXRlbSBvdmVydmlldywgd2hlbiB0aGUgZGVsZXRpb24gd2FzIHN1Y2Nlc3NmdWwKICAgICAgICAgICAgcmV0dXJuICRMYXlvdXRPYmplY3QtPlJlZGlyZWN0KAogICAgICAgICAgICAgICAgT1AgPT4gIkFjdGlvbj1BZ2VudElUU01Db25maWdJdGVtIiwKICAgICAgICAgICAgKTsKICAgICAgICB9CiAgICAgICAgZWxzZSB7CgogICAgICAgICAgICAjIHNob3cgZXJyb3IgbWVzc2FnZSwgd2hlbiBkZWxldGUgZmFpbGVkCiAgICAgICAgICAgIHJldHVybiAkTGF5b3V0T2JqZWN0LT5FcnJvclNjcmVlbigKICAgICAgICAgICAgICAgIE1lc3NhZ2UgPT4gJExheW91dE9iamVjdC0+e0xhbmd1YWdlT2JqZWN0fQogICAgICAgICAgICAgICAgICAgIC0+VHJhbnNsYXRlKCAnV2FzIG5vdCBhYmxlIHRvIGRlbGV0ZSB0aGUgY29uZmlnaXRlbSBJRCAlcyEnLCAkQ29uZmlnSXRlbUlEICksCiAgICAgICAgICAgICAgICBDb21tZW50ID0+IFRyYW5zbGF0YWJsZSgnUGxlYXNlIGNvbnRhY3QgdGhlIGFkbWluaXN0cmF0b3IuJyksCiAgICAgICAgICAgICk7CiAgICAgICAgfQogICAgfQoKICAgICMgZ2V0IGxhdGVzdCB2ZXJzaW9uIGRhdGEKICAgIG15ICRWZXJzaW9uID0gJENvbmZpZ0l0ZW1PYmplY3QtPlZlcnNpb25HZXQoCiAgICAgICAgQ29uZmlnSXRlbUlEID0+ICRDb25maWdJdGVtSUQsCiAgICApOwoKICAgIGlmICggISRWZXJzaW9uLT57VmVyc2lvbklEfSApIHsKICAgICAgICByZXR1cm4gJExheW91dE9iamVjdC0+RXJyb3JTY3JlZW4oCiAgICAgICAgICAgIE1lc3NhZ2UgPT4KICAgICAgICAgICAgICAgICRMYXlvdXRPYmplY3QtPntMYW5ndWFnZU9iamVjdH0tPlRyYW5zbGF0ZSggJ05vIHZlcnNpb24gZm91bmQgZm9yIENvbmZpZ0l0ZW1JRCAlcyEnLCAkQ29uZmlnSXRlbUlEICksCiAgICAgICAgICAgIENvbW1lbnQgPT4gVHJhbnNsYXRhYmxlKCdQbGVhc2UgY29udGFjdCB0aGUgYWRtaW5pc3RyYXRvci4nKSwKICAgICAgICApOwogICAgfQoKICAgICMgc2V0IHRoZSBkaWFsb2cgdHlwZS4gQXMgZGVmYXVsdCwgdGhlIGRpYWxvZyB3aWxsIGhhdmUgMiBidXR0b25zOiBZZXMgYW5kIE5vCiAgICBteSAkRGlhbG9nVHlwZSA9ICdDb25maXJtYXRpb24nOwoKICAgICMgb3V0cHV0IGNvbnRlbnQKICAgIG15ICRPdXRwdXQgPSAkTGF5b3V0T2JqZWN0LT5PdXRwdXQoCiAgICAgICAgVGVtcGxhdGVGaWxlID0+ICdBZ2VudElUU01Db25maWdJdGVtRGVsZXRlJywKICAgICAgICBEYXRhICAgICAgICAgPT4gewogICAgICAgICAgICAlUGFyYW0sCiAgICAgICAgICAgICV7JENvbmZpZ0l0ZW19LAogICAgICAgICAgICAleyRWZXJzaW9ufSwKICAgICAgICB9LAogICAgKTsKCiAgICAjIGJ1aWxkIHRoZSByZXR1cm5lZCBkYXRhIHN0cnVjdHVyZQogICAgbXkgJURhdGEgPSAoCiAgICAgICAgSFRNTCAgICAgICA9PiAkT3V0cHV0LAogICAgICAgIERpYWxvZ1R5cGUgPT4gJERpYWxvZ1R5cGUsCiAgICApOwoKICAgICMgcmV0dXJuIEpTT04tU3RyaW5nIGJlY2F1c2Ugb2YgQUpBWC1Nb2RlCiAgICBteSAkT3V0cHV0SlNPTiA9ICRMYXlvdXRPYmplY3QtPkpTT05FbmNvZGUoIERhdGEgPT4gXCVEYXRhICk7CgogICAgcmV0dXJuICRMYXlvdXRPYmplY3QtPkF0dGFjaG1lbnQoCiAgICAgICAgQ29udGVudFR5cGUgPT4gJ2FwcGxpY2F0aW9uL2pzb247IGNoYXJzZXQ9JyAuICRMYXlvdXRPYmplY3QtPntDaGFyc2V0fSwKICAgICAgICBDb250ZW50ICAgICA9PiAkT3V0cHV0SlNPTiwKICAgICAgICBUeXBlICAgICAgICA9PiAnaW5saW5lJywKICAgICAgICBOb0NhY2hlICAgICA9PiAxLAogICAgKTsKfQoKMTsK
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Modules::AgentITSMConfigItemEdit;

use strict;
use warnings;

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

our $ObjectManagerDisabled = 1;

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

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

    return $Self;
}

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

    # my param object
    my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');

    # get configitem id and class id
    my $ConfigItem = {};
    $ConfigItem->{ConfigItemID} = $ParamObject->GetParam( Param => 'ConfigItemID' ) || 0;
    $ConfigItem->{ClassID}      = $ParamObject->GetParam( Param => 'ClassID' )      || 0;
    my $DuplicateID = $ParamObject->GetParam( Param => 'DuplicateID' ) || 0;

    my $HasAccess;

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

    # get config of frontend module
    $Self->{Config} = $ConfigObject->Get("ITSMConfigItem::Frontend::$Self->{Action}");

    # get needed data
    if ( $ConfigItem->{ConfigItemID} && $ConfigItem->{ConfigItemID} ne 'NEW' ) {

        # check access for config item
        $HasAccess = $ConfigItemObject->Permission(
            Scope  => 'Item',
            ItemID => $ConfigItem->{ConfigItemID},
            UserID => $Self->{UserID},
            Type   => $Self->{Config}->{Permission},
        );

        # get config item
        $ConfigItem = $ConfigItemObject->ConfigItemGet(
            ConfigItemID => $ConfigItem->{ConfigItemID},
        );
    }
    elsif ($DuplicateID) {

        # get config item to duplicate
        $ConfigItem = $ConfigItemObject->ConfigItemGet(
            ConfigItemID => $DuplicateID,
        );

        # check access for config item
        $HasAccess = $ConfigItemObject->Permission(
            Scope  => 'Item',
            ItemID => $ConfigItem->{ConfigItemID},
            UserID => $Self->{UserID},
            Type   => $Self->{Config}->{Permission},
        );

        # set config item id and number
        $ConfigItem->{ConfigItemID} = 'NEW';
        $ConfigItem->{Number}       = Translatable('New');
    }
    elsif ( $ConfigItem->{ClassID} ) {

        # set config item id and number
        $ConfigItem->{ConfigItemID} = 'NEW';
        $ConfigItem->{Number}       = Translatable('New');

        # check access for config item
        $HasAccess = $ConfigItemObject->Permission(
            Scope   => 'Class',
            ClassID => $ConfigItem->{ClassID},
            UserID  => $Self->{UserID},
            Type    => $Self->{Config}->{Permission},
        );

        # get class list
        my $ClassList = $GeneralCatalogObject->ItemList(
            Class => 'ITSM::ConfigItem::Class',
        );
        $ConfigItem->{Class} = $ClassList->{ $ConfigItem->{ClassID} };
    }
    else {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('No ConfigItemID, DuplicateID or ClassID is given!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # if user has no access rights show error page
    if ( !$HasAccess ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('No access is given!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # get definition
    my $XMLDefinition = $ConfigItemObject->DefinitionGet(
        ClassID => $ConfigItem->{ClassID},
    );

    # abort, if no definition is defined
    if ( !$XMLDefinition->{DefinitionID} ) {
        return $LayoutObject->ErrorScreen(
            Message => $LayoutObject->{LanguageObject}
                ->Translate( 'No definition was defined for class %s!', $ConfigItem->{Class} ),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # get upload cache object
    my $UploadCacheObject = $Kernel::OM->Get('Kernel::System::Web::UploadCache');

    # get form id
    $Self->{FormID} = $ParamObject->GetParam( Param => 'FormID' );

    # create form id
    if ( !$Self->{FormID} ) {
        $Self->{FormID} = $UploadCacheObject->FormIDCreate();
    }

    # when there's no ClassID it means, an existing config item is edited as the ClassID is only
    # provided as GET param when creating a new config item
    if ( !$ParamObject->GetParam( Param => 'ClassID' ) ) {

        # get all attachments meta data
        my @ExistingAttachments = $ConfigItemObject->ConfigItemAttachmentList(
            ConfigItemID => $ConfigItem->{ConfigItemID},
        );

        # copy all existing attachments to upload cache
        FILENAME:
        for my $Filename (@ExistingAttachments) {

            # get the existing attachment data
            my $AttachmentData = $ConfigItemObject->ConfigItemAttachmentGet(
                ConfigItemID => $ConfigItem->{ConfigItemID},
                Filename     => $Filename,
                UserID       => $Self->{UserID},
            );

            # add attachment to the upload cache
            $UploadCacheObject->FormIDAddFile(
                FormID      => $Self->{FormID},
                Filename    => $AttachmentData->{Filename},
                Content     => $AttachmentData->{Content},
                ContentType => $AttachmentData->{ContentType},
            );
        }
    }

    # get submit save
    my $SubmitSave = $ParamObject->GetParam( Param => 'SubmitSave' );

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    # get xml data
    my $Version = {};
    my $NameDuplicates;
    my $CINameRegexErrorMessage;
    if ( $Self->{Subaction} eq 'VersionSave' ) {

        # get the uploaded attachment
        my %UploadStuff = $ParamObject->GetUploadAll(
            Param  => 'FileUpload',
            Source => 'string',
        );

        if (%UploadStuff) {

            # add attachment to the upload cache
            $UploadCacheObject->FormIDAddFile(
                FormID => $Self->{FormID},
                %UploadStuff,
            );
        }

        my $AllRequired = 1;

        # get general form data
        for my $FormParam (qw(Name DeplStateID InciStateID)) {
            $Version->{$FormParam} = $ParamObject->GetParam( Param => $FormParam );
            if ( !$Version->{$FormParam} ) {
                $AllRequired = 0;
            }
        }

        # get xml form data
        $Version->{XMLData}->[1]->{Version}->[1] = $Self->_XMLFormGet(
            XMLDefinition => $XMLDefinition->{DefinitionRef},
            AllRequired   => \$AllRequired,
            ConfigItemID  => $ConfigItem->{ConfigItemID},
        );

        # check, whether the feature to check for a unique name is enabled
        if (
            IsStringWithData( $Version->{Name} )
            && $ConfigObject->Get('UniqueCIName::EnableUniquenessCheck')
            )
        {

            if ( $ConfigObject->{Debug} > 0 ) {
                $LogObject->Log(
                    Priority => 'debug',
                    Message  => "Checking for duplicate names (ClassID: $ConfigItem->{ClassID}, "
                        . "Name: $Version->{Name}, ConfigItemID: $ConfigItem->{ConfigItemID})",
                );
            }

            $NameDuplicates = $ConfigItemObject->UniqueNameCheck(
                ConfigItemID => $ConfigItem->{ConfigItemID},
                ClassID      => $ConfigItem->{ClassID},
                Name         => $Version->{Name},
            );

            # stop processing if the name is not unique
            if ( IsArrayRefWithData($NameDuplicates) ) {

                $AllRequired = 0;

                # build a string of all duplicate IDs
                my $NameDuplicatesString = join ', ', @{$NameDuplicates};

                $LogObject->Log(
                    Priority => 'error',
                    Message =>
                        "The name $Version->{Name} is already in use by the ConfigItemID(s): "
                        . $NameDuplicatesString,
                );
            }
        }

        # get the config option for the name regex checks
        my $CINameRegexConfig = $ConfigObject->Get("ITSMConfigItem::CINameRegex");

        # check if the CI name is given and should be checked with a regular expression
        if ( IsStringWithData( $Version->{Name} ) && $CINameRegexConfig ) {

            # get class list
            my $ClassList = $GeneralCatalogObject->ItemList(
                Class => 'ITSM::ConfigItem::Class',
            );

            # get the class name
            my $ClassName = $ClassList->{ $ConfigItem->{ClassID} } || '';

            # get the regex for this class
            my $CINameRegex = $CINameRegexConfig->{ $ClassName . '::' . 'CINameRegex' } || '';

            # if a regex is defined and the CI name does not match the regular expression
            if ( $CINameRegex && $Version->{Name} !~ m{ $CINameRegex }xms ) {

                $AllRequired = 0;

                # get the error message for this class
                $CINameRegexErrorMessage = $CINameRegexConfig->{ $ClassName . '::' . 'CINameRegexErrorMessage' } || '';
            }
        }

        # save version to database
        if ( $SubmitSave && $AllRequired ) {

            if ( $ConfigItem->{ConfigItemID} eq 'NEW' ) {
                $ConfigItem->{ConfigItemID} = $ConfigItemObject->ConfigItemAdd(
                    ClassID => $ConfigItem->{ClassID},
                    UserID  => $Self->{UserID},
                );
            }

            # get all attachments from upload cache
            my @Attachments = $UploadCacheObject->FormIDGetAllFilesData(
                FormID => $Self->{FormID},
            );

            # build a lookup lookup hash of the new attachments
            my %NewAttachment;
            for my $Attachment (@Attachments) {

                # the key is the filename + filesize + content type
                my $Key = $Attachment->{Filename}
                    . $Attachment->{Filesize}
                    . $Attachment->{ContentType};

                # store all of the new attachment data
                $NewAttachment{$Key} = $Attachment;
            }

            # get all attachments meta data
            my @ExistingAttachments = $ConfigItemObject->ConfigItemAttachmentList(
                ConfigItemID => $ConfigItem->{ConfigItemID},
            );

            # check the existing attachments
            FILENAME:
            for my $Filename (@ExistingAttachments) {

                # get the existing attachment data
                my $AttachmentData = $ConfigItemObject->ConfigItemAttachmentGet(
                    ConfigItemID => $ConfigItem->{ConfigItemID},
                    Filename     => $Filename,
                    UserID       => $Self->{UserID},
                );

                # the key is the filename + filesize + content type
                # (no content id, as existing attachments don't have it)
                my $Key = $AttachmentData->{Filename}
                    . $AttachmentData->{Filesize}
                    . $AttachmentData->{ContentType};

                # attachment is already existing, we can delete it from the new attachment hash
                if ( $NewAttachment{$Key} ) {
                    delete $NewAttachment{$Key};
                }

                # existing attachment is no longer in new attachments hash
                else {

                    # delete the existing attachment
                    my $DeleteSuccessful = $ConfigItemObject->ConfigItemAttachmentDelete(
                        ConfigItemID => $ConfigItem->{ConfigItemID},
                        Filename     => $Filename,
                        UserID       => $Self->{UserID},
                    );

                    # check error
                    if ( !$DeleteSuccessful ) {
                        return $LayoutObject->FatalError();
                    }
                }
            }

            # write the new attachments
            ATTACHMENT:
            for my $Attachment ( values %NewAttachment ) {

                # add attachment
                my $Success = $ConfigItemObject->ConfigItemAttachmentAdd(
                    %{$Attachment},
                    ConfigItemID => $ConfigItem->{ConfigItemID},
                    UserID       => $Self->{UserID},
                );

                # check error
                if ( !$Success ) {
                    return $LayoutObject->FatalError();
                }
            }

            # add version
            $ConfigItemObject->VersionAdd(
                %{$Version},
                ConfigItemID => $ConfigItem->{ConfigItemID},
                DefinitionID => $XMLDefinition->{DefinitionID},
                UserID       => $Self->{UserID},
            );

            # redirect to zoom mask
            my $ScreenType = $ParamObject->GetParam( Param => 'ScreenType' ) || 0;
            if ($ScreenType) {

                my $URL = "Action=AgentITSMConfigItemZoom;ConfigItemID=$ConfigItem->{ConfigItemID}";

                # return to overview or search results instead if called Duplicate from row action
                if (
                    $Self->{LastScreenView} eq 'Action=AgentITSMConfigItem'
                    || $Self->{LastScreenView} =~ m{\A Action=AgentITSMConfigItem(?: Search)?;}msx
                    )
                {
                    $URL = $Self->{LastScreenView};
                }
                return $LayoutObject->PopupClose(
                    URL => $URL,
                );
            }
            else {
                return $LayoutObject->Redirect(
                    OP => "Action=AgentITSMConfigItemZoom;ConfigItemID=$ConfigItem->{ConfigItemID}",
                );
            }
        }
    }
    elsif ($DuplicateID) {
        my $VersionID = $ParamObject->GetParam( Param => 'VersionID' );
        if ($VersionID) {

            # get version data to duplicate config item
            $Version = $ConfigItemObject->VersionGet(
                VersionID => $VersionID,
            );
        }
        else {

            # get last version data to duplicate config item
            $Version = $ConfigItemObject->VersionGet(
                ConfigItemID => $DuplicateID,
            );
        }
    }
    elsif ( $ConfigItem->{ConfigItemID} ne 'NEW' ) {

        # get last version data
        $Version = $ConfigItemObject->VersionGet(
            ConfigItemID => $ConfigItem->{ConfigItemID},
        );
    }

    my %XMLFormOutputParam;
    if ( $Version->{XMLData}->[1]->{Version}->[1] ) {
        $XMLFormOutputParam{XMLData} = $Version->{XMLData}->[1]->{Version}->[1];
    }

    # output name invalid block
    my $RowNameInvalid = '';
    if ( !$Version->{Name} && $Self->{Subaction} eq 'VersionSave' && $SubmitSave ) {
        $RowNameInvalid = 'ServerError';
    }

    # check for name duplicates
    if ( IsArrayRefWithData($NameDuplicates) ) {
        $RowNameInvalid = 'ServerError';
    }

    # check for not matched name regex
    if ($CINameRegexErrorMessage) {
        $RowNameInvalid = 'ServerError';
    }

    # output name block
    $LayoutObject->Block(
        Name => 'RowName',
        Data => {
            %{$Version},
            RowNameInvalid => $RowNameInvalid,
        },
    );

    if (
        IsStringWithData($RowNameInvalid)
        && !IsArrayRefWithData($NameDuplicates)
        && !$CINameRegexErrorMessage
        )
    {

        if ( $ConfigObject->{Debug} > 0 ) {
            $LogObject->Log(
                Priority => 'debug',
                Message  => "Rendering default error block",
            );
        }

        $LayoutObject->Block(
            Name => 'RowNameErrorDefault',
        );
    }
    elsif ( IsArrayRefWithData($NameDuplicates) ) {

        # build array with CI-Numbers
        my @NameDuplicatesByCINumber;
        for my $ConfigItemID ( @{$NameDuplicates} ) {

            # lookup the CI number
            my $CINumber = $ConfigItemObject->ConfigItemLookup(
                ConfigItemID => $ConfigItemID,
            );

            push @NameDuplicatesByCINumber, $CINumber;
        }

        my $DuplicateString = join ', ', @NameDuplicatesByCINumber;

        if ( $ConfigObject->{Debug} > 0 ) {
            $LogObject->Log(
                Priority => 'debug',
                Message =>
                    "Rendering block for duplicates (CI-Numbers: $DuplicateString) error message",
            );
        }

        $LayoutObject->Block(
            Name => 'RowNameErrorDuplicates',
            Data => {
                Duplicates => $DuplicateString,
            },
        );
    }

    elsif ($CINameRegexErrorMessage) {

        $LayoutObject->Block(
            Name => 'RowNameErrorRegEx',
            Data => {
                RegExErrorMessage => $CINameRegexErrorMessage,
            },
        );
    }

    # get deployment state list
    my $DeplStateList = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    # output deployment state invalid block
    my $RowDeplStateInvalid = '';
    if ( !$Version->{DeplStateID} && $Self->{Subaction} eq 'VersionSave' && $SubmitSave ) {
        $RowDeplStateInvalid = ' ServerError';
    }

    # generate DeplStateOptionStrg
    my $DeplStateOptionStrg = $LayoutObject->BuildSelection(
        Data         => $DeplStateList,
        Name         => 'DeplStateID',
        PossibleNone => 1,
        Class        => 'Validate_Required Modernize' . $RowDeplStateInvalid,
        SelectedID   => $Version->{DeplStateID},
    );

    # output deployment state block
    $LayoutObject->Block(
        Name => 'RowDeplState',
        Data => {
            DeplStateOptionStrg => $DeplStateOptionStrg,
        },
    );

    # get incident state list
    my $InciStateList = $GeneralCatalogObject->ItemList(
        Class       => 'ITSM::Core::IncidentState',
        Preferences => {
            Functionality => [ 'operational', 'incident' ],
        },
    );

    # output incident state invalid block
    my $RowInciStateInvalid = '';
    if ( !$Version->{InciStateID} && $Self->{Subaction} eq 'VersionSave' && $SubmitSave ) {
        $RowInciStateInvalid = ' ServerError';
    }

    # generate InciStateOptionStrg
    my $InciStateOptionStrg = $LayoutObject->BuildSelection(
        Data         => $InciStateList,
        Name         => 'InciStateID',
        PossibleNone => 1,
        Class        => 'Validate_Required Modernize' . $RowInciStateInvalid,
        SelectedID   => $Version->{InciStateID},
    );

    # output incident state block
    $LayoutObject->Block(
        Name => 'RowInciState',
        Data => {
            InciStateOptionStrg => $InciStateOptionStrg,
        },
    );

    # output xml form
    if ( $XMLDefinition->{Definition} ) {
        $Self->{UserSearchItemIDs}     = [];
        $Self->{CustomerSearchItemIDs} = [];
        $Self->_XMLFormOutput(
            XMLDefinition => $XMLDefinition->{DefinitionRef},
            %XMLFormOutputParam,
        );
    }

    # get all attachments meta data
    $Param{AttachmentList} = [
        $UploadCacheObject->FormIDGetAllFilesMeta(
            FormID => $Self->{FormID},
        )
    ];

    my $Output = '';
    if ( ( $ConfigItem->{ConfigItemID} && $ConfigItem->{ConfigItemID} ne 'NEW' ) || $DuplicateID ) {

        # output block
        $LayoutObject->Block(
            Name => 'StartSmall',
            Data => {
                %Param,
                %{$ConfigItem},
            },
        );
        $LayoutObject->Block( Name => 'EndSmall' );

        # output header
        $Output .= $LayoutObject->Header(
            Title => Translatable('Edit'),
            Type  => 'Small',
        );

        # start template output
        $Output .= $LayoutObject->Output(
            TemplateFile => 'AgentITSMConfigItemEdit',
            Data         => {
                %Param,
                %{$ConfigItem},
                DuplicateID => $DuplicateID,
                FormID      => $Self->{FormID},
            },
        );
        $Output .= $LayoutObject->Footer( Type => 'Small' );
    }
    else {

        # Necessary stuff for Add New
        # get class list
        my $ClassList = $GeneralCatalogObject->ItemList(
            Class => 'ITSM::ConfigItem::Class',
        );

        # check for access rights
        for my $ClassID ( sort keys %{$ClassList} ) {
            my $HasAccess = $ConfigItemObject->Permission(
                Type    => $Self->{Config}->{Permission},
                Scope   => 'Class',
                ClassID => $ClassID,
                UserID  => $Self->{UserID},
            );

            delete $ClassList->{$ClassID} if !$HasAccess;
        }

        # generate ClassOptionStrg
        my $ClassOptionStrg = $LayoutObject->BuildSelection(
            Data         => $ClassList,
            Name         => 'ClassID',
            PossibleNone => 1,
            Translation  => 0,
            Class        => 'W100pc',
            SelectedID   => $ConfigItem->{ClassID},
        );

        # End Necessary stuff for Add New

        # output block
        $LayoutObject->Block(
            Name => 'StartNormal',
            Data => {
                ClassOptionStrg => $ClassOptionStrg,
                %Param,
                %{$ConfigItem},
            },
        );

        $LayoutObject->Block( Name => 'EndNormal' );

        # output header
        $Output .= $LayoutObject->Header(
            Title => Translatable('Edit'),
        );
        $Output .= $LayoutObject->NavigationBar();

        # start template output
        $Output .= $LayoutObject->Output(
            TemplateFile => 'AgentITSMConfigItemEdit',
            Data         => {
                %Param,
                %{$ConfigItem},
                DuplicateID => $DuplicateID,
                FormID      => $Self->{FormID},
            },
        );
        $Output .= $LayoutObject->Footer();
    }

    return $Output;
}

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

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if !$Param{AllRequired};
    return if ref $Param{XMLDefinition} ne 'ARRAY';
    return if ref $Param{AllRequired} ne 'SCALAR';
    return if !$Param{ConfigItemID};

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

    my $FormData = {};

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {
        my $CounterInsert = 1;

        COUNTER:
        for my $Counter ( 1 .. $Item->{CountMax} ) {

            # create inputkey and addkey
            my $InputKey = $Item->{Key} . '::' . $Counter;
            my $AddKey   = $Item->{Key} . '::Add';
            if ( $Param{Prefix} ) {
                $InputKey = $Param{Prefix} . '::' . $InputKey;
                $AddKey   = $Param{Prefix} . '::' . $AddKey;
            }

            # get param
            my $FormValues = $LayoutObject->ITSMConfigItemFormDataGet(
                Key          => $InputKey,
                Item         => $Item,
                ConfigItemID => $Param{ConfigItemID},
            );

            if ( defined $FormValues->{Value} ) {

                # check required value
                if ( $FormValues->{Invalid} ) {
                    ${ $Param{AllRequired} } = 0;
                }

                # check delete button
                next COUNTER if $ParamObject->GetParam( Param => $InputKey . '::Delete' );

                # start recursion, if "Sub" was found
                if ( $Item->{Sub} ) {
                    my $SubFormData = $Self->_XMLFormGet(
                        XMLDefinition => $Item->{Sub},
                        Prefix        => $InputKey,
                        AllRequired   => $Param{AllRequired},
                        ConfigItemID  => $Param{ConfigItemID},
                    );
                    $FormData->{ $Item->{Key} }->[$CounterInsert] = $SubFormData;
                }
                $FormData->{ $Item->{Key} }->[$CounterInsert]->{Content} = $FormValues->{Value};
                $CounterInsert++;
            }
            else {

                # check add button
                if ( $ParamObject->GetParam( Param => $AddKey ) ) {
                    if ( $Item->{Sub} ) {
                        $FormData->{ $Item->{Key} }->[$CounterInsert] = $Self->_XMLDefaultSet(
                            XMLDefinition => $Item->{Sub},
                        );
                    }
                    $FormData->{ $Item->{Key} }->[$CounterInsert]->{Content} = '';
                }
                last COUNTER;
            }
        }
    }

    return $FormData;
}

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

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if ref $Param{XMLDefinition} ne 'ARRAY';

    my $DefaultData = {};

    for my $Item ( @{ $Param{XMLDefinition} } ) {
        for my $Counter ( 1 .. $Item->{CountDefault} ) {

            # start recursion, if "Sub" was found
            if ( $Item->{Sub} ) {
                $DefaultData->{ $Item->{Key} }->[$Counter] = $Self->_XMLDefaultSet(
                    XMLDefinition => $Item->{Sub},
                );
            }

            $DefaultData->{ $Item->{Key} }->[$Counter]->{Content} = '';
        }
    }

    return $DefaultData;
}

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

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if ref $Param{XMLDefinition} ne 'ARRAY';

    $Param{Level}  ||= 0;
    $Param{Prefix} ||= '';

    # get submit save
    my $SubmitSave = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => 'SubmitSave' );

    # set data present mode
    my $DataPresentMode = 0;
    if ( $Param{XMLData} && ref $Param{XMLData} eq 'HASH' ) {
        $DataPresentMode = 1;
    }

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

    my $ItemCounter = 1;
    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        # set loop
        my $Loop = $Item->{CountDefault};
        if ($DataPresentMode) {
            $Loop = 0;

            # search the last content
            COUNTER:
            for my $Counter ( 1 .. $Item->{CountMax} ) {
                last COUNTER if !defined $Param{XMLData}->{ $Item->{Key} }->[$Counter]->{Content};
                $Loop = $Counter;
            }

            # set absolut minimum
            if ( $Loop < $Item->{CountMin} ) {
                $Loop = $Item->{CountMin};
            }
        }

        # set delete
        my $Delete = 0;
        if ( $Loop > $Item->{CountMin} ) {
            $Delete = 1;
        }

        # output content rows
        for my $Counter ( 1 .. $Loop ) {

            # output row block
            $LayoutObject->Block( Name => 'XMLRow' );

            if ( !$Param{Level} ) {
                $LayoutObject->Block( Name => 'XMLRowFieldsetStart' );
            }

            # create inputkey and addkey
            my $InputKey = $Item->{Key} . '::' . $Counter;
            if ( $Param{Prefix} ) {
                $InputKey = $Param{Prefix} . '::' . $InputKey;
            }

            # output blue required star
            my $XMLRowValueContentRequired = 0;
            my $LabelClass                 = '';
            if ( $Item->{Input}->{Required} ) {
                $XMLRowValueContentRequired = 1;
                $LabelClass                 = 'Mandatory';
            }

            # output red invalid star
            my $XMLRowValueContentInvalid = 0;
            if ( $Item->{Form}->{$InputKey}->{Invalid} && $SubmitSave ) {
                $XMLRowValueContentInvalid = 1;
            }

            my $ItemID = 'Item' . $ItemCounter++ . $Param{Prefix} . $Param{Level};

            if ( $Item->{Input}->{Type} eq 'User' ) {
                push @{ $Self->{UserSearchItemIDs} }, $ItemID;
            }
            if ( $Item->{Input}->{Type} eq 'Customer' ) {
                push @{ $Self->{CustomerSearchItemIDs} }, $ItemID;
            }

            # create input element
            my $InputString = $LayoutObject->ITSMConfigItemInputCreate(
                Key              => $InputKey,
                Item             => $Item,
                Value            => $Param{XMLData}->{ $Item->{Key} }->[$Counter]->{Content},
                ItemId           => $ItemID,
                Required         => $XMLRowValueContentRequired,
                Invalid          => $XMLRowValueContentInvalid,
                OverrideTimeZone => 1,
            );

            # ID?
            my $LabelFor = $ItemID;
            if ( $Item->{Input}->{Type} eq 'Date' || $Item->{Input}->{Type} eq 'DateTime' ) {
                $LabelFor = '';
            }

            # id needed?
            if ($LabelFor) {
                $LabelFor = 'for="' . $LabelFor . '"';
            }

            # is this a sub field?
            my $Class = '';
            if ( $Param{Level} ) {
                $Class = 'SubElement';
            }

            # class needed?
            if ($LabelClass) {
                $LabelClass = 'class="' . "$Class  $LabelClass" . '"';
            }
            else {
                $LabelClass = 'class="' . $Class . '"';
            }

            # output row value content block
            $LayoutObject->Block(
                Name => 'XMLRowValue',
                Data => {
                    Name        => $Item->{Name},
                    ItemID      => $ItemID,
                    LabelFor    => $LabelFor || '',
                    Description => $Item->{Description} || $Item->{Name},
                    InputString => $InputString,
                    LabelClass  => $LabelClass || '',
                    Class       => $Class || '',
                },
            );

            if ( $Item->{Input}->{Required} ) {
                $LayoutObject->Block( Name => 'XMLRowValueContentRequired' );
            }

            # output delete button
            if ($Delete) {
                $LayoutObject->Block(
                    Name => 'XMLRowValueContentDelete',
                    Data => {
                        InputKey => $InputKey,
                    },
                );
            }

            # the content is invalid
            if ($XMLRowValueContentInvalid) {

                # show regex error message block
                if ( $Item->{Form}->{$InputKey}->{RegExErrorMessage} ) {

                    $LayoutObject->Block(
                        Name => 'XMLRowValueRegExError',
                        Data => {
                            ItemID            => $ItemID,
                            RegExErrorMessage => $Item->{Form}->{$InputKey}->{RegExErrorMessage},
                        },
                    );
                }

                # otherwise show normal server error block
                else {
                    $LayoutObject->Block(
                        Name => 'XMLRowValueServerError',
                        Data => {
                            ItemID => $ItemID,
                        },
                    );
                }
            }

            # start recursion, if "Sub" was found
            if ( $Item->{Sub} ) {
                my %XMLFormOutputParam;
                if (
                    $DataPresentMode
                    && defined $Param{XMLData}->{ $Item->{Key} }->[$Counter]->{Content}
                    )
                {
                    $XMLFormOutputParam{XMLData} = $Param{XMLData}->{ $Item->{Key} }->[$Counter];
                }

                $Self->_XMLFormOutput(
                    XMLDefinition => $Item->{Sub},
                    %XMLFormOutputParam,
                    Level  => $Param{Level} + 1,
                    Prefix => $InputKey,
                );
            }

            if ( !$Param{Level} ) {
                $LayoutObject->Block( Name => 'XMLRowFieldsetEnd' );
            }

            # output row to sort rows correctly
            $LayoutObject->Block( Name => 'XMLRow' );
        }

        # output add button
        if ( $Loop < $Item->{CountMax} ) {

            # if no item should be shown we need to show the add button
            # and therefore we need to show the XMLRow block
            if ( !$Loop ) {
                $LayoutObject->Block( Name => 'XMLRow' );
            }

            my $Class = '';
            if ( $Param{Level} ) {
                $Class = 'class="SubElement"';
            }
            else {
                $LayoutObject->Block( Name => 'XMLRowFieldsetEnd' );
                $LayoutObject->Block( Name => 'XMLRowFieldsetStart' );
            }

            # set prefix
            my $InputKey = $Item->{Key};
            if ( $Param{Prefix} ) {
                $InputKey = $Param{Prefix} . '::' . $InputKey;
            }

            # output row add content block
            $LayoutObject->Block(
                Name => 'XMLRowAddContent',
                Data => {
                    ItemID      => $InputKey . 'Add',
                    Name        => $Item->{Name},
                    Description => $Item->{Description} || $Item->{Name},
                    InputKey    => $InputKey,
                    Class       => $Class,
                },
            );
        }
    }

    if ( IsArrayRefWithData( $Self->{UserSearchItemIDs} ) ) {
        $LayoutObject->AddJSData(
            Key   => 'UserSearchItemIDs',
            Value => $Self->{UserSearchItemIDs},
        );
    }

    if ( IsArrayRefWithData( $Self->{CustomerSearchItemIDs} ) ) {
        $LayoutObject->AddJSData(
            Key   => 'CustomerSearchItemIDs',
            Value => $Self->{CustomerSearchItemIDs},
        );
    }
    return 1;
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Modules::AgentITSMConfigItemHistory;

use strict;
use warnings;

use Kernel::Language qw(Translatable);

our $ObjectManagerDisabled = 1;

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

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

    return $Self;
}

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

    # get needed objects
    my $ParamObject  = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    $Self->{ConfigItemID} = $ParamObject->GetParam( Param => 'ConfigItemID' );

    # check needed stuff
    if ( !$Self->{ConfigItemID} ) {

        # error page
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Can\'t show history, no ConfigItemID is given!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # get neeeded objects
    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $ConfigObject     = $Kernel::OM->Get('Kernel::Config');

    # check for access rights
    my $HasAccess = $ConfigItemObject->Permission(
        Scope  => 'Item',
        ItemID => $Self->{ConfigItemID},
        UserID => $Self->{UserID},
        Type   => $ConfigObject->Get("ITSMConfigItem::Frontend::$Self->{Action}")->{Permission},
    );

    if ( !$HasAccess ) {

        # error page
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Can\'t show history, no access rights given!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # Define translatable history strings.
    my %HistoryStrings = (
        'CIHistory::ConfigItemCreate'      => Translatable('New ConfigItem (ID=%s)'),
        'CIHistory::VersionCreate'         => Translatable('New version (ID=%s)'),
        'CIHistory::DeploymentStateUpdate' => Translatable('Deployment state updated (new=%s, old=%s)'),
        'CIHistory::IncidentStateUpdate'   => Translatable('Incident state updated (new=%s, old=%s)'),
        'CIHistory::ConfigItemDelete'      => Translatable('ConfigItem (ID=%s) deleted'),
        'CIHistory::LinkAdd'               => Translatable('Link to %s (type=%s) added'),
        'CIHistory::LinkDelete'            => Translatable('Link to %s (type=%s) deleted'),
        'CIHistory::DefinitionUpdate'      => Translatable('ConfigItem definition updated (ID=%s)'),
        'CIHistory::NameUpdate'            => Translatable('Name updated (new=%s, old=%s)'),
        'CIHistory::ValueUpdate'           => Translatable('Attribute %s updated from "%s" to "%s"'),
        'CIHistory::VersionDelete'         => Translatable('Version %s deleted'),
    );

    # get all information about the config item
    my $ConfigItem = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => $Self->{ConfigItemID},
    );
    my $ConfigItemName = $ConfigItem->{Number};

    # get all entries in the history for this config item
    my $Lines = $ConfigItemObject->HistoryGet(
        ConfigItemID => $Self->{ConfigItemID},
    );

    # get shown user info
    my @NewLines = @{$Lines};
    if ( $ConfigObject->Get('ITSMConfigItem::Frontend::HistoryOrder') eq 'reverse' ) {
        @NewLines = reverse @{$Lines};
    }

    # get definition for CI's class
    my $Definition = $ConfigItemObject->DefinitionGet(
        ClassID => $ConfigItem->{ClassID},
    );

    my $Table   = '';
    my $Counter = 1;
    my $Version = 0;
    for my $DataTmp (@NewLines) {
        $Counter++;
        my %Data = (
            %{$DataTmp},
            VersionID => $Version,
        );

        # get general catalog object
        my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

        # trim the comment to only show version number
        if ( $Data{HistoryType} eq 'VersionCreate' ) {
            $Data{Comment} =~ s/\D//g;
            $Data{VersionID} = $Data{Comment};
            $Version = $Data{Comment};
        }
        elsif ( $Data{HistoryType} eq 'ValueUpdate' ) {

            # beautify comment
            my @Parts = split /%%/, $Data{Comment};
            $Parts[0] =~ s{ \A \[.*?\] \{'Version'\} \[.*?\] \{' }{}xms;
            $Parts[0] =~ s{ '\} \[.*?\] \{' }{::}xmsg;
            $Parts[0] =~ s{ '\} \[.*?\] \z }{}xms;

            # get definition info about attribute
            my $AttributeInfo = $ConfigItemObject->DefinitionAttributeInfo(
                Definition    => $Definition->{DefinitionRef},
                AttributePath => $Parts[0],
            );

            if ( $AttributeInfo && $AttributeInfo->{Input}->{Type} eq 'GeneralCatalog' ) {
                my $ItemList = $GeneralCatalogObject->ItemList(
                    Class => $AttributeInfo->{Input}->{Class},
                );

                $Parts[1] = $ItemList->{ $Parts[1] || '' } || '';
                $Parts[2] = $ItemList->{ $Parts[2] || '' } || '';
            }

            # assemble parts
            $Data{Comment} = join '%%', @Parts;
        }
        elsif ( $Data{HistoryType} eq 'DeploymentStateUpdate' ) {

            # get deployment state list
            my $DeplStateList = $GeneralCatalogObject->ItemList(
                Class => 'ITSM::ConfigItem::DeploymentState',
            );

            # show names
            my @Parts = split /%%/, $Data{Comment};
            for my $Part (@Parts) {
                $Part = $DeplStateList->{$Part} || '';
            }

            # assemble parts
            $Data{Comment} = join '%%', @Parts;
        }
        elsif ( $Data{HistoryType} eq 'IncidentStateUpdate' ) {

            # get deployment state list
            my $DeplStateList = $GeneralCatalogObject->ItemList(
                Class => 'ITSM::Core::IncidentState',
            );

            # show names
            my @Parts = split /%%/, $Data{Comment};
            for my $Part (@Parts) {
                $Part = $DeplStateList->{$Part} || '';
            }

            # assemble parts
            $Data{Comment} = join '%%', @Parts;
        }

        # replace text
        if ( $Data{Comment} ) {

            my %Info;

            $Data{Comment} =~ s{ \A %% }{}xmsg;
            my @Values = split /%%/, $Data{Comment};

            $Data{Comment} = $LayoutObject->{LanguageObject}->Translate(
                $HistoryStrings{ 'CIHistory::' . $Data{HistoryType} },
                @Values,
            );

            # remove not needed place holder
            $Data{Comment} =~ s/\%s//g;
        }

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

    # build page
    my $Output = $LayoutObject->Header(
        Value => $ConfigItemName,
        Type  => 'Small'
    );
    $Output .= $LayoutObject->Output(
        TemplateFile => 'AgentITSMConfigItemHistory',
        Data         => {
            Name         => $ConfigItemName,
            ConfigItemID => $Self->{ConfigItemID},
            VersionID    => $ParamObject->GetParam( Param => 'VersionID' ),
        },
    );
    $Output .= $LayoutObject->Footer( Type => 'Small' );

    return $Output;
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Modules::AgentITSMConfigItemPrint;

use strict;
use warnings;

use Kernel::Language qw(Translatable);

our $ObjectManagerDisabled = 1;

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

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

    return $Self;
}

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

    # get param object
    my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');

    # get params
    my $ConfigItemID = $ParamObject->GetParam( Param => 'ConfigItemID' );
    my $VersionID    = $ParamObject->GetParam( Param => 'VersionID' );

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

    # check needed stuff
    if ( !$ConfigItemID || !$VersionID ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('No ConfigItemID or VersionID is given!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # get needed objects
    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $ConfigObject     = $Kernel::OM->Get('Kernel::Config');

    # check for access rights
    my $HasAccess = $ConfigItemObject->Permission(
        Scope  => 'Item',
        ItemID => $ConfigItemID,
        UserID => $Self->{UserID},
        Type   => $ConfigObject->Get("ITSMConfigItem::Frontend::$Self->{Action}")->{Permission},
    );

    if ( !$HasAccess ) {

        # error page
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Can\'t show config item, no access rights given!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # get config item
    my $ConfigItem = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => $ConfigItemID,
    );
    if ( !$ConfigItem->{ConfigItemID} ) {
        return $LayoutObject->ErrorScreen(
            Message =>
                $LayoutObject->{LanguageObject}->Translate( 'ConfigItemID %s not found in database!', $ConfigItemID ),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # get version
    my $Version = $ConfigItemObject->VersionGet(
        VersionID => $VersionID,
    );
    if ( !$Version->{VersionID} ) {
        return $LayoutObject->ErrorScreen(
            Message => $LayoutObject->{LanguageObject}->Translate( 'VersionID %s not found in database!', $VersionID ),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # get last version
    my $LastVersion = $ConfigItemObject->VersionGet(
        ConfigItemID => $ConfigItemID,
    );
    $ConfigItem->{CurrentName} = $LastVersion->{Name};

    # get version list
    my $VersionList = $ConfigItemObject->VersionList(
        ConfigItemID => $ConfigItemID,
    );

    my $VersionNumber = 1;
    LISTVERSIONID:
    for my $ListVersionID ( @{$VersionList} ) {

        last LISTVERSIONID if $VersionID eq $ListVersionID;
        $VersionNumber++;
    }

    # get user object
    my $UserObject = $Kernel::OM->Get('Kernel::System::User');

    # get create & change user data
    for my $Key (qw(Create Change)) {
        $ConfigItem->{ $Key . 'ByName' } = $UserObject->UserName(
            UserID => $ConfigItem->{ $Key . 'By' },
        );
    }

    # get user data of version (create by)
    $Version->{CreateByName} = $UserObject->UserName(
        UserID => $Version->{CreateBy},
    );

    # get linked objects
    my $LinkObject = $Kernel::OM->Get('Kernel::System::LinkObject');

    my $LinkListWithData = $LinkObject->LinkListWithData(
        Object => 'ITSMConfigItem',
        Key    => $ConfigItemID,
        State  => 'Valid',
        UserID => $Self->{UserID},
    );

    # get link type list
    my %LinkTypeList = $LinkObject->TypeList(
        UserID => $Self->{UserID},
    );

    # get the link data
    my %LinkData;
    if ( $LinkListWithData && ref $LinkListWithData eq 'HASH' && %{$LinkListWithData} ) {
        %LinkData = $LayoutObject->LinkObjectTableCreate(
            LinkListWithData => $LinkListWithData,
            ViewMode         => 'SimpleRaw',
        );
    }

    # get attachments
    my @Attachments = $ConfigItemObject->ConfigItemAttachmentList(
        ConfigItemID => $ConfigItemID,
    );

    # get pdf object
    my $PDFObject = $Kernel::OM->Get('Kernel::System::PDF');

    # generate pdf output
    my %Page;

    # get maximum number of pages
    $Page{MaxPages} = $ConfigObject->Get('PDF::MaxPages');
    if ( !$Page{MaxPages} || $Page{MaxPages} < 1 || $Page{MaxPages} > 1000 ) {
        $Page{MaxPages} = 100;
    }

    my $Title = $ConfigItem->{CurrentName};

    $Page{MarginTop}    = 30;
    $Page{MarginRight}  = 40;
    $Page{MarginBottom} = 40;
    $Page{MarginLeft}   = 40;
    $Page{HeaderRight}  = $LayoutObject->{LanguageObject}->Translate('ConfigItem') . '#'
        . $ConfigItem->{Number};
    $Page{HeadlineLeft} = $Version->{Name};
    $Page{PageText}     = $LayoutObject->{LanguageObject}->Translate('Page');
    $Page{PageCount}    = 1;

    # create new pdf document
    $PDFObject->DocumentNew(
        Title  => $ConfigObject->Get('Product') . ':' . $Version->{Name},
        Encode => $LayoutObject->{UserCharset},
    );

    # create first pdf page
    $PDFObject->PageNew(
        %Page,
        FooterRight => $Page{PageText} . ' ' . $Page{PageCount},
    );
    $Page{PageCount}++;

    $PDFObject->PositionSet(
        Move => 'relativ',
        Y    => -6,
    );

    # output title
    $PDFObject->Text(
        Text     => $Title,
        FontSize => 13,
    );

    $PDFObject->PositionSet(
        Move => 'relativ',
        Y    => -6,
    );

    # output "printed by"
    $PDFObject->Text(
        Text => $LayoutObject->{LanguageObject}->Translate(
            'printed by %s at %s',
            $Self->{UserFullname},
            $LayoutObject->{Time},
        ),
        FontSize => 9,
    );

    $PDFObject->PositionSet(
        Move => 'relativ',
        Y    => -14,
    );

    # output general information
    $Self->_PDFOutputGeneralInfos(
        Page       => \%Page,
        ConfigItem => $ConfigItem,
    );

    # output linked objects
    if (%LinkData) {
        $Self->_PDFOutputLinkedObjects(
            PageData     => \%Page,
            LinkData     => \%LinkData,
            LinkTypeList => \%LinkTypeList,
        );
    }

    # output attachments
    if (@Attachments) {
        $Self->_PDFOutputAttachments(
            PageData       => \%Page,
            ConfigItemID   => $ConfigItemID,
            AttachmentData => \@Attachments,
        );
    }

    # output version infos
    $Self->_PDFOutputVersionInfos(
        Page          => \%Page,
        Version       => $Version,
        VersionNumber => $VersionNumber,
    );

    # create file name
    my $Filename = $Kernel::OM->Get('Kernel::System::Main')->FilenameCleanUp(
        Filename => $ConfigItem->{Number},
        Type     => 'Attachment',
    );

    # Get current system datetime object.
    my $CurrentSystemDTObj = $Kernel::OM->Create('Kernel::System::DateTime');

    return $LayoutObject->Attachment(
        Filename => sprintf(
            'configitem_%s_%s.pdf',
            $Filename,
            $CurrentSystemDTObj->Format( Format => '%F_%H-%M' ),
        ),
        ContentType => 'application/pdf',
        Content     => $PDFObject->DocumentOutput(),
        Type        => 'inline',
    );
}

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

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

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

    # create left table
    my $TableLeft = [
        {
            Key   => $LayoutObject->{LanguageObject}->Translate('Class') . ':',
            Value => $Param{ConfigItem}->{Class},
        },
        {
            Key   => $LayoutObject->{LanguageObject}->Translate('ConfigItem') . ':',
            Value => $Param{ConfigItem}->{CurrentName},
        },
        {
            Key   => $LayoutObject->{LanguageObject}->Translate('Current Deployment State') . ':',
            Value => $LayoutObject->{LanguageObject}->Translate(
                $Param{ConfigItem}->{CurDeplState},
            ),
        },
        {
            Key   => $LayoutObject->{LanguageObject}->Translate('Current Incident State') . ':',
            Value => $LayoutObject->{LanguageObject}->Translate(
                $Param{ConfigItem}->{CurInciState},
            ),
        },
    ];

    # create right table
    my $TableRight = [
        {
            Key   => $LayoutObject->{LanguageObject}->Translate('Created') . ':',
            Value => $LayoutObject->Output(
                Template => '[% Data.CreateTime | Localize("TimeLong") %]',
                Data     => \%{ $Param{ConfigItem} },
            ),
        },
        {
            Key   => $LayoutObject->{LanguageObject}->Translate('Created by') . ':',
            Value => $Param{ConfigItem}->{CreateByName},
        },
        {
            Key   => $LayoutObject->{LanguageObject}->Translate('Last changed') . ':',
            Value => $LayoutObject->Output(
                Template => '[% Data.ChangeTime | Localize("TimeLong") %]',
                Data     => \%{ $Param{ConfigItem} },
            ),
        },
        {
            Key   => $LayoutObject->{LanguageObject}->Translate('Last changed by') . ':',
            Value => $Param{ConfigItem}->{ChangeByName},
        },
    ];

    my $Rows = @{$TableLeft};
    if ( @{$TableRight} > $Rows ) {
        $Rows = @{$TableRight};
    }

    my %TableParam;
    for my $Row ( 1 .. $Rows ) {
        $Row--;
        $TableParam{CellData}[$Row][0]{Content}         = $TableLeft->[$Row]->{Key};
        $TableParam{CellData}[$Row][0]{Font}            = 'ProportionalBold';
        $TableParam{CellData}[$Row][1]{Content}         = $TableLeft->[$Row]->{Value};
        $TableParam{CellData}[$Row][2]{Content}         = ' ';
        $TableParam{CellData}[$Row][2]{BackgroundColor} = '#FFFFFF';
        $TableParam{CellData}[$Row][3]{Content}         = $TableRight->[$Row]->{Key};
        $TableParam{CellData}[$Row][3]{Font}            = 'ProportionalBold';
        $TableParam{CellData}[$Row][4]{Content}         = $TableRight->[$Row]->{Value};
    }

    $TableParam{ColumnData}[0]{Width} = 80;
    $TableParam{ColumnData}[1]{Width} = 170.5;
    $TableParam{ColumnData}[2]{Width} = 4;
    $TableParam{ColumnData}[3]{Width} = 80;
    $TableParam{ColumnData}[4]{Width} = 170.5;
    $TableParam{Type}                 = 'Cut';
    $TableParam{Border}               = 0;
    $TableParam{FontSize}             = 6;
    $TableParam{BackgroundColorEven}  = '#DDDDDD';
    $TableParam{Padding}              = 1;
    $TableParam{PaddingTop}           = 3;
    $TableParam{PaddingBottom}        = 3;

    # get pdf object
    my $PDFObject = $Kernel::OM->Get('Kernel::System::PDF');

    # output table
    PAGE:
    for ( $Param{Page}->{PageCount} .. $Param{Page}->{MaxPages} ) {

        # output table (or a fragment of it)
        %TableParam = $PDFObject->Table(%TableParam);

        # stop output or output next page
        last PAGE if $TableParam{State};

        $PDFObject->PageNew(
            %{ $Param{Page} },
            FooterRight => $Param{Page}->{PageText} . ' ' . $Param{Page}->{PageCount},
        );
        $Param{Page}->{PageCount}++;
    }

    return 1;
}

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

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

    my %Page     = %{ $Param{PageData} };
    my %TypeList = %{ $Param{LinkTypeList} };
    my %TableParam;
    my $Row = 0;

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

    for my $LinkTypeLinkDirection ( sort { lc $a cmp lc $b } keys %{ $Param{LinkData} } ) {

        # investigate link type name
        my @LinkData     = split q{::}, $LinkTypeLinkDirection;
        my $LinkTypeName = $TypeList{ $LinkData[0] }->{ $LinkData[1] . 'Name' };
        $LinkTypeName = $LayoutObject->{LanguageObject}->Translate($LinkTypeName);

        # define headline
        $TableParam{CellData}[$Row][0]{Content} = $LinkTypeName . ':';
        $TableParam{CellData}[$Row][0]{Font}    = 'ProportionalBold';
        $TableParam{CellData}[$Row][1]{Content} = '';

        # extract object list
        my $ObjectList = $Param{LinkData}->{$LinkTypeLinkDirection};

        for my $Object ( sort { lc $a cmp lc $b } keys %{$ObjectList} ) {

            for my $Item ( @{ $ObjectList->{$Object} } ) {

                $TableParam{CellData}[$Row][0]{Content} ||= '';
                $TableParam{CellData}[$Row][1]{Content} = $Item->{Title} || '';
            }
            continue {
                $Row++;
            }
        }
    }

    $TableParam{ColumnData}[0]{Width} = 80;
    $TableParam{ColumnData}[1]{Width} = 431;

    # get pdf object
    my $PDFObject = $Kernel::OM->Get('Kernel::System::PDF');

    # set new position
    $PDFObject->PositionSet(
        Move => 'relativ',
        Y    => -15,
    );

    # output headline
    $PDFObject->Text(
        Text     => $LayoutObject->{LanguageObject}->Translate('Linked Objects'),
        Height   => 7,
        Type     => 'Cut',
        Font     => 'ProportionalBoldItalic',
        FontSize => 7,
        Color    => '#666666',
    );

    # set new position
    $PDFObject->PositionSet(
        Move => 'relativ',
        Y    => -4,
    );

    # table params
    $TableParam{Type}            = 'Cut';
    $TableParam{Border}          = 0;
    $TableParam{FontSize}        = 6;
    $TableParam{BackgroundColor} = '#DDDDDD';
    $TableParam{Padding}         = 1;
    $TableParam{PaddingTop}      = 3;
    $TableParam{PaddingBottom}   = 3;

    # output table
    PAGE:
    for my $Count ( $Page{PageCount} .. $Page{MaxPages} ) {

        # output table (or a fragment of it)
        %TableParam = $PDFObject->Table(%TableParam);

        # stop output or output next page
        if ( $TableParam{State} ) {
            last PAGE;
        }
        else {
            $PDFObject->PageNew(
                %Page,
                FooterRight => $Page{PageText} . ' ' . $Page{PageCount},
            );
            $Page{PageCount}++;
        }
    }

    return 1;
}

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

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

    my %Page = %{ $Param{PageData} };
    my %TableParam;
    my $Row = 0;

    # attachments are rendered into a separate row
    ATTACHMENT:
    for my $Attachment ( @{ $Param{AttachmentData} } ) {

        # get the metadata of the current attachment
        my $AttachmentData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemAttachmentGet(
            ConfigItemID => $Param{ConfigItemID},
            Filename     => $Attachment,
        );

        # define attachment line
        $TableParam{CellData}[$Row][0]{Content}
            = $AttachmentData->{Filename} . '  (' . $AttachmentData->{Filesize} . ')';

        $Row++;
    }

    $TableParam{ColumnData}[0]{Width} = 80;
    $TableParam{ColumnData}[1]{Width} = 431;

    # get pdf object
    my $PDFObject = $Kernel::OM->Get('Kernel::System::PDF');

    # set new position
    $PDFObject->PositionSet(
        Move => 'relativ',
        Y    => -15,
    );

    # output headline
    $PDFObject->Text(
        Text     => $Kernel::OM->Get('Kernel::Output::HTML::Layout')->{LanguageObject}->Translate('Attachments'),
        Height   => 7,
        Type     => 'Cut',
        Font     => 'ProportionalBoldItalic',
        FontSize => 7,
        Color    => '#666666',
    );

    # set new position
    $PDFObject->PositionSet(
        Move => 'relativ',
        Y    => -4,
    );

    # table params
    $TableParam{Type}            = 'Cut';
    $TableParam{Border}          = 0;
    $TableParam{FontSize}        = 6;
    $TableParam{BackgroundColor} = '#DDDDDD';
    $TableParam{Padding}         = 1;
    $TableParam{PaddingTop}      = 3;
    $TableParam{PaddingBottom}   = 3;

    # output table
    PAGE:
    for my $Count ( $Page{PageCount} .. $Page{MaxPages} ) {

        # output table (or a fragment of it)
        %TableParam = $PDFObject->Table(%TableParam);

        # stop output or output next page
        if ( $TableParam{State} ) {
            last PAGE;
        }
        else {
            $PDFObject->PageNew(
                %Page,
                FooterRight => $Page{PageText} . ' ' . $Page{PageCount},
            );
            $Page{PageCount}++;
        }
    }

    return 1;
}

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

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

    # get pdf object
    my $PDFObject = $Kernel::OM->Get('Kernel::System::PDF');

    # set new position
    $PDFObject->PositionSet(
        Move => 'relativ',
        Y    => -15,
    );

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

    # output headline
    $PDFObject->Text(
        Text => $LayoutObject->{LanguageObject}->Translate('Version') . ' '
            . $Param{VersionNumber},
        Height   => 7,
        Type     => 'Cut',
        Font     => 'ProportionalBoldItalic',
        FontSize => 7,
        Color    => '#666666',
    );

    # set new position
    $PDFObject->PositionSet(
        Move => 'relativ',
        Y    => -4,
    );

    # create table
    my $Table = [
        {
            Key   => $LayoutObject->{LanguageObject}->Translate('Created') . ':',
            Value => $LayoutObject->Output(
                Template => '[% Data.CreateTime | Localize("TimeLong") %]',
                Data     => \%{ $Param{Version} },
            ),
        },
        {
            Key   => $LayoutObject->{LanguageObject}->Translate('Created by') . ':',
            Value => $Param{Version}->{CreateByName},
        },
        {
            Key   => ' ',
            Value => ' ',
        },
        {
            Key   => $LayoutObject->{LanguageObject}->Translate('Name') . ':',
            Value => $Param{Version}->{Name},
        },
        {
            Key   => $LayoutObject->{LanguageObject}->Translate('Deployment State') . ':',
            Value => $LayoutObject->{LanguageObject}->Translate( $Param{Version}->{DeplState} ),
        },
        {
            Key   => $LayoutObject->{LanguageObject}->Translate('Incident State') . ':',
            Value => $LayoutObject->{LanguageObject}->Translate( $Param{Version}->{InciState} ),
        },
    ];

    # add xml data to table
    if (
        ref $Param{Version} eq 'HASH'
        && $Param{Version}->{XMLDefinition}
        && $Param{Version}->{XMLData}
        && ref $Param{Version}->{XMLDefinition} eq 'ARRAY'
        && ref $Param{Version}->{XMLData} eq 'ARRAY'
        && $Param{Version}->{XMLData}->[1]
        && ref $Param{Version}->{XMLData}->[1] eq 'HASH'
        && $Param{Version}->{XMLData}->[1]->{Version}
        && ref $Param{Version}->{XMLData}->[1]->{Version} eq 'ARRAY'
        )
    {
        $Self->_PDFOutputXMLOutput(
            XMLDefinition => $Param{Version}->{XMLDefinition},
            XMLData       => $Param{Version}->{XMLData}->[1]->{Version}->[1],
            TableData     => $Table,
        );
    }

    my %TableParam;
    my $Rows = @{$Table};

    for my $Row ( 1 .. $Rows ) {
        $Row--;
        $TableParam{CellData}[$Row][0]{Content} = $Table->[$Row]->{Key};
        $TableParam{CellData}[$Row][0]{Font}    = 'ProportionalBold';
        $TableParam{CellData}[$Row][1]{Content} = $Table->[$Row]->{Value};
    }

    $TableParam{ColumnData}[0]{Width} = 100;
    $TableParam{ColumnData}[1]{Width} = 411;
    $TableParam{Type}                 = 'Cut';
    $TableParam{Border}               = 0;
    $TableParam{FontSize}             = 6;
    $TableParam{BackgroundColor}      = '#DDDDDD';
    $TableParam{Padding}              = 1;
    $TableParam{PaddingTop}           = 3;
    $TableParam{PaddingBottom}        = 3;

    # output table
    PAGE:
    for my $Count ( $Param{Page}->{PageCount} .. $Param{Page}->{MaxPages} ) {

        # output table (or a fragment of it)
        %TableParam = $PDFObject->Table(%TableParam);

        # stop output or output next page
        last PAGE if $TableParam{State};

        $PDFObject->PageNew(
            %{ $Param{Page} },
            FooterRight => $Param{Page}->{PageText} . ' ' . $Param{Page}->{PageCount},
        );
        $Param{Page}->{PageCount}++;
    }

    return 1;
}

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

    # check needed stuff
    return if !$Param{TableData};
    return if !$Param{XMLData};
    return if !$Param{XMLDefinition};
    return if ref $Param{TableData} ne 'ARRAY';
    return if ref $Param{XMLData} ne 'HASH';
    return if ref $Param{XMLDefinition} ne 'ARRAY';

    $Param{Level} ||= 0;

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {
        COUNTER:
        for my $Counter ( 1 .. $Item->{CountMax} ) {

            # stop loop, if no content was given
            last COUNTER if !defined $Param{XMLData}->{ $Item->{Key} }->[$Counter]->{Content};

            # lookup value
            my $Value = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->XMLValueLookup(
                Item  => $Item,
                Value => $Param{XMLData}->{ $Item->{Key} }->[$Counter]->{Content},
            );

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

            # create output string
            $Value = $LayoutObject->ITSMConfigItemOutputStringCreate(
                Value => $Value,
                Item  => $Item,
                Print => 1,
            );

            # replace newlines with <br/> (fix for bug# 5928)
            $Value =~ s{ \n }{<br/>}gxms;

            # convert value to ascii
            $Value = $Kernel::OM->Get('Kernel::System::HTMLUtils')->ToAscii( String => $Value );

            # new row
            my $NewRow = {
                Key   => $LayoutObject->{LanguageObject}->Translate( $Item->{Name} ) . ':',
                Value => $Value,
            };

            # output space, if level was given
            if ( $Param{Level} ) {
                for my $Level ( 1 .. $Param{Level} ) {
                    $NewRow->{Key}   = '    ' . $NewRow->{Key};
                    $NewRow->{Value} = '    ' . $NewRow->{Value};
                }
            }

            # add row data
            push @{ $Param{TableData} }, $NewRow;

            next COUNTER if !$Item->{Sub};

            # start recursion, if "Sub" was found
            $Self->_PDFOutputXMLOutput(
                XMLDefinition => $Item->{Sub},
                XMLData       => $Param{XMLData}->{ $Item->{Key} }->[$Counter],
                TableData     => $Param{TableData},
                Level         => $Param{Level} + 1,
            );
        }
    }

    return 1;
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Modules::AgentITSMConfigItemSearch;

use strict;
use warnings;

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 );

    return $Self;
}

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

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

    # get config of frontend module
    $Self->{Config} = $ConfigObject->Get("ITSMConfigItem::Frontend::$Self->{Action}");

    # get param object
    my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');

    # get config data
    $Self->{StartHit}    = int( $ParamObject->GetParam( Param => 'StartHit' ) || 1 );
    $Self->{SearchLimit} = $Self->{Config}->{SearchLimit} || 10000;
    $Self->{SortBy}      = $ParamObject->GetParam( Param => 'SortBy' )
        || $Self->{Config}->{'SortBy::Default'}
        || 'Number';
    $Self->{OrderBy} = $ParamObject->GetParam( Param => 'OrderBy' )
        || $Self->{Config}->{'Order::Default'}
        || 'Down';
    $Self->{Profile}     = $ParamObject->GetParam( Param => 'Profile' )     || '';
    $Self->{SaveProfile} = $ParamObject->GetParam( Param => 'SaveProfile' ) || '';
    $Self->{TakeLastSearch} = $ParamObject->GetParam( Param => 'TakeLastSearch' );

    # get general catalog object
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

    # get class list
    my $ClassList = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # get config item object
    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    # check for access rights on the classes
    for my $ClassID ( sort keys %{$ClassList} ) {
        my $HasAccess = $ConfigItemObject->Permission(
            Type    => $Self->{Config}->{Permission},
            Scope   => 'Class',
            ClassID => $ClassID,
            UserID  => $Self->{UserID},
        );

        delete $ClassList->{$ClassID} if !$HasAccess;
    }

    # get class id
    my $ClassID = $ParamObject->GetParam( Param => 'ClassID' );

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

    # check if class id is valid
    if ( $ClassID && !$ClassList->{$ClassID} ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Invalid ClassID!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # get single params
    my %GetParam;

    # get search profile object
    my $SearchProfileObject = $Kernel::OM->Get('Kernel::System::SearchProfile');

    # load profiles string params
    if ( ( $ClassID && $Self->{Profile} ) && $Self->{TakeLastSearch} ) {
        %GetParam = $SearchProfileObject->SearchProfileGet(
            Base      => 'ConfigItemSearch' . $ClassID,
            Name      => $Self->{Profile},
            UserLogin => $Self->{UserLogin},
        );
    }

    # ------------------------------------------------------------ #
    # delete search profiles
    # ------------------------------------------------------------ #
    if ( $Self->{Subaction} eq 'AJAXProfileDelete' ) {

        # remove old profile stuff
        $SearchProfileObject->SearchProfileDelete(
            Base      => 'ConfigItemSearch' . $ClassID,
            Name      => $Self->{Profile},
            UserLogin => $Self->{UserLogin},
        );
        my $Output = $LayoutObject->JSONEncode(
            Data => 1,
        );
        return $LayoutObject->Attachment(
            NoCache     => 1,
            ContentType => 'text/html',
            Content     => $Output,
            Type        => 'inline',
        );
    }

    # ------------------------------------------------------------ #
    # init search dialog (select class)
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'AJAX' ) {

        # generate dropdown for selecting the class
        # automatically show search mask after selecting a class via AJAX
        my $ClassOptionStrg = $LayoutObject->BuildSelection(
            Data         => $ClassList,
            Name         => 'SearchClassID',
            PossibleNone => 1,
            SelectedID   => $ClassID || '',
            Translation  => 1,
            Class        => 'Modernize',
        );

        # html search mask output
        $LayoutObject->Block(
            Name => 'SearchAJAX',
            Data => {
                ClassOptionStrg => $ClassOptionStrg,
                Profile         => $Self->{Profile},
                ClassID         => $ClassID,
            },
        );

        # output template
        $Output = $LayoutObject->Output(
            TemplateFile => 'AgentITSMConfigItemSearch',
        );

        return $LayoutObject->Attachment(
            NoCache     => 1,
            ContentType => 'text/html',
            Content     => $Output,
            Type        => 'inline',
        );
    }

    # ------------------------------------------------------------ #
    # set search fields for selected class
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'AJAXUpdate' ) {

        # ClassID is required for the search mask and for actual searching
        if ( !$ClassID ) {
            return $LayoutObject->ErrorScreen(
                Message => Translatable('No ClassID is given!'),
                Comment => Translatable('Please contact the administrator.'),
            );
        }

        # check if user is allowed to search class
        my $HasAccess = $ConfigItemObject->Permission(
            Type    => $Self->{Config}->{Permission},
            Scope   => 'Class',
            ClassID => $ClassID,
            UserID  => $Self->{UserID},
        );

        # show error screen
        if ( !$HasAccess ) {
            return $LayoutObject->ErrorScreen(
                Message => Translatable('No access rights for this class given!'),
                Comment => Translatable('Please contact the administrator.'),
            );
        }

        # get current definition
        my $XMLDefinition = $ConfigItemObject->DefinitionGet(
            ClassID => $ClassID,
        );

        # abort, if no definition is defined
        if ( !$XMLDefinition->{DefinitionID} ) {
            return $LayoutObject->ErrorScreen(
                Message =>
                    $LayoutObject->{LanguageObject}->Translate( 'No definition was defined for class %s!', $ClassID ),
                Comment => Translatable('Please contact the administrator.'),
            );
        }

        my @XMLAttributes = (
            {
                Key   => 'Number',
                Value => Translatable('Number'),
            },
            {
                Key   => 'Name',
                Value => Translatable('Name'),
            },
            {
                Key   => 'DeplStateIDs',
                Value => Translatable('Deployment State'),
            },
            {
                Key   => 'InciStateIDs',
                Value => Translatable('Incident State'),
            },
        );

        my %GetParam = $SearchProfileObject->SearchProfileGet(
            Base      => 'ConfigItemSearch' . $ClassID,
            Name      => $Self->{Profile},
            UserLogin => $Self->{UserLogin},
        );

        # get attributes to include in attributes string
        if ( $XMLDefinition->{Definition} ) {
            $Self->_XMLSearchAttributesGet(
                XMLDefinition => $XMLDefinition->{DefinitionRef},
                XMLAttributes => \@XMLAttributes,
            );
        }

        # build attributes string for attributes list
        $Param{AttributesStrg} = $LayoutObject->BuildSelection(
            PossibleNone => 1,
            Data         => \@XMLAttributes,
            Name         => 'Attribute',
            Multiple     => 0,
            Class        => 'Modernize',
        );

        # build attributes string for recovery on add or subtract search fields
        $Param{AttributesOrigStrg} = $LayoutObject->BuildSelection(
            PossibleNone => 1,
            Data         => \@XMLAttributes,
            Name         => 'AttributeOrig',
            Multiple     => 0,
            Class        => 'Modernize',
        );

        my %Profiles = $SearchProfileObject->SearchProfileList(
            Base      => 'ConfigItemSearch' . $ClassID,
            UserLogin => $Self->{UserLogin},
        );

        $Param{ProfilesStrg} = $LayoutObject->BuildSelection(
            Data         => \%Profiles,
            Name         => 'Profile',
            ID           => 'SearchProfile',
            SelectedID   => $Self->{Profile},
            Class        => 'Modernize',
            PossibleNone => 1,
        );

        # get deployment state list
        my $DeplStateList = $GeneralCatalogObject->ItemList(
            Class => 'ITSM::ConfigItem::DeploymentState',
        );

        # generate dropdown for selecting the wanted deployment states
        my $CurDeplStateOptionStrg = $LayoutObject->BuildSelection(
            Data       => $DeplStateList,
            Name       => 'DeplStateIDs',
            SelectedID => $GetParam{DeplStateIDs} || [],
            Size       => 5,
            Multiple   => 1,
            Class      => 'Modernize',
        );

        # get incident state list
        my $InciStateList = $GeneralCatalogObject->ItemList(
            Class => 'ITSM::Core::IncidentState',
        );

        # generate dropdown for selecting the wanted incident states
        my $CurInciStateOptionStrg = $LayoutObject->BuildSelection(
            Data       => $InciStateList,
            Name       => 'InciStateIDs',
            SelectedID => $GetParam{InciStateIDs} || [],
            Size       => 5,
            Multiple   => 1,
            Class      => 'Modernize',
        );

        # generate PreviousVersionOptionStrg
        my $PreviousVersionOptionStrg = $LayoutObject->BuildSelection(
            Name => 'PreviousVersionSearch',
            Data => {
                0 => Translatable('No'),
                1 => Translatable('Yes'),
            },
            SelectedID => $GetParam{PreviousVersionSearch} || '0',
            Class      => 'Modernize',
        );

        # build output format string
        $Param{ResultFormStrg} = $LayoutObject->BuildSelection(
            Data => {
                Normal => Translatable('Normal'),
                Print  => Translatable('Print'),
                CSV    => Translatable('CSV'),
                Excel  => Translatable('Excel'),
            },
            Name       => 'ResultForm',
            SelectedID => $GetParam{ResultForm} || 'Normal',
            Class      => 'Modernize',
        );

        $LayoutObject->Block(
            Name => 'AJAXContent',
            Data => {
                ClassID                   => $ClassID,
                CurDeplStateOptionStrg    => $CurDeplStateOptionStrg,
                CurInciStateOptionStrg    => $CurInciStateOptionStrg,
                PreviousVersionOptionStrg => $PreviousVersionOptionStrg,
                AttributesStrg            => $Param{AttributesStrg},
                AttributesOrigStrg        => $Param{AttributesOrigStrg},
                ResultFormStrg            => $Param{ResultFormStrg},
                ProfilesStrg              => $Param{ProfilesStrg},
                Number                    => $GetParam{Number} || '',
                Name                      => $GetParam{Name} || '',
            },
        );

        # output xml search form
        if ( $XMLDefinition->{Definition} ) {
            $Self->_XMLSearchFormOutput(
                XMLDefinition => $XMLDefinition->{DefinitionRef},
                XMLAttributes => \@XMLAttributes,
                GetParam      => \%GetParam,
            );
        }

        my @ProfileAttributes;

        # show attributes
        my $AttributeIsUsed = 0;
        KEY:
        for my $Key ( sort keys %GetParam ) {
            next KEY if !$Key;
            next KEY if !defined $GetParam{$Key};
            next KEY if $GetParam{$Key} eq '';

            $AttributeIsUsed = 1;

            push @ProfileAttributes, $Key;
        }

        # if no attribute is shown, show configitem number
        if ( !$Self->{Profile} ) {
            push @ProfileAttributes, 'Number';
        }

        $LayoutObject->AddJSData(
            Key   => 'ITSMSearchProfileAttributes',
            Value => \@ProfileAttributes,
        );

        # output template
        $Output = $LayoutObject->Output(
            TemplateFile => 'AgentITSMConfigItemSearch',
            AJAX         => 1,
        );

        return $LayoutObject->Attachment(
            NoCache     => 1,
            ContentType => 'text/html',
            Content     => $Output,
            Type        => 'inline',
        );
    }

    # ------------------------------------------------------------ #
    # perform search
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'Search' ) {

        my $SearchDialog = $ParamObject->GetParam( Param => 'SearchDialog' );

        # fill up profile name (e.g. with last-search)
        if ( !$Self->{Profile} || !$Self->{SaveProfile} ) {
            $Self->{Profile} = 'last-search';
        }

        # store last overview screen
        my $URL = "Action=AgentITSMConfigItemSearch;Profile=$Self->{Profile};"
            . "TakeLastSearch=1;StartHit=$Self->{StartHit};Subaction=Search;"
            . "OrderBy=$Self->{OrderBy};SortBy=$Self->{SortBy}";

        if ($ClassID) {
            $URL .= ";ClassID=$ClassID";
        }

        # get session object
        my $SessionObject = $Kernel::OM->Get('Kernel::System::AuthSession');

        $SessionObject->UpdateSessionID(
            SessionID => $Self->{SessionID},
            Key       => 'LastScreenOverview',
            Value     => $URL,
        );
        $SessionObject->UpdateSessionID(
            SessionID => $Self->{SessionID},
            Key       => 'LastScreenView',
            Value     => $URL,
        );

        # ClassID is required for the search mask and for actual searching
        if ( !$ClassID ) {
            return $LayoutObject->ErrorScreen(
                Message => Translatable('No ClassID is given!'),
                Comment => Translatable('Please contact the administrator.'),
            );
        }

        # check if user is allowed to search class
        my $HasAccess = $ConfigItemObject->Permission(
            Type    => $Self->{Config}->{Permission},
            Scope   => 'Class',
            ClassID => $ClassID,
            UserID  => $Self->{UserID},
        );

        # show error screen
        if ( !$HasAccess ) {
            return $LayoutObject->ErrorScreen(
                Message => Translatable('No access rights for this class given!'),
                Comment => Translatable('Please contact the administrator.'),
            );
        }

        # get current definition
        my $XMLDefinition = $ConfigItemObject->DefinitionGet(
            ClassID => $ClassID,
        );

        # abort, if no definition is defined
        if ( !$XMLDefinition->{DefinitionID} ) {
            return $LayoutObject->ErrorScreen(
                Message =>
                    $LayoutObject->{LanguageObject}->Translate( 'No definition was defined for class %s!', $ClassID ),
                Comment => Translatable('Please contact the administrator.'),
            );
        }

        # get scalar search attributes (special handling for Number and Name)
        FORMVALUE:
        for my $FormValue (qw(Number Name)) {

            my $Value = $ParamObject->GetParam( Param => $FormValue );

            # must be defined and not be an empty string
            # BUT the number 0 is an allowed value
            next FORMVALUE if !defined $Value;
            next FORMVALUE if $Value eq '';

            $GetParam{$FormValue} = $Value;
        }

        # get ther scalar search attributes
        FORMVALUE:
        for my $FormValue (qw(PreviousVersionSearch ResultForm)) {

            my $Value = $ParamObject->GetParam( Param => $FormValue );

            next FORMVALUE if !$Value;

            $GetParam{$FormValue} = $Value;
        }

        # get array search attributes
        FORMARRAY:
        for my $FormArray (qw(DeplStateIDs InciStateIDs)) {

            my @Array = $ParamObject->GetArray( Param => $FormArray );

            next FORMARRAY if !@Array;

            $GetParam{$FormArray} = \@Array;
        }

        # get xml search form
        my $XMLFormData = [];
        my $XMLGetParam = [];
        $Self->_XMLSearchFormGet(
            XMLDefinition => $XMLDefinition->{DefinitionRef},
            XMLFormData   => $XMLFormData,
            XMLGetParam   => $XMLGetParam,
            %GetParam,
        );

        if ( @{$XMLFormData} ) {
            $GetParam{What} = $XMLFormData;
        }

        # save search profile (under last-search or real profile name)
        $Self->{SaveProfile} = 1;

        # remember last search values only if search is called from a search dialog
        # not from results page
        if ( $Self->{SaveProfile} && $Self->{Profile} && $SearchDialog ) {

            # remove old profile stuff
            $SearchProfileObject->SearchProfileDelete(
                Base      => 'ConfigItemSearch' . $ClassID,
                Name      => $Self->{Profile},
                UserLogin => $Self->{UserLogin},
            );

            # insert new profile params
            for my $Key ( sort keys %GetParam ) {
                if ( $GetParam{$Key} && $Key ne 'What' ) {
                    $SearchProfileObject->SearchProfileAdd(
                        Base      => 'ConfigItemSearch' . $ClassID,
                        Name      => $Self->{Profile},
                        Key       => $Key,
                        Value     => $GetParam{$Key},
                        UserLogin => $Self->{UserLogin},
                    );
                }
            }

            # insert new profile params also from XMLform
            if ( @{$XMLGetParam} ) {
                for my $Parameter ( @{$XMLGetParam} ) {
                    for my $Key ( sort keys %{$Parameter} ) {
                        if ( $Parameter->{$Key} ) {

                            $SearchProfileObject->SearchProfileAdd(
                                Base      => 'ConfigItemSearch' . $ClassID,
                                Name      => $Self->{Profile},
                                Key       => $Key,
                                Value     => $Parameter->{$Key},
                                UserLogin => $Self->{UserLogin},
                            );

                        }
                    }
                }
            }
        }

        my $SearchResultList = [];

        # start search if called from a search dialog or from a results page
        if ( $SearchDialog || $Self->{TakeLastSearch} ) {

            # start search
            $SearchResultList = $ConfigItemObject->ConfigItemSearchExtended(
                %GetParam,
                OrderBy          => [ $Self->{SortBy} ],
                OrderByDirection => [ $Self->{OrderBy} ],
                Limit            => $Self->{SearchLimit},
                ClassIDs         => [$ClassID],
            );
        }

        # get param only if called from a search dialog, otherwise it must be already in %GetParam
        # from a loaded profile
        if ($SearchDialog) {
            $GetParam{ResultForm} = $ParamObject->GetParam( Param => 'ResultForm' );
        }

        # CSV output
        if (
            $GetParam{ResultForm} eq 'CSV'
            ||
            $GetParam{ResultForm} eq 'Excel'
            )
        {
            my @CSVData;
            my @CSVHead;

            # mapping between header name and data field
            my %Header2Data = (
                'Class'            => 'Class',
                'Incident State'   => 'InciState',
                'Name'             => 'Name',
                'ConfigItemNumber' => 'Number',
                'Deployment State' => 'DeplState',
                'Version'          => 'VersionID',
                'Create Time'      => 'CreateTime',
            );

            CONFIGITEMID:
            for my $ConfigItemID ( @{$SearchResultList} ) {

                # check for access rights
                my $HasAccess = $ConfigItemObject->Permission(
                    Scope  => 'Item',
                    ItemID => $ConfigItemID,
                    UserID => $Self->{UserID},
                    Type   => $Self->{Config}->{Permission},
                );

                next CONFIGITEMID if !$HasAccess;

                # get version
                my $LastVersion = $ConfigItemObject->VersionGet(
                    ConfigItemID => $ConfigItemID,
                    XMLDataGet   => 0,
                );

                # csv quote
                if ( !@CSVHead ) {
                    @CSVHead = @{ $Self->{Config}->{SearchCSVData} };
                }

                # store data
                my @Data;
                for my $Header (@CSVHead) {
                    push @Data, $LastVersion->{ $Header2Data{$Header} };
                }
                push @CSVData, \@Data;
            }

            # csv quote
            # translate non existing header may result in a garbage file
            if ( !@CSVHead ) {
                @CSVHead = @{ $Self->{Config}->{SearchCSVData} };
            }

            # translate headers
            for my $Header (@CSVHead) {

                # replace ConfigItemNumber header with the current ConfigItemNumber hook from sysconfig
                if ( $Header eq 'ConfigItemNumber' ) {
                    $Header = $ConfigObject->Get('ITSMConfigItem::Hook');
                }
                else {
                    $Header = $LayoutObject->{LanguageObject}->Translate($Header);
                }
            }

            my $CSVObject      = $Kernel::OM->Get('Kernel::System::CSV');
            my $CurSysDTObject = $Kernel::OM->Create('Kernel::System::DateTime');
            if ( $GetParam{ResultForm} eq 'CSV' ) {

                # Assemble CSV data.
                my $CSV = $CSVObject->Array2CSV(
                    Head      => \@CSVHead,
                    Data      => \@CSVData,
                    Separator => $Self->{UserCSVSeparator},
                );

                # Return CSV to download.
                return $LayoutObject->Attachment(
                    Filename => sprintf(
                        'change_search_%s.csv',
                        $CurSysDTObject->Format(
                            Format => '%F_%H-%M',
                        ),
                    ),
                    ContentType => "text/csv; charset=" . $LayoutObject->{UserCharset},
                    Content     => $CSV,
                );
            }
            elsif ( $GetParam{ResultForm} eq 'Excel' ) {

                # Assemble Excel data.
                my $Excel = $CSVObject->Array2CSV(
                    Head   => \@CSVHead,
                    Data   => \@CSVData,
                    Format => 'Excel',
                );

                # Return Excel to download.
                return $LayoutObject->Attachment(
                    Filename => sprintf(
                        'change_search_%s.xlsx',
                        $CurSysDTObject->Format(
                            Format => '%F_%H-%M',
                        ),
                    ),
                    ContentType => "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet; charset="
                        . $LayoutObject->{UserCharset},
                    Content => $Excel,
                );
            }
        }

        # print PDF output
        elsif ( $GetParam{ResultForm} eq 'Print' ) {

            my @PDFData;

            # get pdf object
            my $PDFObject = $Kernel::OM->Get('Kernel::System::PDF');

            CONFIGITEMID:
            for my $ConfigItemID ( @{$SearchResultList} ) {

                # check for access rights
                my $HasAccess = $ConfigItemObject->Permission(
                    Scope  => 'Item',
                    ItemID => $ConfigItemID,
                    UserID => $Self->{UserID},
                    Type   => $Self->{Config}->{Permission},
                );

                next CONFIGITEMID if !$HasAccess;

                # get version
                my $LastVersion = $ConfigItemObject->VersionGet(
                    ConfigItemID => $ConfigItemID,
                    XMLDataGet   => 0,
                );

                # set pdf rows
                my @PDFRow;
                for my $StoreData (qw(Class InciState Name Number DeplState VersionID CreateTime)) {
                    push @PDFRow, $LastVersion->{$StoreData};
                }
                push @PDFData, \@PDFRow;

            }

            # PDF Output
            my $Title = $LayoutObject->{LanguageObject}->Translate('Configuration Item') . ' '
                . $LayoutObject->{LanguageObject}->Translate('Search');
            my $PrintedBy = $LayoutObject->{LanguageObject}->Translate('printed by');
            my $Page      = $LayoutObject->{LanguageObject}->Translate('Page');
            my $Time      = $LayoutObject->{Time};

            # get maximum number of pages
            my $MaxPages = $ConfigObject->Get('PDF::MaxPages');
            if ( !$MaxPages || $MaxPages < 1 || $MaxPages > 1000 ) {
                $MaxPages = 100;
            }

            # create the header
            my $CellData;

            # output 'No Result', if no content was given
            if (@PDFData) {
                $CellData->[0]->[0]->{Content} = $LayoutObject->{LanguageObject}->Translate('Class');
                $CellData->[0]->[0]->{Font}    = 'ProportionalBold';
                $CellData->[0]->[1]->{Content} = $LayoutObject->{LanguageObject}->Translate('Incident State');
                $CellData->[0]->[1]->{Font}    = 'ProportionalBold';
                $CellData->[0]->[2]->{Content} = $LayoutObject->{LanguageObject}->Translate('Name');
                $CellData->[0]->[2]->{Font}    = 'ProportionalBold';
                $CellData->[0]->[3]->{Content} = $LayoutObject->{LanguageObject}->Translate('Number');
                $CellData->[0]->[3]->{Font}    = 'ProportionalBold';
                $CellData->[0]->[4]->{Content} = $LayoutObject->{LanguageObject}->Translate('Deployment State');
                $CellData->[0]->[4]->{Font}    = 'ProportionalBold';
                $CellData->[0]->[5]->{Content} = $LayoutObject->{LanguageObject}->Translate('Version');
                $CellData->[0]->[5]->{Font}    = 'ProportionalBold';
                $CellData->[0]->[6]->{Content} = $LayoutObject->{LanguageObject}->Translate('Create Time');
                $CellData->[0]->[6]->{Font}    = 'ProportionalBold';

                # create the content array
                my $CounterRow = 1;
                for my $Row (@PDFData) {
                    my $CounterColumn = 0;
                    for my $Content ( @{$Row} ) {
                        $CellData->[$CounterRow]->[$CounterColumn]->{Content} = $Content;
                        $CounterColumn++;
                    }
                    $CounterRow++;
                }
            }
            else {
                $CellData->[0]->[0]->{Content} = $LayoutObject->{LanguageObject}->Translate('No Result!');
            }

            # page params
            my %PageParam;
            $PageParam{PageOrientation} = 'landscape';
            $PageParam{MarginTop}       = 30;
            $PageParam{MarginRight}     = 40;
            $PageParam{MarginBottom}    = 40;
            $PageParam{MarginLeft}      = 40;
            $PageParam{HeaderRight}     = $Title;

            # table params
            my %TableParam;
            $TableParam{CellData}            = $CellData;
            $TableParam{Type}                = 'Cut';
            $TableParam{FontSize}            = 6;
            $TableParam{Border}              = 0;
            $TableParam{BackgroundColorEven} = '#DDDDDD';
            $TableParam{Padding}             = 1;
            $TableParam{PaddingTop}          = 3;
            $TableParam{PaddingBottom}       = 3;

            # create new pdf document
            $PDFObject->DocumentNew(
                Title  => $ConfigObject->Get('Product') . ': ' . $Title,
                Encode => $LayoutObject->{UserCharset},
            );

            # start table output
            $PDFObject->PageNew(
                %PageParam,
                FooterRight => $Page . ' 1',
            );

            $PDFObject->PositionSet(
                Move => 'relativ',
                Y    => -6,
            );

            # output title
            $PDFObject->Text(
                Text     => $Title,
                FontSize => 13,
            );

            $PDFObject->PositionSet(
                Move => 'relativ',
                Y    => -6,
            );

            # output "printed by"
            $PDFObject->Text(
                Text => $PrintedBy . ' '
                    . $Self->{UserFullname} . ' '
                    . $Time,
                FontSize => 9,
            );

            $PDFObject->PositionSet(
                Move => 'relativ',
                Y    => -14,
            );

            PAGE:
            for my $Count ( 2 .. $MaxPages ) {

                # output table (or a fragment of it)
                %TableParam = $PDFObject->Table(%TableParam);

                # stop output or another page
                if ( $TableParam{State} ) {
                    last PAGE;
                }
                else {
                    $PDFObject->PageNew(
                        %PageParam,
                        FooterRight => $Page . ' ' . $Count,
                    );
                }
            }

            # return the pdf document
            my $CurrentSystemDTObj = $Kernel::OM->Create('Kernel::System::DateTime');
            my $PDFString          = $PDFObject->DocumentOutput();
            return $LayoutObject->Attachment(
                Filename => sprintf(
                    'configitem_search_%s_%s.pdf',
                    $CurrentSystemDTObj->Format( Format => '%F_%H-%M' ),
                ),
                ContentType => "application/pdf",
                Content     => $PDFString,
                Type        => 'inline',
            );
        }

        # normal HTML output
        else {

            # start html page
            my $Output = $LayoutObject->Header();
            $Output .= $LayoutObject->NavigationBar();
            $LayoutObject->Print( Output => \$Output );
            $Output = '';

            $Self->{Filter} = $ParamObject->GetParam( Param => 'Filter' ) || '';
            $Self->{View}   = $ParamObject->GetParam( Param => 'View' )   || '';

            # show config items
            my $LinkPage = 'Filter='
                . $LayoutObject->Ascii2Html( Text => $Self->{Filter} )
                . ';View=' . $LayoutObject->Ascii2Html( Text => $Self->{View} )
                . ';SortBy=' . $LayoutObject->Ascii2Html( Text => $Self->{SortBy} )
                . ';OrderBy='
                . $LayoutObject->Ascii2Html( Text => $Self->{OrderBy} )
                . ';Profile=' . $Self->{Profile} . ';TakeLastSearch=1;Subaction=Search'
                . ';ClassID=' . $ClassID
                . ';';
            my $LinkSort = 'Filter='
                . $LayoutObject->Ascii2Html( Text => $Self->{Filter} )
                . ';View=' . $LayoutObject->Ascii2Html( Text => $Self->{View} )
                . ';Profile=' . $Self->{Profile} . ';TakeLastSearch=1;Subaction=Search'
                . ';ClassID=' . $ClassID
                . ';';
            my $LinkFilter = 'TakeLastSearch=1;Subaction=Search;Profile='
                . $LayoutObject->Ascii2Html( Text => $Self->{Profile} )
                . ';ClassID='
                . $LayoutObject->Ascii2Html( Text => $ClassID )
                . ';';
            my $LinkBack = 'Subaction=LoadProfile;Profile='
                . $LayoutObject->Ascii2Html( Text => $Self->{Profile} )
                . ';ClassID='
                . $LayoutObject->Ascii2Html( Text => $ClassID )
                . ';TakeLastSearch=1;';

            # find out which columns should be shown
            my @ShowColumns;
            if ( $Self->{Config}->{ShowColumns} ) {

                # get all possible columns from config
                my %PossibleColumn = %{ $Self->{Config}->{ShowColumns} };

                # get the column names that should be shown
                COLUMNNAME:
                for my $Name ( sort keys %PossibleColumn ) {
                    next COLUMNNAME if !$PossibleColumn{$Name};
                    push @ShowColumns, $Name;
                }
            }

            # get the configured columns and reorganize them by class name
            if (
                IsArrayRefWithData( $Self->{Config}->{ShowColumnsByClass} )
                && $ClassID
                )
            {

                my %ColumnByClass;

                NAME:
                for my $Name ( @{ $Self->{Config}->{ShowColumnsByClass} } ) {
                    my ( $Class, $Column ) = split /::/, $Name, 2;

                    next NAME if !$Column;

                    push @{ $ColumnByClass{$Class} }, $Column;
                }

                # check if there is a specific column config for the selected class
                my $SelectedClass = $ClassList->{$ClassID};
                if ( $ColumnByClass{$SelectedClass} ) {
                    @ShowColumns = @{ $ColumnByClass{$SelectedClass} };
                }
            }

            my $ClassName = $ClassList->{$ClassID};
            my $Title     = $LayoutObject->{LanguageObject}->Translate('Config Item Search Results')
                . ' '
                . $LayoutObject->{LanguageObject}->Translate('Class')
                . ' '
                . $LayoutObject->{LanguageObject}->Translate($ClassName);

            $Output .= $LayoutObject->ITSMConfigItemListShow(
                ConfigItemIDs => $SearchResultList,
                Total         => scalar @{$SearchResultList},
                View          => $Self->{View},
                Filter        => $ClassID,
                Env           => $Self,
                LinkPage      => $LinkPage,
                LinkSort      => $LinkSort,
                LinkFilter    => $LinkFilter,
                LinkBack      => $LinkBack,
                Profile       => $Self->{Profile},
                TitleName     => $Title,
                ShowColumns   => \@ShowColumns,
                SortBy        => $LayoutObject->Ascii2Html( Text => $Self->{SortBy} ),
                OrderBy       => $LayoutObject->Ascii2Html( Text => $Self->{OrderBy} ),
                ClassID       => $ClassID,
                RequestURL    => $Self->{RequestedURL},
            );

            $LayoutObject->AddJSData(
                Key   => 'ITSMConfigItemSearch',
                Value => {
                    Profile => $Self->{Profile},
                    ClassID => $ClassID,
                    Action  => $Self->{Action},
                },
            );

            # build footer
            $Output .= $LayoutObject->Footer();

            return $Output;
        }
    }

    # ------------------------------------------------------------ #
    # call search dialog from search empty screen
    # ------------------------------------------------------------ #
    else {

        # show default search screen
        $Output = $LayoutObject->Header();
        $Output .= $LayoutObject->NavigationBar();

        $LayoutObject->AddJSData(
            Key   => 'ITSMConfigItemOpenSearchDialog',
            Value => {
                Profile => $Self->{Profile},
                ClassID => $ClassID,
                Action  => $Self->{Action},
            },
        );

        # output template
        $Output .= $LayoutObject->Output(
            TemplateFile => 'AgentITSMConfigItemSearch',
        );

        # output footer
        $Output .= $LayoutObject->Footer();

        return $Output;
    }
}

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

    my %GetParam = %{ $Param{GetParam} };

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if ref $Param{XMLDefinition} ne 'ARRAY';
    return if ref $Param{XMLAttributes} ne 'ARRAY';

    $Param{Level} ||= 0;
    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        # set prefix
        my $InputKey = $Item->{Key};
        my $Name     = $Item->{Name};
        if ( $Param{Prefix} ) {
            $InputKey = $Param{Prefix} . '::' . $InputKey;
            $Name     = $Param{PrefixName} . '::' . $Name;
        }

        # output attribute, if marked as searchable
        if ( $Item->{Searchable} ) {
            my $Value;

            # date type fields must to get all date parameters
            if ( $Item->{Input}->{Type} eq 'Date' ) {
                $Value =
                    {
                    $InputKey                      => $GetParam{$InputKey},
                    $InputKey . '::TimeStart::Day' => $GetParam{ $InputKey . '::TimeStart::Day' },
                    $InputKey
                        . '::TimeStart::Month' => $GetParam{ $InputKey . '::TimeStart::Month' },
                    $InputKey . '::TimeStart::Year' => $GetParam{ $InputKey . '::TimeStart::Year' },
                    $InputKey . '::TimeStop::Day'   => $GetParam{ $InputKey . '::TimeStop::Day' },
                    $InputKey . '::TimeStop::Month' => $GetParam{ $InputKey . '::TimeStop::Month' },
                    $InputKey . '::TimeStop::Year'  => $GetParam{ $InputKey . '::TimeStop::Year' },
                    } || '';
            }

            # date-time type fields must get all date and time parameters
            elsif ( $Item->{Input}->{Type} eq 'DateTime' ) {
                $Value =
                    {
                    $InputKey => $GetParam{$InputKey},
                    $InputKey
                        . '::TimeStart::Minute' => $GetParam{ $InputKey . '::TimeStart::Minute' },
                    $InputKey . '::TimeStart::Hour' => $GetParam{ $InputKey . '::TimeStart::Hour' },
                    $InputKey . '::TimeStart::Day'  => $GetParam{ $InputKey . '::TimeStart::Day' },
                    $InputKey
                        . '::TimeStart::Month' => $GetParam{ $InputKey . '::TimeStart::Month' },
                    $InputKey . '::TimeStart::Year' => $GetParam{ $InputKey . '::TimeStart::Year' },
                    $InputKey
                        . '::TimeStop::Minute' => $GetParam{ $InputKey . '::TimeStop::Minute' },
                    $InputKey . '::TimeStop::Hour'  => $GetParam{ $InputKey . '::TimeStop::Hour' },
                    $InputKey . '::TimeStop::Day'   => $GetParam{ $InputKey . '::TimeStop::Day' },
                    $InputKey . '::TimeStop::Month' => $GetParam{ $InputKey . '::TimeStop::Month' },
                    $InputKey . '::TimeStop::Year'  => $GetParam{ $InputKey . '::TimeStop::Year' },
                    } || '';
            }

            # other kinds of fields can get its value directly
            else {
                $Value = $GetParam{$InputKey} || '';
            }

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

            # create search input element
            my $InputString = $LayoutObject->ITSMConfigItemSearchInputCreate(
                Key   => $InputKey,
                Item  => $Item,
                Value => $Value,
            );

            # output attribute row
            $LayoutObject->Block(
                Name => 'AttributeRow',
                Data => {
                    Key         => $InputKey,
                    Name        => $Name,
                    Description => $Item->{Description} || $Item->{Name},
                    InputString => $InputString,
                },
            );

            push @{ $Param{XMLAttributes} }, {
                Key   => $InputKey,
                Value => $Name,
            };
        }

        next ITEM if !$Item->{Sub};

        # start recursion, if "Sub" was found
        $Self->_XMLSearchFormOutput(
            XMLDefinition => $Item->{Sub},
            XMLAttributes => $Param{XMLAttributes},
            Level         => $Param{Level} + 1,
            Prefix        => $InputKey,
            PrefixName    => $Name,
            GetParam      => \%GetParam,
        );
    }

    return 1;
}

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

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if !$Param{XMLFormData};
    return if !$Param{XMLGetParam};
    return if ref $Param{XMLDefinition} ne 'ARRAY';
    return if ref $Param{XMLFormData} ne 'ARRAY';
    return if ref $Param{XMLGetParam} ne 'ARRAY';

    $Param{Level} ||= 0;

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        # create inputkey
        my $InputKey = $Item->{Key};
        if ( $Param{Prefix} ) {
            $InputKey = $Param{Prefix} . '::' . $InputKey;
        }

        # Date type fields must to get all date parameters.
        if ( $Item->{Input}->{Type} eq 'Date' && $Param{$InputKey} ) {
            $Param{$InputKey} =
                {
                $InputKey                      => $Param{$InputKey},
                $InputKey . '::TimeStart::Day' => $Param{ $InputKey . '::TimeStart::Day' },
                $InputKey
                    . '::TimeStart::Month' => $Param{ $InputKey . '::TimeStart::Month' },
                $InputKey . '::TimeStart::Year' => $Param{ $InputKey . '::TimeStart::Year' },
                $InputKey . '::TimeStop::Day'   => $Param{ $InputKey . '::TimeStop::Day' },
                $InputKey . '::TimeStop::Month' => $Param{ $InputKey . '::TimeStop::Month' },
                $InputKey . '::TimeStop::Year'  => $Param{ $InputKey . '::TimeStop::Year' },
                } || '';
        }

        # Date-time type fields must get all date and time parameters.
        elsif ( $Item->{Input}->{Type} eq 'DateTime' && $Param{$InputKey} ) {
            $Param{$InputKey} =
                {
                $InputKey => $Param{$InputKey},
                $InputKey
                    . '::TimeStart::Minute' => $Param{ $InputKey . '::TimeStart::Minute' },
                $InputKey . '::TimeStart::Hour' => $Param{ $InputKey . '::TimeStart::Hour' },
                $InputKey . '::TimeStart::Day'  => $Param{ $InputKey . '::TimeStart::Day' },
                $InputKey
                    . '::TimeStart::Month' => $Param{ $InputKey . '::TimeStart::Month' },
                $InputKey . '::TimeStart::Year' => $Param{ $InputKey . '::TimeStart::Year' },
                $InputKey
                    . '::TimeStop::Minute' => $Param{ $InputKey . '::TimeStop::Minute' },
                $InputKey . '::TimeStop::Hour'  => $Param{ $InputKey . '::TimeStop::Hour' },
                $InputKey . '::TimeStop::Day'   => $Param{ $InputKey . '::TimeStop::Day' },
                $InputKey . '::TimeStop::Month' => $Param{ $InputKey . '::TimeStop::Month' },
                $InputKey . '::TimeStop::Year'  => $Param{ $InputKey . '::TimeStop::Year' },
                } || '';
        }

        # get search form data
        my $Values = $Kernel::OM->Get('Kernel::Output::HTML::Layout')->ITSMConfigItemSearchFormDataGet(
            Key   => $InputKey,
            Item  => $Item,
            Value => $Param{$InputKey},
        );

        # create search key
        my $SearchKey = $InputKey;
        $SearchKey =~ s{ :: }{\'\}[%]\{\'}xmsg;
        $SearchKey = "[1]{'Version'}[1]{'$SearchKey'}[%]{'Content'}";

        # ITSMConfigItemSearchFormDataGet() can return string, arrayref or hashref
        if ( ref $Values eq 'ARRAY' ) {

            # filter empty elements
            my @SearchValues = grep {$_} @{$Values};

            if (@SearchValues) {
                push @{ $Param{XMLFormData} }, {
                    $SearchKey => \@SearchValues,
                };

                push @{ $Param{XMLGetParam} }, {
                    $InputKey => \@SearchValues,
                };
            }

        }
        elsif ($Values) {

            # e.g. for Date between searches
            push @{ $Param{XMLFormData} }, {
                $SearchKey => $Values,
            };

            if ( ref $Values eq 'HASH' ) {
                if ( $Item->{Input}->{Type} eq 'Date' ) {
                    if ( $Values->{'-between'} ) {

                        # get time elemet values
                        my ( $StartDate, $StopDate ) = @{ $Values->{'-between'} };
                        my ( $StartYear, $StartMonth, $StartDay ) = split( /-/, $StartDate );
                        my ( $StopYear,  $StopMonth,  $StopDay )  = split( /-/, $StopDate );

                        # store time elment values
                        push @{ $Param{XMLGetParam} }, {
                            $InputKey                        => 1,
                            $InputKey . '::TimeStart::Day'   => $StartDay,
                            $InputKey . '::TimeStart::Month' => $StartMonth,
                            $InputKey . '::TimeStart::Year'  => $StartYear,
                            $InputKey . '::TimeStop::Day'    => $StopDay,
                            $InputKey . '::TimeStop::Month'  => $StopMonth,
                            $InputKey . '::TimeStop::Year'   => $StopYear,
                        };
                    }
                }
                elsif ( $Item->{Input}->{Type} eq 'DateTime' ) {
                    if ( $Values->{'-between'} ) {

                        # get time elemet values
                        my ( $StartDateTime, $StopDateTime ) = @{ $Values->{'-between'} };
                        my ( $StartDate,     $StartTime )    = split( /\s/, $StartDateTime );
                        my ( $StartYear, $StartMonth,  $StartDay )    = split( /-/,  $StartDate );
                        my ( $StartHour, $StartMinute, $StartSecond ) = split( /\:/, $StartTime );

                        my ( $StopDate, $StopTime ) = split( /\s/, $StopDateTime );
                        my ( $StopYear, $StopMonth,  $StopDay )    = split( /-/,  $StopDate );
                        my ( $StopHour, $StopMinute, $StopSecond ) = split( /\:/, $StopTime );

                        # store time elment values
                        push @{ $Param{XMLGetParam} }, {
                            $InputKey                         => 1,
                            $InputKey . '::TimeStart::Minute' => $StartMinute,
                            $InputKey . '::TimeStart::Hour'   => $StartHour,
                            $InputKey . '::TimeStart::Day'    => $StartDay,
                            $InputKey . '::TimeStart::Month'  => $StartMonth,
                            $InputKey . '::TimeStart::Year'   => $StartYear,
                            $InputKey . '::TimeStop::Minute'  => $StopMinute,
                            $InputKey . '::TimeStop::Hour'    => $StopHour,
                            $InputKey . '::TimeStop::Day'     => $StopDay,
                            $InputKey . '::TimeStop::Month'   => $StopMonth,
                            $InputKey . '::TimeStop::Year'    => $StopYear,
                        };
                    }
                }
            }
            else {
                push @{ $Param{XMLGetParam} }, {
                    $InputKey => $Values,
                };
            }

        }

        next ITEM if !$Item->{Sub};

        # start recursion, if "Sub" was found
        $Self->_XMLSearchFormGet(
            %Param,
            XMLDefinition => $Item->{Sub},
            XMLFormData   => $Param{XMLFormData},
            XMLGetParam   => $Param{XMLGetParam},
            Level         => $Param{Level} + 1,
            Prefix        => $InputKey,
        );
    }

    return 1;
}

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

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if ref $Param{XMLDefinition} ne 'ARRAY';
    return if ref $Param{XMLAttributes} ne 'ARRAY';

    $Param{Level} ||= 0;
    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        # set prefix
        my $InputKey = $Item->{Key};
        my $Name     = $Item->{Name};
        if ( $Param{Prefix} ) {
            $InputKey = $Param{Prefix} . '::' . $InputKey;
            $Name     = $Param{PrefixName} . '::' . $Name;
        }

        # store attribute, if marked as searchable
        if ( $Item->{Searchable} ) {
            push @{ $Param{XMLAttributes} }, {
                Key   => $InputKey,
                Value => $Name,
            };
        }

        next ITEM if !$Item->{Sub};

        # start recursion, if "Sub" was found
        $Self->_XMLSearchAttributesGet(
            XMLDefinition => $Item->{Sub},
            XMLAttributes => $Param{XMLAttributes},
            Level         => $Param{Level} + 1,
            Prefix        => $InputKey,
            PrefixName    => $Name,
        );
    }

    return 1;
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Modules::AgentITSMConfigItemZoom;

use strict;
use warnings;
use utf8;

use Kernel::Language qw(Translatable);

our $ObjectManagerDisabled = 1;

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

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

    return $Self;
}

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

    # get param object
    my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $LogObject   = $Kernel::OM->Get('Kernel::System::Log');

    # get params
    my $ConfigItemID = $ParamObject->GetParam( Param => 'ConfigItemID' ) || 0;
    my $VersionID    = $ParamObject->GetParam( Param => 'VersionID' )    || 0;

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

    # check needed stuff
    if ( !$ConfigItemID ) {
        return $LayoutObject->ErrorScreen(
            Message => Translatable('No ConfigItemID is given!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # get needed object
    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $ConfigObject     = $Kernel::OM->Get('Kernel::Config');

    # check for access rights
    my $HasAccess = $ConfigItemObject->Permission(
        Scope  => 'Item',
        ItemID => $ConfigItemID,
        UserID => $Self->{UserID},
        Type   => $ConfigObject->Get("ITSMConfigItem::Frontend::$Self->{Action}")->{Permission},
    );

    if ( !$HasAccess ) {

        # error page
        return $LayoutObject->ErrorScreen(
            Message => Translatable('Can\'t show item, no access rights for ConfigItem are given!'),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # set show versions
    $Param{ShowVersions} = 0;
    if ( $ParamObject->GetParam( Param => 'ShowVersions' ) ) {
        $Param{ShowVersions} = 1;
    }

    # get content
    my $ConfigItem = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => $ConfigItemID,
    );
    if ( !$ConfigItem->{ConfigItemID} ) {
        return $LayoutObject->ErrorScreen(
            Message =>
                $LayoutObject->{LanguageObject}->Translate( 'ConfigItemID %s not found in database!', $ConfigItemID ),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # get version list
    my $VersionList = $ConfigItemObject->VersionZoomList(
        ConfigItemID => $ConfigItemID,
    );
    if ( !$VersionList->[0]->{VersionID} ) {
        return $LayoutObject->ErrorScreen(
            Message =>
                $LayoutObject->{LanguageObject}->Translate( 'No version found for ConfigItemID %s!', $ConfigItemID ),
            Comment => Translatable('Please contact the administrator.'),
        );
    }

    # set version id
    if ( !$VersionID ) {
        $VersionID = $VersionList->[-1]->{VersionID};
    }
    if ( $VersionID ne $VersionList->[-1]->{VersionID} ) {
        $Param{ShowVersions} = 1;
    }

    # set version id in param hash (only for menu module)
    if ($VersionID) {
        $Param{VersionID} = $VersionID;
    }

    # run config item menu modules
    if ( ref $ConfigObject->Get('ITSMConfigItem::Frontend::MenuModule') eq 'HASH' ) {
        my %Menus   = %{ $ConfigObject->Get('ITSMConfigItem::Frontend::MenuModule') };
        my $Counter = 0;

        Menu:
        for my $Menu ( sort keys %Menus ) {

            my $Access;
            my $Permission;
            my $Action = $Menus{$Menu}->{Action};

            if ($Action) {
                my $Config = $ConfigObject->Get("ITSMConfigItem::Frontend::$Action") || {};

                if ( $Config && $Config->{Permission} ) {
                    $Permission = $Config->{Permission};
                }
            }

            if ( !$Permission ) {
                $Access = 1;
            }

            if ( !$Access && defined $Permission ) {

                $Access = $ConfigItemObject->Permission(
                    Type   => $Permission,
                    Scope  => 'Item',
                    ItemID => $ConfigItemID,
                    UserID => $Self->{UserID},
                );
            }
            next Menu if !$Access;

            # load module
            if ( $Kernel::OM->Get('Kernel::System::Main')->Require( $Menus{$Menu}->{Module} ) ) {

                my $Object = $Menus{$Menu}->{Module}->new(
                    %{$Self},
                    ConfigItemID => $Self->{ConfigItemID},
                );

                # set classes
                if ( $Menus{$Menu}->{Target} ) {

                    if ( $Menus{$Menu}->{Target} eq 'PopUp' ) {
                        $Menus{$Menu}->{MenuClass} = 'AsPopup';
                    }
                    elsif ( $Menus{$Menu}->{Target} eq 'Back' ) {
                        $Menus{$Menu}->{MenuClass} = 'HistoryBack';
                    }
                }

                # run module
                $Counter = $Object->Run(
                    %Param,
                    ConfigItem => $ConfigItem,
                    Counter    => $Counter,
                    Config     => $Menus{$Menu},
                    MenuID     => $Menu,
                );
            }
            else {
                return $LayoutObject->FatalError();
            }
        }
    }

    # build version tree
    $LayoutObject->Block( Name => 'Tree' );
    my $Counter = 1;
    if ( !$Param{ShowVersions} && $VersionID eq $VersionList->[-1]->{VersionID} ) {
        $Counter     = @{$VersionList};
        $VersionList = [ $VersionList->[-1] ];
    }

    # get last version
    my $LastVersion = $VersionList->[-1];

    # set incident signal
    my %InciSignals = (
        Translatable('operational') => 'greenled',
        Translatable('warning')     => 'yellowled',
        Translatable('incident')    => 'redled',
    );

    # to store the color for the deployment states
    my %DeplSignals;

    # get general catalog object
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

    # get list of deployment states
    my $DeploymentStatesList = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    # set deployment style colors
    my $StyleClasses = '';

    ITEMID:
    for my $ItemID ( sort keys %{$DeploymentStatesList} ) {

        # get deployment state preferences
        my %Preferences = $GeneralCatalogObject->GeneralCatalogPreferencesGet(
            ItemID => $ItemID,
        );

        # check if a color is defined in preferences
        next ITEMID if !$Preferences{Color};

        # get deployment state
        my $DeplState = $DeploymentStatesList->{$ItemID};

        # remove any non ascii word characters
        $DeplState =~ s{ [^a-zA-Z0-9] }{_}msxg;

        # store the original deployment state as key
        # and the ss safe coverted deployment state as value
        $DeplSignals{ $DeploymentStatesList->{$ItemID} } = $DeplState;

        # covert to lower case
        my $DeplStateColor = lc $Preferences{Color};

        # add to style classes string
        $StyleClasses .= "
            .Flag span.$DeplState {
                background-color: $DeplStateColor;
            }
        ";
    }

    # wrap into style tags
    if ($StyleClasses) {
        $StyleClasses = "<style>$StyleClasses</style>";
    }

    # output version tree header
    if ( $Param{ShowVersions} ) {
        $LayoutObject->Block(
            Name => 'Collapse',
            Data => {
                ConfigItemID => $ConfigItemID,
            },
        );
    }
    else {
        $LayoutObject->Block(
            Name => 'Expand',
            Data => {
                ConfigItemID => $ConfigItemID,
            },
        );
    }

    # get user object
    my $UserObject = $Kernel::OM->Get('Kernel::System::User');

    # output version tree
    for my $VersionHash ( @{$VersionList} ) {

        $Param{CreateByUserFullName} = $UserObject->UserName(
            UserID => $VersionHash->{CreateBy},
        );

        $LayoutObject->Block(
            Name => 'TreeItem',
            Data => {
                %Param,
                %{$ConfigItem},
                %{$VersionHash},
                Count      => $Counter,
                InciSignal => $InciSignals{ $VersionHash->{InciStateType} },
                DeplSignal => $DeplSignals{ $VersionHash->{DeplState} },
                Active     => $VersionHash->{VersionID} eq $VersionID ? 'Active' : '',
            },
        );

        $Counter++;
    }

    # output header
    my $Output = $LayoutObject->Header( Value => $ConfigItem->{Number} );
    $Output .= $LayoutObject->NavigationBar();

    # get version
    my $Version = $ConfigItemObject->VersionGet(
        VersionID => $VersionID,
    );

    if (
        $Version
        && ref $Version eq 'HASH'
        && $Version->{XMLDefinition}
        && $Version->{XMLData}
        && ref $Version->{XMLDefinition} eq 'ARRAY'
        && ref $Version->{XMLData} eq 'ARRAY'
        && $Version->{XMLData}->[1]
        && ref $Version->{XMLData}->[1] eq 'HASH'
        && $Version->{XMLData}->[1]->{Version}
        && ref $Version->{XMLData}->[1]->{Version} eq 'ARRAY'
        )
    {

        # transform ascii to html
        $Version->{Name} = $LayoutObject->Ascii2Html(
            Text           => $Version->{Name},
            HTMLResultMode => 1,
            LinkFeature    => 1,
        );

        # output name
        $LayoutObject->Block(
            Name => 'Data',
            Data => {
                Name        => Translatable('Name'),
                Description => Translatable('The name of this config item'),
                Value       => $Version->{Name},
                Identation  => 10,
            },
        );

        # output deployment state
        $LayoutObject->Block(
            Name => 'Data',
            Data => {
                Name        => Translatable('Deployment State'),
                Description => Translatable('The deployment state of this config item'),
                Value       => $LayoutObject->{LanguageObject}->Translate(
                    $Version->{DeplState},
                ),
                Identation => 10,
            },
        );

        # output incident state
        $LayoutObject->Block(
            Name => 'Data',
            Data => {
                Name        => Translatable('Incident State'),
                Description => Translatable('The incident state of this config item'),
                Value       => $LayoutObject->{LanguageObject}->Translate(
                    $Version->{InciState},
                ),
                Identation => 10,
            },
        );

        # start xml output
        $Self->_XMLOutput(
            XMLDefinition => $Version->{XMLDefinition},
            XMLData       => $Version->{XMLData}->[1]->{Version}->[1],
        );
    }

    # get create & change user data
    for my $Key (qw(Create Change)) {
        $ConfigItem->{ $Key . 'ByUserFullName' } = $UserObject->UserName(
            UserID => $ConfigItem->{ $Key . 'By' },
        );
    }

    # output meta block
    $LayoutObject->Block(
        Name => 'Meta',
        Data => {
            %{$LastVersion},
            %{$ConfigItem},
            CurInciSignal => $InciSignals{ $LastVersion->{CurInciStateType} },
            CurDeplSignal => $DeplSignals{ $LastVersion->{DeplState} },
        },
    );

    # get linked objects
    my $LinkListWithData = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkListWithData(
        Object => 'ITSMConfigItem',
        Key    => $ConfigItemID,
        State  => 'Valid',
        UserID => $Self->{UserID},
    );

    # get link table view mode
    my $LinkTableViewMode = $ConfigObject->Get('LinkObject::ViewMode');

    # create the link table
    my $LinkTableStrg = $LayoutObject->LinkObjectTableCreate(
        LinkListWithData => $LinkListWithData,
        ViewMode         => $LinkTableViewMode,
        Object           => 'ITSMConfigItem',
        Key              => $ConfigItemID,
    );

    # output the link table
    if ($LinkTableStrg) {
        $LayoutObject->Block(
            Name => 'LinkTable' . $LinkTableViewMode,
            Data => {
                LinkTableStrg => $LinkTableStrg,
            },
        );
    }

    my @Attachments = $ConfigItemObject->ConfigItemAttachmentList(
        ConfigItemID => $ConfigItemID,
    );

    if (@Attachments) {

        # get the metadata of the 1st attachment
        my $FirstAttachment = $ConfigItemObject->ConfigItemAttachmentGet(
            ConfigItemID => $ConfigItemID,
            Filename     => $Attachments[0],
        );

        $LayoutObject->Block(
            Name => 'Attachments',
            Data => {
                ConfigItemID => $ConfigItemID,
                Filename     => $FirstAttachment->{Filename},
                Filesize     => $FirstAttachment->{Filesize},
            },
        );

        # the 1st attachment was directly rendered into the 1st row's right cell, all further
        # attachments are rendered into a separate row
        ATTACHMENT:
        for my $Attachment (@Attachments) {

            # skip the 1st attachment
            next ATTACHMENT if $Attachment eq $Attachments[0];

            # get the metadata of the current attachment
            my $AttachmentData = $ConfigItemObject->ConfigItemAttachmentGet(
                ConfigItemID => $ConfigItemID,
                Filename     => $Attachment,
            );

            $LayoutObject->Block(
                Name => 'AttachmentRow',
                Data => {
                    ConfigItemID => $ConfigItemID,
                    Filename     => $AttachmentData->{Filename},
                    Filesize     => $AttachmentData->{Filesize},
                },
            );
        }
    }

    # handle DownloadAttachment
    if ( $Self->{Subaction} eq 'DownloadAttachment' ) {

        # get data for attachment
        my $Filename       = $ParamObject->GetParam( Param => 'Filename' );
        my $AttachmentData = $ConfigItemObject->ConfigItemAttachmentGet(
            ConfigItemID => $ConfigItemID,
            Filename     => $Filename,
        );

        # return error if file does not exist
        if ( !$AttachmentData ) {
            $LogObject->Log(
                Message  => "No such attachment ($Filename)!",
                Priority => 'error',
            );
            return $LayoutObject->ErrorScreen();
        }

        return $LayoutObject->Attachment(
            %{$AttachmentData},
            Type => 'attachment',
        );
    }

    # store last screen
    $Kernel::OM->Get('Kernel::System::AuthSession')->UpdateSessionID(
        SessionID => $Self->{SessionID},
        Key       => 'LastScreenView',
        Value     => $Self->{RequestedURL},
    );

    $LayoutObject->AddJSData(
        Key   => 'UserConfigItemZoomTableHeight',
        Value => $Self->{UserConfigItemZoomTableHeight},
    );

    # start template output
    $Output .= $LayoutObject->Output(
        TemplateFile => 'AgentITSMConfigItemZoom',
        Data         => {
            %{$LastVersion},
            %{$ConfigItem},
            CurInciSignal => $InciSignals{ $LastVersion->{CurInciStateType} },
            CurDeplSignal => $DeplSignals{ $LastVersion->{DeplState} },
            StyleClasses  => $StyleClasses,
        },
    );

    # add footer
    $Output .= $LayoutObject->Footer();

    return $Output;
}

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

    # check needed stuff
    return if !$Param{XMLData};
    return if !$Param{XMLDefinition};
    return if ref $Param{XMLData} ne 'HASH';
    return if ref $Param{XMLDefinition} ne 'ARRAY';

    $Param{Level} ||= 0;

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {
        COUNTER:
        for my $Counter ( 1 .. $Item->{CountMax} ) {

            # stop loop, if no content was given
            last COUNTER if !defined $Param{XMLData}->{ $Item->{Key} }->[$Counter]->{Content};

            # lookup value
            my $Value = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->XMLValueLookup(
                Item  => $Item,
                Value => $Param{XMLData}->{ $Item->{Key} }->[$Counter]->{Content},
            );

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

            # create output string
            $Value = $LayoutObject->ITSMConfigItemOutputStringCreate(
                Value => $Value,
                Item  => $Item,
            );

            # calculate indentation for left-padding css based on 15px per level and 10px as default
            my $Indentation = 10;

            if ( $Param{Level} ) {
                $Indentation += 15 * $Param{Level};
            }

            # output data block
            $LayoutObject->Block(
                Name => 'Data',
                Data => {
                    Name        => $Item->{Name},
                    Description => $Item->{Description} || $Item->{Name},
                    Value       => $Value,
                    Indentation => $Indentation,
                },
            );

            # start recursion, if "Sub" was found
            if ( $Item->{Sub} ) {
                $Self->_XMLOutput(
                    XMLDefinition => $Item->{Sub},
                    XMLData       => $Param{XMLData}->{ $Item->{Key} }->[$Counter],
                    Level         => $Param{Level} + 1,
                );
            }
        }
    }

    return 1;
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Output::HTML::Dashboard::ITSMConfigItemGeneric;

use strict;
use warnings;

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 );

    # Get needed parameters.
    for my $Needed (qw(Config Name UserID)) {
        die "Got no $Needed!" if ( !$Self->{$Needed} );
    }

    # Get param object.
    my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');

    # Get current filter.
    my $Name           = $ParamObject->GetParam( Param => 'Name' ) || '';
    my $PreferencesKey = 'DashboardITSMConfigItemGeneric' . $Self->{Name};
    if ( $Self->{Name} eq $Name ) {
        $Self->{Filter} = $ParamObject->GetParam( Param => 'Filter' ) || '';
    }

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

    # Remember filter.
    if ( $Self->{Filter} ) {

        # Update session.
        $Kernel::OM->Get('Kernel::System::AuthSession')->UpdateSessionID(
            SessionID => $Self->{SessionID},
            Key       => $PreferencesKey,
            Value     => $Self->{Filter},
        );

        # Update preferences.
        if ( !$ConfigObject->Get('DemoSystem') ) {
            $Kernel::OM->Get('Kernel::System::User')->SetPreferences(
                UserID => $Self->{UserID},
                Key    => $PreferencesKey,
                Value  => $Self->{Filter},
            );
        }
    }

    # Set default filter if not set yet.
    if ( !$Self->{Filter} ) {
        $Self->{Filter} = $Self->{$PreferencesKey} || $Self->{Config}->{Filter} || 'All';
    }

    # Setup the prefrences keys.
    $Self->{PrefKeyShown}   = 'DashboardITSMConfigItemGeneric' . $Self->{Name} . '-Shown';
    $Self->{PrefKeyRefresh} = 'DashboardITSMConfigItemGeneric' . $Self->{Name} . '-Refresh';

    $Self->{PageShown} = $Kernel::OM->Get('Kernel::Output::HTML::Layout')->{ $Self->{PrefKeyShown} }
        || $Self->{Config}->{Limit} || 10;

    $Self->{PageRefresh} = $Kernel::OM->Get('Kernel::Output::HTML::Layout')->{ $Self->{PrefKeyRefresh} }
        || 1;

    $Self->{StartHit} = int( $ParamObject->GetParam( Param => 'StartHit' ) || 1 );

    return $Self;
}

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

    my @Params = (
        {
            Desc  => Translatable('Shown config items'),
            Name  => $Self->{PrefKeyShown},
            Block => 'Option',
            Data  => {
                5  => ' 5',
                10 => '10',
                15 => '15',
                20 => '20',
                25 => '25',
                30 => '30',
            },
            SelectedID  => $Self->{PageShown},
            Translation => 0,
        },
    );

    return @Params;
}

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

    return (
        %{ $Self->{Config} },

        # Remember, do not allow to use page cache
        # (it's not working because of internal filter)
        CacheKey => undef,
        CacheTTL => undef,
    );
}

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

    # Get param object.
    my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');

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

    # Get sorting parameters.
    $Param{SortBy} = $ParamObject->GetParam( Param => 'SortBy' ) || 'Number';

    # Get ordering parameters.
    $Param{OrderBy} = $ParamObject->GetParam( Param => 'OrderBy' ) || 'Up';

    # Set Sort and Order by as Arrays.
    my @SortByArray  = ( $Param{SortBy} );
    my @OrderByArray = ( $Param{OrderBy} );

    # Get general catalog object.
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

    # Get class list.
    my $ClassList = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # To store the color for the deployment states.
    my %DeplSignals;

    # Get list of deployment states.
    my $DeploymentStatesList = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    # Set deployment style colors.
    my $StyleClasses = '';

    # To store depl state ids.
    my $DeplStateIDs;

    ITEMID:
    for my $ItemID ( sort keys %{$DeploymentStatesList} ) {

        # Get deployment state preferences.
        my %Preferences = $GeneralCatalogObject->GeneralCatalogPreferencesGet(
            ItemID => $ItemID,
        );

        # Check if a color is defined in preferences.
        next ITEMID if !$Preferences{Color};

        # Get deployment state.
        my $DeplState = $DeploymentStatesList->{$ItemID};

        # Remove any non ascii word characters.
        $DeplState =~ s{ [^a-zA-Z0-9] }{_}msxg;

        # Store the original deployment state as key
        # and the ss safe coverted deployment state as value.
        $DeplSignals{ $DeploymentStatesList->{$ItemID} } = $DeplState;

        # Convert to lower case.
        my $DeplStateColor = lc $Preferences{Color};

        # Add to style classes string.
        $StyleClasses .= "
            .Flag span.$DeplState {
                background-color: $DeplStateColor;
            }
        ";

        # Set depl state ids.
        push @{$DeplStateIDs}, $ItemID;
    }

    # Wrap into style tags.
    if ($StyleClasses) {
        $StyleClasses = "<style>$StyleClasses</style>";
    }

    # To store the default class.
    my $ClassIDAuto = '';

    # To store the NavBar filters.
    my %Filters;

    # Define position of the filter in the frontend.
    my $PrioCounter = 1000;

    # To store the total number of config items in all classes that the user has access.
    my $TotalCount;

    # To store all the clases that the user has access, used in search for filter 'All'.
    my $AccessClassList;

    # Get config item object.
    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    # Define incident signals, needed for services.
    my %InciSignals = (
        Translatable('operational') => 'greenled',
        Translatable('warning')     => 'yellowled',
        Translatable('incident')    => 'redled',
    );

    CLASSID:
    for my $ClassID ( sort { ${$ClassList}{$a} cmp ${$ClassList}{$b} } keys %{$ClassList} ) {

        # Check permission.
        my $HasAccess = $ConfigItemObject->Permission(
            Scope   => 'Class',
            ClassID => $ClassID,
            UserID  => $Self->{UserID},
            Type    => $Self->{Config}->{Permission},
        );

        next CLASSID if !$HasAccess;

        # Insert this class to be passed as search parameter for filter 'All'.
        push @{$AccessClassList}, $ClassID;

        # Increase the PrioCounter.
        $PrioCounter++;

        # Add filter with params for the search method.
        $Filters{$ClassID} = {
            Name   => $ClassList->{$ClassID},
            Prio   => $PrioCounter,
            Search => {
                ClassIDs         => [$ClassID],
                DeplStateIDs     => $DeplStateIDs,
                OrderBy          => \@SortByArray,
                OrderByDirection => \@OrderByArray,
                Limit            => $Self->{SearchLimit},
            },
        };

        # Remember the first class id to show this in the overview
        # if no class id was given.
        if ( !$ClassIDAuto ) {
            $ClassIDAuto = $ClassID;
        }
    }

    # If only one filter exists.
    if ( scalar keys %Filters == 1 ) {

        # Get the name of the only filter.
        my ($FilterKey) = keys %Filters;

        # Activate this filter.
        $Self->{Filter} = $Filters{$FilterKey}->{Name};
    }
    else {

        # Add default filter, which shows all items.
        $Filters{All} = {
            Name   => 'All',
            Prio   => 1000,
            Search => {
                ClassIDs         => $AccessClassList,
                DeplStateIDs     => $DeplStateIDs,
                OrderBy          => \@SortByArray,
                OrderByDirection => \@OrderByArray,
                Limit            => $Self->{SearchLimit},
            },
        };

        # If no filter was selected activate the filter for the default class.
        if ( !$Self->{Filter} ) {
            $Self->{Filter} = $ClassIDAuto;
        }
    }

    # Get given filters.
    my @NavBarFilters;
    for my $Prio ( sort keys %Filters ) {
        push @NavBarFilters, $Filters{$Prio};
    }

    # Sort according to prio.
    @NavBarFilters = sort { $a->{Prio} <=> $b->{Prio} } @NavBarFilters;

    # Find out which columns should be shown.
    my @ShowColumns;
    my @XMLShowColumns;
    if ( $Self->{Config}->{DefaultColumns} ) {

        # Get all possible columns from config.
        my %PossibleColumn = %{ $Self->{Config}->{DefaultColumns} };

        # Show column "Class" if filter 'All' is selected.
        if ( $Self->{Filter} eq 'All' ) {
            $PossibleColumn{Class} = '1';
        }

        # Get the column names that should be shown.
        COLUMNNAME:
        for my $Name ( sort keys %PossibleColumn ) {
            next COLUMNNAME if !$PossibleColumn{$Name};
            push @ShowColumns, $Name;
        }
    }

    # Get frontend config.
    my ($Module) = $Self->{Action} =~ m{^Agent(.*)}xms;
    my $ShowColumnsByClassConfig
        = $Kernel::OM->Get('Kernel::Config')->Get( 'Frontend::Agent::' . $Module . '::ITSMConfigItem' );

    my $ShowColumnsByClass = $ShowColumnsByClassConfig->{ShowColumnsByClass} // '';

    # Get the configured columns and reorganize them by class name.
    if (
        IsArrayRefWithData($ShowColumnsByClass)
        && $Self->{Filter}
        && $Self->{Filter} ne 'All'
        )
    {

        my %ColumnByClass;
        NAME:
        for my $Name ( @{$ShowColumnsByClass} ) {
            my ( $Class, $Column ) = split /::/, $Name, 2;

            next NAME if !$Column;

            push @{ $ColumnByClass{$Class} }, $Column;
        }

        # Check if there is a specific column config for the selected class.
        if ( $ColumnByClass{ $Self->{Filter} } ) {
            @ShowColumns = @{ $ColumnByClass{ $Self->{Filter} } };
        }
    }

    if (@ShowColumns) {

        for my $Column (@ShowColumns) {

            # Create needed veriables.
            my $CSS = 'OverviewHeader';
            my $OrderBy;

            # Remove ID if necesary.
            if ( $Param{SortBy} ) {
                $Param{SortBy} = ( $Param{SortBy} eq 'InciStateID' )
                    ? 'CurInciState'
                    : ( $Param{SortBy} eq 'DeplStateID' ) ? 'CurDeplState'
                    : ( $Param{SortBy} eq 'ClassID' )     ? 'Class'
                    : ( $Param{SortBy} eq 'ChangeTime' )  ? 'LastChanged'
                    :                                       $Param{SortBy};
            }

            # Set the correct Set CSS class and order by link.
            if ( $Param{SortBy} && ( $Param{SortBy} eq $Column ) ) {
                if ( $Param{OrderBy} && ( $Param{OrderBy} eq 'Up' ) ) {
                    $OrderBy = 'Down';
                    $CSS .= ' SortDescendingLarge';
                }
                else {
                    $OrderBy = 'Up';
                    $CSS .= ' SortAscendingLarge';
                }
            }
            else {
                $OrderBy = 'Up';
            }

            # Set column.
            $Param{Column} = $Column;

            # Create header block.
            $LayoutObject->Block(
                Name => 'Record' . $Column . 'Header',
                Data => {
                    %Param,
                    CSS     => $CSS,
                    OrderBy => $OrderBy,
                },
            );
        }
    }

    # Get assigned config items (to customer or customer user).
    my $AssignedCIs = $Self->_AssignedCIsGet(
        Filter => $Self->{Filter},
    );
    my @AssignedConfigItemIDs = @{ $AssignedCIs->{ConfigItemIDs} };

    # Get all assigned config items for count.
    my $CountCIs = $Self->_AssignedCIsGet(
        Filter => 'All',
        Count  => 1,
    );

    # Create search with sorting parameters.
    my $AssignedConfigItemIDs;
    if ( IsArrayRefWithData( \@AssignedConfigItemIDs ) ) {
        $AssignedConfigItemIDs = $ConfigItemObject->ConfigItemSearch(
            ConfigItemIDs    => \@AssignedConfigItemIDs,
            OrderBy          => \@SortByArray,
            OrderByDirection => \@OrderByArray,
        );
    }

    # Clear old array and push new data.
    if ( IsArrayRefWithData($AssignedConfigItemIDs) ) {

        @AssignedConfigItemIDs = ();
        push @AssignedConfigItemIDs, @{$AssignedConfigItemIDs};
    }

    if (@ShowColumns) {

        # Get the XML column headers only if the filter is not set to 'all'
        # and if there are CIs to show.
        if ( $Self->{Filter} && $Self->{Filter} ne 'All' && @AssignedConfigItemIDs ) {

            # Get the version data of the first config item, including all the XML data
            # to get the column header names.
            my $ConfigItem = $ConfigItemObject->VersionGet(
                ConfigItemID => $AssignedConfigItemIDs[0],
                XMLDataGet   => 1,
            );

            # Convert the XML data into a hash.
            my $ExtendedVersionData = $LayoutObject->XMLData2Hash(
                XMLDefinition => $ConfigItem->{XMLDefinition},
                XMLData       => $ConfigItem->{XMLData}->[1]->{Version}->[1],
                Attributes    => \@ShowColumns,
            );

            # Get the xml columns (they contain ::).
            @XMLShowColumns = grep {/::/} @ShowColumns;

            COLUMN:
            for my $Column (@XMLShowColumns) {

                # Check if column exists in CI-Data.
                next COLUMN if !$ExtendedVersionData->{$Column}->{Name};

                # Show the xml attribute header.
                $LayoutObject->Block(
                    Name => 'RecordXMLAttributeHeader',
                    Data => {
                        %Param,
                        XMLAttributeHeader => $ExtendedVersionData->{$Column}->{Name},
                    },
                );
            }
        }
    }

    # Build filter content.
    $LayoutObject->Block(
        Name => 'OverviewNavBarFilter',
        Data => {
            %Filters,
        },
    );

    # Set summary for page nav.
    my %Summary;

    # Loop over filters.
    my $Count = 0;
    FILTER:
    for my $Filter (@NavBarFilters) {

        # Get count by assigned config item ids.
        $Filter->{Count} = $CountCIs->{ $Filter->{Name} } // '0';

        $Count++;
        if ( $Count == scalar @NavBarFilters ) {
            $Filter->{CSS} = 'Last';
        }
        $LayoutObject->Block(
            Name => 'OverviewNavBarFilterItem',
            Data => {
                %Param,
                %{$Filter},
            },
        );
        if ( $Filter->{Name} eq $Self->{Filter} ) {
            $LayoutObject->Block(
                Name => 'OverviewNavBarFilterItemSelected',
                Data => {
                    %Param,
                    %{$Filter},
                },
            );
        }
        else {
            $LayoutObject->Block(
                Name => 'OverviewNavBarFilterItemSelectedNot',
                Data => {
                    %Param,
                    %{$Filter},
                },
            );
        }

        # Add filter to new hash.
        $Summary{ $Filter->{Name} } = \%{$Filter};
    }

    # Add page nav bar.
    my $Total = $Summary{ $Self->{Filter} }->{Count} || 0;

    my $LinkPage =
        'Subaction=Element;Name=' . $Self->{Name}
        . ';Filter=' . $Self->{Filter}
        . ';SortBy=' .  ( $Param{SortBy}  || '' )
        . ';OrderBy=' . ( $Param{OrderBy} || '' )
        . ';';

    if ( $Param{CustomerID} ) {
        $LinkPage .= 'CustomerID=' . $LayoutObject->LinkEncode( $Param{CustomerID} ) . ';';
    }
    if ( $Param{CustomerUserID} ) {
        $LinkPage .= 'CustomerUserID=' . $LayoutObject->LinkEncode( $Param{CustomerUserID} ) . ';';
    }

    my %PageNav = $LayoutObject->PageNavBar(
        StartHit    => $Self->{StartHit},
        PageShown   => $Self->{PageShown},
        AllHits     => $Total || 1,
        Action      => 'Action=' . $LayoutObject->{Action},
        Link        => $LinkPage,
        AJAXReplace => 'Dashboard' . $Self->{Name},
        IDPrefix    => 'Dashboard' . $Self->{Name},
        AJAX        => $Param{AJAX},
    );

    $LayoutObject->Block(
        Name => 'ContentLargeITSMConfigItemGenericFilterNavBar',
        Data => {
            %{ $Self->{Config} },
            Name => $Self->{Name},
            %PageNav,
        },
    );

    # Show config items if there are some.
    my $Counter = 0;
    if (@AssignedConfigItemIDs) {

        # To store all data.
        my %Data;

        CONFIGITEMID:
        for my $ConfigItemID (@AssignedConfigItemIDs) {
            $Counter++;
            if (
                $Counter >= $Self->{StartHit}
                && $Counter < ( $Self->{PageShown} + $Self->{StartHit} )
                )
            {

                # Get config item data.
                my $ConfigItem = $ConfigItemObject->VersionGet(
                    ConfigItemID => $ConfigItemID,
                    XMLDataGet   => 1,
                );

                next CONFIGITEMID if !$ConfigItem;

                # Convert the XML data into a hash.
                my $ExtendedVersionData = $LayoutObject->XMLData2Hash(
                    XMLDefinition => $ConfigItem->{XMLDefinition},
                    XMLData       => $ConfigItem->{XMLData}->[1]->{Version}->[1],
                    Attributes    => \@ShowColumns,
                );

                # Store config item data.
                %Data = %{$ConfigItem};

                # Build record block.
                $LayoutObject->Block(
                    Name => 'Record',
                    Data => {
                        %Param,
                        %Data,
                    },
                );

                # Build column record blocks.
                if (@ShowColumns) {

                    COLUMN:
                    for my $Column (@ShowColumns) {
                        $LayoutObject->Block(
                            Name => 'Record' . $Column,
                            Data => {
                                %Param,
                                %Data,
                                CurInciSignal => $InciSignals{ $Data{CurInciStateType} },
                                CurDeplSignal => $DeplSignals{ $Data{CurDeplState} },
                            },
                        );

                        # Show links if available.
                        $LayoutObject->Block(
                            Name => 'Record' . $Column . 'LinkStart',
                            Data => {
                                %Param,
                                %Data,
                            },
                        );
                        $LayoutObject->Block(
                            Name => 'Record' . $Column . 'LinkEnd',
                            Data => {
                                %Param,
                                %Data,
                            },
                        );
                    }

                    COLUMN:
                    for my $Column (@XMLShowColumns) {

                        # Check if column exists in CI-Data.
                        next COLUMN if !$ExtendedVersionData->{$Column}->{Name};

                        # Convert to ascii text in case the value contains html.
                        my $Value = $Kernel::OM->Get('Kernel::System::HTMLUtils')
                            ->ToAscii( String => $ExtendedVersionData->{$Column}->{Value} // '' )
                            // '';

                        # Convert all whitespace and newlines to single spaces.
                        $Value =~ s{ \s+ }{ }gxms;

                        # Show the xml attribute data.
                        $LayoutObject->Block(
                            Name => 'RecordXMLAttribute',
                            Data => {
                                %Param,
                                XMLAttributeData => $Value,
                            },
                        );
                    }
                }
            }
        }
    }

    # If there are no config items to show, a no data found message is displayed in the table.
    else {
        $LayoutObject->Block(
            Name => 'NoDataFoundMsg',
            Data => {
                TotalColumns => scalar @ShowColumns,
            },
        );
    }

    my $NameHTML = $Self->{Name};
    $NameHTML =~ s{-}{_}xmsg;

    # Send data to JS.
    $LayoutObject->AddJSData(
        Key   => 'ITSMConfigItemGeneric',
        Value => {
            Name     => $Self->{Name},
            NameHTML => $NameHTML,
        },
    );

    my $Content = $LayoutObject->Output(
        TemplateFile => 'AgentDashboardITSMConfigItemGeneric',
        Data         => {
            %{ $Self->{Config} },
            Name         => $Self->{Name},
            StyleClasses => $StyleClasses,
        },
        AJAX => $Param{AJAX},
    );

    return $Content;
}

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

    # Define cache key.
    my $CacheKey;
    if ( $Self->{CustomerID} ) {
        $CacheKey = 'AssignedCIs-' . $Param{Filter} . '-' . $Self->{CustomerID};
    }
    elsif ( $Self->{CustomerUserID} ) {
        $CacheKey = 'AssignedCIs-' . $Param{Filter} . '-' . $Self->{CustomerUserID};
    }
    else {
        $CacheKey = 'AssignedCIs-' . $Param{Filter} . '-' . $Self->{UserID};
    }

    my $Content = $Kernel::OM->Get('Kernel::System::Cache')->Get(
        Type => 'DashboardITSMConfigItemGeneric',
        Key  => $CacheKey,
    );
    return $Content if defined $Content && $Param{Count};

    # Get the defined param.
    my @Params = split /;/, $Self->{Config}->{Attributes};

    # Define result.
    my @ConfigItemIDs;
    my %Result;

    # Get key for CI search.
    my $ConfigItemKey = $Self->{Config}->{ConfigItemKey} // '';

    return if !$ConfigItemKey;

    # Get config item object.
    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    STRING:
    for my $String (@Params) {
        next STRING if !$String;
        my ( $Key, $Value ) = split /=/, $String;

        if ( $Key eq 'CustomerCompany' ) {

            CLASS:
            for my $Class ( sort keys %{$ConfigItemKey} ) {

                # Skip other classes if filter is set.
                next CLASS if $Param{Filter} && ( $Param{Filter} ne $Class && $Param{Filter} ne 'All' );

                my $ClassID = $Self->_ClassIDByNameGet(
                    Value => $Class,
                );

                # Skip if we have no class id.
                next CLASS if !$ClassID;

                my @SearchKey = (
                    {
                        "[1]{'Version'}[1]{'$ConfigItemKey->{$Class}'}[%]{'Content'}" => $Self->{CustomerID},
                    }
                );

                # Perform config item search (extended).
                my $ConfigItemIDs = $ConfigItemObject->ConfigItemSearchExtended(
                    ClassIDs => [$ClassID],
                    What     => \@SearchKey,
                );

                next CLASS if !IsArrayRefWithData($ConfigItemIDs);

                push @ConfigItemIDs, @{$ConfigItemIDs};

                # Count config items per class.
                if ( $Param{Count} ) {
                    $Result{$Class} = scalar @{$ConfigItemIDs};
                }
            }
        }
        elsif ( $Key eq 'Customer' ) {

            CLASS:
            for my $Class ( sort keys %{$ConfigItemKey} ) {

                # Skip other classes if filter is set.
                next CLASS if $Param{Filter} && ( $Param{Filter} ne $Class && $Param{Filter} ne 'All' );

                my $ClassID = $Self->_ClassIDByNameGet(
                    Value => $Class,
                );

                # Skip if we have no class id.
                next CLASS if !$ClassID;

                my @SearchKey = (
                    {
                        "[1]{'Version'}[1]{'$ConfigItemKey->{$Class}'}[%]{'Content'}" => $Self->{CustomerUserID},
                    }
                );

                # Perform config item search (extended).
                my $ConfigItemIDs = $ConfigItemObject->ConfigItemSearchExtended(
                    ClassIDs => [$ClassID],
                    What     => \@SearchKey,
                );

                next CLASS if !IsArrayRefWithData($ConfigItemIDs);

                push @ConfigItemIDs, @{$ConfigItemIDs};

                # Count config items per class.
                if ( $Param{Count} ) {
                    $Result{$Class} = scalar @{$ConfigItemIDs};
                }
            }
        }
        else {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => 'Need CustomerCompany or Customer as attribute.',
            );
            return;
        }
    }

    # Remove duplicated config item ids.
    my %TempConfigItemIDs = map { $_ => 1 } @ConfigItemIDs;
    @ConfigItemIDs = sort keys %TempConfigItemIDs;

    $Result{ConfigItemIDs} = \@ConfigItemIDs;

    # Add count for all items.
    if ( $Param{Count} ) {
        $Result{All} = scalar @ConfigItemIDs;
    }

    if ( $Self->{Config}->{CacheTTLLocal} && $Param{Count} ) {
        $Kernel::OM->Get('Kernel::System::Cache')->Set(
            Type  => 'DashboardITSMConfigItemGeneric',
            Key   => $CacheKey,
            Value => \%Result,
            TTL   => 2 * 60,
        );
    }

    return \%Result;
}

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

    # Check needed stuff.
    if ( !$Param{Value} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need class name!",
        );
        return;
    }

    # Get general catalog object.
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

    # Get class list.
    my $ClassList = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # Reverse class list.
    my %ClassName2ID = reverse %{$ClassList};

    # Get class id.
    my $ClassID = $ClassName2ID{ $Param{Value} };

    return if !$ClassID;

    #Initiate permission check.
    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    my $HasAccess = $ConfigItemObject->Permission(
        Scope   => 'Class',
        ClassID => $ClassID,
        UserID  => $Self->{UserID},
        Type    => $Self->{Config}->{Permission},
    );
    return if !$HasAccess;

    # Return class id.
    return $ClassID;
}

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkZpbHRlckVsZW1lbnRQb3N0OjpBZ2VudElUU01Db25maWdJdGVtQ3VzdG9tZXJDSXM7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OkNvbmZpZycsCiAgICAnS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkxheW91dCcsCiAgICAnS2VybmVsOjpTeXN0ZW06Okdyb3VwJywKICAgICdLZXJuZWw6OlN5c3RlbTo6V2ViOjpSZXF1ZXN0JywKICAgICdLZXJuZWw6OlN5c3RlbTo6QWdlbnRJVFNNQ29uZmlnSXRlbUN1c3RvbWVyQ0lzJywKKTsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6VmFyaWFibGVDaGVjayBxdyg6YWxsKTsKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICBteSAkU2VsZiA9IHt9OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgIHJldHVybiAkU2VsZjsKfQoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRDb25maWdPYmplY3QgICAgICAgICAgICAgICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpDb25maWcnKTsKICAgIG15ICRHcm91cE9iamVjdCAgICAgICAgICAgICAgICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06Okdyb3VwJyk7CiAgICBteSAkTGF5b3V0T2JqZWN0ICAgICAgICAgICAgICAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnKTsKICAgIG15ICRQYXJhbU9iamVjdCAgICAgICAgICAgICAgICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OldlYjo6UmVxdWVzdCcpOwogICAgbXkgJElUU01Db25maWdJdGVtQ3VzdG9tZXJDSXNPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW1DdXN0b21lckNJcycpOwoKICAgIG15ICRDb25maWcgPSAkQ29uZmlnT2JqZWN0LT5HZXQoJ0FnZW50SVRTTUNvbmZpZ0l0ZW1DdXN0b21lckNJc1dpZGdldCcpOwoKICAgIG15ICRQZXJtaXNzaW9uOwoKICAgICMgc2hvdyBvbmx5IGlmIHRoZSB1c2VyIGlzIGluIG9uZSBvZiB0aGUgYWxsb3dlZCBncm91cHMKICAgIGlmICggSXNBcnJheVJlZldpdGhEYXRhKCAkQ29uZmlnLT57R3JvdXB9ICkgKSB7CiAgICAgICAgR1JPVVA6CiAgICAgICAgZm9yIG15ICRDdXJyZW50R3JvdXAgKCBAeyAkQ29uZmlnLT57R3JvdXB9IH0gKSB7CiAgICAgICAgICAgIG15ICRIYXNQZXJtaXNzaW9uID0gJEdyb3VwT2JqZWN0LT5QZXJtaXNzaW9uQ2hlY2soCiAgICAgICAgICAgICAgICBVc2VySUQgICAgPT4gJExheW91dE9iamVjdC0+e1VzZXJJRH0sCiAgICAgICAgICAgICAgICBHcm91cE5hbWUgPT4gJEN1cnJlbnRHcm91cCwKICAgICAgICAgICAgICAgIFR5cGUgICAgICA9PiAnbW92ZV9pbnRvJywKICAgICAgICAgICAgKTsKICAgICAgICAgICAgbmV4dCBHUk9VUCBpZiAhJEhhc1Blcm1pc3Npb247CgogICAgICAgICAgICAkUGVybWlzc2lvbiA9IDE7CgogICAgICAgICAgICBsYXN0IEdST1VQOwogICAgICAgIH0KICAgIH0KICAgIHJldHVybiBpZiAhJFBlcm1pc3Npb247CgogICAgaWYgKCAkTGF5b3V0T2JqZWN0LT57QWN0aW9ufSBlcSAnQWdlbnRUaWNrZXRab29tJyApIHsKICAgICAgICAkUGFyYW17VGlja2V0SUR9ID0gJFBhcmFtT2JqZWN0LT5HZXRQYXJhbSggUGFyYW0gPT4gJ1RpY2tldElEJyApIHx8ICcnOwogICAgfQoKICAgIG15ICVEYXRhOwoKICAgICRMYXlvdXRPYmplY3QtPkJsb2NrKAogICAgICAgIE5hbWUgPT4gJ1dpZGdldCcsCiAgICApOwoKICAgICREYXRhe1dpZGdldH0gPSAkTGF5b3V0T2JqZWN0LT5PdXRwdXQoCiAgICAgICAgVGVtcGxhdGVGaWxlID0+ICdBZ2VudElUU01Db25maWdJdGVtQ3VzdG9tZXJDSXNXaWRnZXQnLAogICAgKTsKCiAgICBteSBAQ29uZmlnSXRlbXMgPSAkSVRTTUNvbmZpZ0l0ZW1DdXN0b21lckNJc09iamVjdC0+R2V0UG9zc2libGVDdXN0b21lckNJcygKICAgICAgICBUaWNrZXRJRCA9PiAkUGFyYW17VGlja2V0SUR9LAogICAgKTsKCiAgICBpZiAoQENvbmZpZ0l0ZW1zKSB7CiAgICAgICAgZm9yIG15ICRDb25maWdJdGVtIChAQ29uZmlnSXRlbXMpIHsKICAgICAgICAgICAgJExheW91dE9iamVjdC0+QmxvY2soCiAgICAgICAgICAgICAgICBOYW1lID0+ICdDb25maWdJdGVtcycsCiAgICAgICAgICAgICAgICBEYXRhID0+IHsKICAgICAgICAgICAgICAgICAgICAleyRDb25maWdJdGVtfSwKICAgICAgICAgICAgICAgIH0sCiAgICAgICAgICAgICk7CiAgICAgICAgfQogICAgICAgICREYXRhe0NvbmZpZ0l0ZW1zfSA9ICRMYXlvdXRPYmplY3QtPk91dHB1dCgKICAgICAgICAgICAgVGVtcGxhdGVGaWxlID0+ICdBZ2VudElUU01Db25maWdJdGVtQ3VzdG9tZXJDSXNXaWRnZXQnLAogICAgICAgICAgICBEYXRhICAgICAgICAgPT4gXCVEYXRhLAogICAgICAgICk7CiAgICB9CgogICAgbXkgJEluaXRQYXJhbWV0ZXIgPSAkTGF5b3V0T2JqZWN0LT5KU09ORW5jb2RlKAogICAgICAgIE5vUXVvdGVzID0+IDEsCiAgICAgICAgRGF0YSAgICAgPT4gXCVEYXRhLAogICAgKTsKCiAgICBteSAkSlNCbG9jayA9IDw8IkpTX0JMT0NLIjsKICAgIElUU00uQWdlbnQuQ29uZmlnSXRlbS5DdXN0b21lckNJc1dpZGdldC5Jbml0KCRJbml0UGFyYW1ldGVyKTsKSlNfQkxPQ0sKCiAgICAkTGF5b3V0T2JqZWN0LT5BZGRKU09uRG9jdW1lbnRDb21wbGV0ZUlmTm90RXhpc3RzKAogICAgICAgIEtleSAgPT4gJ0lUU00uQWdlbnQuQ29uZmlnSXRlbS5DdXN0b21lckNJc1dpZGdldCcsCiAgICAgICAgQ29kZSA9PiAkSlNCbG9jaywKICAgICk7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dEJhc2VTZWxlY3RhYmxlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIEtlcm5lbDo6U3lzdGVtOjpWYXJpYWJsZUNoZWNrIHF3KDphbGwpOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkxheW91dCcsCiAgICAnS2VybmVsOjpTeXN0ZW06OkxvZycsCiAgICAnS2VybmVsOjpTeXN0ZW06OldlYjo6UmVxdWVzdCcsCik7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpPdXRwdXQ6OkhUTUw6OklUU01Db25maWdJdGVtOjpMYXlvdXRCYXNlU2VsZWN0YWJsZSAtIGxheW91dCBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIFNZTk9QU0lTCgpCYXNlIGxheW91dCBtb2R1bGUgZm9yIHNlbGVjdC1hYmxlIHZhbHVlcwoKPWhlYWQyIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgbXkgJEJhY2tlbmRPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dEJhc2VTZWxlY3RhYmxlJyk7Cgo9Y3V0CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgI2FsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aGVhZDIgT3V0cHV0U3RyaW5nQ3JlYXRlKCkKCmNyZWF0ZSBvdXRwdXQgc3RyaW5nCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPk91dHB1dFN0cmluZ0NyZWF0ZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgICAgICAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIE91dHB1dFN0cmluZ0NyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRMYXlvdXRPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0Jyk7CgogICAgIyB0cmFuc2Zvcm0gYXNjaWkgdG8gaHRtbAogICAgJFBhcmFte1ZhbHVlfSA9ICRMYXlvdXRPYmplY3QtPkFzY2lpMkh0bWwoCiAgICAgICAgVGV4dCAgICAgICAgICAgPT4gJFBhcmFte1ZhbHVlfSB8fCAnJywKICAgICAgICBIVE1MUmVzdWx0TW9kZSA9PiAxLAogICAgKTsKCiAgICByZXR1cm4gJFBhcmFte1ZhbHVlfTsKfQoKPWhlYWQyIEZvcm1EYXRhR2V0KCkKCmdldCBmb3JtIGRhdGEgYXMgaGFzaAoKICAgIG15ICRGb3JtRGF0YVJlZiA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtRGF0YUdldCgKICAgICAgICBLZXkgID0+ICdJdGVtOjoxOjpOb2RlOjoyLAogICAgICAgIEl0ZW0gPT4gJEl0ZW1SZWYsCiAgICApOwoKPWN1dAoKc3ViIEZvcm1EYXRhR2V0IHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJExvZ09iamVjdCAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpOwogICAgbXkgJFBhcmFtT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OldlYjo6UmVxdWVzdCcpOwoKICAgICNjaGVjayBuZWVkZWQgc3R1ZmYKICAgIEFSR1VNRU5UOgogICAgZm9yIG15ICRBcmd1bWVudCAocXcoS2V5IEl0ZW0pKSB7CiAgICAgICAgbmV4dCBBUkdVTUVOVCBpZiAkUGFyYW17JEFyZ3VtZW50fTsKCiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkQXJndW1lbnQhIgogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIG15ICVGb3JtRGF0YTsKICAgICRGb3JtRGF0YXtWYWx1ZX0gPSAkUGFyYW1PYmplY3QtPkdldFBhcmFtKCBQYXJhbSA9PiAkUGFyYW17S2V5fSApOwoKICAgIHJldHVybiBcJUZvcm1EYXRhIGlmICEkUGFyYW17SXRlbX0tPntJbnB1dH0tPntSZXF1aXJlZH07CiAgICByZXR1cm4gXCVGb3JtRGF0YSBpZiAkRm9ybURhdGF7VmFsdWV9OwoKICAgICRGb3JtRGF0YXtJbnZhbGlkfSA9IDE7CiAgICAkUGFyYW17SXRlbX0tPntGb3JtfS0+eyAkUGFyYW17S2V5fSB9LT57SW52YWxpZH0gPSAxOwoKICAgIHJldHVybiBcJUZvcm1EYXRhOwoKfQoKPWhlYWQyIFNlYXJjaEZvcm1EYXRhR2V0KCkKCmdldCBzZWFyY2ggZm9ybSBkYXRhCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPlNlYXJjaEZvcm1EYXRhR2V0KAogICAgICAgIEtleSA9PiAnSXRlbTo6MTo6Tm9kZTo6MicsCiAgICAgICAgSXRlbSA9PiAkSXRlbVJlZiwKICAgICk7Cgo9Y3V0CgpzdWIgU2VhcmNoRm9ybURhdGFHZXQgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICBteSAkTG9nT2JqZWN0ICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyk7CiAgICBteSAkUGFyYW1PYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6V2ViOjpSZXF1ZXN0Jyk7CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIEFSR1VNRU5UOgogICAgZm9yIG15ICRBcmd1bWVudCAocXcoS2V5IEl0ZW0pKSB7CiAgICAgICAgbmV4dCBBUkdVTUVOVCBpZiAkUGFyYW17JEFyZ3VtZW50fTsKCiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkQXJndW1lbnQhIiwKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAjIGdldCBmb3JtIGRhdGEKICAgIG15ICRWYWx1ZTsKICAgIGlmICggJFBhcmFte1ZhbHVlfSApIHsKICAgICAgICAkVmFsdWUgPSAkUGFyYW17VmFsdWV9OwogICAgfQogICAgZWxzZSB7CiAgICAgICAgJFZhbHVlID0gJFBhcmFtT2JqZWN0LT5HZXRQYXJhbSggUGFyYW0gPT4gJFBhcmFte0tleX0gKTsKICAgIH0KCiAgICByZXR1cm4gJFZhbHVlOwp9Cgo9aGVhZDIgU2VhcmNoSW5wdXRDcmVhdGUoKQoKY3JlYXRlIGEgc2VhcmNoIGlucHV0IHN0cmluZwoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5TZWFyY2hJbnB1dENyZWF0ZSgKICAgICAgICBLZXkgPT4gJ0l0ZW06OjE6Ok5vZGU6OjInLAogICAgKTsKCj1jdXQKCnN1YiBTZWFyY2hJbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRMb2dPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyk7CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIEFSR1VNRU5UOgogICAgZm9yIG15ICRBcmd1bWVudCAocXcoS2V5IEl0ZW0pKSB7CiAgICAgICAgbmV4dCBBUkdVTUVOVCBpZiAkUGFyYW17JEFyZ3VtZW50fTsKCiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkQXJndW1lbnQhIiwKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICBteSAlRm9ybURhdGE7CiAgICBpZiAoICRQYXJhbXtWYWx1ZX0gKSB7CiAgICAgICAgJEZvcm1EYXRhe1ZhbHVlfSA9ICRQYXJhbXtWYWx1ZX07CiAgICB9CgogICAgIyBjcmVhdGUgaW5wdXQgZmllbGQKICAgIG15ICRJbnB1dFN0cmluZyA9ICRTZWxmLT5JbnB1dENyZWF0ZSgKICAgICAgICAlRm9ybURhdGEsCiAgICAgICAgS2V5ICA9PiAkUGFyYW17S2V5fSwKICAgICAgICBJdGVtID0+ICRQYXJhbXtJdGVtfSwKICAgICk7CgogICAgcmV0dXJuICRJbnB1dFN0cmluZzsKfQoKMTsK
# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --
## nofilter(TidyAll::Plugin::Znuny::CodeStyle::GuardClause)

package Kernel::Output::HTML::ITSMConfigItem::LayoutCI;

use strict;
use warnings;

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

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::Language',
    'Kernel::Output::HTML::Layout',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::Log',
    'Kernel::System::Web::Request',
);

=head1 NAME

Kernel::Output::HTML::ITSMConfigItem::LayoutCI - layout backend module

=head1 SYNOPSIS

All layout functions of CIClass objects

=head2 new()

create an object

    my $BackendObject = $Kernel::OM->Get('Kernel::Output::HTML::ITSMConfigItemLayoutCI');

=cut

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

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

    $Self->{CIID} = 0;

    return $Self;
}

=head2 OutputStringCreate()

create output string

    my $Value = $BackendObject->OutputStringCreate(
        Value => 11,       # (optional)
    );

=cut

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

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

    #transform ascii to html
    $Param{Value} = $LayoutObject->Ascii2Html(
        Text           => $Param{Value} || '',
        HTMLResultMode => 1,
    );

    return $Param{Value};
}

=head2 FormDataGet()

get form data as hash

    my $FormDataRef = $BackendObject->FormDataGet(
        Key  => 'Item::1::Node::2,
        Item => $ItemRef,
    );

=cut

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

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

    # check needed stuff
    NEEDED:
    for my $Needed (qw(Key Item)) {

        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my %FormData;

    #get selected CIClass
    $FormData{Value} = $ParamObject->GetParam( Param => $Param{Key} );

    #check search button..
    if ( $ParamObject->GetParam( Param => $Param{Key} . '::ButtonSearch' ) ) {
        $Param{Item}->{Form}->{ $Param{Key} }->{Search} = $ParamObject->GetParam( Param => $Param{Key} . '::Search' );
    }

    #check select button
    elsif ( $ParamObject->GetParam( Param => $Param{Key} . '::ButtonSelect' ) ) {
        $FormData{Value} = $ParamObject->GetParam( Param => $Param{Key} . '::Select' );
    }

    #check clear button
    elsif ( $ParamObject->GetParam( Param => $Param{Key} . '::ButtonClear' ) ) {
        $FormData{Value} = '';
    }
    else {

        #reset value if search field is empty
        if (
            !$ParamObject->GetParam( Param => $Param{Key} . '::Search' )
            && defined $FormData{Value}
            )
        {
            $FormData{Value} = '';
        }

        #check required option
        if ( $Param{Item}->{Input}->{Required} && !$FormData{Value} ) {
            $Param{Item}->{Form}->{ $Param{Key} }->{Invalid} = 1;
            $FormData{Invalid} = 1;
        }
    }

    return \%FormData;
}

=head2 InputCreate()

create a input string

    my $Value = $BackendObject->InputCreate(
        Key   => 'Item::1::Node::3',
        Value => 11,       # (optional)
        Item  => $ItemRef,
    );

=cut

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

    my $LogObject            = $Kernel::OM->Get('Kernel::System::Log');
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
    my $ITSMConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $LayoutObject         = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $LanguageObject       = $Kernel::OM->Get('Kernel::Language');
    my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');

    # check needed stuff
    NEEDED:
    for my $Needed (qw(Key Item)) {

        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $Value = '';
    if ( defined $Param{Value} ) {
        $Value = $Param{Value};
    }
    elsif ( $Param{Item}->{Input}->{ValueDefault} ) {
        $Value = $Param{Item}->{Input}->{ValueDefault};
    }

    if (
        !$Param{Item}->{Input}->{CIClassID}
        && $Param{Item}->{Input}->{CIClassName}
        )
    {

        my $ItemDataRef = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::Class',
            Name  => $Param{Item}->{Input}->{CIClassName},
        );

        if ( !$ItemDataRef || !( $ItemDataRef->{ItemID} ) ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => "ITSMConfigItemLayoutCI: CI-Class <"
                    . $Param{Item}->{Input}->{CIClassName}
                    . "> not defined in GeneralCatalog!",
            );
        }
        else {
            $Self->{CIID} = $ItemDataRef->{ItemID};
        }

    }
    elsif ( $Param{Item}->{Input}->{CIClassID} ) {
        $Self->{CIID} = $Param{Item}->{Input}->{CIClassID};
    }

    my $Size         = $Param{Item}->{Input}->{Size} || 50;
    my $Search       = '';
    my $StringOption = '';
    my $StringSelect = '';

    # AutoComplete CIClass
    my $Class = 'W50pc CIClassSearch';

    my $Required = $Param{Required} || '';
    my $Invalid  = $Param{Invalid}  || '';

    # CIClass search
    if ( $Param{Item}->{Form}->{ $Param{Key} }->{Search} ) {

        # search for name
        my %CISearchList    = ();
        my $CISearchListRef = $ITSMConfigItemObject->ConfigItemSearchExtended(
            Name     => '*' . $Param{Item}->{Form}->{ $Param{Key} }->{Search} . '*',
            ClassIDs => [ $Self->{CIID} ],
        );

        for my $SearchResult ( @{$CISearchListRef} ) {
            my $CurrVersionData = $ITSMConfigItemObject->VersionGet(
                ConfigItemID => $SearchResult,
                XMLDataGet   => 0,
            );

            if ( IsHashRefWithData($CurrVersionData) && $CurrVersionData->{Name} && $CurrVersionData->{Number} ) {
                $CISearchList{$SearchResult} = $CurrVersionData->{Name} . " (" . $CurrVersionData->{Number} . ")";
            }
        }

        # search for number
        $CISearchListRef = $ITSMConfigItemObject->ConfigItemSearchExtended(
            Number   => '*' . $Param{Item}->{Form}->{ $Param{Key} }->{Search} . '*',
            ClassIDs => [ $Self->{CIID} ],
        );

        for my $SearchResult ( @{$CISearchListRef} ) {
            my $CurrVersionData = $ITSMConfigItemObject->VersionGet(
                ConfigItemID => $SearchResult,
                XMLDataGet   => 0,
            );
            if (
                $CurrVersionData
                &&
                ( ref($CurrVersionData) eq 'HASH' ) &&
                $CurrVersionData->{Name} &&
                $CurrVersionData->{Number}
                )
            {
                $CISearchList{$SearchResult} = $CurrVersionData->{Name}
                    . " ("
                    . $CurrVersionData->{Number}
                    . ")";
            }
        }

        # build search result presentation
        if ( %CISearchList && scalar( keys %CISearchList ) > 1 ) {

            #create option list
            $StringOption = $LayoutObject->BuildSelection(
                Name  => $Param{Key} . '::Select',
                Data  => \%CISearchList,
                Class => 'Modernize W50pc',
            );
            $StringOption .= '<br>';

            # create select button
            $StringSelect = '<input class="button" type="submit" name="' . $Param{Key} . '::ButtonSelect" '
                . 'value="' . $LanguageObject->Translate("Select") . '">&nbsp;';

            # set search
            $Search = $Param{Item}->{Form}->{ $Param{Key} }->{Search};
        }
        elsif (%CISearchList) {

            $Value = ( keys %CISearchList )[0];
            my $CIVersionDataRef = $ITSMConfigItemObject->VersionGet(
                ConfigItemID => $Value,
                XMLDataGet   => 0,
            );
            my $CIName = "";

            if ( IsHashRefWithData($CIVersionDataRef) && $CIVersionDataRef->{Name} && $CIVersionDataRef->{Number} ) {
                $CIName = $CIVersionDataRef->{Name} . " (" . $CIVersionDataRef->{Number} . ")";
            }

            #transform ascii to html
            $Search = $LayoutObject->Ascii2Html(
                Text           => $CIName || '',
                HTMLResultMode => 1,
            );
        }
    }

    # create CIClass string
    elsif ($Value) {

        my $CIVersionDataRef = $ITSMConfigItemObject->VersionGet(
            ConfigItemID => $Value,
            XMLDataGet   => 0,
        );
        my $CIName = "";

        if ( IsHashRefWithData($CIVersionDataRef) && $CIVersionDataRef->{Name} && $CIVersionDataRef->{Number} ) {
            $CIName = $CIVersionDataRef->{Name} . " (" . $CIVersionDataRef->{Number} . ")";
        }

        #transform ascii to html
        $Search = $LayoutObject->Ascii2Html(
            Text           => $CIName || '',
            HTMLResultMode => 1,
        );
    }

    my $String = '<input type="hidden" name="' . $Param{Key} . '" value="' . $Value . '">'
        . '<input type="Text" name="'
        . $Param{Key}
        . '::Search" class="'
        . $Class
        . '" size="'
        . $Size
        . '" value="'
        . $Search . '">'
        . '<br>'
        . $StringOption
        . $StringSelect
        . '<input class="button" type="submit" name="'
        . $Param{Key}
        . '::ButtonSearch" value="'
        . $LanguageObject->Translate("Search") . '">';

    if ($Search) {
        $String .= '&nbsp;' . '<input class="button" type="submit" name="' . $Param{Key} . '::ButtonClear" value="'
            . $LanguageObject->Translate("Clear") . '">';
    }

    return $String;
}

=head2 SearchFormDataGet()

get search form data

    my $Value = $BackendObject->SearchFormDataGet(
        Key  => 'Item::1::Node::2',
        Item => $ItemRef,
    );

=cut

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

    my $LogObject            = $Kernel::OM->Get('Kernel::System::Log');
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
    my $ParamObject          = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $ITSMConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    # check needed stuff
    NEEDED:
    for my $Needed (qw(Key Item)) {

        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    if ( $Param{Item}->{Input}->{SearchInputType} && $Param{Item}->{Input}->{SearchInputType} eq 'Pattern' ) {
        my @Values = qw{};
        if ( !$Self->{CIID} ) {

            my $ItemDataRef = $GeneralCatalogObject->ItemGet(
                Class => 'ITSM::ConfigItem::Class',
                Name  => $Param{Item}->{Input}->{CIClassName},
            );

            if ( !$ItemDataRef || !( $ItemDataRef->{ItemID} ) ) {
                $LogObject->Log(
                    Priority => 'error',
                    Message  => "ITSMConfigItemLayoutCI: CI-Class <"
                        . $Param{Item}->{Input}->{CIClassName}
                        . "> not defined in GeneralCatalog!",
                );
            }
            else {
                $Self->{CIID} = $ItemDataRef->{ItemID};
            }
        }

        my @SearchValues = $ParamObject->GetParam( Param => $Param{Key} );

        SEARCHVALUE:
        for my $CurrSearchValue (@SearchValues) {
            next SEARCHVALUE if ( !$CurrSearchValue );

            my @SearchValueParts = split( '\|\|', $CurrSearchValue );

            SEARCHVALUEPART:
            for my $CurrSearchValuePart (@SearchValueParts) {
                next SEARCHVALUEPART if ( !$CurrSearchValuePart );

                # check pattern for id
                if ( $CurrSearchValuePart =~ m/^\[ID\]([0-9]+)$/i ) {
                    push( @Values, $1 );

                    next SEARCHVALUEPART;
                }

                if ( $CurrSearchValuePart =~ m/^\[Number\]([0-9*]+)$/i ) {
                    my $CISearchListRef = $ITSMConfigItemObject->ConfigItemSearchExtended(
                        Number   => $1,
                        ClassIDs => [ $Self->{CIID} ],
                    );
                    for my $SearchResult ( @{$CISearchListRef} ) {
                        push( @Values, $SearchResult );
                    }

                    next SEARCHVALUEPART;
                }

                my $CISearchListRef = $ITSMConfigItemObject->ConfigItemSearchExtended(
                    Name     => $CurrSearchValuePart,
                    ClassIDs => [ $Self->{CIID} ],
                );
                for my $SearchResult ( @{$CISearchListRef} ) {
                    push( @Values, $SearchResult );
                }
            }
        }

        if ( scalar(@Values) < 1 || !$Values[0] ) {
            @Values = @SearchValues;
        }

        return \@Values;
    }
    else {

        # get form data
        my $Value;
        if ( $Param{Value} ) {
            $Value = $Param{Value};
        }
        else {
            $Value = $ParamObject->GetParam( Param => $Param{Key} );
        }

        return $Value;
    }
}

=head2 SearchInputCreate()

create a search input string

    my $Value = $BackendObject->SearchInputCreate(
        Key => 'Item::1::Node::2',
    );

=cut

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

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    # check needed stuff
    NEEDED:
    for my $Needed (qw(Key Item)) {

        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $InputString = '';

    if ( $Param{Item}->{Input}->{SearchInputType} && $Param{Item}->{Input}->{SearchInputType} eq 'Pattern' ) {
        my $Value = $Param{Value};
        if ( ref( $Param{Value} ) eq 'ARRAY' ) {
            for my $ItemValue ( @{ $Param{Value} } ) {
                if ($Value) {
                    $Value .= "||";
                }
                if ( $ItemValue =~ m/^[0-9]+$/ ) {
                    $Value .= "[ID]" . $ItemValue;
                }
                else {
                    $Value .= $ItemValue;
                }
            }
        }
        $InputString = "<input type=\"Text\" name=\"$Param{Key}\" class=\"W50pc\" value=\"$Value\">";
    }
    else {
        # hash with values for the input field
        my %FormData;

        if ( $Param{Value} ) {
            $FormData{Value} = $Param{Value};
        }

        # create input field
        $InputString = $Self->InputCreate(
            %FormData,
            Key  => $Param{Key},
            Item => $Param{Item},
        );
    }

    return $InputString;
}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Output::HTML::ITSMConfigItem::LayoutCIAttachment;

use strict;
use warnings;

use Digest::MD5 qw(md5 md5_hex md5_base64);

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

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::Output::HTML::Layout',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::ITSMConfigItem::CIAttachment',
    'Kernel::System::Log',
    'Kernel::System::Time',
    'Kernel::System::Web::Request',
    'Kernel::System::Web::UploadCache',
);

use parent 'Kernel::Output::HTML::ITSMConfigItem::LayoutBaseSelectable';

=head1 NAME

Kernel::Output::HTML::ITSMConfigItem::LayoutCIAttachment - layout backend module

=head1 SYNOPSIS

All layout functions of CIAttachment objects

=head2 OutputStringCreate()

create output string

    my $Value = $BackendObject->OutputStringCreate(
        Value => 11,       # (optional)
        Item => $ItemRef,
    );

=cut

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

    my $LogObject          = $Kernel::OM->Get('Kernel::System::Log');
    my $LayoutObject       = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $ParamObject        = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $CIAttachmentObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem::CIAttachment');

    my $Value = $Param{Value};
    return if !$Value;

    my %Attachment = $CIAttachmentObject->CIAttachmentGet(
        AttachmentID => $Param{Value},
    );

    my $String = '<a href="' . $LayoutObject->{Baselink} . 'Action=AgentZnuny4OTRSCIAttachment;AttachmentID=' . $Value;

    # add session id if needed
    if ( !$LayoutObject->{SessionIDCookie} ) {
        $String .= ';' . $LayoutObject->{SessionName} . "=" . $LayoutObject->{SessionID};
    }

    $String .= '">' . $Attachment{Filename} . '</a> (' . $Attachment{Filesize} . ')';

    return $String;
}

=head2 FormDataGet()

get form data as hash reference

    my $FormDataRef = $BackendObject->FormDataGet(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

    my $LogObject          = $Kernel::OM->Get('Kernel::System::Log');
    my $ParamObject        = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $CIAttachmentObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem::CIAttachment');
    my $UploadCacheObject  = $Kernel::OM->Get('Kernel::System::Web::UploadCache');
    my $ConfigItemObject   = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $LayoutObject       = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    # check needed stuff
    NEEDED:
    for my $Needed (qw(Key Item)) {

        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed in FormDataGet!",
        );
        return;
    }

    my $DisplayValue     = $ParamObject->GetParam( Param => $Param{Key} );
    my $AttachmentDelete = $ParamObject->GetParam( Param => $Param{Key} . "::Delete" );
    my $AttachmentIndex  = $Self->AttachmentIndexGet(%Param);
    my $AttachmentKey    = $Self->AttachmentKeyGet(%Param);

    for my $GetParam (qw(ConfigItemID ClassID SubmitSave)) {
        $Param{$GetParam} = $ParamObject->GetParam( Param => $GetParam );
    }

    my %FormData;
    $FormData{Value} = undef;

    my $FormID = $Self->FormIDGet(
        AttachmentKey => $AttachmentKey,
    );

    # get the upload file
    my %UploadFile = $ParamObject->GetUploadAll(
        Param  => $Param{Key} . "::Upload",
        Source => 'string',
    );

    if (%UploadFile) {

        # generate new id for the attachment which will be saved
        # in the xml of the config item version
        $DisplayValue = $Self->AttachmentIDGenerate();

        # add attachment to temp cache
        $UploadCacheObject->FormIDAddFile(
            %UploadFile,
            FormID => $FormID,
        );
    }
    elsif ($AttachmentDelete) {

        # remove attachment from temp cache if the user removes it from the form data
        $Self->AttachmentCacheDelete(
            AttachmentIndex => $AttachmentIndex,
            FormID          => $FormID,
        );

        $FormData{Value} = '';
    }

    if ($DisplayValue) {
        $FormData{Value} = $DisplayValue;
    }

    # set invalid param...
    return \%FormData if !$Param{Item}->{Input}->{Required};
    return \%FormData if $FormData{Value};

    $FormData{Invalid} = 1;
    $Param{Item}->{Form}->{ $Param{Key} }->{Invalid} = 1;

    return \%FormData;
}

=head2 InputCreate()

create a input string

    my $Value = $BackendObject->InputCreate(
        Key => 'Item::1::Node::3',
        Value => 11,       # (optional)
        Item => $ItemRef,
    );

=cut

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

    my $LayoutObject       = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $LogObject          = $Kernel::OM->Get('Kernel::System::Log');
    my $CIAttachmentObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem::CIAttachment');
    my $UploadCacheObject  = $Kernel::OM->Get('Kernel::System::Web::UploadCache');

    # check needed stuff
    ARGUMENT:
    for my $Argument (qw(Key Item)) {
        next ARGUMENT if $Param{$Argument};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Need $Argument!"
        );
        return;
    }

    my $Value     = $Param{Value} || '';
    my $Key       = $Param{Key};
    my $KeyUpload = $Param{Key} . "::Upload";

    my $Filename;
    if ($Value) {

        # find the attachment index for the field key in the temp cache
        my $AttachmentIndex = $Self->AttachmentIndexGet(%Param);

        # get the attachment field base key
        my $AttachmentKey = $Self->AttachmentKeyGet(%Param);

        # get the form id for the field where the attachments are stored
        my $FormID = $Self->FormIDGet(
            AttachmentKey => $AttachmentKey,
        );

        if ( defined $AttachmentIndex && defined $AttachmentKey && defined $FormID ) {

            # find the cached temp attachment for the field
            my %Attachment = $Self->AttachmentCacheGet(
                Index  => $AttachmentIndex,
                FormID => $FormID,
            );

            if (%Attachment) {
                $Filename = $Attachment{Filename};
            }
            else {

                # get attachment from virtual fs
                my %CIAttachmentGet = $CIAttachmentObject->CIAttachmentGet(
                    AttachmentID => $Value,
                );

                if (%CIAttachmentGet) {
                    $Value = $Self->AttachmentIDGenerate();

                    # add attachment to temp cache
                    $UploadCacheObject->FormIDAddFile(
                        Filename    => $CIAttachmentGet{Filename},
                        ContentType => $CIAttachmentGet{ContentType},
                        Content     => $CIAttachmentGet{Content},
                        FormID      => $FormID,
                    );

                    return $Self->InputCreate(
                        %Param,
                        Value => $Value,
                    );
                }
            }
        }
    }

    my $Output = '<input type="hidden" name="'
        . $Key
        . '::CIAttachment" value="1"/><input type="hidden" name="'
        . $Key
        . '" value="'
        . $Value . '"/> ';
    if ($Filename) {
        $Output .= $Filename;
    }
    else {
        $Output .= '<input type="file" id="' . $KeyUpload . '" name="' . $KeyUpload . '" class="fixed" /> ';
    }

    $LayoutObject->AddJSOnDocumentComplete( Code => <<"ZNUUNY");

// use getElementById because id has colons
\$(document.getElementById("$KeyUpload")).on('change', function() {
    var \$Form = \$(document.getElementById("$KeyUpload")).closest('form');
    Core.Form.Validate.DisableValidation(\$Form);
    \$Form.submit();
});
ZNUUNY

    return $Output;
}

=head2 SearchFormDataGet()

get search form data

    my $Value = $BackendObject->SearchFormDataGet(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

    return;
}

=head2 SearchInputCreate()

create a search input string

    my $Value = $BackendObject->SearchInputCreate(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

    return;
}

=head2 AttachmentIDGenerate()

This function returns an unique ID for the attachment.

    my $Success = $BackendObject->AttachmentIDGenerate();

Returns:

    my $Success = 1;

=cut

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

    my $TimeObject = $Kernel::OM->Get('Kernel::System::Time');

    my $SystemTime = $TimeObject->SystemTime();

    my $Rand = rand(1_000_000_000);

    return md5_hex( $SystemTime . $Rand );
}

=head2 AttachmentIndexGet()

This function returns the attachment index for the field key.

    my $AttachmentIndex = $BackendObject->AttachmentIndexGet(
        Key => 'field::1',
    );

Returns:

    my $AttachmentIndex = 0;

=cut

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

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    # check needed stuff
    NEEDED:
    for my $Needed (qw(Key)) {

        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my @FieldSplit = split /\:\:/, $Param{Key};
    return int( $FieldSplit[-1] ) - 1;
}

=head2 AttachmentKeyGet()

This function returns the attachment index for the field key.

    my $AttachmentKey = $BackendObject->AttachmentKeyGet(
        Key => 'field::1',
    );

Returns:

    my $AttachmentKey = 'field';

=cut

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

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    # check needed stuff
    NEEDED:
    for my $Needed (qw(Key)) {

        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my @FieldSplit = split /\:\:/, $Param{Key};
    pop @FieldSplit;

    return join( '::', @FieldSplit );
}

=head2 FormIDGet()

This function returns the form id for the attachment field.

    my $FormID = $BackendObject->FormIDGet(
        AttachmentKey => 123,
    );

Returns:

    my $FormID = '111.12.3.123.1.23';

=cut

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

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

    # check needed stuff
    NEEDED:
    for my $Needed (qw(AttachmentKey)) {

        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $AttachmentKey = $Param{AttachmentKey};

    my $FormID = $ParamObject->GetParam( Param => 'FormID' );
    $FormID ||= $WebUploadCacheObject->{LastFormID};

    return if !$FormID;

    $FormID .= '.CIAttachment';
    $FormID .= '.' . $AttachmentKey;    # Node

    return $FormID;
}

=head2 AttachmentCacheGet()

This function returns the temporarily attachment data of the attachment field.

    my %Attachment = $BackendObject->AttachmentCacheGet(
        Index  => 0,
        FormID => '123',
    );

Returns:

    my %Attachment = (
        Filename    => 'asdf.txt',
        ContentType => 'text/plain',
        Content     => 'blub',
    );

=cut

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

    my $LogObject         = $Kernel::OM->Get('Kernel::System::Log');
    my $UploadCacheObject = $Kernel::OM->Get('Kernel::System::Web::UploadCache');

    # check needed stuff
    NEEDED:
    for my $Needed (qw(Index FormID)) {

        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $FormID          = $Param{FormID};
    my $AttachmentIndex = $Param{Index};

    my @Data = $UploadCacheObject->FormIDGetAllFilesData(
        FormID => $FormID,
    );

    return if !@Data;

    my $Attachment = $Data[$AttachmentIndex];

    return if !IsHashRefWithData($Attachment);

    return %{$Attachment};
}

=head2 AttachmentCacheDelete()

This function deletes the temporarily attachment data of the attachment field.

    my $Success = $BackendObject->AttachmentCacheDelete(
        AttachmentIndex => 1,
        FormID          => '123',
    );

Returns:

    my $Success = 1;

=cut

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

    my $LogObject         = $Kernel::OM->Get('Kernel::System::Log');
    my $UploadCacheObject = $Kernel::OM->Get('Kernel::System::Web::UploadCache');

    # check needed stuff
    NEEDED:
    for my $Needed (qw(AttachmentIndex FormID)) {

        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $FormID          = $Param{FormID};
    my $AttachmentIndex = $Param{AttachmentIndex};

    my @Data = $UploadCacheObject->FormIDGetAllFilesData(
        FormID => $FormID,
    );

    return if !@Data;

    my $Attachment = $Data[$AttachmentIndex];
    return if !IsHashRefWithData($Attachment);

    # remove the attachment from the upload cache
    $UploadCacheObject->FormIDRemoveFile(
        FormID => $FormID,
        FileID => $Attachment->{FileID},
    );

    return 1;
}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --
package Kernel::Output::HTML::ITSMConfigItem::LayoutCISimple;

use strict;
use warnings;

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

our @ObjectDependencies = (
    'Kernel::Output::HTML::Layout',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::Log',
    'Kernel::System::Web::Request',
);

=head1 NAME

Kernel::Output::HTML::ITSMConfigItem::LayoutCISimple - layout backend module

=head1 SYNOPSIS

All layout functions of CIClass objects

=head2 new()

create an object

    my $BackendObject = $Kernel::OM->Get('Kernel::Output::HTML::ITSMConfigItemLayoutCISimple');

=cut

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

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

    $Self->{CIID} = 0;

    return $Self;
}

=head2 OutputStringCreate()

create output string

    my $Value = $BackendObject->OutputStringCreate(
        Value => 11,       # (optional)
    );

=cut

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

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

    #transform ascii to html
    $Param{Value} = $LayoutObject->Ascii2Html(
        Text           => $Param{Value} || '',
        HTMLResultMode => 1,
    );

    return $Param{Value};
}

=head2 FormDataGet()

get form data as hash

    my $FormDataRef = $BackendObject->FormDataGet(
        Key  => 'Item::1::Node::2,
        Item => $ItemRef,
    );

=cut

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

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

    # check needed stuff
    ARGUMENT:
    for my $Argument (qw(Key Item)) {
        next ARGUMENT if $Param{$Argument};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Need $Argument!"
        );
        return;
    }

    my %FormData;

    # get selected CIClass
    $FormData{Value} = $ParamObject->GetParam( Param => $Param{Key} );

    # check required option
    return \%FormData if !$Param{Item}->{Input}->{Required};
    return \%FormData if $FormData{Value};

    $Param{Item}->{Form}->{ $Param{Key} }->{Invalid} = 1;
    $FormData{Invalid} = 1;

    return \%FormData;
}

=head2 InputCreate()

create a input string

    my $Value = $BackendObject->InputCreate(
        Key   => 'Item::1::Node::3',
        Value => 11,       # (optional)
        Item  => $ItemRef,
    );

=cut

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

    my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
    my $LayoutObject         = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $LogObject            = $Kernel::OM->Get('Kernel::System::Log');

    # check needed stuff
    ARGUMENT:
    for my $Argument (qw(Key Item)) {
        next ARGUMENT if $Param{$Argument};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Need $Argument!"
        );
        return;
    }

    my $Value = '';
    if ( defined $Param{Value} ) {
        $Value = $Param{Value};
    }
    elsif ( $Param{Item}->{Input}->{ValueDefault} ) {
        $Value = $Param{Item}->{Input}->{ValueDefault};
    }

    if (
        !$Param{Item}->{Input}->{CIClassID}
        && $Param{Item}->{Input}->{CIClassName}
        )
    {

        my $ItemDataRef = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::Class',
            Name  => $Param{Item}->{Input}->{CIClassName},
        );

        if ( !$ItemDataRef || !( $ItemDataRef->{ItemID} ) ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => "ITSMConfigItemLayoutCISimple: CI-Class <"
                    . $Param{Item}->{Input}->{CIClassName}
                    . "> not defined in GeneralCatalog!",
            );
        }
        else {
            $Self->{CIID} = $ItemDataRef->{ItemID};
        }

    }
    elsif ( $Param{Item}->{Input}->{CIClassID} ) {
        $Self->{CIID} = $Param{Item}->{Input}->{CIClassID};
    }

    my $ConfigItemListRef = $ConfigItemObject->ConfigItemResultList(
        ClassID => $Self->{CIID},
        Start   => 0,
        Limit   => 1_000_000,
    );

    my @SelectionValues;
    for my $Item ( @{ $ConfigItemListRef || [] } ) {
        push @SelectionValues, {
            Key   => $Item->{ConfigItemID},
            Value => $Item->{Name}
                . " ("
                . $Item->{Number}
                . ")",
        };
    }

    my $String = $LayoutObject->BuildSelection(
        Data         => \@SelectionValues,
        Name         => $Param{Key},
        SelectedID   => $Value,
        PossibleNone => 1,
        Class        => 'Modernize',
    );

    return $String;
}

=head2 SearchFormDataGet()

get search form data

    my $Value = $BackendObject->SearchFormDataGet(
        Key  => 'Item::1::Node::2',
        Item => $ItemRef,
    );

=cut

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

    my $LogObject            = $Kernel::OM->Get('Kernel::System::Log');
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
    my $ITSMConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $ParamObject          = $Kernel::OM->Get('Kernel::System::Web::Request');

    # check needed stuff
    ARGUMENT:
    for my $Argument (qw(Key Item)) {
        next ARGUMENT if $Param{$Argument};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Need $Argument!"
        );
        return;
    }

    if ( $Param{Item}->{Input}->{SearchInputType} && $Param{Item}->{Input}->{SearchInputType} eq 'Pattern' ) {
        my @Values = qw{};
        if ( !$Self->{CIID} ) {

            my $ItemDataRef = $GeneralCatalogObject->ItemGet(
                Class => 'ITSM::ConfigItem::Class',
                Name  => $Param{Item}->{Input}->{CIClassName},
            );

            if ( !$ItemDataRef || !( $ItemDataRef->{ItemID} ) ) {
                $LogObject->Log(
                    Priority => 'error',
                    Message  => "ITSMConfigItemLayoutCISimple: CI-Class <"
                        . $Param{Item}->{Input}->{CIClassName}
                        . "> not defined in GeneralCatalog!",
                );
            }
            else {
                $Self->{CIID} = $ItemDataRef->{ItemID};
            }

        }

        my @SearchValues = $ParamObject->GetParam( Param => $Param{Key} );

        SEARCHVALUE:
        for my $CurrSearchValue (@SearchValues) {
            next SEARCHVALUE if ( !$CurrSearchValue );

            my @SearchValueParts = split( '\|\|', $CurrSearchValue );

            SEARCHVALUEPART:
            for my $CurrSearchValuePart (@SearchValueParts) {
                next SEARCHVALUEPART if ( !$CurrSearchValuePart );

                # check pattern for id
                if ( $CurrSearchValuePart =~ m/^\[ID\]([0-9]+)$/i ) {
                    push( @Values, $1 );

                    next SEARCHVALUEPART;
                }

                if ( $CurrSearchValuePart =~ m/^\[Number\]([0-9*]+)$/i ) {
                    my $CISearchListRef = $ITSMConfigItemObject->ConfigItemSearchExtended(
                        Number   => $1,
                        ClassIDs => [ $Self->{CIID} ],
                    );
                    for my $SearchResult ( @{$CISearchListRef} ) {
                        push( @Values, $SearchResult );
                    }

                    next SEARCHVALUEPART;
                }

                my $CISearchListRef = $ITSMConfigItemObject->ConfigItemSearchExtended(
                    Name     => $CurrSearchValuePart,
                    ClassIDs => [ $Self->{CIID} ],
                );
                for my $SearchResult ( @{$CISearchListRef} ) {
                    push( @Values, $SearchResult );
                }
            }
        }    # no guard clause ;)

        if ( scalar(@Values) < 1 || !$Values[0] ) {
            @Values = @SearchValues;
        }

        return \@Values;
    }
    else {

        # get form data
        my $Value;
        if ( $Param{Value} ) {
            $Value = $Param{Value};
        }
        else {
            $Value = $ParamObject->GetParam( Param => $Param{Key} );
        }

        return $Value;
    }
}

=head2 SearchInputCreate()

create a search input string

    my $Value = $BackendObject->SearchInputCreate(
        Key => 'Item::1::Node::2',
    );

=cut

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

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    # check needed stuff
    ARGUMENT:
    for my $Argument (qw(Key Item)) {
        next ARGUMENT if $Param{$Argument};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Need $Argument!"
        );
        return;
    }

    my $InputString = '';

    if ( $Param{Item}->{Input}->{SearchInputType} && $Param{Item}->{Input}->{SearchInputType} eq 'Pattern' ) {
        my $Value = '';
        if ( ref( $Param{Value} ) eq 'ARRAY' ) {
            for my $ItemValue ( @{ $Param{Value} } ) {
                if ($Value) {
                    $Value .= "||";
                }
                if ( $ItemValue =~ m/^[0-9]+$/ ) {
                    $Value .= "[ID]" . $ItemValue;
                }
                else {
                    $Value .= $ItemValue;
                }
            }
        }
        $InputString = "<input type=\"Text\" name=\"$Param{Key}\" size=\"30\" value=\"$Value\">";
    }
    else {
        # hash with values for the input field
        my %FormData;

        if ( $Param{Value} ) {
            $FormData{Value} = $Param{Value};
        }

        # create input field
        $InputString = $Self->InputCreate(
            %FormData,
            Key  => $Param{Key},
            Item => $Param{Item},
        );
    }

    return $InputString;
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Output::HTML::ITSMConfigItem::LayoutCustomer;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::System::Log',
    'Kernel::Output::HTML::Layout',
    'Kernel::System::Web::Request',
    'Kernel::System::CustomerUser',
);

=head1 NAME

Kernel::Output::HTML::ITSMConfigItem::LayoutCustomer - layout backend module

=head1 DESCRIPTION

All layout functions of customer objects

=head2 new()

create an object

    $BackendObject = Kernel::Output::HTML::ITSMConfigItem::LayoutCustomer->new(
        %Param,
    );

=cut

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

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

    return $Self;
}

=head2 OutputStringCreate()

create output string

    my $Value = $BackendObject->OutputStringCreate(
        Value => 11,       # (optional)
    );

=cut

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

    # transform ascii to html
    $Param{Value} = $Kernel::OM->Get('Kernel::Output::HTML::Layout')->Ascii2Html(
        Text           => $Param{Value} || '',
        HTMLResultMode => 1,
    );

    return $Param{Value};
}

=head2 FormDataGet()

get form data as hash reference

    my $FormDataRef = $BackendObject->FormDataGet(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

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

    my %FormData;

    # get param object
    my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');

    # get selected customer user
    $FormData{Value} = $ParamObject->GetParam( Param => $Param{Key} );

    # check search button
    if ( $ParamObject->GetParam( Param => $Param{Key} . '::ButtonSearch' ) ) {
        $Param{Item}->{Form}->{ $Param{Key} }->{Search} = $ParamObject->GetParam( Param => $Param{Key} . '::Search' );
    }

    # check select button
    elsif ( $ParamObject->GetParam( Param => $Param{Key} . '::ButtonSelect' ) ) {
        $FormData{Value} = $ParamObject->GetParam( Param => $Param{Key} . '::Select' );
    }

    # check clear button
    elsif ( $ParamObject->GetParam( Param => $Param{Key} . '::ButtonClear' ) ) {
        $FormData{Value} = '';
    }
    else {

        # reset value if search field is empty
        if (
            !$ParamObject->GetParam( Param => $Param{Key} . '::Search' )
            && defined $FormData{Value}
            )
        {
            $FormData{Value} = '';
        }

        # check required option
        if ( $Param{Item}->{Input}->{Required} && !$FormData{Value} ) {
            $Param{Item}->{Form}->{ $Param{Key} }->{Invalid} = 1;
            $FormData{Invalid} = 1;
        }
    }

    return \%FormData;
}

=head2 InputCreate()

create a input string

    my $Value = $BackendObject->InputCreate(
        Key   => 'Item::1::Node::3',
        Value => 11,       # (optional)
        Item  => $ItemRef,
    );

=cut

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

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

    my $Value = '';
    if ( defined $Param{Value} ) {
        $Value = $Param{Value};
    }
    elsif ( $Param{Item}->{Input}->{ValueDefault} ) {
        $Value = $Param{Item}->{Input}->{ValueDefault};
    }
    my $Class    = 'W50pc ITSMCustomerSearch';
    my $Search   = '';
    my $Required = $Param{Required} || '';
    my $Invalid  = $Param{Invalid} || '';
    my $ItemId   = $Param{ItemId} || '';

    if ($Required) {
        $Class .= ' Validate_Required';
    }

    if ($Invalid) {
        $Class .= ' ServerError';
    }

    # create customer string
    if ($Value) {

        # get customer data
        my %CustomerSearchList = $Kernel::OM->Get('Kernel::System::CustomerUser')->CustomerSearch(
            Search => $Value,
        );

        # transform ascii to html
        $Search = $Kernel::OM->Get('Kernel::Output::HTML::Layout')->Ascii2Html(
            Text           => $CustomerSearchList{$Value} || '',
            HTMLResultMode => 1,
        );
    }

    # create string
    my $String = '<input type="hidden" name="'
        . $Param{Key}
        . '" value="'
        . $Value
        . '" id="'
        . $ItemId . 'Selected'
        . '"/>'
        . '<input type="text" name="'
        . $Param{Key}
        . '::Search" class="'
        . $Class
        . '" id="'
        . $ItemId
        . '" value="'
        . $Search . '"/>';

    return $String;
}

=head2 SearchFormDataGet()

get search form data

    my $Value = $BackendObject->SearchFormDataGet(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

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

    # get form data
    my $Value;
    if ( $Param{Value} ) {
        $Value = $Param{Value};
    }
    else {
        $Value = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => $Param{Key} );
    }
    return $Value;
}

=head2 SearchInputCreate()

create a search input string

    my $Value = $BackendObject->SearchInputCreate(
        Key => 'Item::1::Node::3',
    );

=cut

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

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

    my $Key = $Param{Key};

    # hash with values for the input field
    my %FormData;

    if ( $Param{Value} ) {
        $FormData{Value} = $Param{Value};
    }

    # create input field
    my $InputString = $Self->InputCreate(
        %FormData,
        Key    => $Param{Key},
        Item   => $Param{Item},
        ItemId => $Param{Key},
    );

    return $InputString;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dEN1c3RvbWVyQ29tcGFueTsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpMb2cnLAogICAgJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpXZWI6OlJlcXVlc3QnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpDdXN0b21lckNvbXBhbnknLAopOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJVFNNQ29uZmlnSXRlbTo6TGF5b3V0Q3VzdG9tZXJDb21wYW55IC0gbGF5b3V0IGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgREVTQ1JJUFRJT04KCkFsbCBsYXlvdXQgZnVuY3Rpb25zIG9mIGN1c3RvbWVyIGNvbXBhbnkgb2JqZWN0cwoKPWhlYWQyIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgJEJhY2tlbmRPYmplY3QgPSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dEN1c3RvbWVyQ29tcGFueS0+bmV3KAogICAgICAgICVQYXJhbSwKICAgICk7Cgo9Y3V0CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHt9OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgIHJldHVybiAkU2VsZjsKfQoKPWhlYWQyIE91dHB1dFN0cmluZ0NyZWF0ZSgpCgpjcmVhdGUgb3V0cHV0IHN0cmluZwoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5PdXRwdXRTdHJpbmdDcmVhdGUoCiAgICAgICAgVmFsdWUgPT4gMTEsICAgICAgICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBPdXRwdXRTdHJpbmdDcmVhdGUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIHRyYW5zZm9ybSBhc2NpaSB0byBodG1sCiAgICAkUGFyYW17VmFsdWV9ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkxheW91dCcpLT5Bc2NpaTJIdG1sKAogICAgICAgIFRleHQgICAgICAgICAgID0+ICRQYXJhbXtWYWx1ZX0gfHwgJycsCiAgICAgICAgSFRNTFJlc3VsdE1vZGUgPT4gMSwKICAgICk7CgogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBGb3JtRGF0YUdldCgpCgpnZXQgZm9ybSBkYXRhIGFzIGhhc2ggcmVmZXJlbmNlCgogICAgbXkgJEZvcm1EYXRhUmVmID0gJEJhY2tlbmRPYmplY3QtPkZvcm1EYXRhR2V0KAogICAgICAgIEtleSA9PiAnSXRlbTo6MTo6Tm9kZTo6MycsCiAgICAgICAgSXRlbSA9PiAkSXRlbVJlZiwKICAgICk7Cgo9Y3V0CgpzdWIgRm9ybURhdGFHZXQgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICBORUVERUQ6CiAgICBmb3IgbXkgJE5lZWRlZCAocXcoS2V5IEl0ZW0pKSB7CgogICAgICAgIG5leHQgTkVFREVEIGlmIGRlZmluZWQgJFBhcmFteyROZWVkZWR9OwoKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gIk5lZWQgJE5lZWRlZCEiLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIG15ICVGb3JtRGF0YTsKCiAgICAjIGdldCBmb3JtIGRhdGEKICAgICRGb3JtRGF0YXtWYWx1ZX0gPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6V2ViOjpSZXF1ZXN0JyktPkdldFBhcmFtKCBQYXJhbSA9PiAkUGFyYW17S2V5fSApOwoKICAgICMgc2V0IGludmFsaWQgcGFyYW0KICAgIGlmICggJFBhcmFte0l0ZW19LT57SW5wdXR9LT57UmVxdWlyZWR9ICYmICEkRm9ybURhdGF7VmFsdWV9ICkgewogICAgICAgICRGb3JtRGF0YXtJbnZhbGlkfSA9IDE7CiAgICAgICAgJFBhcmFte0l0ZW19LT57Rm9ybX0tPnsgJFBhcmFte0tleX0gfS0+e0ludmFsaWR9ID0gMTsKICAgIH0KCiAgICByZXR1cm4gXCVGb3JtRGF0YTsKfQoKPWhlYWQyIElucHV0Q3JlYXRlKCkKCmNyZWF0ZSBhIGlucHV0IHN0cmluZwoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5JbnB1dENyZWF0ZSgKICAgICAgICBLZXkgICA9PiAnSXRlbTo6MTo6Tm9kZTo6MycsCiAgICAgICAgVmFsdWUgPT4gMTEsICAgICAgICMgKG9wdGlvbmFsKQogICAgICAgIEl0ZW0gID0+ICRJdGVtUmVmLAogICAgKTsKCj1jdXQKCnN1YiBJbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBmb3IgbXkgJEFyZ3VtZW50IChxdyhLZXkgSXRlbSkpIHsKICAgICAgICBpZiAoICEkUGFyYW17JEFyZ3VtZW50fSApIHsKICAgICAgICAgICAgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpLT5Mb2coCiAgICAgICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICAgICAgTWVzc2FnZSAgPT4gIk5lZWQgJEFyZ3VtZW50ISIsCiAgICAgICAgICAgICk7CiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICB9CgogICAgbXkgJFNlbGVjdGVkSUQgPSAkUGFyYW17VmFsdWV9IHx8ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1ZhbHVlRGVmYXVsdH0gfHwgJyc7CgogICAgbXkgJENTU0NsYXNzID0gJ01vZGVybml6ZSc7CiAgICBteSAkUmVxdWlyZWQgPSAkUGFyYW17UmVxdWlyZWR9OwogICAgbXkgJEludmFsaWQgID0gJFBhcmFte0ludmFsaWR9OwogICAgbXkgJEl0ZW1JZCAgID0gJFBhcmFte0l0ZW1JZH07CgogICAgaWYgKCRSZXF1aXJlZCkgewogICAgICAgICRDU1NDbGFzcyAuPSAnIFZhbGlkYXRlX1JlcXVpcmVkJzsKICAgIH0KCiAgICBpZiAoJEludmFsaWQpIHsKICAgICAgICAkQ1NTQ2xhc3MgLj0gJyBTZXJ2ZXJFcnJvcic7CiAgICB9CgogICAgIyBnZXQgY2xhc3MgbGlzdAogICAgbXkgJUNvbXBhbnlMaXN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkN1c3RvbWVyQ29tcGFueScpLT5DdXN0b21lckNvbXBhbnlMaXN0KAogICAgICAgIExpbWl0ID0+IDAsICAgICMgRGlzcGxheSBhbGwgQ3VzdG9tZXIgQ29tcGFuaWVzCiAgICApOwoKICAgICMgZ2VuZXJhdGUgc3RyaW5nCiAgICBteSAkU3RyaW5nID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkxheW91dCcpLT5CdWlsZFNlbGVjdGlvbigKICAgICAgICBEYXRhICAgICAgICAgPT4gXCVDb21wYW55TGlzdCwKICAgICAgICBOYW1lICAgICAgICAgPT4gJFBhcmFte0tleX0sCiAgICAgICAgSUQgICAgICAgICAgID0+ICRJdGVtSWQsCiAgICAgICAgUG9zc2libGVOb25lID0+IDEsCiAgICAgICAgVHJhbnNsYXRpb24gID0+IDAsCiAgICAgICAgU2VsZWN0ZWRJRCAgID0+ICRTZWxlY3RlZElELAogICAgICAgIENsYXNzICAgICAgICA9PiAkQ1NTQ2xhc3MsCiAgICApOwoKICAgIHJldHVybiAkU3RyaW5nOwp9Cgo9aGVhZDIgU2VhcmNoRm9ybURhdGFHZXQoKQoKZ2V0IHNlYXJjaCBmb3JtIGRhdGEKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+U2VhcmNoRm9ybURhdGFHZXQoCiAgICAgICAgS2V5ID0+ICdJdGVtOjoxOjpOb2RlOjozJywKICAgICAgICBJdGVtID0+ICRJdGVtUmVmLAogICAgKTsKCj1jdXQKCnN1YiBTZWFyY2hGb3JtRGF0YUdldCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17S2V5fSApIHsKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgS2V5IScsCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBnZXQgZm9ybSBkYXRhCiAgICBteSBAVmFsdWVzOwogICAgaWYgKCAkUGFyYW17VmFsdWV9ICkgewogICAgICAgIEBWYWx1ZXMgPSBAeyAkUGFyYW17VmFsdWV9IH07CiAgICB9CiAgICBlbHNlIHsKICAgICAgICBAVmFsdWVzID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OldlYjo6UmVxdWVzdCcpLT5HZXRBcnJheSggUGFyYW0gPT4gJFBhcmFte0tleX0gKTsKICAgIH0KCiAgICByZXR1cm4gXEBWYWx1ZXM7Cn0KCj1oZWFkMiBTZWFyY2hJbnB1dENyZWF0ZSgpCgpjcmVhdGUgYSBzZWFyY2ggaW5wdXQgc3RyaW5nCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPlNlYXJjaElucHV0Q3JlYXRlKAogICAgICAgIEtleSA9PiAnSXRlbTo6MTo6Tm9kZTo6MycsCiAgICApOwoKPWN1dAoKc3ViIFNlYXJjaElucHV0Q3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGZvciBteSAkQXJndW1lbnQgKHF3KEtleSBJdGVtKSkgewogICAgICAgIGlmICggISRQYXJhbXskQXJndW1lbnR9ICkgewogICAgICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkQXJndW1lbnQhIiwKICAgICAgICAgICAgKTsKICAgICAgICAgICAgcmV0dXJuOwogICAgICAgIH0KICAgIH0KCiAgICBteSAkVmFsdWVzID0gJFNlbGYtPlNlYXJjaEZvcm1EYXRhR2V0KCVQYXJhbSk7CgogICAgIyBnZXQgY29tcGFueSBkYXRhCiAgICBteSAlQ29tcGFueUxpc3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6Q3VzdG9tZXJDb21wYW55JyktPkN1c3RvbWVyQ29tcGFueUxpc3QoCiAgICAgICAgTGltaXQgPT4gMCwKICAgICk7CgogICAgIyBnZW5lcmF0ZSBzdHJpbmcKICAgIG15ICRTdHJpbmcgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0JyktPkJ1aWxkU2VsZWN0aW9uKAogICAgICAgIERhdGEgICAgICAgID0+IFwlQ29tcGFueUxpc3QsCiAgICAgICAgTmFtZSAgICAgICAgPT4gJFBhcmFte0tleX0sCiAgICAgICAgU2l6ZSAgICAgICAgPT4gNSwKICAgICAgICBNdWx0aXBsZSAgICA9PiAxLAogICAgICAgIFRyYW5zbGF0aW9uID0+IDAsCiAgICAgICAgU2VsZWN0ZWRJRCAgPT4gJFZhbHVlcywKICAgICAgICBDbGFzcyAgICAgICA9PiAnTW9kZXJuaXplJywKICAgICk7CgogICAgcmV0dXJuICRTdHJpbmc7Cn0KCjE7Cgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKClRoaXMgc29mdHdhcmUgaXMgcGFydCBvZiB0aGUgT1RSUyBwcm9qZWN0IChMPGh0dHBzOi8vb3Rycy5vcmcvPikuCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIEw8aHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dD4uCgo9Y3V0Cg==
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Output::HTML::ITSMConfigItem::LayoutDate;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::System::Log',
    'Kernel::Output::HTML::Layout',
    'Kernel::System::Web::Request',
    'Kernel::System::DateTime',
);

=head1 NAME

Kernel::Output::HTML::ITSMConfigItem::LayoutDate - layout backend module

=head1 DESCRIPTION

All layout functions of date objects.

=cut

=head2 new()

create an object

    $BackendObject = Kernel::Output::HTML::ITSMConfigItem::LayoutDate->new(
        %Param,
    );

=cut

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

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

    return $Self;
}

=head2 OutputStringCreate()

create output string

    my $Value = $BackendObject->OutputStringCreate(
        Value => '2007-01-01',  # (optional)
    );

=cut

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

    return '' if !$Param{Value};

    $Param{Value} = $Kernel::OM->Get('Kernel::Output::HTML::Layout')->Output(
        Template => '[% Data.Date | Localize("Date") %]',
        Data     => {
            Date => $Param{Value} . ' 00:00:00',
        },
    );
    $Param{Value} =~ s/00:00:00//;

    return $Param{Value} || '';
}

=head2 FormDataGet()

get form data as hash reference

    my $FormDataRef = $BackendObject->FormDataGet(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

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

    my %FormData;

    # get param object
    my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');

    # get form data
    my $Day   = $ParamObject->GetParam( Param => $Param{Key} . '::Day' );
    my $Month = $ParamObject->GetParam( Param => $Param{Key} . '::Month' );
    my $Year  = $ParamObject->GetParam( Param => $Param{Key} . '::Year' );

    if ( $Day && $Month && $Year ) {
        $FormData{Value} = sprintf '%02d-%02d-%02d', $Year, $Month, $Day;
    }

    # set invalid param
    if ( $Param{Item}->{Input}->{Required} && !$FormData{Value} ) {
        $FormData{Invalid} = 1;
        $Param{Item}->{Form}->{ $Param{Key} }->{Invalid} = 1;
    }

    # Sanity check of the assembled timestamp
    if ( $FormData{Value} ) {

        my $DateTimeObject = $Kernel::OM->Create(
            'Kernel::System::DateTime',
            ObjectParams => {
                String => $FormData{Value} . ' 00:00:00',
            },
        );

        if ( !$DateTimeObject ) {
            $FormData{Invalid} = 1;
            $Param{Item}->{Form}->{ $Param{Key} }->{Invalid} = 1;
        }
    }

    return \%FormData;
}

=head2 InputCreate()

create a input string

    my $Value = $BackendObject->InputCreate(
        Key => 'Item::1::Node::3',
        Value => '2007-03-26',      # (optional)
        Item => $ItemRef,
    );

=cut

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

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

    my $Invalid = $Param{Invalid};

    my %Values;
    my $Class;
    if ( $Param{Value} || $Param{Item}->{Input}->{ValueDefault} ) {
        my $Value = $Param{Value} || $Param{Item}->{Input}->{ValueDefault};

        if ( $Value =~ /^(\d\d\d\d)-(\d\d|\d)-(\d\d|\d)$/i ) {
            $Values{ $Param{Key} . '::Year' }  = $1;
            $Values{ $Param{Key} . '::Month' } = $2;
            $Values{ $Param{Key} . '::Day' }   = $3;

            if ($Invalid) {
                $Class = 'ServerError';
            }
        }
    }

    my $String = $Kernel::OM->Get('Kernel::Output::HTML::Layout')->BuildDateSelection(
        Prefix           => $Param{Key} . '::',
        Format           => 'DateInputFormat',
        YearPeriodPast   => $Param{Item}->{Input}->{YearPeriodPast} || 10,
        YearPeriodFuture => $Param{Item}->{Input}->{YearPeriodFuture} || 10,
        %Values,
        $Param{Key} . '::' . 'Class' => $Class,
        Validate                     => 1,
        OverrideTimeZone             => $Param{OverrideTimeZone},
    );

    return $String;
}

=head2 SearchFormDataGet()

get search form data

    my $Value = $BackendObject->SearchFormDataGet(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

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

    # get form data
    my $Used;
    my $StartDay;
    my $StartMonth;
    my $StartYear;
    my $StopDay;
    my $StopMonth;
    my $StopYear;

    # get param object
    my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');

    if ( $Param{Value} && ref $Param{Value} eq 'HASH' ) {
        $Used       = $Param{Value}->{ $Param{Key} };
        $StartDay   = $Param{Value}->{ $Param{Key} . '::TimeStart::Day' };
        $StartMonth = $Param{Value}->{ $Param{Key} . '::TimeStart::Month' };
        $StartYear  = $Param{Value}->{ $Param{Key} . '::TimeStart::Year' };
        $StopDay    = $Param{Value}->{ $Param{Key} . '::TimeStop::Day' };
        $StopMonth  = $Param{Value}->{ $Param{Key} . '::TimeStop::Month' };
        $StopYear   = $Param{Value}->{ $Param{Key} . '::TimeStop::Year' };
    }
    else {
        $Used       = $ParamObject->GetParam( Param => $Param{Key} );
        $StartDay   = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStart::Day' );
        $StartMonth = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStart::Month' );
        $StartYear  = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStart::Year' );
        $StopDay    = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStop::Day' );
        $StopMonth  = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStop::Month' );
        $StopYear   = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStop::Year' );
    }
    if (
        $Used
        && $StartDay && $StartMonth && $StartYear
        && $StopDay  && $StopMonth  && $StopYear
        )
    {
        my $StartDate = sprintf '%02d-%02d-%02d', $StartYear, $StartMonth, $StartDay;
        my $StopDate  = sprintf '%02d-%02d-%02d', $StopYear,  $StopMonth,  $StopDay;

        return { '-between' => [ $StartDate, $StopDate ] };
    }

    return [];    # no conditions by default
}

=head2 SearchInputCreate()

create a search input string

    my $Value = $BackendObject->SearchInputCreate(
        Key      => 'Item::1::Node::3',
        Item     => $ItemRef,
        Optional => 1,                   # (optional) default 0 (0|1)
    );

=cut

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

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

    # just for convenience
    my $Key         = $Param{Key};
    my $PrefixStart = $Key . '::TimeStart::';
    my $PrefixStop  = $Key . '::TimeStop::';

    # get time related params
    my %GetParam;

    # get param object
    my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');

    if ( $Param{Value} ) {
        %GetParam = %{ $Param{Value} };
    }
    else {
        $GetParam{$Key} = $ParamObject->GetParam( Param => $Key );
        for my $TimeType ( $PrefixStart, $PrefixStop ) {
            for my $Part (qw( Year Month Day )) {
                my $ParamKey = $TimeType . $Part;
                my $ParamVal = $ParamObject->GetParam( Param => $ParamKey );

                # remove white space on the start and end
                if ($ParamVal) {
                    $ParamVal =~ s{ \A \s+ }{}xms;
                    $ParamVal =~ s{ \s+ \z }{}xms;
                }

                # store in %GetParam
                $GetParam{$ParamKey} = $ParamVal;
            }
        }
    }

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

    # Build selection for the start and stop time.
    # Note that searching is by date, while input is by time as well
    my $TimeStartSelectionString = $LayoutObject->BuildDateSelection(
        Prefix           => $PrefixStart,
        Format           => 'DateInputFormat',
        YearPeriodPast   => 10,
        YearPeriodFuture => 10,
        %GetParam,
    );
    my $TimeStopSelectionString = $LayoutObject->BuildDateSelection(
        Optional         => 0,
        Prefix           => $PrefixStop,
        Format           => 'DateInputFormat',
        YearPeriodPast   => 10,
        YearPeriodFuture => 10,
        %GetParam,
    );

    my $Checkbox = qq{<input type="hidden" name="$Key" value="1"/>};
    if ( $Param{Optional} ) {
        $Checkbox = qq{<input type="checkbox" name="$Key" value="1"/>};
    }

    my $Between = $LayoutObject->{LanguageObject}->Translate('Between');
    my $And     = $LayoutObject->{LanguageObject}->Translate('and');

    return "<div> $Checkbox $Between $TimeStartSelectionString </div>"
        . "<span style=\"margin-left: 27px;\">$And</span> $TimeStopSelectionString";
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Output::HTML::ITSMConfigItem::LayoutDateTime;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::System::Log',
    'Kernel::Output::HTML::Layout',
    'Kernel::System::Web::Request',
    'Kernel::System::DateTime',
);

=head1 NAME

Kernel::Output::HTML::ITSMConfigItem::LayoutDateTime - layout backend module

=head1 DESCRIPTION

All layout functions of datetime objects.

=cut

=head2 new()

create an object

    $BackendObject = Kernel::Output::HTML::ITSMConfigItem::LayoutDateTime->new(
        %Param,
    );

=cut

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

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

    return $Self;
}

=head2 OutputStringCreate()

create output string

    my $Value = $BackendObject->OutputStringCreate(
        Value => '2007-01-01 12:00',  # (optional)
    );

=cut

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

    return '' if !$Param{Value};

    $Param{Value} = $Kernel::OM->Get('Kernel::Output::HTML::Layout')->Output(
        Template => '[% Data.Date | Localize("TimeLong") %]',
        Data     => {
            Date => $Param{Value} . ':00',
        },
    );

    return $Param{Value} || '';
}

=head2 FormDataGet()

get form data as hash reference

    my $FormDataRef = $BackendObject->FormDataGet(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

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

    my %FormData;

    # get param object
    my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');

    # get form data
    my $Day    = $ParamObject->GetParam( Param => $Param{Key} . '::Day' );
    my $Month  = $ParamObject->GetParam( Param => $Param{Key} . '::Month' );
    my $Year   = $ParamObject->GetParam( Param => $Param{Key} . '::Year' );
    my $Hour   = $ParamObject->GetParam( Param => $Param{Key} . '::Hour' ) || 0;
    my $Minute = $ParamObject->GetParam( Param => $Param{Key} . '::Minute' ) || 0;

    if ( $Day && $Month && $Year ) {
        $FormData{Value} = sprintf '%02d-%02d-%02d %02d:%02d', $Year, $Month, $Day, $Hour, $Minute;
    }

    # set invalid param
    if ( $Param{Item}->{Input}->{Required} && !$FormData{Value} ) {
        $FormData{Invalid} = 1;
        $Param{Item}->{Form}->{ $Param{Key} }->{Invalid} = 1;
    }

    # Sanity check of the assembled timestamp
    if ( $FormData{Value} ) {

        my $DateTimeObject = $Kernel::OM->Create(
            'Kernel::System::DateTime',
            ObjectParams => {
                String => $FormData{Value} . ':00',
            },
        );

        if ( !$DateTimeObject ) {
            $FormData{Invalid} = 1;
            $Param{Item}->{Form}->{ $Param{Key} }->{Invalid} = 1;
        }
    }

    return \%FormData;
}

=head2 InputCreate()

create a input string

    my $Value = $BackendObject->InputCreate(
        Key => 'Item::1::Node::3',
        Value => '2007-03-26 12:00',  # (optional)
        Item => $ItemRef,
    );

=cut

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

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

    my $Invalid = $Param{Invalid};

    my %Values;
    my $Class;
    if ( $Param{Value} || $Param{Item}->{Input}->{ValueDefault} ) {
        my $Value = $Param{Value} || $Param{Item}->{Input}->{ValueDefault};

        if ( $Value =~ /^(\d\d\d\d)-(\d\d|\d)-(\d\d|\d) (\d\d|\d):(\d\d|\d)$/i ) {
            $Values{ $Param{Key} . '::Year' }   = $1;
            $Values{ $Param{Key} . '::Month' }  = $2;
            $Values{ $Param{Key} . '::Day' }    = $3;
            $Values{ $Param{Key} . '::Hour' }   = $4;
            $Values{ $Param{Key} . '::Minute' } = $5;

            if ($Invalid) {
                $Class = 'ServerError';
            }
        }
    }

    my $String = $Kernel::OM->Get('Kernel::Output::HTML::Layout')->BuildDateSelection(
        Prefix           => $Param{Key} . '::',
        Format           => 'DateInputFormatLong',
        YearPeriodPast   => $Param{Item}->{Input}->{YearPeriodPast} || 10,
        YearPeriodFuture => $Param{Item}->{Input}->{YearPeriodFuture} || 10,
        %Values,
        $Param{Key} . '::' . 'Class' => $Class,
        Validate                     => 1,
    );

    return $String;
}

=head2 SearchFormDataGet()

get search form data

    my $Value = $BackendObject->SearchFormDataGet(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

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

    my $Used;
    my $StartMinute;
    my $StartHour;
    my $StartDay;
    my $StartMonth;
    my $StartYear;
    my $StopMinute;
    my $StopHour;
    my $StopDay;
    my $StopMonth;
    my $StopYear;

    # get param object
    my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');

    # get form data
    if ( $Param{Value} && ref $Param{Value} eq 'HASH' ) {
        $Used        = $Param{Value}->{ $Param{Key} };
        $StartMinute = $Param{Value}->{ $Param{Key} . '::TimeStart::Minute' };
        $StartHour   = $Param{Value}->{ $Param{Key} . '::TimeStart::Hour' };
        $StartDay    = $Param{Value}->{ $Param{Key} . '::TimeStart::Day' };
        $StartMonth  = $Param{Value}->{ $Param{Key} . '::TimeStart::Month' };
        $StartYear   = $Param{Value}->{ $Param{Key} . '::TimeStart::Year' };
        $StopMinute  = $Param{Value}->{ $Param{Key} . '::TimeStop::Minute' };
        $StopHour    = $Param{Value}->{ $Param{Key} . '::TimeStop::Hour' };
        $StopDay     = $Param{Value}->{ $Param{Key} . '::TimeStop::Day' };
        $StopMonth   = $Param{Value}->{ $Param{Key} . '::TimeStop::Month' };
        $StopYear    = $Param{Value}->{ $Param{Key} . '::TimeStop::Year' };
    }
    else {
        $Used        = $ParamObject->GetParam( Param => $Param{Key} );
        $StartMinute = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStart::Minute' ) || 00;
        $StartHour   = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStart::Hour' )
            || 00;
        $StartDay   = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStart::Day' );
        $StartMonth = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStart::Month' );
        $StartYear  = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStart::Year' );
        $StopMinute = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStop::Minute' )
            || 59;
        $StopHour = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStop::Hour' )
            || 23;
        $StopDay   = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStop::Day' );
        $StopMonth = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStop::Month' );
        $StopYear  = $ParamObject->GetParam( Param => $Param{Key} . '::TimeStop::Year' );
    }

    if (
        $Used
        && $StartMinute && $StartHour && $StartDay && $StartMonth && $StartYear
        && $StopMinute  && $StopHour  && $StopDay  && $StopMonth  && $StopYear
        )
    {

        # add hour, minutes and seconds,
        # so that that the first and the last day is selected as well
        my $StartDate = sprintf '%02d-%02d-%02d %02d:%02d:00'
            , $StartYear, $StartMonth, $StartDay, $StartHour, $StartMinute;
        my $StopDate = sprintf '%02d-%02d-%02d %02d:%02d:59'
            , $StopYear, $StopMonth, $StopDay, $StopHour, $StopMinute;

        return { '-between' => [ $StartDate, $StopDate ] };
    }

    return [];    # no conditions by default
}

=head2 SearchInputCreate()

create a search input string

    my $Value = $BackendObject->SearchInputCreate(
        Key      => 'Item::1::Node::3',
        Item     => $ItemRef,
        Optional => 1,                   # (optional) default 0 (0|1)
    );

=cut

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

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

    # just for convenience
    my $Key         = $Param{Key};
    my $PrefixStart = $Key . '::TimeStart::';
    my $PrefixStop  = $Key . '::TimeStop::';

    # get time related params
    my %GetParam;

    # get param object
    my $ParamObject = $Kernel::OM->Get('Kernel::System::Web::Request');

    if ( $Param{Value} ) {
        %GetParam = %{ $Param{Value} };
    }
    else {
        $GetParam{$Key} = $ParamObject->GetParam( Param => $Key );
        for my $TimeType ( $PrefixStart, $PrefixStop ) {
            for my $Part (qw( Year Month Day )) {
                my $ParamKey = $TimeType . $Part;
                my $ParamVal = $ParamObject->GetParam( Param => $ParamKey );

                # remove white space on the start and end
                if ($ParamVal) {
                    $ParamVal =~ s{ \A \s+ }{}xms;
                    $ParamVal =~ s{ \s+ \z }{}xms;
                }

                # store in %GetParam
                $GetParam{$ParamKey} = $ParamVal;
            }
        }
    }

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

    # Build selection for the start and stop time.
    my $TimeStartSelectionString = $LayoutObject->BuildDateSelection(
        Prefix           => $PrefixStart,
        Format           => 'DateInputFormatLong',
        YearPeriodPast   => 10,
        YearPeriodFuture => 10,
        %GetParam,
    );
    my $TimeStopSelectionString = $LayoutObject->BuildDateSelection(
        Optional         => 0,
        Prefix           => $PrefixStop,
        Format           => 'DateInputFormatLong',
        YearPeriodPast   => 10,
        YearPeriodFuture => 10,
        %GetParam,
    );

    my $Checkbox = qq{<input type="hidden" name="$Key" value="1"/>};
    if ( $Param{Optional} ) {
        $Checkbox = qq{<input type="checkbox" name="$Key" value="1"/>};
    }

    my $Between = $LayoutObject->{LanguageObject}->Translate('Between');
    my $And     = $LayoutObject->{LanguageObject}->Translate('and');

    return "<div> $Checkbox $Between $TimeStartSelectionString </div>"
        . "<span style=\"margin-left: 27px;\"> $And </span> $TimeStopSelectionString";
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dER1bW15OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkxheW91dCcsCiAgICAnS2VybmVsOjpTeXN0ZW06OkxvZycsCiAgICAnS2VybmVsOjpTeXN0ZW06OldlYjo6UmVxdWVzdCcsCik7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpPdXRwdXQ6OkhUTUw6OklUU01Db25maWdJdGVtOjpMYXlvdXREdW1teSAtIGxheW91dCBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIERFU0NSSVBUSU9OCgpBbGwgbGF5b3V0IGZ1bmN0aW9ucyBvZiBkdW1teSBvYmplY3RzCgo9aGVhZDIgbmV3KCkKCmNyZWF0ZSBhbiBvYmplY3QKCiAgICAkQmFja2VuZE9iamVjdCA9IEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJVFNNQ29uZmlnSXRlbTo6TGF5b3V0RHVtbXktPm5ldygKICAgICAgICAlUGFyYW0sCiAgICApOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1oZWFkMiBPdXRwdXRTdHJpbmdDcmVhdGUoKQoKY3JlYXRlIG91dHB1dCBzdHJpbmcKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+T3V0cHV0U3RyaW5nQ3JlYXRlKCk7Cgo9Y3V0CgpzdWIgT3V0cHV0U3RyaW5nQ3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgSXRlbSEnLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIGlmICggIWRlZmluZWQgJFBhcmFte1ZhbHVlfSApIHsKICAgICAgICAkUGFyYW17VmFsdWV9ID0gJyc7CiAgICB9CgogICAgIyBnZXQgbGF5b3V0IG9iamVjdAogICAgbXkgJExheW91dE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnKTsKCiAgICAjIHRyYW5zbGF0ZQogICAgaWYgKCAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntUcmFuc2xhdGlvbn0gKSB7CiAgICAgICAgJFBhcmFte1ZhbHVlfSA9ICRMYXlvdXRPYmplY3QtPntMYW5ndWFnZU9iamVjdH0tPlRyYW5zbGF0ZSggJFBhcmFte1ZhbHVlfSApOwogICAgfQoKICAgIG15ICRMaW5rRmVhdHVyZSA9IDE7CgogICAgIyBkbyBub3QgdHJhbnNmb3JtIGxpbmtzIGluIHByaW50IHZpZXcKICAgIGlmICggJFBhcmFte1ByaW50fSApIHsKICAgICAgICAkTGlua0ZlYXR1cmUgPSAwOwogICAgfQoKICAgICMgdHJhbnNmb3JtIGFzY2lpIHRvIGh0bWwKICAgICRQYXJhbXtWYWx1ZX0gPSAkTGF5b3V0T2JqZWN0LT5Bc2NpaTJIdG1sKAogICAgICAgIFRleHQgICAgICAgICAgID0+ICRQYXJhbXtWYWx1ZX0sCiAgICAgICAgSFRNTFJlc3VsdE1vZGUgPT4gMSwKICAgICAgICBMaW5rRmVhdHVyZSAgICA9PiAkTGlua0ZlYXR1cmUsCiAgICApOwoKICAgIHJldHVybiAkUGFyYW17VmFsdWV9Owp9Cgo9aGVhZDIgRm9ybURhdGFHZXQoKQoKZ2V0IGZvcm0gZGF0YSBhcyBoYXNoIHJlZmVyZW5jZQoKICAgIG15ICRGb3JtRGF0YVJlZiA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtRGF0YUdldCgpOwoKPWN1dAoKc3ViIEZvcm1EYXRhR2V0IHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgTkVFREVEOgogICAgZm9yIG15ICROZWVkZWQgKHF3KEtleSBJdGVtKSkgewoKICAgICAgICBuZXh0IE5FRURFRCBpZiBkZWZpbmVkICRQYXJhbXskTmVlZGVkfTsKCiAgICAgICAgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpLT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJOZWVkICROZWVkZWQhIiwKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICBteSAlRm9ybURhdGE7CgogICAgIyBnZXQgZm9ybSBkYXRhCiAgICAkRm9ybURhdGF7VmFsdWV9ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OldlYjo6UmVxdWVzdCcpLT5HZXRQYXJhbSggUGFyYW0gPT4gJFBhcmFte0tleX0gKTsKCiAgICAjIHNldCBpbnZhbGlkIHBhcmFtCiAgICBpZiAoICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1JlcXVpcmVkfSAmJiAhJEZvcm1EYXRhe1ZhbHVlfSApIHsKICAgICAgICAkRm9ybURhdGF7SW52YWxpZH0gPSAxOwogICAgICAgICRQYXJhbXtJdGVtfS0+e0Zvcm19LT57ICRQYXJhbXtLZXl9IH0tPntJbnZhbGlkfSA9IDE7CiAgICB9CgogICAgcmV0dXJuIFwlRm9ybURhdGE7Cn0KCj1oZWFkMiBJbnB1dENyZWF0ZSgpCgpjcmVhdGUgYSBpbnB1dCBzdHJpbmcKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+SW5wdXRDcmVhdGUoKTsKCj1jdXQKCnN1YiBJbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIE5FRURFRDoKICAgIGZvciBteSAkTmVlZGVkIChxdyhLZXkgSXRlbSkpIHsKCiAgICAgICAgbmV4dCBORUVERUQgaWYgZGVmaW5lZCAkUGFyYW17JE5lZWRlZH07CgogICAgICAgICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkTmVlZGVkISIsCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgbXkgJFZhbHVlID0gJFBhcmFte1ZhbHVlfTsKICAgIGlmICggIWRlZmluZWQgJFBhcmFte1ZhbHVlfSApIHsKICAgICAgICAkVmFsdWUgPSAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntWYWx1ZURlZmF1bHR9IHx8ICcnOwogICAgfQoKICAgIG15ICRDbGFzcyAgICA9ICcnOwogICAgbXkgJFNpemUgICAgID0gJ1c1MHBjJzsKICAgIG15ICRSZXF1aXJlZCA9ICRQYXJhbXtSZXF1aXJlZH07CiAgICBteSAkSW52YWxpZCAgPSAkUGFyYW17SW52YWxpZH07CiAgICBteSAkSXRlbUlkICAgPSAkUGFyYW17SXRlbUlkfTsKCiAgICBpZiAoJFJlcXVpcmVkKSB7CiAgICAgICAgJENsYXNzIC49ICcgVmFsaWRhdGVfUmVxdWlyZWQnOwogICAgfQoKICAgIGlmICgkSW52YWxpZCkgewogICAgICAgICRDbGFzcyAuPSAnIFNlcnZlckVycm9yJzsKICAgIH0KICAgICRDbGFzcyAuPSAnICcgLiAkU2l6ZTsKICAgIG15ICRTdHJpbmcgPSAiPHNwYW4gc3R5bGU9XCJkaXNwbGF5OiBpbmxpbmUtYmxvY2s7IGhlaWdodDogMS4zZW07XCI+IjsKICAgICRTdHJpbmcKICAgICAgICAuPSAiPGlucHV0IHN0eWxlPVwiZGlzcGxheTpub25lO1wiIHR5cGU9XCJ0ZXh0XCIgbmFtZT1cIiRQYXJhbXtLZXl9XCIgY2xhc3M9XCIkQ2xhc3NcIiAiOwoKICAgIGlmICgkSXRlbUlkKSB7CiAgICAgICAgJFN0cmluZyAuPSAiaWQ9XCIkSXRlbUlkXCIgIjsKICAgIH0KCiAgICBpZiAoJFZhbHVlKSB7CgogICAgICAgICMgZ2V0IGxheW91dCBvYmplY3QKICAgICAgICBteSAkTGF5b3V0T2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkxheW91dCcpOwoKICAgICAgICAjIHRyYW5zbGF0ZQogICAgICAgIGlmICggJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VHJhbnNsYXRpb259ICkgewogICAgICAgICAgICAkVmFsdWUgPSAkTGF5b3V0T2JqZWN0LT57TGFuZ3VhZ2VPYmplY3R9LT5UcmFuc2xhdGUoJFZhbHVlKTsKICAgICAgICB9CgogICAgICAgICMgdHJhbnNmb3JtIGFzY2lpIHRvIGh0bWwKICAgICAgICAkVmFsdWUgPSAkTGF5b3V0T2JqZWN0LT5Bc2NpaTJIdG1sKAogICAgICAgICAgICBUZXh0ICAgICAgICAgICA9PiAkVmFsdWUsCiAgICAgICAgICAgIEhUTUxSZXN1bHRNb2RlID0+IDEsCiAgICAgICAgKTsKICAgIH0KCiAgICAkU3RyaW5nIC49ICJ2YWx1ZT1cIiRWYWx1ZVwiICI7CgogICAgIyBhZGQgbWF4aW11bSBsZW5ndGgKICAgIGlmICggJFBhcmFte0l0ZW19LT57SW5wdXR9LT57TWF4TGVuZ3RofSApIHsKICAgICAgICAkU3RyaW5nIC49ICJtYXhsZW5ndGg9XCIkUGFyYW17SXRlbX0tPntJbnB1dH0tPntNYXhMZW5ndGh9XCIgIjsKICAgIH0KCiAgICAkU3RyaW5nIC49ICcvPiA8L3NwYW4+JzsKCiAgICByZXR1cm4gJFN0cmluZzsKfQoKPWhlYWQyIFNlYXJjaEZvcm1EYXRhR2V0KCkKCmdldCBzZWFyY2ggZm9ybSBkYXRhCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPlNlYXJjaEZvcm1EYXRhR2V0KCk7Cgo9Y3V0CgpzdWIgU2VhcmNoRm9ybURhdGFHZXQgewogICAgcmV0dXJuIFtdOwp9Cgo9aGVhZDIgU2VhcmNoSW5wdXRDcmVhdGUoKQoKY3JlYXRlIGEgc2VhcmNoIGlucHV0IHN0cmluZwoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5TZWFyY2hJbnB1dENyZWF0ZSgpOwoKPWN1dAoKc3ViIFNlYXJjaElucHV0Q3JlYXRlIHsKICAgIHJldHVybiAnJm5ic3A7JzsKfQoKMTsKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKEw8aHR0cHM6Ly9vdHJzLm9yZy8+KS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0Pi4KCj1jdXQK
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Output::HTML::ITSMConfigItem::LayoutGeneralCatalog;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::System::GeneralCatalog',
    'Kernel::System::Log',
    'Kernel::Output::HTML::Layout',
    'Kernel::System::Web::Request'
);

=head1 NAME

Kernel::Output::HTML::ITSMConfigItem::LayoutGeneralCatalog - layout backend module

=head1 DESCRIPTION

All layout functions of general catalog objects

=head2 new()

create an object

    $BackendObject = Kernel::Output::HTML::ITSMConfigItem::LayoutGeneralCatalog->new(
        %Param,
    );

=cut

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

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

    return $Self;
}

=head2 OutputStringCreate()

create output string

    my $Value = $BackendObject->OutputStringCreate(
        Value => 11,       # (optional)
        Item => $ItemRef,
    );

=cut

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

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

    $Param{Value} //= '';

    # translate
    if ( $Param{Item}->{Input}->{Translation} ) {
        $Param{Value} = $Kernel::OM->Get('Kernel::Output::HTML::Layout')->{LanguageObject}->Translate( $Param{Value} );
    }

    return $Param{Value};
}

=head2 FormDataGet()

get form data as hash reference

    my $FormDataRef = $BackendObject->FormDataGet(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

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

    my %FormData;

    # get form data
    $FormData{Value} = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => $Param{Key} );

    # set invalid param
    if ( $Param{Item}->{Input}->{Required} && !$FormData{Value} ) {
        $FormData{Invalid} = 1;
        $Param{Item}->{Form}->{ $Param{Key} }->{Invalid} = 1;
    }

    return \%FormData;
}

=head2 InputCreate()

create a input string

    my $Value = $BackendObject->InputCreate(
        Key => 'Item::1::Node::3',
        Value => 11,                # (optional)
        Item => $ItemRef,
    );

=cut

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

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

    my $CSSClass = 'Modernize';
    my $Required = $Param{Required};
    my $Invalid  = $Param{Invalid};
    my $ItemId   = $Param{ItemId};

    if ($Required) {
        $CSSClass .= ' Validate_Required';
    }

    if ($Invalid) {
        $CSSClass .= ' ServerError';
    }

    # translation on or off
    my $Translation = 0;
    if ( $Param{Item}->{Input}->{Translation} ) {
        $Translation = 1;
    }

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => $Param{Item}->{Input}->{Class} || '',
    );

    # reverse the class list
    my %ReverseClassList = reverse %{$ClassList};

    my $SelectedID;

    # get the current value
    if ( defined $Param{Value} ) {
        $SelectedID = $Param{Value};
    }

    # get the default id by default value
    else {
        $SelectedID = $ReverseClassList{ $Param{Item}->{Input}->{ValueDefault} || '' } || '';
    }

    # generate string
    my $String = $Kernel::OM->Get('Kernel::Output::HTML::Layout')->BuildSelection(
        Data         => $ClassList,
        Name         => $Param{Key},
        ID           => $ItemId,
        PossibleNone => 1,
        Translation  => $Translation,
        SelectedID   => $SelectedID,
        Class        => $CSSClass,
    );

    return $String;
}

=head2 SearchFormDataGet()

get search form data

    my $Value = $BackendObject->SearchFormDataGet(
        Key => 'Item::1::Node::3',
    );

=cut

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

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

    # get form data
    my @Values;
    if ( $Param{Value} ) {
        @Values = @{ $Param{Value} };
    }
    else {
        @Values = $Kernel::OM->Get('Kernel::System::Web::Request')->GetArray( Param => $Param{Key} );
    }

    return \@Values;
}

=head2 SearchInputCreate()

create a search input string

    my $Value = $BackendObject->SearchInputCreate(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

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

    my $Values = $Self->SearchFormDataGet(%Param);

    # translation on or off
    my $Translation = 0;
    if ( $Param{Item}->{Input}->{Translation} ) {
        $Translation = 1;
    }

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => $Param{Item}->{Input}->{Class} || '',
    );

    # generate string
    my $String = $Kernel::OM->Get('Kernel::Output::HTML::Layout')->BuildSelection(
        Data        => $ClassList,
        Name        => $Param{Key},
        Size        => 5,
        Multiple    => 1,
        Translation => $Translation,
        SelectedID  => $Values,
        Class       => 'Modernize',
    );

    return $String;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Output::HTML::ITSMConfigItem::LayoutInteger;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::System::Log',
    'Kernel::Output::HTML::Layout',
    'Kernel::System::Web::Request'
);

=head1 NAME

Kernel::Output::HTML::ITSMConfigItem::LayoutInteger - layout backend module

=head1 DESCRIPTION

All layout functions of integer objects

=head2 new()

create an object

    $BackendObject = Kernel::Output::HTML::ITSMConfigItem::LayoutInteger->new(
        %Param,
    );

=cut

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

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

    return $Self;
}

=head2 OutputStringCreate()

create output string

    my $Value = $BackendObject->OutputStringCreate(
        Value => 11,       # (optional)
    );

=cut

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

    if ( !defined $Param{Value} ) {
        $Param{Value} = '';
    }

    return $Param{Value};
}

=head2 FormDataGet()

get form data as hash reference

    my $FormDataRef = $BackendObject->FormDataGet(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

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

    my %FormData;

    # get form data
    $FormData{Value} = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => $Param{Key} );

    # set invalid param
    if ( $Param{Item}->{Input}->{Required} && !$FormData{Value} ) {
        $FormData{Invalid} = 1;
        $Param{Item}->{Form}->{ $Param{Key} }->{Invalid} = 1;
    }

    return \%FormData;
}

=head2 InputCreate()

create a input string

    my $Value = $BackendObject->InputCreate(
        Key => 'Item::1::Node::3',
        Value => 11,                # (optional)
        Item => $ItemRef,
    );

=cut

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

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

    # set min, max and default
    my $ValueMin = $Param{Item}->{Input}->{ValueMin} // 1;
    my $ValueMax = $Param{Item}->{Input}->{ValueMax} // 1;
    if ( $ValueMin > $ValueMax ) {
        $ValueMin = $ValueMax;
    }
    if (
        $Param{Item}->{Input}->{ValueDefault}
        && (
            $Param{Item}->{Input}->{ValueDefault} < $ValueMin
            || $Param{Item}->{Input}->{ValueDefault} > $ValueMax
        )
        )
    {
        $Param{Item}->{Input}->{ValueDefault} = '';
    }

    # create data array
    my $IntegerList = [];
    for my $Counter ( $ValueMin .. $ValueMax ) {
        push @{$IntegerList}, $Counter;
    }

    # generate string
    my $String = $Kernel::OM->Get('Kernel::Output::HTML::Layout')->BuildSelection(
        Data         => $IntegerList,
        Name         => $Param{Key},
        PossibleNone => 1,
        Translation  => 0,
        SelectedID   => $Param{Value} || $Param{Item}->{Input}->{ValueDefault} || '',
        Class        => 'Modernize',
    );

    return $String;
}

=head2 SearchFormDataGet()

get search form data

    my $Value = $BackendObject->SearchFormDataGet(
        Key => 'Item::1::Node::3',
    );

=cut

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

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

    # get form data
    my @Values;
    if ( $Param{Value} ) {
        @Values = @{ $Param{Value} };
    }
    else {
        @Values = $Kernel::OM->Get('Kernel::System::Web::Request')->GetArray( Param => $Param{Key} );
    }

    return \@Values;
}

=head2 SearchInputCreate()

create a search input string

    my $Value = $BackendObject->SearchInputCreate(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

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

    # set min, max
    my $ValueMin = $Param{Item}->{Input}->{ValueMin} // 1;
    my $ValueMax = $Param{Item}->{Input}->{ValueMax} // 1;
    if ( $ValueMin > $ValueMax ) {
        $ValueMin = $ValueMax;
    }

    # set preselected value, either from previous selection or the default
    my $Values = $Self->SearchFormDataGet(%Param);

    # check whether the preselected value is within the valid range
    my @FilteredValues;
    VALUE:
    for my $Value ( @{$Values} ) {
        next VALUE if !defined $Value;
        next VALUE if !$Value;
        next VALUE if $Value < $ValueMin;
        next VALUE if $Value > $ValueMax;

        push @FilteredValues, $Value;
    }

    # create data array
    my @IntegerList = ( $ValueMin .. $ValueMax );

    # generate string
    my $String = $Kernel::OM->Get('Kernel::Output::HTML::Layout')->BuildSelection(
        Data        => \@IntegerList,
        Name        => $Param{Key},
        Size        => 5,
        Translation => 0,
        SelectedID  => \@FilteredValues,
        Multiple    => 1,
        Class       => 'Modernize',
    );

    return $String;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dFByaW9yaXR5OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIEtlcm5lbDo6U3lzdGVtOjpWYXJpYWJsZUNoZWNrIHF3KDphbGwpOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkxheW91dCcsCiAgICAnS2VybmVsOjpTeXN0ZW06OkxvZycsCiAgICAnS2VybmVsOjpTeXN0ZW06OlByaW9yaXR5JywKKTsKCnVzZSBwYXJlbnQgJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpJVFNNQ29uZmlnSXRlbTo6TGF5b3V0QmFzZVNlbGVjdGFibGUnOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJVFNNQ29uZmlnSXRlbTo6TGF5b3V0UHJpb3JpdHkgLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKTGF5b3V0IGZ1bmN0aW9ucyBmb3IgQ0kgcHJpb3JpdHkgb2JqZWN0cwoKPWhlYWQyIG5ldygpCgo9aGVhZDIgSW5wdXRDcmVhdGUoKQoKY3JlYXRlIGEgaW5wdXQgc3RyaW5nCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPklucHV0Q3JlYXRlKAogICAgICAgIEtleSAgID0+ICdJdGVtOjoxOjpOb2RlOjozJywKICAgICAgICBWYWx1ZSA9PiAxMSwgICAgICAgIyAob3B0aW9uYWwpCiAgICAgICAgSXRlbSAgPT4gJEl0ZW1SZWYsCiAgICApOwoKPWN1dAoKc3ViIElucHV0Q3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJExvZ09iamVjdCAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpOwogICAgbXkgJFByaW9yaXR5T2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlByaW9yaXR5Jyk7CiAgICBteSAkTGF5b3V0T2JqZWN0ICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0Jyk7CgogICAgI2NoZWNrIG5lZWRlZCBzdHVmZgogICAgQVJHVU1FTlQ6CiAgICBmb3IgbXkgJEFyZ3VtZW50IChxdyhLZXkgSXRlbSkpIHsKICAgICAgICBuZXh0IEFSR1VNRU5UIGlmICRQYXJhbXskQXJndW1lbnR9OwoKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJOZWVkICRBcmd1bWVudCEiCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgbXkgJENsYXNzID0gJ1c1MHBjIE1vZGVybml6ZSc7CiAgICBpZiAoICRQYXJhbXtSZXF1aXJlZH0gKSB7CiAgICAgICAgJENsYXNzIC49ICcgVmFsaWRhdGVfUmVxdWlyZWQnOwogICAgfQoKICAgIGlmICggJFBhcmFte0ludmFsaWR9ICkgewogICAgICAgICRDbGFzcyAuPSAnIFNlcnZlckVycm9yJzsKICAgIH0KCiAgICBteSAkVmFsdWUgPSAnJzsKICAgIGlmICggZGVmaW5lZCAkUGFyYW17VmFsdWV9ICkgewogICAgICAgICRWYWx1ZSA9ICRQYXJhbXtWYWx1ZX07CiAgICB9CiAgICBlbHNpZiAoICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1ZhbHVlRGVmYXVsdH0gKSB7CiAgICAgICAgJFZhbHVlID0gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VmFsdWVEZWZhdWx0fTsKICAgIH0KCiAgICBteSAlUHJpb3JpdGllcyAgICAgICAgPSAkUHJpb3JpdHlPYmplY3QtPlByaW9yaXR5TGlzdCgpOwogICAgbXkgJFByaW9yaXR5U2VsZWN0aW9uID0gJExheW91dE9iamVjdC0+QnVpbGRTZWxlY3Rpb24oCiAgICAgICAgTmFtZSAgICAgICAgID0+ICRQYXJhbXtLZXl9LAogICAgICAgIElEICAgICAgICAgICA9PiAkUGFyYW17SXRlbUlkfSwKICAgICAgICBEYXRhICAgICAgICAgPT4gXCVQcmlvcml0aWVzLAogICAgICAgIENsYXNzICAgICAgICA9PiAkQ2xhc3MsCiAgICAgICAgVHJlZVZpZXcgICAgID0+IDEsCiAgICAgICAgU2VsZWN0ZWRJRCAgID0+ICRWYWx1ZSwKICAgICAgICBQb3NzaWJsZU5vbmUgPT4gMSwKICAgICk7CgogICAgcmV0dXJuICRQcmlvcml0eVNlbGVjdGlvbjsKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dFF1ZXVlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIEtlcm5lbDo6U3lzdGVtOjpWYXJpYWJsZUNoZWNrIHF3KDphbGwpOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkxheW91dCcsCiAgICAnS2VybmVsOjpTeXN0ZW06OkxvZycsCiAgICAnS2VybmVsOjpTeXN0ZW06OlF1ZXVlJywKKTsKCnVzZSBwYXJlbnQgJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpJVFNNQ29uZmlnSXRlbTo6TGF5b3V0QmFzZVNlbGVjdGFibGUnOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJVFNNQ29uZmlnSXRlbTo6TGF5b3V0UXVldWUgLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKTGF5b3V0IGZ1bmN0aW9ucyBmb3IgQ0kgcXVldWUgb2JqZWN0cwoKPWhlYWQyIG5ldygpCgo9aGVhZDIgSW5wdXRDcmVhdGUoKQoKY3JlYXRlIGEgaW5wdXQgc3RyaW5nCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPklucHV0Q3JlYXRlKAogICAgICAgIEtleSAgID0+ICdJdGVtOjoxOjpOb2RlOjozJywKICAgICAgICBWYWx1ZSA9PiAxMSwgICAgICAgIyAob3B0aW9uYWwpCiAgICAgICAgSXRlbSAgPT4gJEl0ZW1SZWYsCiAgICApOwoKPWN1dAoKc3ViIElucHV0Q3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJExvZ09iamVjdCAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKTsKICAgIG15ICRRdWV1ZU9iamVjdCAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6UXVldWUnKTsKICAgIG15ICRMYXlvdXRPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0Jyk7CgogICAgI2NoZWNrIG5lZWRlZCBzdHVmZgogICAgQVJHVU1FTlQ6CiAgICBmb3IgbXkgJEFyZ3VtZW50IChxdyhLZXkgSXRlbSkpIHsKICAgICAgICBuZXh0IEFSR1VNRU5UIGlmICRQYXJhbXskQXJndW1lbnR9OwoKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJOZWVkICRBcmd1bWVudCEiCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgbXkgJENsYXNzID0gJ1c1MHBjIE1vZGVybml6ZSc7CiAgICBpZiAoICRQYXJhbXtSZXF1aXJlZH0gKSB7CiAgICAgICAgJENsYXNzIC49ICcgVmFsaWRhdGVfUmVxdWlyZWQnOwogICAgfQoKICAgIGlmICggJFBhcmFte0ludmFsaWR9ICkgewogICAgICAgICRDbGFzcyAuPSAnIFNlcnZlckVycm9yJzsKICAgIH0KCiAgICBteSAkVmFsdWUgPSAnJzsKICAgIGlmICggZGVmaW5lZCAkUGFyYW17VmFsdWV9ICkgewogICAgICAgICRWYWx1ZSA9ICRQYXJhbXtWYWx1ZX07CiAgICB9CiAgICBlbHNpZiAoICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1ZhbHVlRGVmYXVsdH0gKSB7CiAgICAgICAgJFZhbHVlID0gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VmFsdWVEZWZhdWx0fTsKICAgIH0KCiAgICBteSAlUXVldWVzICAgICAgICAgPSAkUXVldWVPYmplY3QtPlF1ZXVlTGlzdCgpOwogICAgbXkgJFF1ZXVlU2VsZWN0aW9uID0gJExheW91dE9iamVjdC0+QnVpbGRTZWxlY3Rpb24oCiAgICAgICAgTmFtZSAgICAgICAgID0+ICRQYXJhbXtLZXl9LAogICAgICAgIElEICAgICAgICAgICA9PiAkUGFyYW17SXRlbUlkfSwKICAgICAgICBEYXRhICAgICAgICAgPT4gXCVRdWV1ZXMsCiAgICAgICAgQ2xhc3MgICAgICAgID0+ICRDbGFzcywKICAgICAgICBUcmVlVmlldyAgICAgPT4gMSwKICAgICAgICBTZWxlY3RlZElEICAgPT4gJFZhbHVlLAogICAgICAgIFBvc3NpYmxlTm9uZSA9PiAxLAogICAgKTsKCiAgICByZXR1cm4gJFF1ZXVlU2VsZWN0aW9uOwp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dFNMQTsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6VmFyaWFibGVDaGVjayBxdyg6YWxsKTsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAogICAgJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpMb2cnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpTTEEnLAopOwoKdXNlIHBhcmVudCAnS2VybmVsOjpPdXRwdXQ6OkhUTUw6OklUU01Db25maWdJdGVtOjpMYXlvdXRCYXNlU2VsZWN0YWJsZSc7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpPdXRwdXQ6OkhUTUw6OklUU01Db25maWdJdGVtOjpMYXlvdXRTTEEgLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKTGF5b3V0IGZ1bmN0aW9ucyBmb3IgQ0kgU0xBIG9iamVjdHMKCj1oZWFkMiBuZXcoKQoKPWhlYWQyIElucHV0Q3JlYXRlKCkKCmNyZWF0ZSBhIGlucHV0IHN0cmluZwoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5JbnB1dENyZWF0ZSgKICAgICAgICBLZXkgICA9PiAnSXRlbTo6MTo6Tm9kZTo6MycsCiAgICAgICAgVmFsdWUgPT4gMTEsICAgICAgICMgKG9wdGlvbmFsKQogICAgICAgIEl0ZW0gID0+ICRJdGVtUmVmLAogICAgKTsKCj1jdXQKCnN1YiBJbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRMb2dPYmplY3QgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyk7CiAgICBteSAkTGF5b3V0T2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkxheW91dCcpOwogICAgbXkgJFNMQU9iamVjdCAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpTTEEnKTsKCiAgICAjY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBBUkdVTUVOVDoKICAgIGZvciBteSAkQXJndW1lbnQgKHF3KEtleSBJdGVtKSkgewogICAgICAgIG5leHQgQVJHVU1FTlQgaWYgJFBhcmFteyRBcmd1bWVudH07CgogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gIk5lZWQgJEFyZ3VtZW50ISIKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICBteSAkQ2xhc3MgPSAnVzUwcGMgTW9kZXJuaXplJzsKICAgIGlmICggJFBhcmFte1JlcXVpcmVkfSApIHsKICAgICAgICAkQ2xhc3MgLj0gJyBWYWxpZGF0ZV9SZXF1aXJlZCc7CiAgICB9CgogICAgaWYgKCAkUGFyYW17SW52YWxpZH0gKSB7CiAgICAgICAgJENsYXNzIC49ICcgU2VydmVyRXJyb3InOwogICAgfQoKICAgIG15ICRWYWx1ZSA9ICcnOwogICAgaWYgKCBkZWZpbmVkICRQYXJhbXtWYWx1ZX0gKSB7CiAgICAgICAgJFZhbHVlID0gJFBhcmFte1ZhbHVlfTsKICAgIH0KICAgIGVsc2lmICggJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VmFsdWVEZWZhdWx0fSApIHsKICAgICAgICAkVmFsdWUgPSAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntWYWx1ZURlZmF1bHR9OwogICAgfQoKICAgIG15ICVTTEFzICAgICAgICAgPSAkU0xBT2JqZWN0LT5TTEFMaXN0KCBVc2VySUQgPT4gJExheW91dE9iamVjdC0+e1VzZXJJRH0gKTsKICAgIG15ICRTTEFTZWxlY3Rpb24gPSAkTGF5b3V0T2JqZWN0LT5CdWlsZFNlbGVjdGlvbigKICAgICAgICBOYW1lICAgICAgICAgPT4gJFBhcmFte0tleX0sCiAgICAgICAgSUQgICAgICAgICAgID0+ICRQYXJhbXtJdGVtSWR9LAogICAgICAgIERhdGEgICAgICAgICA9PiBcJVNMQXMsCiAgICAgICAgQ2xhc3MgICAgICAgID0+ICRDbGFzcywKICAgICAgICBUcmVlVmlldyAgICAgPT4gMSwKICAgICAgICBTZWxlY3RlZElEICAgPT4gJFZhbHVlLAogICAgICAgIFBvc3NpYmxlTm9uZSA9PiAxLAogICAgKTsKCiAgICByZXR1cm4gJFNMQVNlbGVjdGlvbjsKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dFNlcnZpY2U7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgS2VybmVsOjpTeXN0ZW06OlZhcmlhYmxlQ2hlY2sgcXcoOmFsbCk7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0JywKICAgICdLZXJuZWw6OlN5c3RlbTo6TG9nJywKICAgICdLZXJuZWw6OlN5c3RlbTo6U2VydmljZScsCik7Cgp1c2UgcGFyZW50ICdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dEJhc2VTZWxlY3RhYmxlJzsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dFNlcnZpY2UgLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKTGF5b3V0IGZ1bmN0aW9ucyBmb3IgQ0kgc2VydmljZSBvYmplY3RzCgo9aGVhZDIgbmV3KCkKCj1oZWFkMiBJbnB1dENyZWF0ZSgpCgpjcmVhdGUgYSBpbnB1dCBzdHJpbmcKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+SW5wdXRDcmVhdGUoCiAgICAgICAgS2V5ICAgPT4gJ0l0ZW06OjE6Ok5vZGU6OjMnLAogICAgICAgIFZhbHVlID0+IDExLCAgICAgICAjIChvcHRpb25hbCkKICAgICAgICBJdGVtICA9PiAkSXRlbVJlZiwKICAgICk7Cgo9Y3V0CgpzdWIgSW5wdXRDcmVhdGUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICBteSAkTG9nT2JqZWN0ICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKTsKICAgIG15ICRMYXlvdXRPYmplY3QgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkxheW91dCcpOwogICAgbXkgJFNlcnZpY2VPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6U2VydmljZScpOwoKICAgICNjaGVjayBuZWVkZWQgc3R1ZmYKICAgIEFSR1VNRU5UOgogICAgZm9yIG15ICRBcmd1bWVudCAocXcoS2V5IEl0ZW0pKSB7CiAgICAgICAgbmV4dCBBUkdVTUVOVCBpZiAkUGFyYW17JEFyZ3VtZW50fTsKCiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkQXJndW1lbnQhIgogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIG15ICRDbGFzcyA9ICdXNTBwYyBNb2Rlcm5pemUnOwogICAgaWYgKCAkUGFyYW17UmVxdWlyZWR9ICkgewogICAgICAgICRDbGFzcyAuPSAnIFZhbGlkYXRlX1JlcXVpcmVkJzsKICAgIH0KCiAgICBpZiAoICRQYXJhbXtJbnZhbGlkfSApIHsKICAgICAgICAkQ2xhc3MgLj0gJyBTZXJ2ZXJFcnJvcic7CiAgICB9CgogICAgbXkgJFZhbHVlID0gJyc7CiAgICBpZiAoIGRlZmluZWQgJFBhcmFte1ZhbHVlfSApIHsKICAgICAgICAkVmFsdWUgPSAkUGFyYW17VmFsdWV9OwogICAgfQogICAgZWxzaWYgKCAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntWYWx1ZURlZmF1bHR9ICkgewogICAgICAgICRWYWx1ZSA9ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1ZhbHVlRGVmYXVsdH07CiAgICB9CgogICAgbXkgJVNlcnZpY2VzICAgICAgICAgPSAkU2VydmljZU9iamVjdC0+U2VydmljZUxpc3QoIFVzZXJJRCA9PiAkTGF5b3V0T2JqZWN0LT57VXNlcklEfSApOwogICAgbXkgJFNlcnZpY2VTZWxlY3Rpb24gPSAkTGF5b3V0T2JqZWN0LT5CdWlsZFNlbGVjdGlvbigKICAgICAgICBOYW1lICAgICAgICAgPT4gJFBhcmFte0tleX0sCiAgICAgICAgSUQgICAgICAgICAgID0+ICRQYXJhbXtJdGVtSWR9LAogICAgICAgIERhdGEgICAgICAgICA9PiBcJVNlcnZpY2VzLAogICAgICAgIENsYXNzICAgICAgICA9PiAkQ2xhc3MsCiAgICAgICAgVHJlZVZpZXcgICAgID0+IDEsCiAgICAgICAgU2VsZWN0ZWRJRCAgID0+ICRWYWx1ZSwKICAgICAgICBQb3NzaWJsZU5vbmUgPT4gMSwKICAgICk7CgogICAgcmV0dXJuICRTZXJ2aWNlU2VsZWN0aW9uOwp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dFN0YXRlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIEtlcm5lbDo6U3lzdGVtOjpWYXJpYWJsZUNoZWNrIHF3KDphbGwpOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkxheW91dCcsCiAgICAnS2VybmVsOjpTeXN0ZW06OkxvZycsCiAgICAnS2VybmVsOjpTeXN0ZW06OlN0YXRlJywKKTsKCnVzZSBwYXJlbnQgJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpJVFNNQ29uZmlnSXRlbTo6TGF5b3V0QmFzZVNlbGVjdGFibGUnOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJVFNNQ29uZmlnSXRlbTo6TGF5b3V0U3RhdGUgLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKTGF5b3V0IGZ1bmN0aW9ucyBmb3IgQ0kgU3RhdGUgb2JqZWN0cwoKPWhlYWQyIG5ldygpCgo9aGVhZDIgSW5wdXRDcmVhdGUoKQoKY3JlYXRlIGEgaW5wdXQgc3RyaW5nCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPklucHV0Q3JlYXRlKAogICAgICAgIEtleSAgID0+ICdJdGVtOjoxOjpOb2RlOjozJywKICAgICAgICBWYWx1ZSA9PiAxMSwgICAgICAgIyAob3B0aW9uYWwpCiAgICAgICAgSXRlbSAgPT4gJEl0ZW1SZWYsCiAgICApOwoKPWN1dAoKc3ViIElucHV0Q3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJExvZ09iamVjdCAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKTsKICAgIG15ICRMYXlvdXRPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0Jyk7CiAgICBteSAkU3RhdGVPYmplY3QgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlN0YXRlJyk7CgogICAgI2NoZWNrIG5lZWRlZCBzdHVmZgogICAgQVJHVU1FTlQ6CiAgICBmb3IgbXkgJEFyZ3VtZW50IChxdyhLZXkgSXRlbSkpIHsKICAgICAgICBuZXh0IEFSR1VNRU5UIGlmICRQYXJhbXskQXJndW1lbnR9OwoKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJOZWVkICRBcmd1bWVudCEiCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgbXkgJENsYXNzID0gJ1c1MHBjIE1vZGVybml6ZSc7CiAgICBpZiAoICRQYXJhbXtSZXF1aXJlZH0gKSB7CiAgICAgICAgJENsYXNzIC49ICcgVmFsaWRhdGVfUmVxdWlyZWQnOwogICAgfQoKICAgIGlmICggJFBhcmFte0ludmFsaWR9ICkgewogICAgICAgICRDbGFzcyAuPSAnIFNlcnZlckVycm9yJzsKICAgIH0KCiAgICBteSAkVmFsdWUgPSAnJzsKICAgIGlmICggZGVmaW5lZCAkUGFyYW17VmFsdWV9ICkgewogICAgICAgICRWYWx1ZSA9ICRQYXJhbXtWYWx1ZX07CiAgICB9CiAgICBlbHNpZiAoICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1ZhbHVlRGVmYXVsdH0gKSB7CiAgICAgICAgJFZhbHVlID0gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VmFsdWVEZWZhdWx0fTsKICAgIH0KCiAgICBteSAlU3RhdGVzICAgICAgICAgPSAkU3RhdGVPYmplY3QtPlN0YXRlTGlzdCggVXNlcklEID0+ICRMYXlvdXRPYmplY3QtPntVc2VySUR9ICk7CiAgICBteSAkU3RhdGVTZWxlY3Rpb24gPSAkTGF5b3V0T2JqZWN0LT5CdWlsZFNlbGVjdGlvbigKICAgICAgICBOYW1lICAgICAgICAgPT4gJFBhcmFte0tleX0sCiAgICAgICAgSUQgICAgICAgICAgID0+ICRQYXJhbXtJdGVtSWR9LAogICAgICAgIERhdGEgICAgICAgICA9PiBcJVN0YXRlcywKICAgICAgICBDbGFzcyAgICAgICAgPT4gJENsYXNzLAogICAgICAgIFRyZWVWaWV3ICAgICA9PiAxLAogICAgICAgIFNlbGVjdGVkSUQgICA9PiAkVmFsdWUsCiAgICAgICAgUG9zc2libGVOb25lID0+IDEsCiAgICApOwoKICAgIHJldHVybiAkU3RhdGVTZWxlY3Rpb247Cn0KCjE7Cg==
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Output::HTML::ITSMConfigItem::LayoutText;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::Output::HTML::Layout',
    'Kernel::System::Log',
    'Kernel::System::Web::Request',
);

=head1 NAME

Kernel::Output::HTML::ITSMConfigItem::LayoutText - layout backend module

=head1 DESCRIPTION

All layout functions of text objects

=head2 new()

create an object

    $BackendObject = Kernel::Output::HTML::ITSMConfigItem::LayoutText->new(
        %Param,
    );

=cut

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

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

    return $Self;
}

=head2 OutputStringCreate()

create output string

    my $Value = $BackendObject->OutputStringCreate(
        Value => 11,       # (optional)
        Item => $ItemRef,
    );

=cut

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

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

    if ( !defined $Param{Value} ) {
        $Param{Value} = '';
    }

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

    # translate
    if ( $Param{Item}->{Input}->{Translation} ) {
        $Param{Value} = $LayoutObject->{LanguageObject}->Translate( $Param{Value} );
    }

    my $LinkFeature = 1;

    # do not transform links in print view
    if ( $Param{Print} ) {
        $LinkFeature = 0;
    }

    # transform ascii to html
    $Param{Value} = $LayoutObject->Ascii2Html(
        Text           => $Param{Value},
        HTMLResultMode => 1,
        LinkFeature    => $LinkFeature,
    );

    return $Param{Value};
}

=head2 FormDataGet()

get form data as hash reference

    my $FormDataRef = $BackendObject->FormDataGet(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

    NEEDED:
    for my $Needed (qw(Key Item)) {

        next NEEDED if defined $Param{$Needed};

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need $Needed!",
        );
        return;
    }

    my %FormData;

    # get form data
    $FormData{Value} = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => $Param{Key} );

    # set invalid param
    if ( $Param{Item}->{Input}->{Required} && !defined $FormData{Value} ) {
        $FormData{Invalid} = 1;
        $Param{Item}->{Form}->{ $Param{Key} }->{Invalid} = 1;
    }

    # value was entered in the form, a regex is defined and the value does not match the regex
    if (
        $FormData{Value}
        && $Param{Item}->{Input}->{RegEx}
        && $FormData{Value} !~ m{ $Param{Item}->{Input}->{RegEx} }xms
        )
    {

        $FormData{Invalid}                                         = 1;
        $Param{Item}->{Form}->{ $Param{Key} }->{Invalid}           = 1;
        $Param{Item}->{Form}->{ $Param{Key} }->{RegExErrorMessage} = $Param{Item}->{Input}->{RegExErrorMessage};
    }

    return \%FormData;
}

=head2 InputCreate()

create a input string

    my $Value = $BackendObject->InputCreate(
        Key => 'Item::1::Node::3',
        Value => 11,                # (optional)
        Item => $ItemRef,
    );

=cut

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

    NEEDED:
    for my $Needed (qw(Key Item)) {

        next NEEDED if defined $Param{$Needed};

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need $Needed!",
        );
        return;
    }

    my $Value = $Param{Value};
    if ( !defined $Param{Value} ) {
        $Value = $Param{Item}->{Input}->{ValueDefault} || '';
    }

    my $Class    = '';
    my $Size     = 'W50pc';
    my $Required = $Param{Required};
    my $Invalid  = $Param{Invalid};
    my $ItemId   = $Param{ItemId};

    if ($Required) {
        $Class .= ' Validate_Required';
    }

    if ($Invalid) {
        $Class .= ' ServerError';
    }
    $Class .= ' ' . $Size;
    my $String = "<input type=\"text\" name=\"$Param{Key}\" class=\"$Class\" ";

    if ($ItemId) {
        $String .= "id=\"$ItemId\" ";
    }

    if ($Value) {

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

        # translate
        if ( $Param{Item}->{Input}->{Translation} ) {
            $Value = $LayoutObject->{LanguageObject}->Translate($Value);
        }

        # transform ascii to html
        $Value = $LayoutObject->Ascii2Html(
            Text           => $Value,
            HTMLResultMode => 1,
        );
    }

    $String .= "value=\"$Value\" ";

    # add maximum length
    if ( $Param{Item}->{Input}->{MaxLength} ) {
        $String .= "maxlength=\"$Param{Item}->{Input}->{MaxLength}\" ";
    }

    $String .= '/> ';

    return $String;
}

=head2 SearchFormDataGet()

get search form data

    my $Value = $BackendObject->SearchFormDataGet(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

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

    # get form data
    my $Value;
    if ( $Param{Value} ) {
        $Value = $Param{Value};
    }
    else {
        $Value = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => $Param{Key} );
    }
    return $Value;
}

=head2 SearchInputCreate()

create a search input string

    my $Value = $BackendObject->SearchInputCreate(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

    NEEDED:
    for my $Needed (qw(Key Item)) {

        next NEEDED if defined $Param{$Needed};

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need $Needed!",
        );
        return;
    }

    my $Value = $Self->SearchFormDataGet(%Param);
    if ( !defined $Value ) {
        $Value = '';
    }

    my $String = qq{<input type="text" name="$Param{Key}" value="$Value" class="W50pc" >};

    return $String;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Output::HTML::ITSMConfigItem::LayoutTextArea;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::Output::HTML::Layout',
    'Kernel::System::Log',
    'Kernel::System::Web::Request',
);

=head1 NAME

Kernel::Output::HTML::ITSMConfigItem::LayoutTextArea - layout backend module

=head1 DESCRIPTION

All layout functions of textarea objects

=head2 new()

create an object

    $BackendObject = Kernel::Output::HTML::ITSMConfigItem::LayoutTextArea->new(
        %Param,
    );

=cut

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

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

    return $Self;
}

=head2 OutputStringCreate()

create output string

    my $Value = $BackendObject->OutputStringCreate(
        Value => 11,       # (optional)
        Item => $ItemRef,
    );

=cut

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

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

    if ( !defined $Param{Value} ) {
        $Param{Value} = '';
    }

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

    # translate
    if ( $Param{Item}->{Input}->{Translation} ) {
        $Param{Value} = $LayoutObject->{LanguageObject}->Translate( $Param{Value} );
    }

    my $LinkFeature    = 1;
    my $HTMLResultMode = 1;

    # do not transform links in print view
    if ( $Param{Print} ) {
        $LinkFeature    = 0;
        $HTMLResultMode = 0;
    }

    # transform ascii to html
    $Param{Value} = $LayoutObject->Ascii2Html(
        Text           => $Param{Value},
        HTMLResultMode => $HTMLResultMode,
        LinkFeature    => $LinkFeature,
    );

    return $Param{Value};
}

=head2 FormDataGet()

get form data as hash reference

    my $FormDataRef = $BackendObject->FormDataGet(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

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

    my %FormData;

    # get form data
    $FormData{Value} = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => $Param{Key} );

    # set invalid param
    if ( $Param{Item}->{Input}->{Required} && !defined $FormData{Value} ) {
        $FormData{Invalid} = 1;
        $Param{Item}->{Form}->{ $Param{Key} }->{Invalid} = 1;
    }

    # value was entered in the form, a regex is defined and the value does not match the regex
    if (
        $FormData{Value}
        && $Param{Item}->{Input}->{RegEx}
        && $FormData{Value} !~ m{ $Param{Item}->{Input}->{RegEx} }xms
        )
    {

        $FormData{Invalid}                                         = 1;
        $Param{Item}->{Form}->{ $Param{Key} }->{Invalid}           = 1;
        $Param{Item}->{Form}->{ $Param{Key} }->{RegExErrorMessage} = $Param{Item}->{Input}->{RegExErrorMessage};
    }

    return \%FormData;
}

=head2 InputCreate()

create a input string

    my $Value = $BackendObject->InputCreate(
        Key => 'Item::1::Node::3',
        Value => 11,                # (optional)
        Item => $ItemRef,
    );

=cut

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

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

    my $Cols = $Param{Item}->{Input}->{Cols} || 58;
    my $Rows = $Param{Item}->{Input}->{Rows} || 10;

    my $Value = $Param{Value};
    if ( !defined $Param{Value} ) {
        $Value = $Param{Item}->{Input}->{ValueDefault} || '';
    }

    my $Class    = 'W50pc';
    my $Required = $Param{Required};
    my $Invalid  = $Param{Invalid};
    my $ItemId   = $Param{ItemId};

    if ($Required) {
        $Class .= ' Validate_Required';
    }

    if ($Invalid) {
        $Class .= ' ServerError';
    }

    # translate
    if ( $Param{Item}->{Input}->{Translation} ) {
        $Value = $Kernel::OM->Get('Kernel::Output::HTML::Layout')->{LanguageObject}->Translate($Value);
    }
    my $String
        = "<textarea name=\"$Param{Key}\" id=\"$ItemId\" cols=\"$Cols\" rows=\"$Rows\" class=\"$Class\">$Value</textarea>";

    return $String;
}

=head2 SearchFormDataGet()

get search form data

    my $Value = $BackendObject->SearchFormDataGet(
        Key => 'Item::1::Node::3',
    );

=cut

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

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

    # get form data
    my $Value;
    if ( $Param{Value} ) {
        $Value = $Param{Value};
    }
    else {
        $Value = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => $Param{Key} );
    }
    return $Value;
}

=head2 SearchInputCreate()

create a search input string

    my $Value = $BackendObject->SearchInputCreate(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

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

    my $Value = $Self->SearchFormDataGet(%Param);
    if ( !defined $Value ) {
        $Value = '';
    }

    my $String = qq{<input type="text" name="$Param{Key}" value="$Value" class="W50pc">};

    return $String;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpPdXRwdXQ6OkhUTUw6OklUU01Db25maWdJdGVtOjpMYXlvdXRUZXh0TGluazsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSBwYXJlbnQgcXcoIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJVFNNQ29uZmlnSXRlbTo6TGF5b3V0VGV4dCApOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoKTsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dFRleHRMaW5rIC0gbGF5b3V0IGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgREVTQ1JJUFRJT04KCkFsbCBsYXlvdXQgZnVuY3Rpb25zIG9mIHRleHRsaW5rIG9iamVjdHMKCj1oZWFkMiBPdXRwdXRTdHJpbmdDcmVhdGUoKQoKY3JlYXRlIG91dHB1dCBzdHJpbmcKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+T3V0cHV0U3RyaW5nQ3JlYXRlKAogICAgICAgIFZhbHVlID0+IDExLCAgICAgICAjIChvcHRpb25hbCkKICAgICAgICBJdGVtID0+ICRJdGVtUmVmLAogICAgKTsKCj1jdXQKCnN1YiBPdXRwdXRTdHJpbmdDcmVhdGUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICBteSAkUmVzdWx0ID0gJFNlbGYtPlNVUEVSOjpPdXRwdXRTdHJpbmdDcmVhdGUoJVBhcmFtKTsKICAgIHJldHVybiAkUmVzdWx0IGlmICEkUmVzdWx0OwoKICAgIG15ICRMaW5rID0gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VVJMfTsKICAgICRMaW5rID1+IHMvPFZBTFVFPi8kUmVzdWx0LzsKCiAgICAkUmVzdWx0ID0gPDwiSFRNTCI7CjxhIGhyZWY9IiRMaW5rIiB0YXJnZXQ9Il9ibGFuayI+JFJlc3VsdDwvYT4KSFRNTAoKICAgIHJldHVybiAkUmVzdWx0Owp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dFR5cGU7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgS2VybmVsOjpTeXN0ZW06OlZhcmlhYmxlQ2hlY2sgcXcoOmFsbCk7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0JywKICAgICdLZXJuZWw6OlN5c3RlbTo6TG9nJywKICAgICdLZXJuZWw6OlN5c3RlbTo6VHlwZScsCik7Cgp1c2UgcGFyZW50ICdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dEJhc2VTZWxlY3RhYmxlJzsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06OkxheW91dFR5cGUgLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKTGF5b3V0IGZ1bmN0aW9ucyBmb3IgQ0kgdHlwZSBvYmplY3RzCgo9aGVhZDIgbmV3KCkKCj1oZWFkMiBJbnB1dENyZWF0ZSgpCgpjcmVhdGUgYSBpbnB1dCBzdHJpbmcKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+SW5wdXRDcmVhdGUoCiAgICAgICAgS2V5ICAgPT4gJ0l0ZW06OjE6Ok5vZGU6OjMnLAogICAgICAgIFZhbHVlID0+IDExLCAgICAgICAjIChvcHRpb25hbCkKICAgICAgICBJdGVtICA9PiAkSXRlbVJlZiwKICAgICk7Cgo9Y3V0CgpzdWIgSW5wdXRDcmVhdGUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICBteSAkTG9nT2JqZWN0ICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpOwogICAgbXkgJExheW91dE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnKTsKICAgIG15ICRUeXBlT2JqZWN0ICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VHlwZScpOwoKICAgICNjaGVjayBuZWVkZWQgc3R1ZmYKICAgIEFSR1VNRU5UOgogICAgZm9yIG15ICRBcmd1bWVudCAocXcoS2V5IEl0ZW0pKSB7CiAgICAgICAgbmV4dCBBUkdVTUVOVCBpZiAkUGFyYW17JEFyZ3VtZW50fTsKCiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkQXJndW1lbnQhIgogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIG15ICRDbGFzcyA9ICdXNTBwYyBNb2Rlcm5pemUnOwogICAgaWYgKCAkUGFyYW17UmVxdWlyZWR9ICkgewogICAgICAgICRDbGFzcyAuPSAnIFZhbGlkYXRlX1JlcXVpcmVkJzsKICAgIH0KCiAgICBpZiAoICRQYXJhbXtJbnZhbGlkfSApIHsKICAgICAgICAkQ2xhc3MgLj0gJyBTZXJ2ZXJFcnJvcic7CiAgICB9CgogICAgbXkgJFZhbHVlID0gJyc7CiAgICBpZiAoIGRlZmluZWQgJFBhcmFte1ZhbHVlfSApIHsKICAgICAgICAkVmFsdWUgPSAkUGFyYW17VmFsdWV9OwogICAgfQogICAgZWxzaWYgKCAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntWYWx1ZURlZmF1bHR9ICkgewogICAgICAgICRWYWx1ZSA9ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1ZhbHVlRGVmYXVsdH07CiAgICB9CgogICAgbXkgJVR5cGVzICAgICAgICAgPSAkVHlwZU9iamVjdC0+VHlwZUxpc3QoKTsKICAgIG15ICRUeXBlU2VsZWN0aW9uID0gJExheW91dE9iamVjdC0+QnVpbGRTZWxlY3Rpb24oCiAgICAgICAgTmFtZSAgICAgICAgID0+ICRQYXJhbXtLZXl9LAogICAgICAgIElEICAgICAgICAgICA9PiAkUGFyYW17SXRlbUlkfSwKICAgICAgICBEYXRhICAgICAgICAgPT4gXCVUeXBlcywKICAgICAgICBDbGFzcyAgICAgICAgPT4gJENsYXNzLAogICAgICAgIFRyZWVWaWV3ICAgICA9PiAxLAogICAgICAgIFNlbGVjdGVkSUQgICA9PiAkVmFsdWUsCiAgICAgICAgUG9zc2libGVOb25lID0+IDEsCiAgICApOwoKICAgIHJldHVybiAkVHlwZVNlbGVjdGlvbjsKfQoKMTsK
# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Output::HTML::ITSMConfigItem::LayoutUser;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::Output::HTML::Layout',
    'Kernel::System::Log',
    'Kernel::System::User',
    'Kernel::System::Web::Request',
);

=head1 NAME

Kernel::Output::HTML::ITSMConfigItem::LayoutUser - layout backend module

=head1 SYNOPSIS

All layout functions of user objects


=head2 new()

create an object

    $BackendObject = Kernel::Output::HTML::ITSMConfigItem::LayoutUser->new(
        %Param,
    );

=cut

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

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

    return $Self;
}

=head2 OutputStringCreate()

create output string

    my $Value = $BackendObject->OutputStringCreate(
        Value => 11,       # (optional)
    );

=cut

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

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

    # transform ASCII to html
    $Param{Value} = $LayoutObject->Ascii2Html(
        Text           => $Param{Value} || '',
        HTMLResultMode => 1,
    );

    return $Param{Value};
}

=head2 FormDataGet()

get form data as hash reference

    my $FormDataRef = $BackendObject->FormDataGet(
        Key  => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

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

    #check needed stuff
    ARGUMENT:
    for my $Argument (qw(Key Item)) {

        next ARGUMENT if $Param{$Argument};
        $LogObject->Log(
            Priority => 'error',
            Message  => "Need $Argument!"
        );
        return;
    }

    my %FormData;

    # get param object

    # get selected customer user
    $FormData{Value} = $ParamObject->GetParam( Param => $Param{Key} );

    # check search button
    if ( $ParamObject->GetParam( Param => $Param{Key} . '::ButtonSearch' ) ) {
        $Param{Item}->{Form}->{ $Param{Key} }->{Search} = $ParamObject->GetParam( Param => $Param{Key} . '::Search' );
    }

    # check select button
    elsif ( $ParamObject->GetParam( Param => $Param{Key} . '::ButtonSelect' ) ) {
        $FormData{Value} = $ParamObject->GetParam( Param => $Param{Key} . '::Select' );
    }

    # check clear button
    elsif ( $ParamObject->GetParam( Param => $Param{Key} . '::ButtonClear' ) ) {
        $FormData{Value} = '';
    }
    else {

        # reset value if search field is empty
        if (
            !$ParamObject->GetParam( Param => $Param{Key} . '::Search' )
            && defined $FormData{Value}
            )
        {
            $FormData{Value} = '';
        }

        # check required option
        if ( $Param{Item}->{Input}->{Required} && !$FormData{Value} ) {
            $Param{Item}->{Form}->{ $Param{Key} }->{Invalid} = 1;
            $FormData{Invalid} = 1;
        }
    }

    return \%FormData;
}

=head2 InputCreate()

create a input string

    my $Value = $BackendObject->InputCreate(
        Key   => 'Item::1::Node::3',
        Value => 11,       # (optional)
        Item  => $ItemRef,
    );

=cut

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

    my $LogObject    = $Kernel::OM->Get('Kernel::System::Log');
    my $UserObject   = $Kernel::OM->Get('Kernel::System::User');
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    # check needed stuff
    ARGUMENT:
    for my $Argument (qw(Key Item)) {
        next ARGUMENT if $Param{$Argument};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Need $Argument!"
        );
        return;
    }

    my $Value = '';
    if ( defined $Param{Value} ) {
        $Value = $Param{Value};
    }
    elsif ( $Param{Item}->{Input}->{ValueDefault} ) {
        $Value = $Param{Item}->{Input}->{ValueDefault};
    }

    my $Class = 'W50pc ITSMUserSearch';

    my $Search   = '';
    my $Required = $Param{Required} || '';
    my $Invalid  = $Param{Invalid} || '';
    my $ItemId   = $Param{ItemId} || '';

    if ($Required) {
        $Class .= ' Validate_Required';
    }

    if ($Invalid) {
        $Class .= ' ServerError';
    }

    # create customer string
    if ($Value) {

        # get user data
        my %User = $UserObject->GetUserData(
            UserID => $Value,
        );
        if (%User) {

            my $UserValue = sprintf '"%s %s" <%s>',
                $User{UserFirstname},
                $User{UserLastname},
                $User{UserEmail};

            # transform ASCII to html
            $Search = $LayoutObject->Ascii2Html(
                Text           => $UserValue,
                HTMLResultMode => 1,
            );
        }
    }

    # create string
    my $String = '<input type="hidden" name="'
        . $Param{Key}
        . '" value="'
        . $Value
        . '" id="'
        . $ItemId . 'Selected'
        . '"/>'
        . '<input type="text" name="'
        . $Param{Key}
        . '::Search" class="'
        . $Class
        . '" id="'
        . $ItemId
        . '" value="'
        . $Search . '"/>';

    return $String;
}

=head2 SearchFormDataGet()

get search form data

    my $Value = $BackendObject->SearchFormDataGet(
        Key  => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

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

    # check needed stuff
    ARGUMENT:
    for my $Argument (qw(Key )) {
        next ARGUMENT if $Param{$Argument};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Need $Argument!"
        );
        return;
    }

    # get form data
    my $Value;
    if ( $Param{Value} ) {
        $Value = $Param{Value};
    }
    else {
        $Value = $ParamObject->GetParam( Param => $Param{Key} );
    }
    return $Value;
}

=head2 SearchInputCreate()

create a search input string

    my $Value = $BackendObject->SearchInputCreate(
        Key => 'Item::1::Node::3',
    );

=cut

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

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    # check needed stuff
    ARGUMENT:
    for my $Argument (qw(Key Item)) {
        next ARGUMENT if $Param{$Argument};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Need $Argument!"
        );
        return;
    }

    my $Key = $Param{Key};

    # hash with values for the input field
    my %FormData;

    if ( $Param{Value} ) {
        $FormData{Value} = $Param{Value};
    }

    # create input field
    my $InputString = $Self->InputCreate(
        %FormData,
        Key    => $Param{Key},
        Item   => $Param{Item},
        ItemId => $Param{Key},
    );

    return $InputString;
}

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTUNvbmZpZ0l0ZW06Ok1lbnVHZW5lcmljOwp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgcGFyZW50KCdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6QmFzZScpOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpDb25maWcnLAogICAgJ0tlcm5lbDo6TGFuZ3VhZ2UnLAogICAgJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpMb2cnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpHcm91cCcsCik7CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtDb25maWdJdGVtfSApIHsKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgQ29uZmlnSXRlbSEnLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICMgZ3JhbnQgYWNjZXNzIGJ5IGRlZmF1bHQKICAgIG15ICRBY2Nlc3MgPSAxOwoKICAgICMgZ2V0IGFjdGlvbgogICAgbXkgJEFjdGlvbiA9ICRQYXJhbXtDb25maWd9LT57QWN0aW9ufTsKICAgIGlmICggJEFjdGlvbiBlcSAnQWdlbnRMaW5rT2JqZWN0JyApIHsKCiAgICAgICAgIyBUaGUgTGluay1saW5rIGlzIGEgc3BlY2lhbCBjYXNlLCBhcyBpdCBpcyBub3Qgc3BlY2lmaWMgdG8gSVRTTUNvbmZpZ0l0ZW0uCiAgICAgICAgIyBBcyBhIHdvcmthcm91bmQgd2UgaGFyZGNvZGUgdGhhdCBBZ2VudExpbmtPYmplY3QgaXMgdHJlYXRlZCBsaWtlIEFnZW50SVRTTUNvbmZpZ0l0ZW1FZGl0CiAgICAgICAgJEFjdGlvbiA9ICdBZ2VudElUU01Db25maWdJdGVtRWRpdCc7CiAgICB9CgogICAgIyBnZXQgY29uZmlnIG9iamVjdAogICAgbXkgJENvbmZpZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyk7CgogICAgIyBnZXQgZ3JvdXBzCiAgICBteSAkR3JvdXBzUm8gPSAkQ29uZmlnT2JqZWN0LT5HZXQoJ0Zyb250ZW5kOjpNb2R1bGUnKS0+eyRBY3Rpb259LT57R3JvdXBSb30gfHwgW107CiAgICBteSAkR3JvdXBzUncgPSAkQ29uZmlnT2JqZWN0LT5HZXQoJ0Zyb250ZW5kOjpNb2R1bGUnKS0+eyRBY3Rpb259LT57R3JvdXB9ICAgfHwgW107CgogICAgIyBnZXQgbGF5b3V0IG9iamVjdAogICAgbXkgJExheW91dE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnKTsKCiAgICAjIGNoZWNrIHBlcm1pc3Npb24KICAgIGlmICggJEFjdGlvbiAmJiAoIEB7JEdyb3Vwc1JvfSB8fCBAeyRHcm91cHNSd30gKSApIHsKCiAgICAgICAgIyBkZW55IGFjY2VzcyBieSBkZWZhdWx0LCB3aGVuIHRoZXJlIGFyZSBncm91cHMgdG8gY2hlY2sKICAgICAgICAkQWNjZXNzID0gMDsKCiAgICAgICAgIyBjaGVjayByZWFkIG9ubHkgZ3JvdXBzCiAgICAgICAgUk9HUk9VUDoKICAgICAgICBmb3IgbXkgJFJvR3JvdXAgKCBAeyRHcm91cHNSb30gKSB7CgogICAgICAgICAgICBuZXh0IFJPR1JPVVAgaWYgISRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHcm91cCcpLT5QZXJtaXNzaW9uQ2hlY2soCiAgICAgICAgICAgICAgICBVc2VySUQgICAgPT4gJFNlbGYtPntVc2VySUR9LAogICAgICAgICAgICAgICAgR3JvdXBOYW1lID0+ICRSb0dyb3VwLAogICAgICAgICAgICAgICAgVHlwZSAgICAgID0+ICdybycsCiAgICAgICAgICAgICk7CgogICAgICAgICAgICAjIHNldCBhY2Nlc3MKICAgICAgICAgICAgJEFjY2VzcyA9IDE7CiAgICAgICAgICAgIGxhc3QgUk9HUk9VUDsKICAgICAgICB9CgogICAgICAgICMgY2hlY2sgcmVhZCB3cml0ZSBncm91cHMKICAgICAgICBSV0dST1VQOgogICAgICAgIGZvciBteSAkUndHcm91cCAoIEB7JEdyb3Vwc1J3fSApIHsKCiAgICAgICAgICAgIG5leHQgUldHUk9VUCBpZiAhJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06Okdyb3VwJyktPlBlcm1pc3Npb25DaGVjaygKICAgICAgICAgICAgICAgIFVzZXJJRCAgICA9PiAkU2VsZi0+e1VzZXJJRH0sCiAgICAgICAgICAgICAgICBHcm91cE5hbWUgPT4gJFJ3R3JvdXAsCiAgICAgICAgICAgICAgICBUeXBlICAgICAgPT4gJ3J3JywKICAgICAgICAgICAgKTsKCiAgICAgICAgICAgICMgc2V0IGFjY2VzcwogICAgICAgICAgICAkQWNjZXNzID0gMTsKICAgICAgICAgICAgbGFzdCBSV0dST1VQOwogICAgICAgIH0KICAgIH0KCiAgICByZXR1cm4gJFBhcmFte0NvdW50ZXJ9IGlmICEkQWNjZXNzOwoKICAgICMgb3V0cHV0IG1lbnUgYmxvY2sKICAgICRMYXlvdXRPYmplY3QtPkJsb2NrKCBOYW1lID0+ICdNZW51JyApOwoKICAgICMgb3V0cHV0IG1lbnUgaXRlbQogICAgJExheW91dE9iamVjdC0+QmxvY2soCiAgICAgICAgTmFtZSA9PiAnTWVudUl0ZW0nLAogICAgICAgIERhdGEgPT4gewogICAgICAgICAgICAlUGFyYW0sCiAgICAgICAgICAgICV7ICRQYXJhbXtDb25maWdJdGVtfSB9LAogICAgICAgICAgICAleyAkUGFyYW17Q29uZmlnfSB9LAogICAgICAgIH0sCiAgICApOwoKICAgICMgY2hlY2sgaWYgYSBkaWFsb2cgaGFzIHRvIGJlIHNob3duCiAgICBpZiAoICRQYXJhbXtDb25maWd9LT57RGlhbG9nVGl0bGV9ICkgewogICAgICAgIG15ICRDb25maWdPYmplY3QgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyk7CiAgICAgICAgbXkgJExhbmd1YWdlT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpMYW5ndWFnZScpOwoKICAgICAgICAjIFJlcGxhY2UgdGhlIHRlbXBsYXRlIHRvb2xraXQgZXhwcmVzc2lvbnMgd2l0aCB0aGUgZmluYWwgdmFsdWUuCiAgICAgICAgbXkgJUpTRGF0YSA9ICgKICAgICAgICAgICAgJVBhcmFtLAogICAgICAgICAgICAleyAkUGFyYW17Q29uZmlnSXRlbX0gfSwKICAgICAgICAgICAgJXsgJFBhcmFte0NvbmZpZ30gfSwKICAgICAgICApOwoKICAgICAgICBkZWxldGUgJEpTRGF0YXtDb25maWd9OwogICAgICAgIGRlbGV0ZSAkSlNEYXRhe0NvbmZpZ0l0ZW19OwoKICAgICAgICAkSlNEYXRhe0VsZW1lbnRTZWxlY3Rvcn0gICAgICAgICAgICA9fiBzL1xbJVxzKkRhdGFcLk1lbnVJRFxzKlx8XHMqaHRtbFxzKiVcXS8kSlNEYXRhe01lbnVJRH0vaTsKICAgICAgICAkSlNEYXRhe0RpYWxvZ0NvbnRlbnRRdWVyeVN0cmluZ30gICA9fiBzL1xbJVxzKkRhdGFcLkNvbmZpZ0l0ZW1JRFxzKlx8XHMqaHRtbFxzKiVcXS8kSlNEYXRhe0NvbmZpZ0l0ZW1JRH0vaTsKICAgICAgICAkSlNEYXRhe0NvbmZpcm1lZEFjdGlvblF1ZXJ5U3RyaW5nfSA9fiBzL1xbJVxzKkRhdGFcLkNvbmZpZ0l0ZW1JRFxzKlx8XHMqaHRtbFxzKiVcXS8kSlNEYXRhe0NvbmZpZ0l0ZW1JRH0vaTsKCiAgICAgICAgJEpTRGF0YXtEaWFsb2dUaXRsZX0gPX4gcy9cWyVccypUcmFuc2xhdGVcKCIoLiopIlwpXHMqXHxccypodG1sXHMqJVxdLyRMYW5ndWFnZU9iamVjdC0+VHJhbnNsYXRlKCQxKS9laTsKICAgICAgICAkSlNEYXRhe0RpYWxvZ1RpdGxlfSA9fiBzL1xbJVxzKkNvbmZpZ1woIiguKikiXClccyolXF0vJENvbmZpZ09iamVjdC0+R2V0KCQxKS9laTsKICAgICAgICAkSlNEYXRhe0RpYWxvZ1RpdGxlfSA9fiBzL1xbJVxzKkRhdGEuTnVtYmVyXHMqXHxccypodG1sXHMqJVxdLyRKU0RhdGF7TnVtYmVyfS9laTsKCiAgICAgICAgJEpTRGF0YXtNZW51SUR9ID0gJ01lbnUnIC4gJEpTRGF0YXtNZW51SUR9OwoKICAgICAgICAkTGF5b3V0T2JqZWN0LT5BZGRKU0RhdGEoCiAgICAgICAgICAgIEtleSAgID0+ICdJVFNNU2hvd0NvbmZpcm1EaWFsb2cuJyAuICRQYXJhbXtNZW51SUR9LAogICAgICAgICAgICBWYWx1ZSA9PiBcJUpTRGF0YSwKICAgICAgICApOwogICAgfQoKICAgICRQYXJhbXtDb3VudGVyfSsrOwoKICAgIHJldHVybiAkUGFyYW17Q291bnRlcn07Cn0KCjE7Cg==
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Output::HTML::ITSMConfigItem::OverviewSmall;

use strict;
use warnings;

use Kernel::Language qw(Translatable);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::Output::HTML::Layout',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::Group',
    'Kernel::System::HTMLUtils',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::Log',
    'Kernel::System::Main',
);

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

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

    return $Self;
}

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

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    NEEDED:
    for my $Needed (qw(PageShown StartHit)) {
        next NEEDED if defined $Param{$Needed};
        $LogObject->Log(
            Priority => 'error',
            Message  => "Need $Needed!",
        );
        return;
    }

    # need ConfigItemIDs
    if ( !$Param{ConfigItemIDs} ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => 'Need the ConfigItemIDs!',
        );
        return;
    }

    # define incident signals, needed for services
    my %InciSignals = (
        Translatable('operational') => 'greenled',
        Translatable('warning')     => 'yellowled',
        Translatable('incident')    => 'redled',
    );

    # to store the color for the deployment states
    my %DeplSignals;

    # get general catalog object
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

    # get list of deployment states
    my $DeploymentStatesList = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    # set deployment style colors
    my $StyleClasses = '';

    ITEMID:
    for my $ItemID ( sort keys %{$DeploymentStatesList} ) {

        # get deployment state preferences
        my %Preferences = $GeneralCatalogObject->GeneralCatalogPreferencesGet(
            ItemID => $ItemID,
        );

        # check if a color is defined in preferences
        next ITEMID if !$Preferences{Color};

        # get deployment state
        my $DeplState = $DeploymentStatesList->{$ItemID};

        # remove any non ascii word characters
        $DeplState =~ s{ [^a-zA-Z0-9] }{_}msxg;

        # store the original deployment state as key
        # and the ss safe coverted deployment state as value
        $DeplSignals{ $DeploymentStatesList->{$ItemID} } = $DeplState;

        # covert to lower case
        my $DeplStateColor = lc $Preferences{Color};

        # add to style classes string
        $StyleClasses .= "
            .Flag span.$DeplState {
                background-color: $DeplStateColor;
            }
        ";
    }

    # wrap into style tags
    if ($StyleClasses) {
        $StyleClasses = "<style>$StyleClasses</style>";
    }

    # store either ConfigItem IDs Locally
    my @ConfigItemIDs = @{ $Param{ConfigItemIDs} };

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

    # check if bulk feature is enabled
    my $BulkFeature = 0;
    if ( $ConfigObject->Get('ITSMConfigItem::Frontend::BulkFeature') ) {
        my @Groups;
        if ( $ConfigObject->Get('ITSMConfigItem::Frontend::BulkFeatureGroup') ) {
            @Groups = @{ $ConfigObject->Get('ITSMConfigItem::Frontend::BulkFeatureGroup') };
        }
        if ( !@Groups ) {
            $BulkFeature = 1;
        }
        else {
            GROUP:
            for my $Group (@Groups) {
                next GROUP if !$Kernel::OM->Get('Kernel::System::Group')->PermissionCheck(
                    UserID    => $Self->{UserID},
                    GroupName => $Group,
                    Type      => 'rw',
                );

                $BulkFeature = 1;
                last GROUP;
            }
        }
    }

    # get config item pre menu modules
    my @ActionItems;
    if ( ref $ConfigObject->Get('ITSMConfigItem::Frontend::PreMenuModule') eq 'HASH' ) {
        my %Menus = %{ $ConfigObject->Get('ITSMConfigItem::Frontend::PreMenuModule') };

        MENU:
        for my $MenuKey ( sort keys %Menus ) {

            # load module
            if ( $Kernel::OM->Get('Kernel::System::Main')->Require( $Menus{$MenuKey}->{Module} ) ) {
                my $Object = $Menus{$MenuKey}->{Module}->new(
                    %{$Self},
                );

                # check if the menu is available
                next MENU if ref $Menus{$MenuKey} ne 'HASH';

                # set classes
                if ( $Menus{$MenuKey}->{Target} ) {

                    if ( $Menus{$MenuKey}->{Target} eq 'PopUp' ) {
                        $Menus{$MenuKey}->{MenuClass} = 'AsPopup';
                        $Menus{$MenuKey}->{PopupType} = 'ITSMConfigItemAction';
                    }
                    else {
                        $Menus{$MenuKey}->{MenuClass} = '';
                        $Menus{$MenuKey}->{PopupType} = '';
                    }
                }

                # grant access by default
                my $Access = 1;

                my $Action = $Menus{$MenuKey}->{Action};

                # can not execute the module due to a ConfigItem is required, then just check the
                # permissions as in the MenuModuleGeneric
                my $GroupsRo = $ConfigObject->Get('Frontend::Module')->{$Action}->{GroupRo} || [];
                my $GroupsRw = $ConfigObject->Get('Frontend::Module')->{$Action}->{Group}   || [];

                # check permission
                if ( $Action && ( @{$GroupsRo} || @{$GroupsRw} ) ) {

                    # deny access by default, when there are groups to check
                    $Access = 0;

                    # check read only groups
                    ROGROUP:
                    for my $RoGroup ( @{$GroupsRo} ) {
                        next ROGROUP if !$Kernel::OM->Get('Kernel::System::Group')->PermissionCheck(
                            UserID    => $Self->{UserID},
                            GroupName => $RoGroup,
                            Type      => 'ro',
                        );

                        # set access
                        $Access = 1;
                        last ROGROUP;
                    }

                    # check read write groups
                    RWGROUP:
                    for my $RwGroup ( @{$GroupsRw} ) {
                        next RWGROUP if !$Kernel::OM->Get('Kernel::System::Group')->PermissionCheck(
                            UserID    => $Self->{UserID},
                            GroupName => $RwGroup,
                            Type      => 'rw',
                        );

                        # set access
                        $Access = 1;
                        last RWGROUP;
                    }
                }

                # return if there is no access to the module
                next MENU if !$Access;

                # translate Name and Description
                my $Description = $LayoutObject->{LanguageObject}->Translate( $Menus{$MenuKey}->{Description} );
                my $Name        = $LayoutObject->{LanguageObject}->Translate( $Menus{$MenuKey}->{Description} );

                # generarte a web safe link
                my $Link = $LayoutObject->{Baselink} . $Menus{$MenuKey}->{Link};

                # sanity check
                if ( !defined $Menus{$MenuKey}->{MenuClass} ) {
                    $Menus{$MenuKey}->{MenuClass} = '';
                }

                # generate HTML for the menu item
                my $MenuHTML = << "END";
<li>
    <a href="$Link" class="$Menus{$MenuKey}->{MenuClass}" title="$Description">$Name</a>
</li>
END

                $MenuHTML =~ s/\n+//g;
                $MenuHTML =~ s/\s+/ /g;
                $MenuHTML =~ s/<\!--.+?-->//g;

                $Menus{$MenuKey}->{ID} = $Menus{$MenuKey}->{Name};
                $Menus{$MenuKey}->{ID} =~ s/(\s|&|;)//ig;

                push @ActionItems, {
                    HTML        => $MenuHTML,
                    ID          => $Menus{$MenuKey}->{ID},
                    Link        => $Link,
                    Target      => $Menus{$MenuKey}->{Target},
                    PopupType   => $Menus{$MenuKey}->{PopupType},
                    Description => $Description,
                };
            }
        }
    }

    # check ShowColumns parameter
    my @ShowColumns;
    my @XMLShowColumns;
    if ( $Param{ShowColumns} && ref $Param{ShowColumns} eq 'ARRAY' ) {
        @ShowColumns = @{ $Param{ShowColumns} };
    }

    # get config item object
    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    # build column header blocks
    if (@ShowColumns) {

        # show the bulk action button checkboxes if feature is enabled
        if ($BulkFeature) {
            push @ShowColumns, 'BulkAction';
        }

        for my $Column (@ShowColumns) {

            # create needed veriables
            my $CSS = 'OverviewHeader';
            my $OrderBy;

            # remove ID if necesary
            if ( $Param{SortBy} ) {
                $Param{SortBy} = ( $Param{SortBy} eq 'InciStateID' )
                    ? 'CurInciState'
                    : ( $Param{SortBy} eq 'DeplStateID' ) ? 'CurDeplState'
                    : ( $Param{SortBy} eq 'ClassID' )     ? 'Class'
                    : ( $Param{SortBy} eq 'ChangeTime' )  ? 'LastChanged'
                    :                                       $Param{SortBy};
            }

            # set the correct Set CSS class and order by link
            if ( $Param{SortBy} && ( $Param{SortBy} eq $Column ) ) {
                if ( $Param{OrderBy} && ( $Param{OrderBy} eq 'Up' ) ) {
                    $OrderBy = 'Down';
                    $CSS .= ' SortDescendingLarge';
                }
                else {
                    $OrderBy = 'Up';
                    $CSS .= ' SortAscendingLarge';
                }
            }
            else {
                $OrderBy = 'Up';
            }

            $LayoutObject->Block(
                Name => 'Record' . $Column . 'Header',
                Data => {
                    %Param,
                    CSS     => $CSS,
                    OrderBy => $OrderBy,
                },
            );
        }

        # get the XML column headers only if the filter is not set to 'all'
        # and if there are CIs to show
        if ( $Param{Filter} && $Param{Filter} ne 'All' && @ConfigItemIDs ) {

            # get the version data of the first config item, including all the XML data
            # to get the column header names
            my $ConfigItem = $ConfigItemObject->VersionGet(
                ConfigItemID => $ConfigItemIDs[0],
                XMLDataGet   => 1,
            );

            # convert the XML data into a hash
            my $ExtendedVersionData = $LayoutObject->XMLData2Hash(
                XMLDefinition => $ConfigItem->{XMLDefinition},
                XMLData       => $ConfigItem->{XMLData}->[1]->{Version}->[1],
                Attributes    => \@ShowColumns,
                Print         => 1,
            );

            # get the xml columns (they contain ::)
            @XMLShowColumns = grep {/::/} @ShowColumns;

            COLUMN:
            for my $Column (@XMLShowColumns) {

                # check if column exists in CI-Data
                next COLUMN if !$ExtendedVersionData->{$Column}->{Name};

                # show the xml attribute header
                $LayoutObject->Block(
                    Name => 'RecordXMLAttributeHeader',
                    Data => {
                        %Param,
                        XMLAttributeHeader => $ExtendedVersionData->{$Column}->{Name},
                    },
                );
            }
        }
    }

    my $Output  = '';
    my $Counter = 0;

    # show config items if there are some
    if (@ConfigItemIDs) {

        # to store all data
        my %Data;

        CONFIGITEMID:
        for my $ConfigItemID (@ConfigItemIDs) {
            $Counter++;
            if (
                $Counter >= $Param{StartHit}
                && $Counter < ( $Param{PageShown} + $Param{StartHit} )
                )
            {

                # check for access rights
                my $HasAccess = $ConfigItemObject->Permission(
                    Scope  => 'Item',
                    ItemID => $ConfigItemID,
                    UserID => $Self->{UserID},
                    Type   => $Self->{Config}->{Permission},
                );

                next CONFIGITEMID if !$HasAccess;

                # get config item data
                my $ConfigItem = $ConfigItemObject->VersionGet(
                    ConfigItemID => $ConfigItemID,
                    XMLDataGet   => 1,
                );

                next CONFIGITEMID if !$ConfigItem;

                # convert the XML data into a hash
                my $ExtendedVersionData = $LayoutObject->XMLData2Hash(
                    XMLDefinition => $ConfigItem->{XMLDefinition},
                    XMLData       => $ConfigItem->{XMLData}->[1]->{Version}->[1],
                    Attributes    => \@ShowColumns,
                    Print         => 1,
                );

                # store config item data,
                %Data = %{$ConfigItem};

                # Get config item data for 'CreateTime' and 'ChangeTime'.
                my $ConfigItemData = $ConfigItemObject->ConfigItemGet(
                    ConfigItemID => $ConfigItemID,
                );
                $Data{CreateTime} = $ConfigItemData->{CreateTime};
                $Data{ChangeTime} = $ConfigItemData->{ChangeTime};

                # build record block
                $LayoutObject->Block(
                    Name => 'Record',
                    Data => {
                        %Param,
                        %Data,
                    },
                );

                # build column record blocks
                if (@ShowColumns) {

                    COLUMN:
                    for my $Column (@ShowColumns) {
                        $LayoutObject->Block(
                            Name => 'Record' . $Column,
                            Data => {
                                %Param,
                                %Data,
                                CurInciSignal => $InciSignals{ $Data{CurInciStateType} },
                                CurDeplSignal => $DeplSignals{ $Data{CurDeplState} },
                            },
                        );
                    }

                    COLUMN:
                    for my $Column (@XMLShowColumns) {

                        # check if column exists in CI-Data
                        next COLUMN if !$ExtendedVersionData->{$Column}->{Name};

                        # Convert to ascii text in case the value contains html.
                        my $Value = $Kernel::OM->Get('Kernel::System::HTMLUtils')
                            ->ToAscii( String => $ExtendedVersionData->{$Column}->{Value} // '' ) // '';

                        # convert all whitespace and newlines to single spaces
                        $Value =~ s{ \s+ }{ }gxms;

                        # show the xml attribute data
                        $LayoutObject->Block(
                            Name => 'RecordXMLAttribute',
                            Data => {
                                %Param,
                                XMLAttributeData => $Value,
                            },
                        );
                    }
                }

                # make a deep copy of the action items to avoid changing the definition
                my $ClonedActionItems = Storable::dclone( \@ActionItems );

                # substitute TT variables
                for my $ActionItem ( @{$ClonedActionItems} ) {
                    $ActionItem->{HTML} =~ s{ \Q[% Data.ConfigItemID | html %]\E }{$ConfigItemID}xmsg;
                    $ActionItem->{HTML} =~ s{ \Q[% Data.VersionID | html %]\E }{$ConfigItem->{VersionID}}xmsg;
                    $ActionItem->{Link} =~ s{ \Q[% Data.ConfigItemID | html %]\E }{$ConfigItemID}xmsg;
                    $ActionItem->{Link} =~ s{ \Q[% Data.VersionID | html %]\E }{$ConfigItem->{VersionID}}xmsg;
                }

                #my $JSON = $LayoutObject->JSONEncode(
                #    Data => $ClonedActionItems,
                #);

                #$LayoutObject->Block(
                #    Name => 'DocumentReadyActionRowAdd',
                #    Data => {
                #        ConfigItemID => $ConfigItemID,
                #        Data         => $JSON,
                #    },
                #);

                $LayoutObject->AddJSData(
                    Key   => 'ITSMConfigItemActionRow.' . $ConfigItemID,
                    Value => $ClonedActionItems,
                );
            }
        }
    }

    # if there are no config items to show, a no data found message is displayed in the table
    else {
        $LayoutObject->Block(
            Name => 'NoDataFoundMsg',
            Data => {
                TotalColumns => scalar @ShowColumns,
            },
        );
    }

    # use template
    $Output .= $LayoutObject->Output(
        TemplateFile => 'AgentITSMConfigItemOverviewSmall',
        Data         => {
            %Param,
            Type         => $Self->{ViewType},
            ColumnCount  => scalar @ShowColumns,
            StyleClasses => $StyleClasses,
        },
    );

    return $Output;
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Output::HTML::Layout::ITSMConfigItem;

use strict;
use warnings;

our $ObjectManagerDisabled = 1;

=head1 NAME

Kernel::Output::HTML::Layout::ITSMConfigItem - all ConfigItem-related HTML functions

=head1 DESCRIPTION

All ITSM Configuration Management-related HTML functions

=head1 PUBLIC INTERFACE

=head2 ITSMConfigItemOutputStringCreate()

returns an output string

    my $String = $LayoutObject->ITSMConfigItemOutputStringCreate(
        Value => 11,       # (optional)
        Item  => $ItemRef,
        Print => 1,        # (optional, default 0)
    );

=cut

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

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

    # load backend
    my $BackendObject = $Self->_ITSMLoadLayoutBackend(
        Type => $Param{Item}->{Input}->{Type},
    );

    return '' if !$BackendObject;

    # generate output string
    my $String = $BackendObject->OutputStringCreate(%Param);

    return $String;
}

=head2 ITSMConfigItemFormDataGet()

returns the values from the html form as hash reference

    my $FormDataRef = $LayoutObject->ITSMConfigItemFormDataGet(
        Key          => 'Item::1::Node::3',
        Item         => $ItemRef,
        ConfigItemID => 123,
    );

=cut

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

    NEEDED:
    for my $Needed (qw(Key Item ConfigItemID)) {

        next NEEDED if defined $Param{$Needed};

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need $Needed!",
        );
        return;
    }

    # load backend
    my $BackendObject = $Self->_ITSMLoadLayoutBackend(
        Type => $Param{Item}->{Input}->{Type},
    );

    return {} if !$BackendObject;

    # get form data
    my $FormData = $BackendObject->FormDataGet(%Param);

    return $FormData;
}

=head2 ITSMConfigItemInputCreate()

returns a input field html string

    my $String = $LayoutObject->ITSMConfigItemInputCreate(
        Key => 'Item::1::Node::3',
        Value => 11,                # (optional)
        Item => $ItemRef,
    );

=cut

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

    NEEDED:
    for my $Needed (qw(Key Item)) {

        next NEEDED if defined $Param{$Needed};

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need $Needed!",
        );
        return;
    }

    # load backend
    my $BackendObject = $Self->_ITSMLoadLayoutBackend(
        Type => $Param{Item}->{Input}->{Type},
    );

    return '' if !$BackendObject;

    # lookup item value
    my $String = $BackendObject->InputCreate(%Param);

    return $String;
}

=head2 ITSMConfigItemSearchFormDataGet()

returns the values from the search html form

    my $ArrayRef = $LayoutObject->ITSMConfigItemSearchFormDataGet(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

    NEEDED:
    for my $Needed (qw(Key Item)) {

        next NEEDED if defined $Param{$Needed};

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need $Needed!",
        );
        return;
    }

    # load backend
    my $BackendObject = $Self->_ITSMLoadLayoutBackend(
        Type => $Param{Item}->{Input}->{Type},
    );

    return [] if !$BackendObject;

    # get form data
    my $Values = $BackendObject->SearchFormDataGet(%Param);

    return $Values;
}

=head2 ITSMConfigItemSearchInputCreate()

returns a search input field html string

    my $String = $LayoutObject->ITSMConfigItemSearchInputCreate(
        Key => 'Item::1::Node::3',
        Item => $ItemRef,
    );

=cut

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

    NEEDED:
    for my $Needed (qw(Key Item)) {

        next NEEDED if defined $Param{$Needed};

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need $Needed!",
        );
        return;
    }

    # load backend
    my $BackendObject = $Self->_ITSMLoadLayoutBackend(
        Type => $Param{Item}->{Input}->{Type},
    );

    return '' if !$BackendObject;

    # lookup item value
    my $String = $BackendObject->SearchInputCreate(%Param);

    return $String;
}

=head2 _ITSMLoadLayoutBackend()

load a input type backend module

    $BackendObject = $LayoutObject->_ITSMLoadLayoutBackend(
        Type => 'GeneralCatalog',
    );

=cut

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

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    if ( !$Param{Type} ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => 'Need Type!',
        );
        return;
    }

    my $GenericModule = "Kernel::Output::HTML::ITSMConfigItem::Layout$Param{Type}";

    # load the backend module
    if ( !$Kernel::OM->Get('Kernel::System::Main')->Require($GenericModule) ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => "Can't load backend module $Param{Type}!"
        );
        return;
    }

    # create new instance
    my $BackendObject = $GenericModule->new(
        %{$Self},
        %Param,
        LayoutObject => $Self,
    );

    if ( !$BackendObject ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => "Can't create a new instance of backend module $Param{Type}!",
        );
        return;
    }

    return $BackendObject;
}

=head2 ITSMConfigItemListShow()

Returns a list of configuration items with sort and pagination capabilities.

This function is similar to L<Kernel::Output::HTML::LayoutTicket::TicketListShow()>
in F<Kernel/Output/HTML/LayoutTicket.pm>.

    my $Output = $LayoutObject->ITSMConfigItemListShow(
        ConfigItemIDs => $ConfigItemIDsRef,                  # total list of config item ids, that can be listed
        Total         => scalar @{ $ConfigItemIDsRef },      # total number of list items, config items in this case
        View          => $Self->{View},                      # optional, the default value is 'Small'
        Filter        => 'All',
        Filters       => \%NavBarFilter,
        FilterLink    => $LinkFilter,
        TitleName     => 'Overview: Config Item: Computer',
        TitleValue    => $Self->{Filter},
        Env           => $Self,
        LinkPage      => $LinkPage,
        LinkSort      => $LinkSort,
        Frontend      => 'Agent',                           # optional (Agent|Customer), default: Agent, indicates from which frontend this function was called
    );

=cut

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

    # take object ref to local, remove it from %Param (prevent memory leak)
    my $Env = delete $Param{Env};

    # lookup latest used view mode
    if ( !$Param{View} && $Self->{ 'UserITSMConfigItemOverview' . $Env->{Action} } ) {
        $Param{View} = $Self->{ 'UserITSMConfigItemOverview' . $Env->{Action} };
    }

    # set frontend
    my $Frontend = $Param{Frontend} || 'Agent';

    # set defaut view mode to 'small'
    my $View = $Param{View} || 'Small';

    # store latest view mode
    $Kernel::OM->Get('Kernel::System::AuthSession')->UpdateSessionID(
        SessionID => $Self->{SessionID},
        Key       => 'UserITSMConfigItemOverview' . $Env->{Action},
        Value     => $View,
    );

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

    # get backend from config
    my $Backends = $ConfigObject->Get('ITSMConfigItem::Frontend::Overview');
    if ( !$Backends ) {
        return $LayoutObject->FatalError(
            Message => 'Need config option ITSMConfigItem::Frontend::Overview',
        );
    }

    # check for hash-ref
    if ( ref $Backends ne 'HASH' ) {
        return $LayoutObject->FatalError(
            Message => 'Config option ITSMConfigItem::Frontend::Overview needs to be a HASH ref!',
        );
    }

    # check for config key
    if ( !$Backends->{$View} ) {
        return $LayoutObject->FatalError(
            Message => "No config option found for the view '$View'!",
        );
    }

    # nav bar
    my $StartHit = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam(
        Param => 'StartHit',
    ) || 1;

    # get personal page shown count
    my $PageShownPreferencesKey = 'UserConfigItemOverview' . $View . 'PageShown';
    my $PageShown               = $Self->{$PageShownPreferencesKey} || 10;
    my $Group                   = 'ConfigItemOverview' . $View . 'PageShown';

    # check start option, if higher then elements available, set
    # it to the last overview page (Thanks to Stefan Schmidt!)
    if ( $StartHit > $Param{Total} ) {
        my $Pages = int( ( $Param{Total} / $PageShown ) + 0.99999 );
        $StartHit = ( ( $Pages - 1 ) * $PageShown ) + 1;
    }

    # get data selection
    my %Data;
    my $Config = $ConfigObject->Get('PreferencesGroups');
    if ( $Config && $Config->{$Group} && $Config->{$Group}->{Data} ) {
        %Data = %{ $Config->{$Group}->{Data} };
    }

    # set page limit and build page nav
    my $Limit   = $Param{Limit} || 20_000;
    my %PageNav = $LayoutObject->PageNavBar(
        Limit     => $Limit,
        StartHit  => $StartHit,
        PageShown => $PageShown,
        AllHits   => $Param{Total} || 0,
        Action    => 'Action=' . $Env->{Action},
        Link      => $Param{LinkPage},
    );

    # build shown ticket a page
    $Param{RequestedURL}    = $Param{RequestedURL} || "Action=$Self->{Action};$Param{LinkPage}";
    $Param{Group}           = $Group;
    $Param{PreferencesKey}  = $PageShownPreferencesKey;
    $Param{PageShownString} = $Self->BuildSelection(
        Name        => $PageShownPreferencesKey,
        SelectedID  => $PageShown,
        Data        => \%Data,
        Translation => 0,
        Sort        => 'NumericValue',
        Class       => 'Modernize',
    );

    # build navbar content
    $LayoutObject->Block(
        Name => 'OverviewNavBar',
        Data => \%Param,
    );

    # back link
    if ( $Param{LinkBack} ) {
        $LayoutObject->Block(
            Name => 'OverviewNavBarPageBack',
            Data => \%Param,
        );

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

    # get filters
    if ( $Param{Filters} ) {

        # get given filters
        my @NavBarFilters;
        for my $Prio ( sort keys %{ $Param{Filters} } ) {
            push @NavBarFilters, $Param{Filters}->{$Prio};
        }

        # build filter content
        $LayoutObject->Block(
            Name => 'OverviewNavBarFilter',
            Data => {
                %Param,
            },
        );

        # loop over filters
        my $Count = 0;
        for my $Filter (@NavBarFilters) {

            # increment filter count and build filter item
            $Count++;
            $LayoutObject->Block(
                Name => 'OverviewNavBarFilterItem',
                Data => {
                    %Param,
                    %{$Filter},
                },
            );

            # filter is selected
            if ( $Filter->{Filter} eq $Param{Filter} ) {
                $LayoutObject->Block(
                    Name => 'OverviewNavBarFilterItemSelected',
                    Data => {
                        %Param,
                        %{$Filter},
                    },
                );

            }
            else {
                $LayoutObject->Block(
                    Name => 'OverviewNavBarFilterItemSelectedNot',
                    Data => {
                        %Param,
                        %{$Filter},
                    },
                );

            }
        }
    }

    # loop over configured backends
    for my $Backend ( sort keys %{$Backends} ) {

        # build navbar view mode
        $LayoutObject->Block(
            Name => 'OverviewNavBarViewMode',
            Data => {
                %Param,
                %{ $Backends->{$Backend} },
                Filter => $Param{Filter},
                View   => $Backend,
            },
        );

        # current view is configured in backend
        if ( $View eq $Backend ) {
            $LayoutObject->Block(
                Name => 'OverviewNavBarViewModeSelected',
                Data => {
                    %Param,
                    %{ $Backends->{$Backend} },
                    Filter => $Param{Filter},
                    View   => $Backend,
                },
            );
        }
        else {
            $LayoutObject->Block(
                Name => 'OverviewNavBarViewModeNotSelected',
                Data => {
                    %Param,
                    %{ $Backends->{$Backend} },
                    Filter => $Param{Filter},
                    View   => $Backend,
                },
            );
        }
    }

    # check if page nav is available
    if (%PageNav) {
        $LayoutObject->Block(
            Name => 'OverviewNavBarPageNavBar',
            Data => \%PageNav,
        );

        # don't show context settings in AJAX case (e. g. in customer ticket history),
        #   because the submit with page reload will not work there
        if ( !$Param{AJAX} ) {
            $LayoutObject->Block(
                Name => 'ContextSettings',
                Data => {
                    %PageNav,
                    %Param,
                },
            );
        }
    }

    # check if bulk feature is enabled
    my $BulkFeature = 0;
    if ( $ConfigObject->Get('ITSMConfigItem::Frontend::BulkFeature') ) {
        my @Groups;
        if ( $ConfigObject->Get('ITSMConfigItem::Frontend::BulkFeatureGroup') ) {
            @Groups = @{ $ConfigObject->Get('ITSMConfigItem::Frontend::BulkFeatureGroup') };
        }
        if ( !@Groups ) {
            $BulkFeature = 1;
        }
        else {
            GROUP:
            for my $Group (@Groups) {
                next GROUP if !$Kernel::OM->Get('Kernel::System::Group')->PermissionCheck(
                    UserID    => $Self->{UserID},
                    GroupName => $Group,
                    Type      => 'rw',
                );

                $BulkFeature = 1;
                last GROUP;
            }
        }
    }

    # show the bulk action button if feature is enabled
    if ($BulkFeature) {
        $LayoutObject->Block(
            Name => 'BulkAction',
            Data => {
                %PageNav,
                %Param,
            },
        );
    }

    # build html content
    my $OutputNavBar = $LayoutObject->Output(
        TemplateFile => 'AgentITSMConfigItemOverviewNavBar',
        Data         => {%Param},
    );

    # create output
    my $OutputRaw = '';
    if ( !$Param{Output} ) {
        $LayoutObject->Print(
            Output => \$OutputNavBar,
        );
    }
    else {
        $OutputRaw .= $OutputNavBar;
    }

    # load module
    if ( !$Kernel::OM->Get('Kernel::System::Main')->Require( $Backends->{$View}->{Module} ) ) {
        return $LayoutObject->FatalError();
    }

    # check for backend object
    my $Object = $Backends->{$View}->{Module}->new( %{$Env} );
    return if !$Object;

    # run module
    my $Output = $Object->Run(
        %Param,
        Limit     => $Limit,
        StartHit  => $StartHit,
        PageShown => $PageShown,
        AllHits   => $Param{Total} || 0,
        Frontend  => $Frontend,
    );

    # create output
    if ( !$Param{Output} ) {
        $LayoutObject->Print(
            Output => \$Output,
        );
    }
    else {
        $OutputRaw .= $Output;
    }

    # create overview nav bar
    $LayoutObject->Block(
        Name => 'OverviewNavBar',
        Data => {%Param},
    );

    # return content if available
    return $OutputRaw;
}

=head2 XMLData2Hash()

returns a hash reference with the requested attributes data for a config item

Return

    $Data = {
        'HardDisk::2' => {
            Value => 'HD2',
            Name  => 'Hard Disk',
         },
        'CPU::1' => {
            Value => '',
            Name  => 'CPU',
        },
        'HardDisk::2::Capacity::1' => {
            Value => '780 GB',
            Name  => 'Capacity',
        },
    };

    my $Data = $LayoutObject->XMLData2Hash(
        XMLDefinition => $Version->{XMLDefinition},
        XMLData       => $Version->{XMLData}->[1]->{Version}->[1],
        Attributes    => ['CPU::1', 'HardDrive::2::Capacity::1', ...],
        Data          => \%DataHashRef,                                 # optional
        Prefix        => 'HardDisk::1',                                 # optional
    );

=cut

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

    # check needed stuff
    return if !$Param{XMLData};
    return if !$Param{XMLDefinition};
    return if !$Param{Attributes};
    return if ref $Param{XMLData} ne 'HASH';
    return if ref $Param{XMLDefinition} ne 'ARRAY';
    return if ref $Param{Attributes} ne 'ARRAY';

    # to store the return data
    my $Data = $Param{Data} || {};

    # create a lookup structure
    my %RelevantAttributes = map { $_ => 1 } @{ $Param{Attributes} };

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        my $CountMax = $Item->{CountMax} || 1;

        COUNTER:
        for my $Counter ( 1 .. $CountMax ) {

            # add prefix
            my $Prefix = $Item->{Key} . '::' . $Counter;
            if ( $Param{Prefix} ) {
                $Prefix = $Param{Prefix} . '::' . $Prefix;
            }

            # skip not needed elements and sub elements
            next COUNTER if !grep { $_ =~ m{\A$Prefix} } @{ $Param{Attributes} };

            # lookup value
            my $Value = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->XMLValueLookup(
                Item  => $Item,
                Value => $Param{XMLData}->{ $Item->{Key} }->[$Counter]->{Content} // '',
            );

            # only if value is defined
            if ( defined $Value ) {

                # create output string
                $Value = $Self->ITSMConfigItemOutputStringCreate(
                    Value => $Value,
                    Item  => $Item,
                    Print => $Param{Print},
                );
            }

            if ( $RelevantAttributes{$Prefix} ) {

                # store the item in hash
                $Data->{$Prefix} = {
                    Name  => $Item->{Name},
                    Value => $Value // '',
                };
            }

            # start recursion, if "Sub" was found
            if ( $Item->{Sub} ) {
                $Data = $Self->XMLData2Hash(
                    XMLDefinition => $Item->{Sub},
                    XMLData       => $Param{XMLData}->{ $Item->{Key} }->[$Counter],
                    Prefix        => $Prefix,
                    Data          => $Data,
                    Attributes    => $Param{Attributes},
                );
            }
        }
    }

    return $Data;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Output::HTML::LinkObject::ITSMConfigItem;

use strict;
use warnings;

use Kernel::Output::HTML::Layout;

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::HTMLUtils',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::Log',
    'Kernel::System::Web::Request',
);

=head1 NAME

Kernel::Output::HTML::LinkObject::ITSMConfigItem - layout backend module

=head1 DESCRIPTION

All layout functions of link object (config item)

=head2 new()

create an object

    $BackendObject = Kernel::Output::HTML::LinkObject::ITSMConfigItem->new(
        UserLanguage => 'en',
        UserID       => 1,
    );

=cut

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

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

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

    # We need our own LayoutObject instance to avoid blockdata collisions
    #   with the main page.
    $Self->{LayoutObject} = Kernel::Output::HTML::Layout->new( %{$Self} );

    # define needed variables
    $Self->{ObjectData} = {
        Object     => 'ITSMConfigItem',
        Realname   => 'ConfigItem',
        ObjectName => 'SourceObjectID',
    };

    return $Self;
}

=head2 TableCreateComplex()

return an array with the block data

Return

    @BlockData = (
        {

            ObjectName  => 'ConfigItemID',
            ObjectID    => '123',

            Object    => 'ITSMConfigItem',
            Blockname => 'ConfigItem Computer',
            Headline  => [
                {
                    Content => '',
                    Width   => 20,
                },
                {
                    Content => 'ConfigItem#',
                    Width   => 100,
                },
                {
                    Content => 'Name',
                },
                {
                    Content => 'Deployment State',
                    Width   => 130,
                },
                {
                    Content => 'Created',
                    Width   => 130,
                },
            ],
            ItemList => [
                [
                    {
                        Type             => 'CurInciSignal',
                        Key              => '123',
                        Content          => 'Incident',
                        CurInciStateType => 'incident',
                    },
                    {
                        Type    => 'Link',
                        Content => '123',
                        Link    => 'Action=AgentITSMConfigItemZoom;ConfigItemID=123',
                    },
                    {
                        Type      => 'Text',
                        Content   => 'The Name of the Config Item',
                        MaxLength => 50,
                    },
                    {
                        Type      => 'Text',
                        Content   => 'In Repair',
                        Translate => 1,
                    },
                    {
                        Type    => 'TimeLong',
                        Content => '2008-01-01 12:12:00',
                    },
                ],
                [
                    {
                        Type             => 'CurInciSignal',
                        Key              => '234',
                        Content          => 'Incident',
                        CurInciStateType => 'incident',
                    },
                    {
                        Type    => 'Link',
                        Content => '234',
                        Link    => 'Action=AgentITSMConfigItemZoom;ConfigItemID=234',
                    },
                    {
                        Type      => 'Text',
                        Content   => 'The Name of the Config Item 234',
                        MaxLength => 50,
                    },
                    {
                        Type      => 'Text',
                        Content   => 'Productive',
                        Translate => 1,
                    },
                    {
                        Type    => 'TimeLong',
                        Content => '2007-11-11 12:12:00',
                    },
                ],
            ],
        },
    );

    @BlockData = $LinkObject->TableCreateComplex(
        ObjectLinkListWithData => $ObjectLinkListRef,
    );

=cut

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

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

    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
    my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    # get and remember the Deployment state colors
    my $DeploymentStatesList = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    ITEMID:
    for my $ItemID ( sort keys %{$DeploymentStatesList} ) {

        # get deployment state preferences
        my %Preferences = $GeneralCatalogObject->GeneralCatalogPreferencesGet(
            ItemID => $ItemID,
        );

        # check if a color is defined in preferences
        next ITEMID if !$Preferences{Color};

        # add color style definition
        my $DeplState = $DeploymentStatesList->{$ItemID};

        # remove any non ascii word characters
        $DeplState =~ s{ [^a-zA-Z0-9] }{_}msxg;

        # covert to lower case
        $Self->{DeplStateColors}->{$DeplState} = lc $Preferences{Color};
    }

    # Get the columns config.
    my $ColumnsConfig = $Kernel::OM->Get('Kernel::Config')->Get('LinkObject::ITSMConfigItem::ShowColumns');

    # Get the columns by class config.
    my $ColumnsByClassConfig
        = $Kernel::OM->Get('Kernel::Config')->Get('LinkObject::ITSMConfigItem::ShowColumnsByClass');

    # Get configured columns and reorganize them by class name.
    my %ColumnByClass;
    my %SignalColumnList;
    if ( $ColumnsByClassConfig && ref $ColumnsByClassConfig eq 'ARRAY' && @{$ColumnsByClassConfig} ) {

        NAME:
        for my $Name ( @{$ColumnsByClassConfig} ) {
            my ( $Class, $Column ) = split /::/, $Name, 2;

            next NAME if !$Column;

            # If signal columns are configured just for certain classes, add them to the beginning of ItemColumns array,
            # not to the array with other columns.
            if ( $Column eq 'CurInciSignal' || $Column eq 'CurDeplSignal' ) {
                $SignalColumnList{$Class}->{$Column} = 1;
                next NAME;
            }

            push @{ $ColumnByClass{$Class} }, $Column;
        }
    }

    # convert the list
    my %LinkList;
    for my $LinkType ( sort keys %{ $Param{ObjectLinkListWithData} } ) {

        # extract link type List
        my $LinkTypeList = $Param{ObjectLinkListWithData}->{$LinkType};

        for my $Direction ( sort keys %{$LinkTypeList} ) {

            # extract direction list
            my $DirectionList = $Param{ObjectLinkListWithData}->{$LinkType}->{$Direction};

            CONFIGITEMID:
            for my $ConfigItemID ( sort keys %{$DirectionList} ) {

                # extract class
                my $Class = $DirectionList->{$ConfigItemID}->{Class} || '';

                next CONFIGITEMID if !$Class;

                $LinkList{$Class}->{$ConfigItemID}->{Data} = $DirectionList->{$ConfigItemID};
            }
        }
    }

    my @BlockData;
    for my $Class ( sort { lc $a cmp lc $b } keys %LinkList ) {

        # extract config item data
        my $ConfigItemList = $LinkList{$Class};

        # to store the column headline
        my @ShowColumnsHeadlines;

        # create the item list
        my @ItemList;
        for my $ConfigItemID (
            sort { $ConfigItemList->{$a}->{Data}->{Name} cmp $ConfigItemList->{$b}->{Data}->{Name} }
            keys %{$ConfigItemList}
            )
        {
            my $ConfigItemData = $ConfigItemObject->ConfigItemGet(
                ConfigItemID => $ConfigItemID,
            );

            # extract version data
            my $Version = $ConfigItemList->{$ConfigItemID}->{Data};

            # make sure the column headline array is empty for each loop
            @ShowColumnsHeadlines = ();

            # get the version data, including all the XML data
            my $VersionXMLData = $ConfigItemObject->VersionGet(
                ConfigItemID => $ConfigItemID,
                XMLDataGet   => 1,
            );

            my @ItemColumns = (
                {
                    Type    => 'Link',
                    Content => $Version->{Number},
                    Link    => $Self->{LayoutObject}->{Baselink}
                        . 'Action=AgentITSMConfigItemZoom;ConfigItemID='
                        . $ConfigItemID,
                    Title => "ConfigItem# $Version->{Number} ($Version->{Class}): $Version->{Name}",
                },
            );

            # Add columns from 'LinkObject::ITSMConfigItem::ShowColumns' setting to the current CI class
            # if it does not contain them. Signal columns is saved in hash because they will be added later.
            COLUMNCONFIG:
            for my $Column ( @{$ColumnsConfig} ) {

                if ( $Column eq 'CurInciSignal' || $Column eq 'CurDeplSignal' ) {
                    $SignalColumnList{$Class}->{$Column} = 1;
                }
                elsif ( !grep { $_ eq $Column } @{ $ColumnByClass{$Class} } ) {
                    push @{ $ColumnByClass{$Class} }, $Column;
                }
            }

            # Add signal columns to the beginning of the array if needed.
            if ( $SignalColumnList{$Class}->{CurDeplSignal} ) {
                unshift @ItemColumns, {
                    Type    => 'CurDeplSignal',
                    Key     => $ConfigItemID,
                    Content => $Version->{CurDeplState},
                };
            }
            if ( $SignalColumnList{$Class}->{CurInciSignal} ) {
                unshift @ItemColumns, {
                    Type             => 'CurInciSignal',
                    Key              => $ConfigItemID,
                    Content          => $Version->{CurInciState},
                    CurInciStateType => $Version->{CurInciStateType},
                };
            }

            # these columns will be added if no class based column config is defined
            my @AdditionalDefaultItemColumns = (
                {
                    Type      => 'Text',
                    Content   => $Version->{Name},
                    MaxLength => 50,
                },
                {
                    Type      => 'Text',
                    Content   => $Version->{CurDeplState},
                    Translate => 1,
                },
                {
                    Type    => 'TimeLong',
                    Content => $ConfigItemData->{CreateTime},
                },
            );

            # individual column config for this class exists
            if ( $ColumnByClass{$Class} ) {

                # convert the XML data into a hash
                my $ExtendedVersionData = $Self->{LayoutObject}->XMLData2Hash(
                    XMLDefinition => $VersionXMLData->{XMLDefinition},
                    XMLData       => $VersionXMLData->{XMLData}->[1]->{Version}->[1],
                    Attributes    => $ColumnByClass{$Class},
                );

                COLUMN:
                for my $Column ( @{ $ColumnByClass{$Class} } ) {

                    # process some non-xml attributes
                    if ( $Version->{$Column} ) {

                        # handle the CI name
                        if ( $Column eq 'Name' ) {

                            # add the column
                            push @ItemColumns, {
                                Type      => 'Text',
                                Content   => $Version->{Name},
                                MaxLength => 50,
                            };

                            # add the headline
                            push @ShowColumnsHeadlines, {
                                Content => 'Name',
                            };
                        }

                        # special translation handling
                        elsif ( $Column eq 'CurDeplState' ) {

                            # add the column
                            push @ItemColumns, {
                                Type      => 'Text',
                                Content   => $Version->{$Column},
                                Translate => 1,
                            };

                            # add the headline
                            push @ShowColumnsHeadlines, {
                                Content => 'Deployment State',
                            };
                        }

                        # special translation handling
                        elsif ( $Column eq 'CurInciState' ) {

                            # add the column
                            push @ItemColumns, {
                                Type      => 'Text',
                                Content   => $Version->{$Column},
                                Translate => 1,
                            };

                            # add the headline
                            push @ShowColumnsHeadlines, {
                                Content => 'Incident State',
                            };
                        }

                        # special translation handling
                        elsif ( $Column eq 'Class' ) {

                            # add the column
                            push @ItemColumns, {
                                Type      => 'Text',
                                Content   => $Version->{$Column},
                                Translate => 1,
                            };

                            # add the headline
                            push @ShowColumnsHeadlines, {
                                Content => 'Class',
                            };
                        }

                        # special date/time handling
                        elsif ( $Column eq 'CreateTime' ) {

                            # add the column
                            push @ItemColumns, {
                                Type    => 'TimeLong',
                                Content => $Version->{CreateTime},
                            };

                            # add the headline
                            push @ShowColumnsHeadlines, {
                                Content => 'Created',
                            };
                        }

                        next COLUMN;
                    }

                    # convert to ascii text in case the value contains html
                    my $Value = $Kernel::OM->Get('Kernel::System::HTMLUtils')->ToAscii(
                        String => $ExtendedVersionData->{$Column}->{Value},
                    ) || '';

                    # convert all whitespace and newlines to single spaces
                    $Value =~ s{ \s+ }{ }gxms;

                    # add the column
                    push @ItemColumns, {
                        Type    => 'Text',
                        Content => $Value,
                    };

                    # add the headline
                    push @ShowColumnsHeadlines, {
                        Content => $ExtendedVersionData->{$Column}->{Name} || '',
                    };
                }
            }

            # individual column config for this class does not exist,
            # so the default columns will be used
            else {

                # add the default columns
                push @ItemColumns, @AdditionalDefaultItemColumns;

                # add the default column headlines
                @ShowColumnsHeadlines = (
                    {
                        Content => 'Name',
                    },
                    {
                        Content => 'Deployment State',
                        Width   => 130,
                    },
                    {
                        Content => 'Created',
                        Width   => 130,
                    },
                );
            }

            push @ItemList, \@ItemColumns;
        }

        return if !@ItemList;

        # define the block data
        my %Block = (
            Object    => $Self->{ObjectData}->{Object},
            Blockname => $Self->{ObjectData}->{Realname} . ' (' . $Class . ')',
            Headline  => [
                {
                    Content => 'ConfigItem#',
                    Width   => 100,
                },
            ],
            ItemList => \@ItemList,
        );

        # Add 'CurInciSignal' and 'CurDeplSignal' columns to the beginning of headline if needed.
        if ( $SignalColumnList{$Class}->{CurDeplSignal} ) {
            unshift @{ $Block{Headline} }, {
                Content => 'Deployment State',
                Width   => 20,
            };
        }
        if ( $SignalColumnList{$Class}->{CurInciSignal} ) {
            unshift @{ $Block{Headline} }, {
                Content => 'Incident State',
                Width   => 20,
            };
        }

        # add the column headlines
        push @{ $Block{Headline} }, @ShowColumnsHeadlines;

        push @BlockData, \%Block;
    }

    return @BlockData;
}

=head2 TableCreateSimple()

return a hash with the link output data

Return

    %LinkOutputData = (
        Normal::Source => {
            ITSMConfigItem => [
                {
                    Type    => 'Link',
                    Content => 'CI:55555',
                    Title   => 'ConfigItem# 555555: The config item name',
                    Css     => 'style="text-decoration: line-through"',
                },
                {
                    Type    => 'Link',
                    Content => 'CI:22222',
                    Title   => 'ConfigItem# 22222: Title of config name 22222',
                },
            ],
        },
        ParentChild::Target => {
            ITSMConfigItem => [
                {
                    Type    => 'Link',
                    Content => 'CI:77777',
                    Title   => 'ConfigItem# 77777: ConfigItem name',
                },
            ],
        },
    );

    %LinkOutputData = $LinkObject->TableCreateSimple(
        ObjectLinkListWithData => $ObjectLinkListRef,
    );

=cut

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

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

    my %LinkOutputData;
    for my $LinkType ( sort keys %{ $Param{ObjectLinkListWithData} } ) {

        # extract link type List
        my $LinkTypeList = $Param{ObjectLinkListWithData}->{$LinkType};

        for my $Direction ( sort keys %{$LinkTypeList} ) {

            # extract direction list
            my $DirectionList = $Param{ObjectLinkListWithData}->{$LinkType}->{$Direction};

            my @ItemList;
            for my $ConfigItemID ( sort { $a <=> $b } keys %{$DirectionList} ) {

                # extract config item data
                my $Version = $DirectionList->{$ConfigItemID};

                # define item data
                my %Item = (
                    Type    => 'Link',
                    Content => 'CI:' . $Version->{Number},
                    Title   => "ConfigItem# $Version->{Number} ($Version->{Class}): $Version->{Name}",
                    Link    => $Self->{LayoutObject}->{Baselink}
                        . 'Action=AgentITSMConfigItemZoom;ConfigItemID='
                        . $ConfigItemID,
                );

                push @ItemList, \%Item;
            }

            # add item list to link output data
            $LinkOutputData{ $LinkType . '::' . $Direction }->{ITSMConfigItem} = \@ItemList;
        }
    }

    return %LinkOutputData;
}

=head2 ContentStringCreate()

return a output string

    my $String = $LayoutObject->ContentStringCreate(
        ContentData => $HashRef,
    );

=cut

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

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

    # extract content
    my $Content = $Param{ContentData};

    if ( $Content->{Type} ne 'CurInciSignal' && $Content->{Type} ne 'CurDeplSignal' ) {
        return;
    }

    my $String;
    if ( $Content->{Type} eq 'CurInciSignal' ) {

        # set incident signal
        my %InciSignals = (
            incident    => 'redled',
            operational => 'greenled',
            unknown     => 'grayled',
            warning     => 'yellowled',
        );

        # investigate current incident signal
        $Content->{CurInciStateType} ||= 'unknown';
        my $CurInciSignal = $InciSignals{ $Content->{CurInciStateType} };
        $CurInciSignal ||= $InciSignals{unknown};

        $String = $Self->{LayoutObject}->Output(
            Template => '<div class="Flag Small" title="[% Translate(Data.CurInciState) | html %]"> '
                . '<span class="[% Data.CurInciSignal | html %]"></span> </div>',
            Data => {
                CurInciSignal => $CurInciSignal,
                CurInciState  => $Content->{Content} || '',
            },
        );
    }
    elsif ( $Content->{Type} eq 'CurDeplSignal' ) {

        # convert deployment state to a web safe CSS class
        my $DeplState = $Content->{Content} || '';
        $DeplState =~ s{ [^a-zA-Z0-9] }{_}msxg;

        # get the color of the deplyment state if defined
        my $DeplStateColor = $Self->{DeplStateColors}->{$DeplState} || '';

        my $Template = '<div class="Flag Small" title="[% Translate(Data.CurDeplState) | html %]"> ';

        # check if color is defined and set the style class
        if ($DeplStateColor) {
            $Template .= << "END";
<style>
    .Flag span.$DeplState {
        background-color: $DeplStateColor;
    }
</style>
END
        }

        $Template .= "<span class=\"DeplState $DeplState\"></span> </div>";

        $String = $Self->{LayoutObject}->Output(
            Template => $Template,
            Data     => {
                CurDeplState => $Content->{Content} || '',
            },
        );
    }

    return $String;
}

=head2 SelectableObjectList()

return an array hash with C<selectable> objects

Return

    @SelectableObjectList = (
        {
            Key      => '-',
            Value    => 'ConfigItem',
            Disabled => 1,
        },
        {
            Key   => 'ITSMConfigItem::25',
            Value => 'ConfigItem::Computer',
        },
        {
            Key   => 'ITSMConfigItem::26',
            Value => 'ConfigItem::Software',
        },
        {
            Key   => 'ITSMConfigItem::27',
            Value => 'ConfigItem::Network',
        },
    );

    @SelectableObjectList = $LinkObject->SelectableObjectList(
        Selected => $Identifier,  # (optional)
    );

=cut

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

    # define headline
    my @ObjectSelectList;

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    return if !$ClassList;
    return if ref $ClassList ne 'HASH';

    # get the config with the default subobjects
    my $DefaultSubobject = $Kernel::OM->Get('Kernel::Config')->Get('LinkObject::DefaultSubObject') || {};

    CLASSID:
    for my $ClassID ( sort { lc $ClassList->{$a} cmp lc $ClassList->{$b} } keys %{$ClassList} ) {

        # show class only if user has access rights
        my $HasAccess = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->Permission(
            Scope   => 'Class',
            ClassID => $ClassID,
            UserID  => $Self->{UserID},
            Type    => 'ro',
        );

        next CLASSID if !$HasAccess;

        my $Class      = $ClassList->{$ClassID} || '';
        my $Identifier = $Self->{ObjectData}->{Object} . '::' . $ClassID;

        # set selected flag
        my $Selected;
        if ( $Param{Selected} ) {

            if ( $Param{Selected} eq $Identifier ) {
                $Selected = 1;
            }
            elsif (
                $Param{Selected} eq $Self->{ObjectData}->{Object}
                && $DefaultSubobject->{ $Self->{ObjectData}->{Object} }
                )
            {

                # extract default class name
                my $DefaultClass = $DefaultSubobject->{ $Self->{ObjectData}->{Object} } || '';

                # check class
                if ( $DefaultClass eq $Class ) {
                    $Selected = 1;
                }
            }
        }

        # create row
        my %Row = (
            Key      => $Identifier,
            Value    => $Self->{ObjectData}->{Realname} . '::' . $Class,
            Selected => $Selected,
        );

        push @ObjectSelectList, \%Row;
    }

    # only add headline if there are configitem classes
    # where the user has the permission to use them
    if (@ObjectSelectList) {

        # add headline as first array element
        unshift @ObjectSelectList, {
            Key      => '-',
            Value    => $Self->{ObjectData}->{Realname},
            Disabled => 1,
        };
    }

    return @ObjectSelectList;
}

=head2 SearchOptionList()

return an array hash with search options

Return

    @SearchOptionList = (
        {
            Key       => 'Number',
            Name      => 'ConfigItem#',
            InputStrg => $FormString,
            FormData  => '1234',
        },
        {
            Key       => 'Name',
            Name      => 'Name',
            InputStrg => $FormString,
            FormData  => 'BlaBla',
        },
    );

    @SearchOptionList = $LinkObject->SearchOptionList(
        SubObject => '25',  # (optional)
    );

=cut

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

    # search option list
    my @SearchOptionList = (
        {
            Key  => 'Number',
            Name => 'ConfigItem#',
            Type => 'Text',
        },
        {
            Key  => 'Name',
            Name => 'Name',
            Type => 'Text',
        },
        {
            Key  => 'DeplStateIDs',
            Name => 'Deployment State',
            Type => 'List',
        },
        {
            Key  => 'InciStateIDs',
            Name => 'Incident State',
            Type => 'List',
        },
    );

    # add object dependence attributes
    #if ( $Param{SubObject} ) {
    #
    #    # get class list
    #    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
    #        Class => 'ITSM::ConfigItem::Class',
    #    );
    #
    #    if ( $ClassList && $ClassList eq 'HASH' ) {
    #
    #        # add here the search attributes of the subobject!
    #    }
    #}

    # add formkey
    for my $Row (@SearchOptionList) {
        $Row->{FormKey} = 'SEARCH::' . $Row->{Key};
    }

    # add form data and input string
    ROW:
    for my $Row (@SearchOptionList) {

        # prepare text input fields
        if ( $Row->{Type} eq 'Text' ) {

            # get form data
            $Row->{FormData} = $Kernel::OM->Get('Kernel::System::Web::Request')->GetParam( Param => $Row->{FormKey} );

            # parse the input text block
            $Self->{LayoutObject}->Block(
                Name => 'InputText',
                Data => {
                    Key   => $Row->{FormKey},
                    Value => $Row->{FormData} || '',
                },
            );

            # add the input string
            $Row->{InputStrg} = $Self->{LayoutObject}->Output(
                TemplateFile => 'LinkObject',
            );

            next ROW;
        }

        # prepare list boxes
        if ( $Row->{Type} eq 'List' ) {

            # get form data
            my @FormData = $Kernel::OM->Get('Kernel::System::Web::Request')->GetArray( Param => $Row->{FormKey} );
            $Row->{FormData} = \@FormData;

            # prepare deployment state list
            my %ListData;
            if ( $Row->{Key} eq 'DeplStateIDs' ) {

                # get deployment state list
                my $DeplStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
                    Class => 'ITSM::ConfigItem::DeploymentState',
                );

                # add list
                if ( $DeplStateList && ref $DeplStateList eq 'HASH' ) {
                    %ListData = %{$DeplStateList};
                }
            }

            # prepare incident state list
            elsif ( $Row->{Key} eq 'InciStateIDs' ) {

                # get incident state list
                my $InciStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
                    Class => 'ITSM::Core::IncidentState',
                );

                # add list
                if ( $InciStateList && ref $InciStateList eq 'HASH' ) {
                    %ListData = %{$InciStateList};
                }
            }

            # add the input string
            $Row->{InputStrg} = $Self->{LayoutObject}->BuildSelection(
                Data       => \%ListData,
                Name       => $Row->{FormKey},
                SelectedID => $Row->{FormData},
                Size       => 3,
                Multiple   => 1,
                Class      => 'Modernize',
            );

            next ROW;
        }
    }

    return @SearchOptionList;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --
<div class="MainBox ARIARoleMain LayoutFixedSidebar SidebarFirst">
    <h1 class="InvisibleText">
        [% Translate("Dynamic Fields") | html %] - [% Translate(Data.ObjectTypeName) | html %]:
        [% USE Title = String(Data.BreadcrumbText) %]
        [% Title | html %]
    </h1>

    [% BreadcrumbPath = [
            {
                Name => Translate('Dynamic Fields Management'),
                Link => 'AdminDynamicField',
            },
        ]
    %]

    [% USE BreadcrumbName = String(Translate(Data.ObjectTypeName)) %]

    [% SWITCH Data.Mode %]
        [% CASE 'Add' %]
            [% BreadcrumbPath.push({ Name => BreadcrumbName.append( ': ', Title ) }) %]
        [% CASE 'Change' %]
            [% BreadcrumbPath.push({ Name => BreadcrumbName.append( ': ', Title, ' - ', Data.Name ) }) %]
    [% END %]

    [% INCLUDE "Breadcrumb.tt" Path = BreadcrumbPath %]
    <div class="Clear"></div>

    <div class="SidebarColumn">
        <div class="WidgetSimple">
            <div class="Header">
                <h2>[% Translate("Actions") | html %]</h2>
            </div>
            <div class="Content">
                <ul class="ActionList">
                    <li>
                        <a href="[% Env("Baselink") %]Action=AdminDynamicField" class="CallForAction"><span>[% Translate("Go back to overview") | html %]</span></a>
                    </li>
                </ul>
            </div>
        </div>
    </div>
    <div class="ContentColumn">
        <form action="[% Env("CGIHandle") %]" method="post" class="Validate PreventMultipleSubmits">
            <input type="hidden" name="Action" value="AdminDynamicFieldConfigItem">
            <input type="hidden" name="Subaction" value="[% Data.Mode | html %]Action">
            <input type="hidden" name="ObjectType" value="[% Data.ObjectType | html %]">
            <input type="hidden" name="FieldType" value="[% Data.FieldType | html %]">
            <input type="hidden" name="ID" value="[% Data.ID | html %]">
            [% IF Data.Mode == 'Change' %]
                <input type="hidden" name="ContinueAfterSave" id="ContinueAfterSave" value=""/>
            [% END %]

            <div class="WidgetSimple">
                <div class="Header">
                    <h2>[% Translate("General") | html %]</h2>
                </div>
                <div class="Content">
                    <div class="LayoutGrid ColumnsWithSpacing">
                        <div class="Size1of2">
                            <fieldset class="TableLike">
                                <label class="Mandatory" for="Name"><span class="Marker">*</span> [% Translate("Name") | html %]:</label>
                                <div class="Field">
                                    <input id="Name" class="W50pc [% Data.NameServerError | html %] [% Data.ShowWarning | html %]  Validate_Alphanumeric" type="text" maxlength="200" value="[% Data.Name | html %]" name="Name" [% Data.ReadonlyInternalField | html %]>
                                    <div id="NameError" class="TooltipErrorMessage"><p>[% Translate("This field is required and the value must be alphanumeric.") | html %]</p></div>
                                    <div id="NameServerError" class="TooltipErrorMessage"><p>[% Translate(Data.NameServerErrorMessage) | html %]</p></div>
                                    <p class="FieldExplanation">[% Translate("Must be unique and only accepts alphanumeric characters.") | html %]</p>
                                    <p class="Warning Hidden">[% Translate("Changing this value will require manual changes in the system.") | html %]</p>
                                </div>
                                <div class="Clear"></div>

                                <label class="Mandatory" for="Label"><span class="Marker">*</span> [% Translate("Label") | html %]:</label>
                                <div class="Field">
                                    <input id="Label" class="W50pc [% Data.LabelServerError | html %] Validate_Required" type="text" maxlength="200" value="[% Data.Label | html %]" name="Label">
                                    <div id="LabelError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                                    <div id="LabelServerError" class="TooltipErrorMessage"><p>[% Translate(Data.LabelServerErrorMessage) | html %]</p></div>
                                    <p class="FieldExplanation">[% Translate("This is the name to be shown on the screens where the field is active.") | html %]</p>
                                </div>
                                <div class="Clear"></div>

                                <label class="Mandatory" for="FieldOrder"><span class="Marker">*</span> [% Translate("Field order") | html %]:</label>
                                <div class="Field">
                                    [% Data.DynamicFieldOrderStrg %]
                                    <div id="FieldOrderError" class="TooltipErrorMessage"><p>[% Translate("This field is required and must be numeric.") | html %]</p></div>
                                    <div id="FieldOrderServerError" class="TooltipErrorMessage"><p>[% Translate(Data.FieldOrderServerErrorMessage) | html %]</p></div>
                                    <p class="FieldExplanation">[% Translate("This is the order in which this field will be shown on the screens where it is active.") | html %]</p>
                                </div>
                                <div class="Clear"></div>
                            </fieldset>
                        </div>
                        <div class="Size1of2">
                            <fieldset class="TableLike">
                                <label for="ValidID">[% Translate("Validity") | html %]:</label>
                                <div class="Field">
                                    [% Data.ValidityStrg %]
                                </div>
                                <div class="Clear"></div>

                                <div class="SpacingTop"></div>
                                <label for="FieldTypeName">[% Translate("Field type") | html %]:</label>
                                <div class="Field">
                                    <input id="FieldTypeName" readonly class="W50pc" type="text" maxlength="200" value="[% Translate(Data.FieldTypeName) | html %]" name="FieldTypeName">
                                    <div class="Clear"></div>
                                </div>

                                <div class="SpacingTop"></div>
                                <label for="ObjectTypeName">[% Translate("Object type") | html %]:</label>
                                <div class="Field">
                                    <input id="ObjectTypeName" readonly class="W50pc" type="text" maxlength="200" value="[% Translate(Data.ObjectTypeName) | html %]" name="ObjectTypeName">
                                    <div class="Clear"></div>
                                </div>

[% RenderBlockStart("InternalField") %]
                                <div class="SpacingTop"></div>
                                <label for="InternalField">[% Translate("Internal field") | html %]:</label>
                                <div class="Field">
                                    <input id="InternalField" readonly class="W50pc" type="text" maxlength="1" value="[% Data.InternalField | html %]" name="InternalField">
                                    <p class="FieldExplanation">
                                        [% Translate("This field is protected and can't be deleted.") | html %]
                                    </p>
                                    <div class="Clear"></div>
                                </div>
[% RenderBlockEnd("InternalField") %]
                            </fieldset>
                        </div>
                    </div>
                </div>
            </div>
            <div class="WidgetSimple">
                <div class="Header">
                    <h2>[% Translate("Field Settings") | html %]: [% Translate(Data.FieldTypeName) | html %]</h2>
                </div>
                <div class="Content">
                    <fieldset class="TableLike">
[% INCLUDE "AdminDynamicFieldConfigItem/Config.tt" %]
                    </fieldset>
                </div>
            </div>
            <fieldset class="TableLike">
                <div class="Field SpacingTop">
                    [% IF Data.Mode == 'Change' %]
                        <button class="CallForAction Primary" id="SubmitAndContinue" type="button" value="[% Translate("Save") | html %]"><span>[% Translate("Save") | html %]</span></button>
                        [% Translate("or") | html %]
                        <button class="CallForAction Primary" id="Submit" type="submit" value="[% Translate("Save") | html %]"><span>[% Translate("Save and finish") | html %]</span></button>
                    [% ELSE %]
                        <button class="CallForAction Primary" id="Submit" type="submit" value="[% Translate("Save") | html %]"><span>[% Translate("Save") | html %]</span></button>
                    [% END %]
                    [% Translate("or") | html %]
                    <a href="[% Env("Baselink") %]Action=AdminDynamicField"><span>[% Translate("Cancel") | html %]</span></a>
                </div>
                <div class="Clear"></div>
            </fieldset>
        </form>
    </div>
</div>

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

<label class="Mandatory" for="ConfigItemClass"><span class="Marker">*</span> [% Translate("Config item class") | html %]:</label>
<div class="Field">
    [% Data.ConfigItemClassStrg %]
    [% IF Data.SelectedConfigItemClassIsInvalid %]
        <p class="FieldExplanation">[% Translate("The selected config item class is invalid or does not exist.") | html %]</p>
    [% END %]
    <div id="ConfigItemClassError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
    <div id="ConfigItemClassServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
</div>
<div class="Clear"></div>

<div class="SpacingTop"></div>
<label for="DeplStateIDs">[% Translate("Config item deployment states") | html %]:</label>
<div class="Field">
    [% Data.DeplStateSelectionHTML %]
</div>
<div class="Clear"></div>

<div class="SpacingTop"></div>
<label for="ConfigItemLinkType">[% Translate("Config item link type") | html %]:</label>
<div class="Field">
    [% Data.ConfigItemLinkTypeSelectionHTML %]
    <p class="FieldExplanation">[% Translate("Select a link type to enable linking between the ticket and the selected config items. Note that linking is only availabe for dynamic fields of tickets.") | html %]</p>
</div>
<div class="Clear"></div>

<div class="SpacingTop"></div>
<label for="ConfigItemLinkSource">[% Translate("Config item link source") | html %]:</label>
<div class="Field">
    [% Data.ConfigItemLinkSourceSelectionHTML %]
</div>
<div class="Clear"></div>

<div class="SpacingTop"></div>
<label for="ConfigItemLinkRemoval">[% Translate("Config item link removal") | html %]:</label>
<div class="Field">
    [% Data.ConfigItemLinkRemovalSelectionHTML %]
    <p class="FieldExplanation">[% Translate("Activate this option to remove links between the ticket and config items that are removed from this field.") | html %]</p>
</div>
<div class="Clear"></div>

[% RenderBlockStart("AdditionalDFStorage") %]
<label>[% Translate("Additional dynamic field storage") | html %]:</label>
<div class="Field ValueInsert AdditionalDFStorageRow" data-formelement-restore-destination='AdditionalDFStorage'  data-formelement-add-destination="AdditionalDFStorage">
[% RenderBlockStart("AdditionalDFStorageValueCounter") %]
    <input type="hidden" name="AdditionalDFStorageValueCounter" data-formelement-add-counter="AdditionalDFStorageCounter" value="[% Data.AdditionalDFStorageValueCounter | html %]" id="AdditionalDFStorageValueCounter" class="ValueCounter" />
[% RenderBlockEnd("AdditionalDFStorageValueCounter") %]
[% RenderBlockStart("AdditionalDFStorageRow") %]
    <div class="ValueRow" data-formelement-remove-destination="AdditionalDFStorageRow_[% Data.AdditionalDFStorageValueCounter | html %]" >

        <label class="Mandatory" for="DynamicField_[% Data.AdditionalDFStorageValueCounter | html %]"><span class="Marker">*</span>[% Translate("Dynamic field") | html %]:</label>
        [% Data.DynamicFieldSelection %]
        <div id="DynamicField_[% Data.AdditionalDFStorageValueCounter | html %]Error" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
        <div id="DynamicField_[% Data.AdditionalDFStorageValueCounter | html %]ServerError" class="TooltipErrorMessage"><p>[% Translate(Data.DynamicFieldErrorMessage) | html %]</p></div>

        <label class="Mandatory" for="ConfigItemKey_[% Data.AdditionalDFStorageValueCounter | html %]"><span class="Marker">*</span>[% Translate("Config item key") | html %]:</label>
        <input name="ConfigItemKey_[% Data.AdditionalDFStorageValueCounter | html %]" id="ConfigItemKey_[% Data.AdditionalDFStorageValueCounter | html %]" class="DefaultValueItem Validate_Required DataTable VariableWidth [% Data.ConfigItemKeyError | html %]" type="text" maxlength="100" value="[% Data.ConfigItemKey | html %]"/>
        <div id="ConfigItemKey_[% Data.AdditionalDFStorageValueCounter | html %]Error" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
        <div id="ConfigItemKey_[% Data.AdditionalDFStorageValueCounter | html %]ServerError" class="TooltipErrorMessage"><p>[% Translate(Data.ConfigItemKeyErrorMessage) | html %]</p></div>

        [% Data.TypeOption %]

        <a href="#" id="RemoveValue_[% Data.AdditionalDFStorageValueCounter | html %]" class="RemoveButton ValueRemove" data-formelement-remove='AdditionalDFStorage'  data-formelement-remove-destination-name="AdditionalDFStorageRow_[% Data.AdditionalDFStorageValueCounter | html %]"><i class="fa fa-minus-square-o"></i><span class="InvisibleText">[% Translate("Remove value") | html %]</span></a>
        <div class="SpacingTopMini" ></div>
    </div>
[% RenderBlockEnd("AdditionalDFStorageRow") %]
[% RenderBlockStart("AdditionalDFStorageTemplate") %]
    <div class="ValueTemplate Hidden" data-formelement-add-source="AdditionalDFStorageTemplate">

        <label class="Mandatory" for="DynamicField"><span class="Marker">*</span>[% Translate("Dynamic field") | html %]:</label>
        [% Data.DynamicFieldSelectionTemplate %]
        <div id="DynamicFieldError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
        <div id="DynamicFieldServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>

        <label class="Mandatory" for="ConfigItemKey"><span class="Marker">*</span>[% Translate("Config item key") | html %]:</label>
        <input name="ConfigItemKey" id="ConfigItemKey" class="DefaultValueItem DataTable VariableWidth" type="text" maxlength="100" value=""/>
        <div id="ConfigItemKeyError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
        <div id="ConfigItemKeyServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>

        [% Data.TypeOption %]

        <a href="#" id="RemoveValue" class="RemoveButton ValueRemove" data-formelement-remove='AdditionalDFStorage' data-formelement-remove-destination-name="AdditionalDFStorageRow"><i class="fa fa-minus-square-o"></i><span class="InvisibleText">[% Translate("Remove value") | html %]</span></a>
        <div class="SpacingTopMini" ></div>
    </div>
[% RenderBlockEnd("AdditionalDFStorageTemplate") %]
</div>
<div class="Clear"></div>

<label for="AddValue">[% Translate("Add dynamic field") | html %]:</label>
<div class="Field">
    <a href="#" class="AddValue" id="AddValue" class="AddButton"
        data-formelement-add='AdditionalDFStorage'
        data-formelement-add-destination-name="AdditionalDFStorage"
        data-formelement-add-source-name="AdditionalDFStorageTemplate"
        data-formelement-add-counter-name="AdditionalDFStorageCounter"
        data-formelement-add-method='append'><i class="fa fa-plus-square-o"></i><span class="InvisibleText">[% Translate("Add Value") | html %]</span></a>
    <a href="#" id="Restore" class="Restore"
    data-formelement-restore
    data-formelement-restore-destination-name='AdditionalDFStorage'
    ><i class="fa fa-undo"></i><span class="InvisibleText">[% Translate("Restore values") | html %]</span></a>
    <p class="FieldExplanation">[% Translate("These dynamic fields will be filled with values of the same selected config item(s).") | html %]</p>
</div>
<div class="Clear SpacingBottom"></div>
[% RenderBlockEnd("AdditionalDFStorage") %]

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

[% RenderBlockStart("Overview") %]
<div class="MainBox ARIARoleMain LayoutFixedSidebar SidebarFirst">
    <h1>[% Translate("Config Item Management") | html %]</h1>

    [% BreadcrumbPath = [
            {
                Name => Translate('Config Item Management'),
                Link => 'AdminITSMConfigItem',
            },
        ]
    %]

    [% IF (Data.ClassSelected) %]
        [% BreadcrumbPath.push({
            Name => Data.ClassSelected.Name,
            Link => 'AdminITSMConfigItem;Subaction=DefinitionList;ClassID=' _ Data.ClassSelected.ID
        }) %]
    [% END %]

    [% IF (Data.VersionSelected) %]
        [% BreadcrumbPath.push({
            Name => 'Version ' _ Data.VersionSelected.Version,
            Link => 'AdminITSMConfigItem;Subaction=DefinitionView;DefinitionID=' _ Data.VersionSelected.ID
        }) %]
    [% END %]

    [% IF (Data.Edit) %]
        [% BreadcrumbPath.push({
            Name => 'Edit definition',
        }) %]
    [% END %]

    [% INCLUDE "Breadcrumb.tt" Path = BreadcrumbPath %]

    <div class="SidebarColumn">
        <div class="WidgetSimple">
            <div class="Header">
                <h2>[% Translate("Actions") | html %]</h2>
            </div>
            <div class='Content'>
                <ul class="ActionList">
                    <li>
                        <form action="[% Env("CGIHandle") %]" method="get">
                            <input type="hidden" name="Action" value="[% Env("Action") %]"/>
                            <input type="hidden" name="Subaction" value="DefinitionChange"/>

                            <div>[% Data.ClassOptionStrg %]</div>
                            <button class="CallForAction Fullsize Center" type="submit" value="[% Translate("Add") | html %]">
                                <span>[% Translate("Change class definition") | html %]</span>
                            </button>
                        </form>
                    </li>
[% RenderBlockStart("ActionOverview") %]
                    <li>
                        <a href="[% Env("Baselink") %]Action=[% Env("Action") %]" class="CallForAction Fullsize Center"><span><i class="fa fa-caret-left"></i>[% Translate("Go to overview") | html %]</span></a>
                    </li>
[% RenderBlockEnd("ActionOverview") %]
                </ul>
            </div>
        </div>
    </div>

    <div class="ContentColumn">
        <div class="WidgetSimple">
[% RenderBlockStart("OverviewList") %]
            <div class="Header">
                <h2>[% Translate("List") | html %]</h2>
            </div>
            <div class="Content">
                <table class="DataTable">
                    <thead>
                        <tr>
                            <th>[% Translate("Config Item Class") | html %]</th>
                        </tr>
                    </thead>
                    <tbody>
[% RenderBlockStart("OverviewListRow") %]
                        <tr>
                            <td>
                                <a class="AsBlock" href="[% Env("Baselink") %]Action=[% Env("Action") %];Subaction=DefinitionList;ClassID=[% Data.ClassID | uri %]">[% Data.Name | html %]</a>
                            </td>
                        </tr>
[% RenderBlockEnd("OverviewListRow") %]
                    </tbody>
                </table>
            </div>
[% RenderBlockEnd("OverviewList") %]
[% RenderBlockStart("DefinitionList") %]
            <div class="Header">
                <h2>[% Translate("List") | html %]</h2>
            </div>
            <div class="Content">
                <table class="DataTable">
                    <thead>
                        <tr>
                            <th class="W50pc">[% Translate("Config Item Class") | html %]</th>
                            <th>[% Translate("Version") | html %]</th>
                            <th>[% Translate("Created by") | html %]</th>
                            <th>[% Translate("Created") | html %]</th>
                        </tr>
                    </thead>
                    <tbody>
[% RenderBlockStart("DefinitionListRow") %]
                        <tr>
                            <td>
                                <a class="AsBlock" href="[% Env("Baselink") %]Action=[% Env("Action") %];Subaction=DefinitionView;DefinitionID=[% Data.DefinitionID | uri %]">[% Data.Class | html %]</a>
                            </td>
                            <td>[% Data.Version | html %]</td>
                            <td>[% Data.CreateByUser | html %]</td>
                            <td>[% Data.CreateTime | Localize("TimeLong") %]</td>
                        </tr>
[% RenderBlockEnd("DefinitionListRow") %]
                    </tbody>
                </table>
            </div>
[% RenderBlockEnd("DefinitionList") %]
[% RenderBlockStart("DefinitionView") %]
            <div class="Header">
                <h2>[% Translate("View") | html %]: </h2>
            </div>
            <div class="Content">
                <fieldset class="TableLike">

                    <label>[% Translate("Config Item Class") | html %]: </label>
                    <div class="Field">
                        [% Data.Class | html %]
                    </div>
                    <div class="Clear"></div>

                    <label>[% Translate("Version") | html %]: </label>
                    <div class="Field">
                        [% Data.Version | html %]
                    </div>
                    <div class="Clear"></div>

                    <label>[% Translate("Created by") | html %]: </label>
                    <div class="Field">
                        [% Data.CreateByUser | html %]
                    </div>
                    <div class="Clear"></div>

                    <label>[% Translate("Created") | html %]: </label>
                    <div class="Field">
                        [% Data.CreateTime | Localize("TimeLong") %]
                    </div>
                    <div class="Clear"></div>

                    <label>[% Translate("Definition") | html %]: </label>
                    <div class="Field">
                        <pre>[% Data.DefinitionString %]</pre>
                    </div>
                    <div class="Clear"></div>

                </fieldset>
            </div>
[% RenderBlockEnd("DefinitionView") %]
[% RenderBlockStart("DefinitionChange") %]
            <div class="Header">
                <h2>[% Translate("Change") | html %]: </h2>
            </div>
            <div class="Content">
                <form action="[% Env("CGIHandle") %]" method="post">
                    <input type="hidden" name="Action" value="[% Env("Action") %]"/>
                    <input type="hidden" name="Subaction" value="DefinitionSave"/>
                    <input type="hidden" name="ClassID" value="[% Data.ClassID | html %]"/>
                    <input type="hidden" name="ContinueAfterSave" id="ContinueAfterSave" value=""/>
                    <fieldset class="TableLike">


                        <label>[% Translate("Config Item Class") | html %]: </label>
                        <div class="Field">
                            [% Data.Class | html %]
                        </div>
                        <div class="Clear"></div>

                        <label for="Definition">[% Translate("Definition") | html %]: </label>
                        <div class="Field">
                            <textarea name="Definition" id="Definition" rows="[% Data.Rows | html %]" cols="80" wrap="off">[% Data.Definition | html %]</textarea>
                        </div>
                        <div class="Clear"></div>

                        <div class="Field SpacingTop">
                            <button name="SubmitAndContinue" class="CallForAction Primary" id="SubmitAndContinue" type="button" value="Submit">
                                <span>[% Translate("Save") | html %]</span>
                            </button>
                            [% Translate("or") | html %]
                            <button name="Submit" class="CallForAction Primary" id="Submit" type="submit" value="Submit">
                                <span>[% Translate("Save and finish") | html %]</span>
                            </button>
                            [% Translate("or") | html %]
                            <a href="[% Env("Baselink") %]Action=[% Env("Action") %]">[% Translate("Cancel") | html %]</a>
                        </div>
                        <div class="Clear"></div>

                    </fieldset>
                </form>
            </div>
[% RenderBlockEnd("DefinitionChange") %]
        </div>
    </div>
    <div class="Clear"></div>
</div>
[% RenderBlockEnd("Overview") %]

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

<div class="DashboardActions">

    <ul class="ITSMConfigItemFilter Tab Actions">
[% RenderBlockStart("OverviewNavBarFilter") %]
[% RenderBlockStart("OverviewNavBarFilterItem") %]
[% RenderBlockStart("OverviewNavBarFilterItemSelected") %]
            <li class="Selected [% Data.CSS | html %] [% Data.Name | html %]">
                <a href="#" id="DashboardAdditionalFilter[% Data.Name | html %]" data-filter="[% Data.Name | html %]" class="DashboardITSMConfigItemGenericFilter">[% Translate(Data.Name) | html %] ([% Data.Count | html %]) </a>
            </li>
[% RenderBlockEnd("OverviewNavBarFilterItemSelected") %]
[% RenderBlockStart("OverviewNavBarFilterItemSelectedNot") %]
            <li class="[% Data.CSS | html %] [% Data.Name | html %]">
                <a href="#" id="DashboardAdditionalFilter[% Data.Name | html %]" data-filter="[% Data.Name | html %]" class="DashboardITSMConfigItemGenericFilter">[% Translate(Data.Name) | html %] ([% Data.Count | html %]) </a>
            </li>
[% RenderBlockEnd("OverviewNavBarFilterItemSelectedNot") %]
[% RenderBlockEnd("OverviewNavBarFilterItem") %]
[% RenderBlockEnd("OverviewNavBarFilter") %]
    </ul>

</div>

[% RenderBlockStart("ContentLargeITSMConfigItemGenericFilterNavBar") %]
<span class="Pagination">
    [% Data.SiteNavBar %]
</span>
[% RenderBlockEnd("ContentLargeITSMConfigItemGenericFilterNavBar") %]

#<!-- This form will not be submitted, we need it for the AJAX calls. -->
<form action="[% Env("CGIHandle") %]" method="post" enctype="multipart/form-data">
    <input type="hidden" name="CustomerID" value="[% Data.CustomerID | html %]"/>
    <input type="hidden" name="CustomerUserID" value="[% Data.CustomerUserID | html %]"/>
    <input type="hidden" name="Filter[% Data.Name | html %]" id="Filter[% Data.Name | html %]" value="[% Data.FilterValue | html %]" />
    <input type="hidden" name="AdditionalFilter[% Data.Name | html %]" id="AdditionalFilter[% Data.Name | html %]" value="[% Data.AdditionalFilterValue | html %]" />
    <table class="DataTable">
         [% Data.StyleClasses %]
        <thead>
            <tr>
[% RenderBlockStart("RecordCurInciSignalHeader") %]
                <th>
                    <span>[% Translate("Incident State") | html %]</span>
                </th>
[% RenderBlockEnd("RecordCurInciSignalHeader") %]
[% RenderBlockStart("RecordCurDeplSignalHeader") %]
                <th>
                    <span>[% Translate("Deployment State") | html %]</span>
                </th>
[% RenderBlockEnd("RecordCurDeplSignalHeader") %]
[% RenderBlockStart("RecordNumberHeader") %]
                <th class="DashboardHeader Number [% Data.CSS | html %]">
                    <a id="OverviewControl[% Data.Column | html %]" name="OverviewControl[% Data.Column | html %]" href="#" data-column="Number">[% Config("ITSMConfigItem::Hook") %]</a>
                </th>
[% RenderBlockEnd("RecordNumberHeader") %]
[% RenderBlockStart("RecordNameHeader") %]
                <th class="DashboardHeader Name [% Data.CSS | html %]">
                    <a id="OverviewControl[% Data.Column | html %]" name="OverviewControl[% Data.Column | html %]" href="#" data-column="Name">[% Translate("Name") | html %]</a>
                </th>
[% RenderBlockEnd("RecordNameHeader") %]
[% RenderBlockStart("RecordClassHeader") %]
                <th class="DashboardHeader Class [% Data.CSS | html %]">
                    <a id="OverviewControl[% Data.Column | html %]" name="OverviewControl[% Data.Column | html %]" href="#" data-column="ClassID">[% Translate("Class") | html %]</a>
                </th>
[% RenderBlockEnd("RecordClassHeader") %]
[% RenderBlockStart("RecordCurDeplStateHeader") %]
                <th class="DashboardHeader CurDeplState [% Data.CSS | html %]">
                    <a id="OverviewControl[% Data.Column | html %]" name="OverviewControl[% Data.Column | html %]" href="#" data-column="DeplStateID">[% Translate("Deployment State") | html %]</a>
                </th>
[% RenderBlockEnd("RecordCurDeplStateHeader") %]
[% RenderBlockStart("RecordCurDeplStateTypeHeader") %]
                <th>
                    <span>[% Translate("Deployment State Type") | html %]</span>
                </th>
[% RenderBlockEnd("RecordCurDeplStateTypeHeader") %]
[% RenderBlockStart("RecordCurInciStateHeader") %]
                <th class="DashboardHeader CurInciState [% Data.CSS | html %]">
                    <a id="OverviewControl[% Data.Column | html %]" name="OverviewControl[% Data.Column | html %]" href="#" data-column="InciStateID">[% Translate("Current Incident State") | html %]</a>
                </th>
[% RenderBlockEnd("RecordCurInciStateHeader") %]
[% RenderBlockStart("RecordCurInciStateTypeHeader") %]
                <th>
                    <span>[% Translate("Current Incident State Type") | html %]</span>
                </th>
[% RenderBlockEnd("RecordCurInciStateTypeHeader") %]
[% RenderBlockStart("RecordXMLAttributeHeader") %]
                <th>
                    <span>[% Translate(Data.XMLAttributeHeader) | html %]</span>
                </th>
[% RenderBlockEnd("RecordXMLAttributeHeader") %]
[% RenderBlockStart("RecordLastChangedHeader") %]
                <th class="DashboardHeader LastChanged [% Data.CSS | html %]">
                    <a id="OverviewControl[% Data.Column | html %]" name="OverviewControl[% Data.Column | html %]" href="#" data-column="LastChanged">[% Translate("Last changed") | html %]</a>
                </th>
[% RenderBlockEnd("RecordLastChangedHeader") %]
            </tr>
        </thead>
        <tbody>
[% RenderBlockStart("NoDataFoundMsg") %]
            <tr>
                <td colspan="[% Data.TotalColumns | html %]">
                    [% Translate("No data found.") | html %]
                </td>
            </tr>
[% RenderBlockEnd("NoDataFoundMsg") %]
[% RenderBlockStart("Record") %]
            <tr id="ConfigItemID_[% Data.ConfigItemID | html %]" class="MasterAction">
[% RenderBlockStart("RecordCurInciSignal") %]
                <td title="[% Translate(Data.CurInciState) | html %]">
                    <div class="Flag Small">
                        <span class="[% Data.CurInciSignal | html %]">"[% Translate(Data.CurInciState) | html %]"</span>
                    </div>
                </td>
[% RenderBlockEnd("RecordCurInciSignal") %]
[% RenderBlockStart("RecordCurDeplSignal") %]
                <td title="[% Translate(Data.CurDeplState) | html %]">
                    <div class="Flag Small">
                        <span class="[% Data.CurDeplSignal | html %]">"[% Translate(Data.CurDeplState) | html %]"</span>
                    </div>
                </td>
[% RenderBlockEnd("RecordCurDeplSignal") %]
[% RenderBlockStart("RecordNumber") %]
                <td>
[% RenderBlockStart("RecordNumberLinkStart") %]
                    <a href="[% Env("Baselink") %]Action=AgentITSMConfigItemZoom;ConfigItemID=[% Data.ConfigItemID | uri %]" title="[% Data.Number | uri %]" class="MasterActionLink">
[% RenderBlockEnd("RecordNumberLinkStart") %]
                        [% Data.Number | html %]
[% RenderBlockStart("RecordNumberLinkEnd") %]
                    </a>
[% RenderBlockEnd("RecordNumberLinkEnd") %]
                </td>
[% RenderBlockEnd("RecordNumber") %]
[% RenderBlockStart("RecordName") %]
                <td>
[% RenderBlockStart("RecordNameLinkStart") %]
                    <a href="[% Env("Baselink") %]Action=AgentITSMConfigItemZoom;ConfigItemID=[% Data.ConfigItemID | uri %]" title="[% Data.Name | uri %]" class="MasterActionLink">
[% RenderBlockEnd("RecordNameLinkStart") %]
                        <div title="[% Data.Name | html %]">[% Data.Name | truncate(40) | html %]</div>
[% RenderBlockStart("RecordNameLinkEnd") %]
                    </a>
[% RenderBlockEnd("RecordNameLinkEnd") %]
                </td>
[% RenderBlockEnd("RecordName") %]
[% RenderBlockStart("RecordClass") %]
                <td>
                    <div title="[% Translate(Data.Class) | html %]">[% Translate(Data.Class) | html %]</div>
                </td>
[% RenderBlockEnd("RecordClass") %]
[% RenderBlockStart("RecordCurDeplState") %]
                <td>
                    <div title="[% Translate(Data.CurDeplState) | html %]">[% Translate(Data.CurDeplState) | html %]</div>
                </td>
[% RenderBlockEnd("RecordCurDeplState") %]
[% RenderBlockStart("RecordCurDeplStateType") %]
                <td>
                    <div title="[% Translate(Data.CurDeplStateType) | html %]">[% Translate(Data.CurDeplStateType) | html %]</div>
                </td>
[% RenderBlockEnd("RecordCurDeplStateType") %]
[% RenderBlockStart("RecordCurInciState") %]
                <td>
                    <div title="[% Translate(Data.CurInciState) | html %]">[% Translate(Data.CurInciState) | html %]</div>
                </td>
[% RenderBlockEnd("RecordCurInciState") %]
[% RenderBlockStart("RecordCurInciStateType") %]
                <td>
                    <div title="[% Translate(Data.CurInciStateType) | html %]">[% Translate(Data.CurInciStateType) | html %]</div>
                </td>
[% RenderBlockEnd("RecordCurInciStateType") %]
[% RenderBlockStart("RecordXMLAttribute") %]
                <td>
[% RenderBlockStart("RecordXMLAttributeLinkStart") %]
                    <a href="[% Env("Baselink") %]Action=AgentITSMConfigItemZoom;ConfigItemID=[% Data.ConfigItemID | uri %]" title="[% Data.Number | uri %]" class="MasterActionLink">
[% RenderBlockEnd("RecordXMLAttributeLinkStart") %]
                        <div title="[% Data.XMLAttributeData | html %]">[% Data.XMLAttributeData | html %]</div>
[% RenderBlockStart("RecordXMLAttributeLinkEnd") %]
                    </a>
[% RenderBlockEnd("RecordXMLAttributeLinkEnd") %]
                </td>
[% RenderBlockEnd("RecordXMLAttribute") %]
[% RenderBlockStart("RecordLastChanged") %]
                <td>
                    <div title="[% Data.CreateTime | Localize("TimeLong") %]">[% Data.CreateTime | Localize("TimeLong") %]</div>
                </td>
[% RenderBlockEnd("RecordLastChanged") %]
            </tr>
[% RenderBlockEnd("Record") %]
        </tbody>
    </table>
</form>

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKPGRpdiBjbGFzcz0iTWFpbkJveCBBUklBUm9sZU1haW4gTGF5b3V0Rml4ZWRTaWRlYmFyIFNpZGViYXJGaXJzdCI+CiAgICA8aDE+WyUgVHJhbnNsYXRlKCJBZGQiKSB8IGh0bWwgJV06IFslIFRyYW5zbGF0ZSgiQ29uZmlnIEl0ZW0iKSB8IGh0bWwgJV08L2gxPgoKICAgIDxkaXYgY2xhc3M9IlNpZGViYXJDb2x1bW4iPgoKICAgICAgICA8ZGl2IGNsYXNzPSJXaWRnZXRTaW1wbGUiPgogICAgICAgICAgICA8ZGl2IGNsYXNzPSJIZWFkZXIiPgogICAgICAgICAgICAgICAgPGgyPjxsYWJlbCBmb3I9IkZpbHRlckNsYXNzZXMiPlslIFRyYW5zbGF0ZSgiRmlsdGVyIGZvciBDbGFzc2VzIikgfCBodG1sICVdPC9sYWJlbD48L2gyPgogICAgICAgICAgICA8L2Rpdj4KICAgICAgICAgICAgPGRpdiBjbGFzcz0iQ29udGVudCI+CiAgICAgICAgICAgICAgICA8aW5wdXQgdHlwZT0idGV4dCIgaWQ9IkZpbHRlckNsYXNzZXMiIGNsYXNzPSJGaWx0ZXJCb3giIG5hbWU9IkZpbHRlckNsYXNzZXMiIHZhbHVlPSIiIHRpdGxlPSJbJSBUcmFuc2xhdGUoIkZpbHRlciBmb3IgQ2xhc3NlcyIpIHwgaHRtbCAlXSIgcGxhY2Vob2xkZXI9IlslIFRyYW5zbGF0ZSgiSnVzdCBzdGFydCB0eXBpbmcgdG8gZmlsdGVyLi4uIikgfCBodG1sICVdIiAvPgogICAgICAgICAgICA8L2Rpdj4KICAgICAgICA8L2Rpdj4KCiAgICAgICAgPGRpdiBjbGFzcz0iV2lkZ2V0U2ltcGxlIj4KICAgICAgICAgICAgPGRpdiBjbGFzcz0iSGVhZGVyIj4KICAgICAgICAgICAgICAgIDxoMj5bJSBUcmFuc2xhdGUoIkhpbnQiKSB8IGh0bWwgJV08L2gyPgogICAgICAgICAgICA8L2Rpdj4KICAgICAgICAgICAgPGRpdiBjbGFzcz0iQ29udGVudCI+CiAgICAgICAgICAgICAgICA8cCBjbGFzcz0iRmllbGRFeHBsYW5hdGlvbiI+CiAgICAgICAgICAgICAgICAgICAgWyUgVHJhbnNsYXRlKCJTZWxlY3QgYSBDbGFzcyBmcm9tIHRoZSBsaXN0IHRvIGNyZWF0ZSBhIG5ldyBDb25maWcgSXRlbS4iKSB8IGh0bWwgJV0KICAgICAgICAgICAgICAgIDwvcD4KICAgICAgICAgICAgPC9kaXY+CiAgICAgICAgPC9kaXY+CgogICAgPC9kaXY+CgogICAgPGRpdiBjbGFzcz0iQ29udGVudENvbHVtbiI+CiAgICAgICAgPGRpdiBjbGFzcz0iV2lkZ2V0U2ltcGxlIj4KICAgICAgICAgICAgPGRpdiBjbGFzcz0iSGVhZGVyIj4KICAgICAgICAgICAgICAgIDxoMj5bJSBUcmFuc2xhdGUoIkxpc3QiKSB8IGh0bWwgJV08L2gyPgogICAgICAgICAgICA8L2Rpdj4KICAgICAgICAgICAgPGRpdiBjbGFzcz0iQ29udGVudCI+CiAgICAgICAgICAgICAgICA8dGFibGUgY2xhc3M9IkRhdGFUYWJsZSIgaWQ9IkNsYXNzZXMiPgogICAgICAgICAgICAgICAgICAgIDx0aGVhZD4KICAgICAgICAgICAgICAgICAgICAgICAgPHRyPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRoIGNsYXNzPSJXMTAwcGMiPlslIFRyYW5zbGF0ZSgiQ2xhc3MiKSB8IGh0bWwgJV08L3RoPgogICAgICAgICAgICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICAgICAgICAgIDwvdGhlYWQ+CiAgICAgICAgICAgICAgICAgICAgPHRib2R5PgogICAgICAgICAgICAgICAgICAgICAgICA8dHIgY2xhc3M9IkZpbHRlck1lc3NhZ2UgSGlkZGVuIj4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0ZD5bJSBUcmFuc2xhdGUoIk5vIG1hdGNoZXMgZm91bmQuIikgfCBodG1sICVdPC90ZD4KICAgICAgICAgICAgICAgICAgICAgICAgPC90cj4KWyUgUmVuZGVyQmxvY2tTdGFydCgiT3ZlcnZpZXdJdGVtTGlzdCIpICVdCiAgICAgICAgICAgICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0ZD4KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8YSBocmVmPSJbJSBFbnYoIkJhc2VsaW5rIikgJV1BY3Rpb249QWdlbnRJVFNNQ29uZmlnSXRlbUVkaXQ7Q2xhc3NJRD1bJSBEYXRhLkNsYXNzSUQgfCB1cmkgJV0iPlslIFRyYW5zbGF0ZShEYXRhLk5hbWUpIHwgaHRtbCAlXTwvYT4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDwvdGQ+CiAgICAgICAgICAgICAgICAgICAgICAgIDwvdHI+ClslIFJlbmRlckJsb2NrRW5kKCJPdmVydmlld0l0ZW1MaXN0IikgJV0KICAgICAgICAgICAgICAgICAgICA8L3Rib2R5PgogICAgICAgICAgICAgICAgPC90YWJsZT4KICAgICAgICAgICAgPC9kaXY+CiAgICAgICAgPC9kaXY+CiAgICA8L2Rpdj4KICAgIDxkaXYgY2xhc3M9IkNsZWFyIj48L2Rpdj4KPC9kaXY+Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKWyUgUmVuZGVyQmxvY2tTdGFydCgiQnVsa0FjdGlvbiIpICVdCjxmb3JtIGFjdGlvbj0iWyUgRW52KCJDR0lIYW5kbGUiKSAlXSIgbWV0aG9kPSJwb3N0IiBlbmN0eXBlPSJtdWx0aXBhcnQvZm9ybS1kYXRhIiBuYW1lPSJjb21wb3NlIiBjbGFzcz0iVmFsaWRhdGUiPgogICAgPGlucHV0IHR5cGU9ImhpZGRlbiIgbmFtZT0iQWN0aW9uIiB2YWx1ZT0iWyUgRW52KCJBY3Rpb24iKSAlXSIvPgogICAgPGlucHV0IHR5cGU9ImhpZGRlbiIgbmFtZT0iU3ViYWN0aW9uIiB2YWx1ZT0iRG8iLz4KCiAgICA8ZGl2IGNsYXNzPSJMYXlvdXRQb3B1cCBBUklBUm9sZU1haW4iPgogICAgICAgIDxkaXYgY2xhc3M9IkhlYWRlciI+CiAgICAgICAgICAgIDxoMT5bJSBUcmFuc2xhdGUoIklUU00gQ29uZmlnSXRlbSBCdWxrIEFjdGlvbiIpIHwgaHRtbCAlXTwvaDE+CiAgICAgICAgICAgIDxwPgogICAgICAgICAgICAgICAgPGEgY2xhc3M9IkNhbmNlbENsb3NlUG9wdXAiIGhyZWY9IiMiPlslIFRyYW5zbGF0ZSgiQ2FuY2VsICYgY2xvc2UiKSB8IGh0bWwgJV08L2E+CiAgICAgICAgICAgIDwvcD4KICAgICAgICA8L2Rpdj4KICAgICAgICA8ZGl2IGNsYXNzPSJDb250ZW50Ij4KCiAgICAgICAgICAgIDxkaXYgY2xhc3M9IkNvbnRlbnQiPgpbJSBSZW5kZXJCbG9ja1N0YXJ0KCJVc2VkQ29uZmlnSXRlbUlEIikgJV0KICAgICAgICAgICAgICAgIDxpbnB1dCB0eXBlPSJoaWRkZW4iIG5hbWU9IkNvbmZpZ0l0ZW1JRCIgdmFsdWU9IlslIERhdGEuQ29uZmlnSXRlbUlEIHwgaHRtbCAlXSIgLz4KWyUgUmVuZGVyQmxvY2tFbmQoIlVzZWRDb25maWdJdGVtSUQiKSAlXQoKICAgICAgICAgICAgICAgIDxmaWVsZHNldCBjbGFzcz0iVGFibGVMaWtlIEZpeGVkTGFiZWwiPgoKWyUgUmVuZGVyQmxvY2tTdGFydCgiRGVwbFN0YXRlIikgJV0KICAgICAgICAgICAgICAgICAgICA8bGFiZWwgZm9yPSJEZXBsU3RhdGVJRCI+WyUgVHJhbnNsYXRlKCJEZXBsb3ltZW50IHN0YXRlIikgfCBodG1sICVdOjwvbGFiZWw+CiAgICAgICAgICAgICAgICAgICAgPGRpdiBjbGFzcz0iRmllbGQiPgogICAgICAgICAgICAgICAgICAgICAgICBbJSBEYXRhLkRlcGxTdGF0ZVN0cmcgJV0KICAgICAgICAgICAgICAgICAgICA8L2Rpdj4KICAgICAgICAgICAgICAgICAgICA8ZGl2IGNsYXNzPSJDbGVhciI+PC9kaXY+ClslIFJlbmRlckJsb2NrRW5kKCJEZXBsU3RhdGUiKSAlXQpbJSBSZW5kZXJCbG9ja1N0YXJ0KCJJbmNpU3RhdGUiKSAlXQogICAgICAgICAgICAgICAgICAgIDxsYWJlbCBmb3I9IkluY2lTdGF0ZUlEIj5bJSBUcmFuc2xhdGUoIkluY2lkZW50IHN0YXRlIikgfCBodG1sICVdOjwvbGFiZWw+CiAgICAgICAgICAgICAgICAgICAgPGRpdiBjbGFzcz0iRmllbGQiPgogICAgICAgICAgICAgICAgICAgICAgICBbJSBEYXRhLkluY2lTdGF0ZVN0cmcgJV0KICAgICAgICAgICAgICAgICAgICA8L2Rpdj4KICAgICAgICAgICAgICAgICAgICA8ZGl2IGNsYXNzPSJDbGVhciI+PC9kaXY+ClslIFJlbmRlckJsb2NrRW5kKCJJbmNpU3RhdGUiKSAlXQoKICAgICAgICAgICAgICAgICAgICA8ZGl2IGNsYXNzPSJTcGFjaW5nVG9wIj48L2Rpdj4KICAgICAgICAgICAgICAgICAgICA8bGFiZWwgZm9yPSJMaW5rVG9nZXRoZXIiPlslIFRyYW5zbGF0ZSgiTGluayB0b2dldGhlciIpIHwgaHRtbCAlXTo8L2xhYmVsPgogICAgICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9IkZpZWxkIj4KICAgICAgICAgICAgICAgICAgICAgICAgWyUgRGF0YS5MaW5rVG9nZXRoZXJZZXNOb09wdGlvbiAlXQogICAgICAgICAgICAgICAgICAgICAgICBbJSBEYXRhLkxpbmtUb2dldGhlckxpbmtUeXBlU3RyZyAlXQogICAgICAgICAgICAgICAgICAgIDwvZGl2PgogICAgICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9IkNsZWFyIj48L2Rpdj4KCiAgICAgICAgICAgICAgICAgICAgPGxhYmVsIGZvcj0iTGlua1RvZ2V0aGVyQW5vdGhlciI+WyUgVHJhbnNsYXRlKCJMaW5rIHRvIGFub3RoZXIiKSB8IGh0bWwgJV06PC9sYWJlbD4KICAgICAgICAgICAgICAgICAgICA8ZGl2IGNsYXNzPSJGaWVsZCI+CiAgICAgICAgICAgICAgICAgICAgICAgIDxpbnB1dCB0eXBlPSJ0ZXh0IiBjbGFzcz0iWyUgRGF0YS5MaW5rVG9nZXRoZXJBbm90aGVySW52YWxpZCB8IGh0bWwgJV0gVzI1cGMiIGlkPSJMaW5rVG9nZXRoZXJBbm90aGVyIiBuYW1lPSJMaW5rVG9nZXRoZXJBbm90aGVyIiB2YWx1ZT0iWyUgRGF0YS5MaW5rVG9nZXRoZXJBbm90aGVyIHwgaHRtbCAlXSIvPgogICAgICAgICAgICAgICAgICAgICAgICA8ZGl2IGlkPSJMaW5rVG9nZXRoZXJBbm90aGVyU2VydmVyRXJyb3IiIGNsYXNzPSJUb29sdGlwRXJyb3JNZXNzYWdlIj48cD5bJSBUcmFuc2xhdGUoIkludmFsaWQgQ29uZmlndXJhdGlvbiBJdGVtIG51bWJlciEiKSB8IGh0bWwgJV08L3A+PC9kaXY+CiAgICAgICAgICAgICAgICAgICAgICAgIFslIERhdGEuTGlua1R5cGVTdHJnICVdCiAgICAgICAgICAgICAgICAgICAgICAgIDxwIGNsYXNzPSJGaWVsZEV4cGxhbmF0aW9uIj5bJSBUcmFuc2xhdGUoIlRoZSBudW1iZXIgb2YgYW5vdGhlciBDb25maWd1cmF0aW9uIEl0ZW0gdG8gbGluayB3aXRoLiIpIHwgaHRtbCAlXTwvcD4KICAgICAgICAgICAgICAgICAgICA8L2Rpdj4KICAgICAgICAgICAgICAgICAgICA8ZGl2IGNsYXNzPSJDbGVhciI+PC9kaXY+CiAgICAgICAgICAgICAgICA8L2ZpZWxkc2V0PgogICAgICAgICAgICA8L2Rpdj4KICAgICAgICA8L2Rpdj4KICAgICAgICA8ZGl2IGNsYXNzPSJGb290ZXIiPgogICAgICAgICAgICA8YnV0dG9uIGNsYXNzPSJQcmltYXJ5IENhbGxGb3JBY3Rpb24iIGlkPSJzdWJtaXRSaWNoVGV4dCIgYWNjZXNza2V5PSJnIiB0aXRsZT0iWyUgVHJhbnNsYXRlKCJTdWJtaXQiKSB8IGh0bWwgJV0gKGcpIiB0eXBlPSJzdWJtaXQiIHZhbHVlPSJbJSBUcmFuc2xhdGUoIlN1Ym1pdCIpIHwgaHRtbCAlXSI+PHNwYW4+WyUgVHJhbnNsYXRlKCJTdWJtaXQiKSB8IGh0bWwgJV08L3NwYW4+PC9idXR0b24+CiAgICAgICAgPC9kaXY+CiAgICA8L2Rpdj4KPC9mb3JtPgpbJSBSZW5kZXJCbG9ja0VuZCgiQnVsa0FjdGlvbiIpICVdCg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KClslIFJlbmRlckJsb2NrU3RhcnQoIldpZGdldCIpICVdCjxkaXYgY2xhc3M9IldpZGdldFNpbXBsZSBFeHBhbmRlZCIgaWQ9IkFnZW50SVRTTUNvbmZpZ0l0ZW1DdXN0b21lckNJc1dpZGdldCI+CiAgICA8ZGl2IGNsYXNzPSJIZWFkZXIiPgogICAgICAgIDxoMj5bJSBUcmFuc2xhdGUoIkN1c3RvbWVyIHJlbGF0ZWQgY29uZmlnIGl0ZW1zIikgfCBodG1sICVdPC9oMj4KICAgIDwvZGl2PgogICAgPGRpdiBjbGFzcz0iQ29udGVudCI+CiAgICAgICAgPGZpZWxkc2V0IGlkPSdBZ2VudElUU01Db25maWdJdGVtQ3VzdG9tZXJDSXMnIGNsYXNzPSJUYWJsZUxpa2UgRml4ZWRMYWJlbFNtYWxsIE5hcnJvdyI+CiAgICAgICAgICAgIFslIFRyYW5zbGF0ZSgibm9uZSIpIHwgaHRtbCAlXQogICAgICAgICA8L2ZpZWxkc2V0PgogICAgPC9kaXY+CjwvZGl2PgpbJSBSZW5kZXJCbG9ja0VuZCgiV2lkZ2V0IikgJV0KClslIFJlbmRlckJsb2NrU3RhcnQoIkNvbmZpZ0l0ZW1zIikgJV0KICAgICAgICAgICAgPHNwYW4+CiAgICAgICAgICAgICAgICA8bGFiZWwgZm9yPSJDb25maWdJdGVtLVslIERhdGEuTmFtZSB8IGh0bWwgJV0iIGNsYXNzPSJDaGVja2VkIj4KICAgICAgICAgICAgICAgICAgICA8aW5wdXQgdHlwZT0iY2hlY2tib3giIGNsYXNzPSJDb25maWdJdGVtTGluayIgbmFtZT0iQ29uZmlnSXRlbUxpbmsiIGlkPSJDb25maWdJdGVtTGluayIgdmFsdWU9IlslIERhdGEuQ29uZmlnSXRlbUlEIHwgaHRtbCAlXSIgWyUgSUYgRGF0YS5MaW5rZWQgJV1jaGVja2VkWyUgRU5EICVdIC8+CiAgICAgICAgICAgICAgICAgICAgPGEgaHJlZj0iWyUgRW52KCJCYXNlbGluayIpICVdQWN0aW9uPUFnZW50SVRTTUNvbmZpZ0l0ZW1ab29tO0NvbmZpZ0l0ZW1JRD1bJSBEYXRhLkNvbmZpZ0l0ZW1JRCB8IGh0bWwgJV0iIHRhcmdldD0iX2JsYW5rIj5bJSBJRiBEYXRhLkljb24gJV08aSBjbGFzcz0iZmEgWyUgRGF0YS5JY29uICVdIj48L2k+Jm5ic3A7WyUgRU5EICVdWyUgRGF0YS5OYW1lIHwgaHRtbCAlXSAoWyUgRGF0YS5OdW1iZXIgfCBodG1sICVdKTwvYT4KICAgICAgICAgICAgICAgIDwvbGFiZWw+CiAgICAgICAgICAgIDwvc3Bhbj4KICAgICAgICAgICAgPGRpdiBjbGFzcz0iQ2xlYXIiPjwvZGl2PgpbJSBSZW5kZXJCbG9ja0VuZCgiQ29uZmlnSXRlbXMiKSAlXQo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQo8ZmllbGRzZXQgY2xhc3M9IlRhYmxlTGlrZSBXMzAwcHgiPgoKICAgIDxsYWJlbD5bJSBUcmFuc2xhdGUoIk5hbWUiKSB8IGh0bWwgJV06PC9sYWJlbD4KICAgIDxwIGNsYXNzPSJWYWx1ZSI+CiAgICAgICAgWyUgRGF0YS5OYW1lIHwgaHRtbCAlXQogICAgPC9wPgogICAgPGRpdiBjbGFzcz0iQ2xlYXIiPjwvZGl2PgoKICAgIDxsYWJlbD5bJSBUcmFuc2xhdGUoIk51bWJlciIpIHwgaHRtbCAlXTo8L2xhYmVsPgogICAgPHAgY2xhc3M9IlZhbHVlIj4KICAgICAgICBbJSBUcmFuc2xhdGUoRGF0YS5OdW1iZXIpIHwgaHRtbCAlXQogICAgPC9wPgogICAgPGRpdiBjbGFzcz0iQ2xlYXIiPjwvZGl2PgoKCjwvZmllbGRzZXQ+CjxwIGNsYXNzPSJTcGFjaW5nVG9wIENlbnRlciI+CiAgICBbJSBUcmFuc2xhdGUoIkRvIHlvdSByZWFsbHkgd2FudCB0byBkZWxldGUgdGhpcyBjb25maWcgaXRlbT8iKSB8IGh0bWwgJV0KPC9wPgo=
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

[% RenderBlockStart("StartNormal") %]
<div class="MainBox ARIARoleMain LayoutFixedSidebar SidebarLast">
    <h1>[% Translate("Edit") | html %]: [% Translate("Config Item") | html %]: [% Translate(Data.Number) | html %] - [% Translate("Class") | html %]: [% Translate(Data.Class) | html %]</h1>
    <div class="ContentColumn">
[% RenderBlockEnd("StartNormal") %]
        <form action="[% Env("CGIHandle") %]" method="post" enctype="multipart/form-data" name="ClassItem" class="Validate PreventMultipleSubmits" id="ClassItem">
[% RenderBlockStart("StartSmall") %]
            <div class="LayoutPopup ARIARoleMain">
                <div class="Header">
                    <h1>[% Translate("Edit") | html %]: [% Translate("Config Item") | html %]: [% Translate(Data.Number) | html %] - [% Translate("Class") | html %]: [% Translate(Data.Class) | html %]</h1>
                    <p>
                        <a class="CancelClosePopup" href="#">[% Translate("Cancel & close") | html %]</a>
                    </p>
                </div>
                <input type="hidden" name="ScreenType" value="Popup"/>
[% RenderBlockEnd("StartSmall") %]
                <div class="Content">
                    <input type="hidden" name="Action" value="[% Env("Action") %]"/>
                    <input type="hidden" name="Subaction" value="VersionSave"/>
                    <input type="hidden" name="ClassID" value="[% Data.ClassID | html %]"/>
                    <input type="hidden" name="ConfigItemID" value="[% Data.ConfigItemID | html %]"/>
                    <input type="hidden" name="DuplicateID" value="[% Data.DuplicateID | html %]"/>
                    <input type="hidden" name="FormID" value="[% Data.FormID | html %]"/>
                    <input type="hidden" name="SubmitSave" value="[% Data.SubmitSave | html %]"/>
                    <fieldset class="TableLike">
[% RenderBlockStart("RowName") %]
                        <label class="Mandatory" for="Name"><span class="Marker">*</span> [% Translate("Name") | html %]: </label>
                        <div class="Field">
                            <input type="text" name="Name" id="Name" class="W50pc Validate_Required [% Data.RowNameInvalid | html %]" maxlength="250" value="[% Data.Name | html %]" title="[% Translate("The name of this config item") | html %]"/>
                            <div id="NameError" class="TooltipErrorMessage" ><p>[% Translate("This field is required.") | html %]</p></div>
[% RenderBlockStart("RowNameErrorDefault") %]
                            <div id="NameServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
[% RenderBlockEnd("RowNameErrorDefault") %]
[% RenderBlockStart("RowNameErrorDuplicates") %]
                            <div id="NameServerError" class="TooltipErrorMessage"><p>[% Translate("Name is already in use by the ConfigItems with the following Number(s): %s", Data.Duplicates) | html %]</p></div>
[% RenderBlockEnd("RowNameErrorDuplicates") %]
[% RenderBlockStart("RowNameErrorRegEx") %]
                            <div id="NameServerError" class="TooltipErrorMessage"><p>[% Translate(Data.RegExErrorMessage) | html %]</p></div>
[% RenderBlockEnd("RowNameErrorRegEx") %]
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("RowName") %]
[% RenderBlockStart("RowDeplState") %]
                        <label class="Mandatory" for="DeplStateID"><span class="Marker">*</span> [% Translate("Deployment State") | html %]: </label>
                        <div class="Field">
                            [% Data.DeplStateOptionStrg %]
                            <div id="DeplStateIDError" class="TooltipErrorMessage" ><p>[% Translate("This field is required.") | html %]</p></div>
                            <div id="DeplStateIDServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("RowDeplState") %]
[% RenderBlockStart("RowInciState") %]
                        <label class="Mandatory" for="InciStateID"><span class="Marker">*</span> [% Translate("Incident State") | html %]: </label>
                        <div class="Field">
                            [% Data.InciStateOptionStrg %]
                            <div id="InciStateIDError" class="TooltipErrorMessage" ><p>[% Translate("This field is required.") | html %]</p></div>
                            <div id="InciStateIDServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                        </div>
                        <div class="Clear"></div>
[% RenderBlockEnd("RowInciState") %]
                    </fieldset>

[% RenderBlockStart("XMLRow") %]
[% RenderBlockStart("XMLRowFieldsetStart") %]
                    <fieldset class="TableLike">
[% RenderBlockEnd("XMLRowFieldsetStart") %]
[% RenderBlockStart("XMLRowValue") %]
                        <label [% Data.LabelClass %] [% Data.LabelFor %]>
[% RenderBlockStart("XMLRowValueContentRequired") %]
                            <span class="Marker">*</span>
[% RenderBlockEnd("XMLRowValueContentRequired") %]
                            [% Translate(Data.Name) | html %]:
                        </label>
                        <div class="[% Data.Class %] Field">
                            [% Data.InputString %]
[% RenderBlockStart("XMLRowValueContentDelete") %]
                            <button class="DisableValidation Remove" type="submit" name="[% Data.InputKey | html %]::Delete" value="[% Translate("Delete") | html %]">[% Translate("Delete") | html %]</button>
[% RenderBlockEnd("XMLRowValueContentDelete") %]
                            <div id="[% Data.ItemID | html %]Error" class="TooltipErrorMessage" ><p>[% Translate("This field is required.") | html %]</p></div>
[% RenderBlockStart("XMLRowValueServerError") %]
                            <div id="[% Data.ItemID | html %]ServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
[% RenderBlockEnd("XMLRowValueServerError") %]
[% RenderBlockStart("XMLRowValueRegExError") %]
                            <div id="[% Data.ItemID | html %]ServerError" class="TooltipErrorMessage"><p>[% Translate(Data.RegExErrorMessage) | html %]</p></div>
[% RenderBlockEnd("XMLRowValueRegExError") %]
                        </div>
[% RenderBlockEnd("XMLRowValue") %]

[% RenderBlockStart("XMLRowAddContent") %]
                        <label [% Data.Class %] for="[% Data.ItemID | html %]">[% Translate(Data.Name) | html %]: </label>
                        <div class="Field">
                            <button class="DisableValidation Add" id="[% Data.ItemID | html %]" type="submit" name="[% Data.InputKey | html %]::Add" value="[% Translate("Add") | html %]" title="[% Translate(Data.Description) | html %]" ></button>
                        </div>
[% RenderBlockEnd("XMLRowAddContent") %]

[% RenderBlockStart("XMLRowFieldsetEnd") %]
                    </fieldset>
[% RenderBlockEnd("XMLRowFieldsetEnd") %]
[% RenderBlockEnd("XMLRow") %]

                    <fieldset class="TableLike">
                        <label>[% Translate("Attachments") | html %]:</label>
                        <div class="Field">
                            [% INCLUDE "FormElements/AttachmentList.tt" %]
                        </div>
                        <div class="Clear"></div>
                    </fieldset>

                    <fieldset class="TableLike">
                        <div class="Field SpacingTop">
                            <button name="Submit" class="CallForAction Primary" id="SubmitButton" type="submit" value="Submit">
                                <span>[% Translate("Save") | html %]</span>
                            </button>
                            [% Translate("or") | html %]
                            <a id="CancelButton" href="[% Env("Baselink") %]Action=AgentITSMConfigItemAdd">[% Translate("Cancel") | html %]</a>
                        </div>
                    </fieldset>
                </div>

[% RenderBlockStart("EndSmall") %]
            </div>
[% RenderBlockEnd("EndSmall") %]
        </form>
[% RenderBlockStart("EndNormal") %]
    </div>
    <div class="Clear"></div>
</div>
[% RenderBlockEnd("EndNormal") %]

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKPCEtLSBzdGFydCBmb3JtIC0tPgo8ZGl2IGNsYXNzPSJMYXlvdXRQb3B1cCBBUklBUm9sZU1haW4iPgoKICAgIDxkaXYgY2xhc3M9IkhlYWRlciI+CiAgICAgICAgPGgxPgogICAgICAgICAgICBbJSBUcmFuc2xhdGUoIkhpc3Rvcnkgb2YgQ29uZmlnIEl0ZW06ICVzIiwgRGF0YS5OYW1lKSB8IGh0bWwgJV0KICAgICAgICA8L2gxPgogICAgICAgIDxwPgogICAgICAgICAgICA8YSBocmVmPSIjIiBjbGFzcz0iQ2FuY2VsQ2xvc2VQb3B1cCI+WyUgVHJhbnNsYXRlKCJDYW5jZWwgJiBjbG9zZSIpIHwgaHRtbCAlXTwvYT4KICAgICAgICA8L3A+CiAgICA8L2Rpdj4KCiAgICA8ZGl2IGNsYXNzPSJDb250ZW50Ij4KICAgICAgICA8ZGl2IGNsYXNzPSJXaWRnZXRTaW1wbGUiPgogICAgICAgICAgICA8ZGl2IGNsYXNzPSJIZWFkZXIiPgogICAgICAgICAgICAgICAgPGgyPgogICAgICAgICAgICAgICAgICAgIFslIFRyYW5zbGF0ZSgiSGlzdG9yeSBDb250ZW50IikgfCBodG1sICVdCiAgICAgICAgICAgICAgICA8L2gyPgogICAgICAgICAgICA8L2Rpdj4KICAgICAgICAgICAgPGRpdiBjbGFzcz0iQ29udGVudCI+CiAgICAgICAgICAgICAgICA8dGFibGUgc3VtbWFyeT0iQ29uZmlndXJhdGlvbiBJdGVtIEhpc3RvcnkiIGNsYXNzPSJEYXRhVGFibGUiPgogICAgICAgICAgICAgICAgICAgIDx0aGVhZD4KICAgICAgICAgICAgICAgICAgICAgICAgPHRyPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRoPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFslIFRyYW5zbGF0ZSgiQWN0aW9uIikgfCBodG1sICVdCiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8L3RoPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRoPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFslIFRyYW5zbGF0ZSgiQ29tbWVudCIpIHwgaHRtbCAlXQogICAgICAgICAgICAgICAgICAgICAgICAgICAgPC90aD4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0aD4KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbJSBUcmFuc2xhdGUoIlpvb20iKSB8IGh0bWwgJV0KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDwvdGg+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8dGg+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgWyUgVHJhbnNsYXRlKCJVc2VyIikgfCBodG1sICVdCiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8L3RoPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRoPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFslIFRyYW5zbGF0ZSgiQ3JlYXRldGltZSIpIHwgaHRtbCAlXQogICAgICAgICAgICAgICAgICAgICAgICAgICAgPC90aD4KICAgICAgICAgICAgICAgICAgICAgICAgPC90cj4KICAgICAgICAgICAgICAgICAgICA8L3RoZWFkPgogICAgICAgICAgICAgICAgICAgIDx0Ym9keT4KWyUgUmVuZGVyQmxvY2tTdGFydCgiUm93IikgJV0KICAgICAgICAgICAgICAgICAgICAgICAgPHRyPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRkPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFslIERhdGEuSGlzdG9yeVR5cGUgfCBodG1sICVdCiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8L3RkPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRkPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxkaXY+WyUgRGF0YS5Db21tZW50IHwgaHRtbCAlXTwvZGl2PgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iQ2VudGVyIj4KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8YSAgY2xhc3M9Ikxpbmtab29tVmlldyBBc0Jsb2NrIiBocmVmPSJbJSBFbnYoIkJhc2VsaW5rIikgJV1BY3Rpb249QWdlbnRJVFNNQ29uZmlnSXRlbVpvb207Q29uZmlnSXRlbUlEPVslIERhdGEuQ29uZmlnSXRlbUlEICVdO1ZlcnNpb25JRD1bJSBEYXRhLlZlcnNpb25JRCAlXSI+WyUgVHJhbnNsYXRlKCJab29tIHZpZXciKSB8IGh0bWwgJV08L2E+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8L3RkPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRkPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFslIERhdGEuVXNlckZ1bGxuYW1lIHwgaHRtbCAlXQogICAgICAgICAgICAgICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0ZD4KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbJSBEYXRhLkNyZWF0ZVRpbWUgfCBMb2NhbGl6ZSgiVGltZUxvbmciKSAlXQogICAgICAgICAgICAgICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICAgICAgICAgICAgPC90cj4KWyUgUmVuZGVyQmxvY2tFbmQoIlJvdyIpICVdCiAgICAgICAgICAgICAgICAgICAgPC90Ym9keT4KICAgICAgICAgICAgICAgIDwvdGFibGU+CiAgICAgICAgICAgIDwvZGl2PgogICAgICAgIDwvZGl2PgogICAgPC9kaXY+CiAgICA8ZGl2IGNsYXNzPSJGb290ZXIiPgogICAgPC9kaXY+CjwvZGl2Pgo8IS0tIGVuZCBmb3JtIC0tPgo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQpbJSBSZW5kZXJCbG9ja1N0YXJ0KCJPdmVydmlld05hdkJhciIpICVdCjxkaXYgY2xhc3M9Ik92ZXJ2aWV3Qm94IEFSSUFSb2xlTWFpbiBbJSBEYXRhLlZpZXcgfCBodG1sICVdIj4KICAgIDxoMT5bJSBUcmFuc2xhdGUoRGF0YS5UaXRsZU5hbWUpIHwgaHRtbCAlXTogWyUgVHJhbnNsYXRlKERhdGEuVGl0bGVWYWx1ZSkgfCBodG1sICVdPC9oMT4KCiAgICBbJSBEYXRhLkRhdGFJblRoZU1pZGRsZSAlXQoKICAgIDxkaXYgY2xhc3M9Ik92ZXJ2aWV3Q29udHJvbCIgaWQ9Ik92ZXJ2aWV3Q29udHJvbCI+CiAgICAgICAgPGRpdj4KICAgICAgICAgICAgPGRpdiBjbGFzcz0iQ29udHJvbFJvdyI+ClslIFJlbmRlckJsb2NrU3RhcnQoIk92ZXJ2aWV3TmF2QmFyRmlsdGVyIikgJV0KICAgICAgICAgICAgICAgIDx1bCBjbGFzcz0iVGFicyI+ClslIFJlbmRlckJsb2NrU3RhcnQoIk92ZXJ2aWV3TmF2QmFyRmlsdGVySXRlbSIpICVdClslIFJlbmRlckJsb2NrU3RhcnQoIk92ZXJ2aWV3TmF2QmFyRmlsdGVySXRlbVNlbGVjdGVkIikgJV0KICAgICAgICAgICAgICAgICAgICA8bGkgY2xhc3M9IkFjdGl2ZSBbJSBEYXRhLkNTUyB8IGh0bWwgJV0iPgogICAgICAgICAgICAgICAgICAgICAgICA8YSBocmVmPSJbJSBFbnYoIkJhc2VsaW5rIikgJV1BY3Rpb249WyUgRW52KCJBY3Rpb24iKSAlXTtbJSBEYXRhLkZpbHRlckxpbmsgJV1GaWx0ZXI9WyUgRGF0YS5GaWx0ZXIgfCB1cmkgJV0iPlslIFRyYW5zbGF0ZShEYXRhLk5hbWUpIHwgaHRtbCAlXTxzcGFuPlslIERhdGEuQ291bnQgfCBodG1sICVdPC9zcGFuPjwvYT4KICAgICAgICAgICAgICAgICAgICA8L2xpPgpbJSBSZW5kZXJCbG9ja0VuZCgiT3ZlcnZpZXdOYXZCYXJGaWx0ZXJJdGVtU2VsZWN0ZWQiKSAlXQpbJSBSZW5kZXJCbG9ja1N0YXJ0KCJPdmVydmlld05hdkJhckZpbHRlckl0ZW1TZWxlY3RlZE5vdCIpICVdCiAgICAgICAgICAgICAgICAgICAgPGxpIGNsYXNzPSJbJSBEYXRhLkNTUyB8IGh0bWwgJV0iPgogICAgICAgICAgICAgICAgICAgICAgICA8YSBocmVmPSJbJSBFbnYoIkJhc2VsaW5rIikgJV1BY3Rpb249WyUgRW52KCJBY3Rpb24iKSAlXTtbJSBEYXRhLkZpbHRlckxpbmsgJV1GaWx0ZXI9WyUgRGF0YS5GaWx0ZXIgfCB1cmkgJV0iPlslIFRyYW5zbGF0ZShEYXRhLk5hbWUpIHwgaHRtbCAlXTxzcGFuPlslIERhdGEuQ291bnQgfCBodG1sICVdPC9zcGFuPjwvYT4KICAgICAgICAgICAgICAgICAgICA8L2xpPgpbJSBSZW5kZXJCbG9ja0VuZCgiT3ZlcnZpZXdOYXZCYXJGaWx0ZXJJdGVtU2VsZWN0ZWROb3QiKSAlXQpbJSBSZW5kZXJCbG9ja1N0YXJ0KCJPdmVydmlld05hdkJhckZpbHRlckl0ZW1TcGxpdCIpICVdCiAgICAgICAgICAgICAgICAgICAgLQpbJSBSZW5kZXJCbG9ja0VuZCgiT3ZlcnZpZXdOYXZCYXJGaWx0ZXJJdGVtU3BsaXQiKSAlXQpbJSBSZW5kZXJCbG9ja0VuZCgiT3ZlcnZpZXdOYXZCYXJGaWx0ZXJJdGVtIikgJV0KICAgICAgICAgICAgICAgIDwvdWw+ClslIFJlbmRlckJsb2NrRW5kKCJPdmVydmlld05hdkJhckZpbHRlciIpICVdClslIFJlbmRlckJsb2NrU3RhcnQoIk92ZXJ2aWV3TmF2QmFyUGFnZUJhY2siKSAlXQogICAgICAgICAgICAgICAgPHVsIGNsYXNzPSJUYWJzIj4KICAgICAgICAgICAgICAgICAgICA8bGk+CiAgICAgICAgICAgICAgICAgICAgICAgIDxhIGlkPSJJVFNNQ29uZmlnSXRlbVNlYXJjaCIgaHJlZj0iWyUgRW52KCJCYXNlbGluayIpICVdQWN0aW9uPVslIEVudigiQWN0aW9uIikgJV07WyUgRGF0YS5MaW5rQmFjayAlXSI+WyUgVHJhbnNsYXRlKCJDaGFuZ2Ugc2VhcmNoIG9wdGlvbnMiKSB8IGh0bWwgJV08L2E+CiAgICAgICAgICAgICAgICAgICAgPC9saT4KICAgICAgICAgICAgICAgIDwvdWw+ClslIFJlbmRlckJsb2NrRW5kKCJPdmVydmlld05hdkJhclBhZ2VCYWNrIikgJV0KClslIFJlbmRlckJsb2NrU3RhcnQoIkNvbnRleHRTZXR0aW5ncyIpICVdCiAgICAgICAgICAgICAgICA8dWwgY2xhc3M9IkNvbnRleHRGdW5jdGlvbnMiPgogICAgICAgICAgICAgICAgICAgIDxsaSBjbGFzcz0iQ29udGV4dFNldHRpbmdzIj4KICAgICAgICAgICAgICAgICAgICAgICAgPGEgaHJlZj0iIyIgaWQ9IlNob3dDb250ZXh0U2V0dGluZ3NEaWFsb2ciIHRpdGxlPSJbJSBUcmFuc2xhdGUoIkNvbnRleHQgU2V0dGluZ3MiKSB8IGh0bWwgJV0iPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPGkgY2xhc3M9ImZhIGZhLWNvZyI+PC9pPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHNwYW4+WyUgVHJhbnNsYXRlKCJDb250ZXh0IFNldHRpbmdzIikgfCBodG1sICVdPC9zcGFuPgogICAgICAgICAgICAgICAgICAgICAgICA8L2E+CiAgICAgICAgICAgICAgICAgICAgPC9saT4KICAgICAgICAgICAgICAgIDwvdWw+CgogICAgICAgICAgICAgICAgPGRpdiBpZD0iQ29udGV4dFNldHRpbmdzRGlhbG9nQ29udGFpbmVyIiBjbGFzcz0iSGlkZGVuIj4KICAgICAgICAgICAgICAgICAgICA8Zm9ybSBhY3Rpb249IlslIEVudigiQ0dJSGFuZGxlIikgJV0iIG1ldGhvZD0icG9zdCIgaWQ9IkNvbnRleHRTZXR0aW5nc0RpYWxvZyI+CiAgICAgICAgICAgICAgICAgICAgICAgIDxpbnB1dCB0eXBlPSJoaWRkZW4iIG5hbWU9IkFjdGlvbiIgdmFsdWU9IkFnZW50UHJlZmVyZW5jZXMiLz4KICAgICAgICAgICAgICAgICAgICAgICAgPGlucHV0IHR5cGU9ImhpZGRlbiIgbmFtZT0iU3ViYWN0aW9uIiB2YWx1ZT0iVXBkYXRlIi8+CiAgICAgICAgICAgICAgICAgICAgICAgIDxpbnB1dCB0eXBlPSJoaWRkZW4iIG5hbWU9Ikdyb3VwIiB2YWx1ZT0iWyUgRGF0YS5Hcm91cCB8IGh0bWwgJV0iLz4KICAgICAgICAgICAgICAgICAgICAgICAgPGlucHV0IHR5cGU9ImhpZGRlbiIgbmFtZT0iUmVkaXJlY3RVUkwiIHZhbHVlPSJbJSBEYXRhLlJlcXVlc3RlZFVSTCB8IGh0bWwgJV0iLz4KICAgICAgICAgICAgICAgICAgICAgICAgPGZpZWxkc2V0IGNsYXNzPSJUYWJsZUxpa2UiPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPGxhYmVsIGZvcj0iWyUgRGF0YS5QcmVmZXJlbmNlc0tleSB8IGh0bWwgJV0iPlslIFRyYW5zbGF0ZSgiQ29uZmlnIEl0ZW1zIHBlciBwYWdlIikgfCBodG1sICVdOjwvbGFiZWw+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8ZGl2IGNsYXNzPSJGaWVsZCI+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgWyUgRGF0YS5QYWdlU2hvd25TdHJpbmcgJV0KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDwvZGl2PgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPGRpdiBjbGFzcz0iQ2xlYXIiPjwvZGl2PgogICAgICAgICAgICAgICAgICAgICAgICA8L2ZpZWxkc2V0PgogICAgICAgICAgICAgICAgICAgIDwvZm9ybT4KICAgICAgICAgICAgICAgIDwvZGl2PgpbJSBSZW5kZXJCbG9ja0VuZCgiQ29udGV4dFNldHRpbmdzIikgJV0KCiAgICAgICAgICAgIDwvZGl2PgogICAgICAgICAgICA8ZGl2IGNsYXNzPSJBY3Rpb25Sb3ciPgogICAgICAgICAgICAgICAgPGRpdiBjbGFzcz0iT3ZlcnZpZXdBY3Rpb25zIj4KClslIFJlbmRlckJsb2NrU3RhcnQoIkJ1bGtBY3Rpb24iKSAlXQogICAgICAgICAgICAgICAgICAgIDx1bCBjbGFzcz0iQWN0aW9ucyI+CiAgICAgICAgICAgICAgICAgICAgICAgIDxsaSBpZD0iQ29uZmlnSXRlbUJ1bGtBY3Rpb24iIGNsYXNzPSJCdWxrIEluYWN0aXZlIj4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxhIGhyZWY9IiMiPlslIFRyYW5zbGF0ZSgiQnVsayIpIHwgaHRtbCAlXTwvYT4KICAgICAgICAgICAgICAgICAgICAgICAgPC9saT4KICAgICAgICAgICAgICAgICAgICA8L3VsPgpbJSBSZW5kZXJCbG9ja0VuZCgiQnVsa0FjdGlvbiIpICVdCgogICAgICAgICAgICAgICAgICAgIFslIERhdGEuQWN0aW9uUm93ICVdCiAgICAgICAgICAgICAgICAgICAgPHVsIGNsYXNzPSJPdmVydmlld1pvb20iPgoKWyUgUmVuZGVyQmxvY2tTdGFydCgiT3ZlcnZpZXdOYXZCYXJWaWV3TW9kZSIpICVdClslIFJlbmRlckJsb2NrU3RhcnQoIk92ZXJ2aWV3TmF2QmFyVmlld01vZGVTZWxlY3RlZCIpICVdCiAgICAgICAgICAgICAgICAgICAgICAgIDxsaSBjbGFzcz0iQWN0aXZlIj4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxhIGNsYXNzPSJbJSBEYXRhLk5hbWUgfCBodG1sICVdIiBuYW1lPSJPdmVydmlld0NvbnRyb2wiIGhyZWY9IlslIEVudigiQmFzZWxpbmsiKSAlXUFjdGlvbj1bJSBFbnYoIkFjdGlvbiIpICVdO0ZpbHRlcj1bJSBEYXRhLkZpbHRlciB8IHVyaSAlXTtWaWV3PVslIERhdGEuVmlldyB8IHVyaSAlXTtbJSBEYXRhLkxpbmtGaWx0ZXIgJV0iIHRpdGxlPSJbJSBUcmFuc2xhdGUoRGF0YS5OYW1lKSB8IGh0bWwgJV0iPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFslIFRyYW5zbGF0ZShEYXRhLk5hbWVTaG9ydCkgfCBodG1sICVdCiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8L2E+CiAgICAgICAgICAgICAgICAgICAgICAgIDwvbGk+ClslIFJlbmRlckJsb2NrRW5kKCJPdmVydmlld05hdkJhclZpZXdNb2RlU2VsZWN0ZWQiKSAlXQpbJSBSZW5kZXJCbG9ja1N0YXJ0KCJPdmVydmlld05hdkJhclZpZXdNb2RlTm90U2VsZWN0ZWQiKSAlXQogICAgICAgICAgICAgICAgICAgICAgICA8bGk+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8YSBjbGFzcz0iWyUgRGF0YS5OYW1lIHwgaHRtbCAlXSIgbmFtZT0iT3ZlcnZpZXdDb250cm9sIiBocmVmPSJbJSBFbnYoIkJhc2VsaW5rIikgJV1BY3Rpb249WyUgRW52KCJBY3Rpb24iKSAlXTtGaWx0ZXI9WyUgRGF0YS5GaWx0ZXIgfCB1cmkgJV07Vmlldz1bJSBEYXRhLlZpZXcgfCB1cmkgJV07WyUgRGF0YS5MaW5rRmlsdGVyICVdIiB0aXRsZT0iWyUgVHJhbnNsYXRlKERhdGEuTmFtZSkgfCBodG1sICVdIj4KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbJSBUcmFuc2xhdGUoRGF0YS5OYW1lU2hvcnQpIHwgaHRtbCAlXQogICAgICAgICAgICAgICAgICAgICAgICAgICAgPC9hPgogICAgICAgICAgICAgICAgICAgICAgICA8L2xpPgpbJSBSZW5kZXJCbG9ja0VuZCgiT3ZlcnZpZXdOYXZCYXJWaWV3TW9kZU5vdFNlbGVjdGVkIikgJV0KWyUgUmVuZGVyQmxvY2tFbmQoIk92ZXJ2aWV3TmF2QmFyVmlld01vZGUiKSAlXQogICAgICAgICAgICAgICAgICAgIDwvdWw+CiAgICAgICAgICAgICAgICAgICAgPHNwYW4gY2xhc3M9IlBhZ2luYXRpb24iPgpbJSBSZW5kZXJCbG9ja1N0YXJ0KCJPdmVydmlld05hdkJhclBhZ2VOYXZCYXIiKSAlXQogICAgICAgICAgICAgICAgICAgICAgICBbJSBEYXRhLlJlc3VsdExvbmcgJV0KICAgICAgICAgICAgICAgICAgICAgICAgWyUgRGF0YS5TaXRlTmF2QmFyTG9uZyAlXQpbJSBSZW5kZXJCbG9ja0VuZCgiT3ZlcnZpZXdOYXZCYXJQYWdlTmF2QmFyIikgJV0KICAgICAgICAgICAgICAgICAgICA8L3NwYW4+CiAgICAgICAgICAgICAgICAgICAgPGRpdiBjbGFzcz0iQ2xlYXJMZWZ0Ij48L2Rpdj4KICAgICAgICAgICAgICAgIDwvZGl2PgojIHNvcnQgb3JkZXIgYmFyIGlmIG5lZWRlZAogICAgICAgICAgICAgICAgWyUgRGF0YS5Tb3J0T3JkZXJCYXIgJV0KICAgICAgICAgICAgPC9kaXY+CiAgICAgICAgPC9kaXY+CiAgICA8L2Rpdj4KPC9kaXY+ClslIFJlbmRlckJsb2NrRW5kKCJPdmVydmlld05hdkJhciIpICVdCg==
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

<div id="OverviewBody" class="Overview TicketList">
    <table class="TableSmall" summary="[% Translate("A generic ITSM Configuration Item table") | html %]">
        [% Data.StyleClasses %]
        <thead>
            <tr>
[% RenderBlockStart("RecordBulkActionHeader") %]
                <th class="Checkbox Fixed">
                    <input type="checkbox" class="Checkbox" title="[% Translate("Select all") | html %]" id="SelectAllConfigItems" autocomplete="off" />
                </th>
[% RenderBlockEnd("RecordBulkActionHeader") %]
[% RenderBlockStart("RecordCurInciSignalHeader") %]
                <th>
                    <span>[% Translate("Incident State") | html %]</span>
                </th>
[% RenderBlockEnd("RecordCurInciSignalHeader") %]
[% RenderBlockStart("RecordCurDeplSignalHeader") %]
                <th>
                    <span>[% Translate("Deployment State") | html %]</span>
                </th>
[% RenderBlockEnd("RecordCurDeplSignalHeader") %]
[% RenderBlockStart("RecordNumberHeader") %]
                <th class="Number [% Data.CSS | html %]">
                    <a href="[% Env("Baselink") %]Action=[% Env("Action") %];[% Data.LinkSort %];SortBy=Number;OrderBy=[% Data.OrderBy | uri %]">[% Config("ITSMConfigItem::Hook") %]</a>
                </th>
[% RenderBlockEnd("RecordNumberHeader") %]
[% RenderBlockStart("RecordNameHeader") %]
                <th class="Name [% Data.CSS | html %]">
                    <a href="[% Env("Baselink") %]Action=[% Env("Action") %];[% Data.LinkSort %];SortBy=Name;OrderBy=[% Data.OrderBy | uri %]">[% Translate("Name") | html %]</a>
                </th>
[% RenderBlockEnd("RecordNameHeader") %]
[% RenderBlockStart("RecordClassHeader") %]
                <th class="Class [% Data.CSS | html %]">
                    <a href="[% Env("Baselink") %]Action=[% Env("Action") %];[% Data.LinkSort %];SortBy=ClassID;OrderBy=[% Data.OrderBy | uri %]">[% Translate("Class") | html %]</a>
                </th>
[% RenderBlockEnd("RecordClassHeader") %]
[% RenderBlockStart("RecordCurDeplStateHeader") %]
                <th class="CurDeplState [% Data.CSS | html %]">
                    <a href="[% Env("Baselink") %]Action=[% Env("Action") %];[% Data.LinkSort %];SortBy=DeplStateID;OrderBy=[% Data.OrderBy | uri %]">[% Translate("Deployment State") | html %]</a>
                </th>
[% RenderBlockEnd("RecordCurDeplStateHeader") %]
[% RenderBlockStart("RecordCurDeplStateTypeHeader") %]
                <th>
                    <span>[% Translate("Deployment State Type") | html %]</span>
                </th>
[% RenderBlockEnd("RecordCurDeplStateTypeHeader") %]
[% RenderBlockStart("RecordCurInciStateHeader") %]
                <th class="CurInciState [% Data.CSS | html %]">
                    <a href="[% Env("Baselink") %]Action=[% Env("Action") %];[% Data.LinkSort %];SortBy=InciStateID;OrderBy=[% Data.OrderBy | uri %]">[% Translate("Current Incident State") | html %]</a>
                </th>
[% RenderBlockEnd("RecordCurInciStateHeader") %]
[% RenderBlockStart("RecordCurInciStateTypeHeader") %]
                <th>
                    <span>[% Translate("Current Incident State Type") | html %]</span>
                </th>
[% RenderBlockEnd("RecordCurInciStateTypeHeader") %]
[% RenderBlockStart("RecordXMLAttributeHeader") %]
                <th>
                    <span>[% Translate(Data.XMLAttributeHeader) | html %]</span>
                </th>
[% RenderBlockEnd("RecordXMLAttributeHeader") %]
[% RenderBlockStart("RecordLastChangedHeader") %]
                <th class="LastChanged [% Data.CSS | html %]">
                    <a href="[% Env("Baselink") %]Action=[% Env("Action") %];[% Data.LinkSort %];SortBy=ChangeTime;OrderBy=[% Data.OrderBy | uri %]">[% Translate("Last changed") | html %]</a>
                </th>
[% RenderBlockEnd("RecordLastChangedHeader") %]
[% RenderBlockStart("RecordCreateTimeHeader") %]
                <th class="[% Data.CSS | html %]">
                    <a href="[% Env("Baselink") %]Action=[% Env("Action") %];[% Data.LinkSort %];SortBy=CreateTime;OrderBy=[% Data.OrderBy | uri %]">[% Translate("Create Time") | html %]</a>
                </th>
[% RenderBlockEnd("RecordCreateTimeHeader") %]
                <th class="Hidden">
                </th>
            </tr>
        </thead>
        <tbody>
[% RenderBlockStart("NoDataFoundMsg") %]
            <tr>
                <td colspan="[% Data.TotalColumns | html %]">
                    [% Translate("No data found.") | html %]
                </td>
            </tr>
[% RenderBlockEnd("NoDataFoundMsg") %]
[% RenderBlockStart("Record") %]
            <tr id="ConfigItemID_[% Data.ConfigItemID | html %]" class="MasterAction">
[% RenderBlockStart("RecordBulkAction") %]
                <td class="Checkbox Fixed NonTextContent">
                    <input class="Checkbox" type="checkbox" name="ConfigItemID" value="[% Data.ConfigItemID | html %]"  autocomplete="off" />
                </td>
[% RenderBlockEnd("RecordBulkAction") %]
[% RenderBlockStart("RecordCurInciSignal") %]
                <td title="[% Translate(Data.CurInciState) | html %]">
                    <div class="Flag Small">
                        <span class="[% Data.CurInciSignal | html %]">"[% Translate(Data.CurInciState) | html %]"</span>
                    </div>
                </td>
[% RenderBlockEnd("RecordCurInciSignal") %]
[% RenderBlockStart("RecordCurDeplSignal") %]
                <td title="[% Translate(Data.CurDeplState) | html %]">
                    <div class="Flag Small">
                        <span class="[% Data.CurDeplSignal | html %]">"[% Translate(Data.CurDeplState) | html %]"</span>
                    </div>
                </td>
[% RenderBlockEnd("RecordCurDeplSignal") %]
[% RenderBlockStart("RecordNumber") %]
                <td>
                    <div title="[% Data.Number | html %]">
                        <a href="[% Env("Baselink") %]Action=AgentITSMConfigItemZoom;ConfigItemID=[% Data.ConfigItemID | uri %]" class="MasterActionLink">[% Data.Number | html %]</a>
                    </div>
                </td>
[% RenderBlockEnd("RecordNumber") %]
[% RenderBlockStart("RecordName") %]
                <td>
                    <div title="[% Data.Name | html %]">
                        <a href="[% Env("Baselink") %]Action=AgentITSMConfigItemZoom;ConfigItemID=[% Data.ConfigItemID | uri %]" class="MasterActionLink">[% Data.Name | html %]</a>
                    </div>
                </td>
[% RenderBlockEnd("RecordName") %]
[% RenderBlockStart("RecordClass") %]
                <td>
                    <div title="[% Translate(Data.Class) | html %]">[% Translate(Data.Class) | html %]</div>
                </td>
[% RenderBlockEnd("RecordClass") %]
[% RenderBlockStart("RecordCurDeplState") %]
                <td>
                    <div title="[% Translate(Data.CurDeplState) | html %]">[% Translate(Data.CurDeplState) | html %]</div>
                </td>
[% RenderBlockEnd("RecordCurDeplState") %]
[% RenderBlockStart("RecordCurDeplStateType") %]
                <td>
                    <div title="[% Translate(Data.CurDeplStateType) | html %]">[% Translate(Data.CurDeplStateType) | html %]</div>
                </td>
[% RenderBlockEnd("RecordCurDeplStateType") %]
[% RenderBlockStart("RecordCurInciState") %]
                <td>
                    <div title="[% Translate(Data.CurInciState) | html %]">[% Translate(Data.CurInciState) | html %]</div>
                </td>
[% RenderBlockEnd("RecordCurInciState") %]
[% RenderBlockStart("RecordCurInciStateType") %]
                <td>
                    <div title="[% Translate(Data.CurInciStateType) | html %]">[% Translate(Data.CurInciStateType) | html %]</div>
                </td>
[% RenderBlockEnd("RecordCurInciStateType") %]
[% RenderBlockStart("RecordXMLAttribute") %]
                <td>
                    <div title="[% Data.XMLAttributeData | html %]">
                        <a href="[% Env("Baselink") %]Action=AgentITSMConfigItemZoom;ConfigItemID=[% Data.ConfigItemID | uri %]" class="MasterActionLink">[% Data.XMLAttributeData | html %]</a>
                    </div>
                </td>
[% RenderBlockEnd("RecordXMLAttribute") %]
[% RenderBlockStart("RecordLastChanged") %]
                <td>
                    <div title="[% Data.ChangeTime | Localize("TimeLong") %]">[% Data.ChangeTime | Localize("TimeLong") %]</div>
                </td>
[% RenderBlockEnd("RecordLastChanged") %]
[% RenderBlockStart("RecordCreateTime") %]
                <td>
                    <div title="[% Data.CreateTime | Localize("TimeLong") %]">[% Data.CreateTime | Localize("TimeLong") %]</div>
                </td>
[% RenderBlockEnd("RecordCreateTime") %]
                <td class="Hidden">
                    <a href="[% Env("Baselink") %]Action=AgentITSMConfigItemZoom;ConfigItemID=[% Data.ConfigItemID | uri %]" class="MasterActionLink">
                </td>
            </tr>
[% RenderBlockEnd("Record") %]
        </tbody>
    </table>
</div>

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

[% RenderBlockStart("SearchAJAX") %]
<div class="InnerContent">
    <fieldset class="TableLike">
        <div class="Clear"></div>
        <label for="Class">[% Translate("Class") | html %]</label>
        <div class="Field" id='DivClassID'>
            [% Data.ClassOptionStrg %]
        </div>
    </fieldset>
</div>

<div class="InnerContent" id="AJAXUpdate">

    <form action="[% Env("CGIHandle") %]" method="post" id="SearchForm">
        <input type="hidden" name="Action" value="[% Env("Action") %]" id="SearchAction"/>
        <input type="hidden" name="ClassID" value="[% Data.ClassID | html %]" id="SearchClassID"/>
        <input type="hidden" name="Subaction" value="Search"/>

        <fieldset class="TableLike Hidden" id='ITSMSearchProfile'>
            <div class="Clear"></div>
            <legend><span>[% Translate("Templates") | html %]</span></legend>
            <label for="SearchProfile">[% Translate("Search template") | html %]:</label>
            <div class="Field">
                <input type="hidden" id="SearchProfile" name="SearchProfile" value="[% Data.Profile | html %]"/>
                <div id="SearchProfileAddBlock">
                    <input type="text" name="ProfileName" id="SearchProfileAddName"/>
                    <button class="CallForAction" type="button" title="[% Translate("Create Template") | html %]" id="SearchProfileAddAction"><span>[% Translate("Add") | html %]</span></button>
                    <input type="hidden" name="SaveProfile" value="1"/>
                </div>
            </div>
            <div class="Field">
                <button id="SearchProfileNew" class="CallForAction" value="[% Translate("Create New") | html %]"><span>[% Translate("Create New") | html %]</span></button>
                <button id="SearchProfileDelete" class="CallForAction Hidden" value="[% Translate("Delete") | html %]"><span>[% Translate("Delete") | html %]</span></button>
                <button id="SearchProfileAsLink" class="CallForAction Hidden" value="[% Translate("Profile link") | html %]"><span>[% Translate("Profile link") | html %]</span></button>
            </div>
        </fieldset>

    </form>
</div>

<div class="ContentFooter Center">
    <button id="SearchFormSubmit" class="CallForAction Primary Hidden" value="Run Search"><span><i class="fa fa-search"></i>[% Translate("Run Search") | html %]</span></button>
</div>
[% RenderBlockEnd("SearchAJAX") %]

[% RenderBlockStart("AJAXContent") %]
<form action="[% Env("CGIHandle") %]" method="post" id="SearchForm">
    <input type="hidden" name="Action" value="[% Env("Action") %]" id="SearchAction"/>
    <input type="hidden" name="Subaction" value="Search"/>
    <input type="hidden" name="ClassID" value="[% Data.ClassID | html %]" id="SearchClassID">
    <input type="hidden" name="SearchDialog" value='1'>
    <input type="hidden" name="EmptySearch" value="[% Data.EmptySearch | html %]" id="EmptySearch"/>
    <fieldset class="TableLike Hidden" id='ITSMSearchProfile'>
        <legend><span>[% Translate("Templates") | html %]</span></legend>
        <label for="SearchProfile">[% Translate("Search template") | html %]:</label>
        <div class="Field">
            [% Data.ProfilesStrg %]
            <div id="SearchProfileAddBlock">
                <input type="text" name="ProfileName" id="SearchProfileAddName"/>
                <button class="CallForAction" type="button" title="[% Translate("Create Template") | html %]" id="SearchProfileAddAction"><span>[% Translate("Add") | html %]</span></button>
            </div>
        </div>
        <div class="Field">
            <button id="SearchProfileNew" class="CallForAction" value="[% Translate("Create New") | html %]"><span>[% Translate("Create New") | html %]</span></button>
            <button id="SearchProfileDelete" class="CallForAction Hidden" value="[% Translate("Delete") | html %]"><span>[% Translate("Delete") | html %]</span></button>
            <button id="SearchProfileAsLink" class="CallForAction Hidden" value="[% Translate("Profile link") | html %]"><span>[% Translate("Profile link") | html %]</span></button>
        </div>
        <div class="Clear"></div>
        <label>[% Translate("Save changes in template") | html %]:</label>
        <div class="Field">
            <input type="checkbox" name="SaveProfile" id="SaveProfile" value="1"/>
        </div>
    </fieldset>

    <fieldset class="TableLike Hidden" id='ITSMSearchFields'>
        <fieldset class="TableLike" id="SearchInsert">
            <legend><span>[% Translate("Filters in use") | html %]</span></legend>
        </fieldset>
        <fieldset class="TableLike">
            <legend><span>[% Translate("Additional filters") | html %]</span></legend>
            <label>[% Translate("Add another attribute") | html %]:</label>
            <div class="Field">
                [% Data.AttributesStrg %]
            </div>

            <div class="Clear"></div>
            <label for="PreviousVersionSearch">[% Translate("Also search in previous versions?") | html %]</label>
            <div class="Field">
                [% Data.PreviousVersionOptionStrg %]
            </div>

            <div class="Clear"></div>
            <label>[% Translate("Output") | html %]:</label>
            <div class="Field">
                [% Data.ResultFormStrg %]
            </div>
        </fieldset>
    </fieldset>
</form>
<fieldset class="TableLike Hidden" id="SearchAttributesHidden">
    <div class="Clear"></div>
    <label for="Number" id="LabelNumber">[% Translate("Number") | html %] [% Translate("(e. g. 10*5155 or 105658*)") | html %]:</label>
    <div class="Field">
        <input type="text" name="Number" value="[% Data.Number | html %]" class="W50pc" />
        <a class="RemoveButton" href="#" title="[% Translate("Remove this entry") | html %]"><i class="fa fa-minus-square-o"></i><span class="InvisibleText">[% Translate("Remove") | html %]</span></a>
    </div>

    <div class="Clear"></div>
    <label for="Name" id="LabelName">[% Translate("Name") | html %]:</label>
    <div class="Field">
        <input type="text" name="Name" value="[% Data.Name | html %]" class="W50pc" />
        <a class="RemoveButton" href="#" title="[% Translate("Remove this entry") | html %]"><i class="fa fa-minus-square-o"></i><span class="InvisibleText">[% Translate("Remove") | html %]</span></a>
    </div>

    <div class="Clear"></div>
    <label for="DeplStateIDs" id="LabelDeplStateIDs">[% Translate("Deployment State") | html %]:</label>
    <div class="Field">
        [% Data.CurDeplStateOptionStrg %]
        <a class="RemoveButton" href="#" title="[% Translate("Remove this entry") | html %]"><i class="fa fa-minus-square-o"></i><span class="InvisibleText">[% Translate("Remove") | html %]</span></a>
    </div>

    <div class="Clear"></div>
    <label for="InciStateIDs" id="LabelInciStateIDs">[% Translate("Incident State") | html %]:</label>
    <div class="Field">
        [% Data.CurInciStateOptionStrg %]
        <a class="RemoveButton" href="#" title="[% Translate("Remove this entry") | html %]"><i class="fa fa-minus-square-o"></i><span class="InvisibleText">[% Translate("Remove") | html %]</span></a>
    </div>

[% RenderBlockStart("AttributeRow") %]
    <div class="Clear"></div>
    <label for="[% Translate(Data.Key) | html %]" id="Label[% Data.Key %]" title="[% Translate(Data.Description) | html %]">[% Translate(Data.Name) | html %]:</label>
    <div class="Field">
        [% Data.InputString %]
        <a class="RemoveButton" href="#" title="[% Translate("Remove this entry") | html %]"><i class="fa fa-minus-square-o"></i><span class="InvisibleText">[% Translate("Remove") | html %]</span></a>
    </div>
[% RenderBlockEnd("AttributeRow") %]
    <div class="Clear"></div>
    [% Data.AttributesOrigStrg %]
</fieldset>
[% RenderBlockEnd("AJAXContent") %]

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

<div class="MainBox ARIARoleMain">
    <div class="ClearLeft"></div>
    <div class="Headline">
        <div class="Flag" title="[% Translate(Data.CurInciState) | html %]">
            <span class="[% Data.CurInciSignal | html %]"></span>
        </div>
        <h1 title="[% Translate("Configuration Item") | html %]: [% Data.Number | html %] &ndash; [% Data.Name | truncate(40) | html %]">
            [% Translate("Configuration Item") | html %]: [% Data.Number | html %] <span>&mdash;</span> [% Data.Name | truncate(60) | html %]
        </h1>
    </div>
    <div class="LayoutFixedSidebar SidebarLast">

        [% Data.StyleClasses %]

        <div class="SidebarColumn">
[% RenderBlockStart("Meta") %]
            <div class="WidgetSimple">
                <div class="Header">
                    <div class="WidgetAction Toggle">
                        <a href="#" title="[% Translate("Show or hide the content") | html %]"><i class="fa fa-caret-right"></i><i class="fa fa-caret-down"></i></a>
                    </div>
                    <h2>[% Translate("Configuration Item Information") | html %]</h2>
                </div>
                <div class="Content">
                    <fieldset class="TableLike FixedLabelSmall Tight">

                        <label>[% Translate("Class") | html %]:</label>
                        <p class="Value">[% Translate(Data.Class) | html %]</p>
                        <div class="Clear"></div>

                        <label>[% Translate("Name") | html %]:</label>
                        <p class="Value">[% Data.Name | truncate(25) | html %]</p>
                        <div class="Clear"></div>

                        <label>[% Translate("Current Deployment State") | html %]:</label>
                        <div class="Value" title="[% Translate(Data.CurDeplState) | html %]">
                            <div class="Flag Small">
                                <span class="[% Data.CurDeplSignal | html %]"></span>
                            </div>
                            [% Translate(Data.CurDeplState) | html %]
                        </div>
                        <div class="Clear"></div>

                        <label>[% Translate("Current Incident State") | html %]:</label>
                        <div class="Value" title="[% Translate(Data.CurInciState) | html %]">
                            <div class="Flag Small">
                                <span class="[% Data.CurInciSignal | html %]"></span>
                            </div>
                            [% Translate(Data.CurInciState) | html %]
                        </div>
                        <div class="Clear"></div>

                        <label>[% Translate("Created") | html %]:</label>
                        <p class="Value">[% Data.CreateTime | Localize("TimeLong") %]</p>
                        <div class="Clear"></div>

                        <label>[% Translate("Created by") | html %]:</label>
                        <p class="Value">
                            [% Data.CreateByUserFullName | html %]
                        </p>
                        <div class="Clear"></div>

                        <label>[% Translate("Last changed") | html %]:</label>
                        <p class="Value">[% Data.ChangeTime | Localize("TimeLong") %]</p>
                        <div class="Clear"></div>

                        <label>[% Translate("Last changed by") | html %]:</label>
                        <p class="Value">
                            [% Data.ChangeByUserFullName | html %]
                        </p>
                        <div class="Clear"></div>
                    </fieldset>
                </div>
            </div>
[% RenderBlockEnd("Meta") %]
[% RenderBlockStart("LinkTableSimple") %]
            <div class="WidgetSimple DontPrint">
                <div class="Header">
                    <h2>[% Translate("Linked Objects") | html %]</h2>
                </div>
                <div class="Content">
                    [% Data.LinkTableStrg %]
                </div>
            </div>
[% RenderBlockEnd("LinkTableSimple") %]
        </div>
        <div class="ContentColumn">
            <div class="ControlRow">
                <h2></h2>
            </div>
            <div class="ActionRow">
                <ul class="Actions">
[% RenderBlockStart("Menu") %]
[% RenderBlockStart("MenuItem") %]
                    <li>
                        <a href="[% Env("Baselink") %][% Data.Link | Interpolate %]" id="Menu[% Data.MenuID | html %]" class="[% Data.MenuClass | html %]" title="[% Translate(Data.Description) | html %]">[% Translate(Data.Name) | html %]</a>
                    </li>
[% RenderBlockEnd("MenuItem") %]
[% RenderBlockEnd("Menu") %]
                </ul>
            </div>
            <div id="ITSMTree">
[% RenderBlockStart("Tree") %]
                <div id="ITSMTable"  class="TicketList">
                    <div class="ControlRow">
                        <div class="ITSMItemView Icons">
[% RenderBlockStart("Expand") %]
                            <a class="OneITSMItem Active" href="[% Env("Baselink") %]Action=AgentITSMConfigItemZoom;ConfigItemID=[% Data.ConfigItemID | uri %];ShowVersions=0" title="[% Translate("Show one version") | html %]">
                                <i class="fa fa-minus"></i>
                                <span>[% Translate("Show one version") | html %]</span>
                            </a>
                            <a class="AllITSMItems" href="[% Env("Baselink") %]Action=AgentITSMConfigItemZoom;ConfigItemID=[% Data.ConfigItemID | uri %];ShowVersions=1" title="[% Translate("Show all versions") | html %]">
                                <i class="fa fa-reorder"></i>
                                <span>[% Translate("Show all versions") | html %]</span>
                            </a>
[% RenderBlockEnd("Expand") %]
[% RenderBlockStart("Collapse") %]
                            <a class="OneITSMItem" href="[% Env("Baselink") %]Action=AgentITSMConfigItemZoom;ConfigItemID=[% Data.ConfigItemID | uri %];ShowVersions=0" title="[% Translate("Show one version") | html %]">
                                <i class="fa fa-minus"></i>
                                <span>[% Translate("Show one version") | html %]</span>
                            </a>
                            <a class="AllITSMItems Active" href="[% Env("Baselink") %]Action=AgentITSMConfigItemZoom;ConfigItemID=[% Data.ConfigItemID | uri %];ShowVersions=1" title="[% Translate("Show all versions") | html %]">
                                <i class="fa fa-reorder"></i>
                                <span>[% Translate("Show all versions") | html %]</span>
                            </a>
[% RenderBlockEnd("Collapse") %]
                        </div>
                    </div>
                    <div id="ITSMTableBody" class="ActionRow">
                        <div class="Scroller">
                            <table class="TableSmall">
                                <thead>
                                    <tr>
                                        <th><span>[% Translate("Version Incident State") | html %]</span></th>
                                        <th><span>[% Translate("Version Deployment State") | html %]</span></th>
                                        <th><span>[% Translate("Version Number") | html %]</span></th>
                                        <th><span>[% Translate("Name") | html %]</span></th>
                                        <th><span>[% Translate("Created by") | html %]</span></th>
                                        <th><span>[% Translate("Changed") | html %]</span></th>
                                    </tr>
                                </thead>
                                <tbody>
[% RenderBlockStart("TreeItem") %]
                                    <tr class="[% Data.Active | html %] MasterAction">
                                        <td title="[% Translate(Data.InciState) | html %]">
                                            <div class="Flag Small">
                                                <span class="[% Data.InciSignal | html %]">[% Data.InciState | html %]</span>
                                            </div>
                                        </td>
                                        <td title="[% Translate(Data.DeplState) | html %]">
                                            <div class="Flag Small">
                                                <span class="[% Data.DeplSignal | html %]">[% Data.DeplState | html %]</span>
                                            </div>
                                        </td>
                                        <td>
                                            [% Data.Count | html %].
                                        </td>
                                        <td>
                                            <a class="AsBlock MasterActionLink" href="[% Env("Baselink") %]Action=AgentITSMConfigItemZoom;ConfigItemID=[% Data.ConfigItemID | uri %];VersionID=[% Data.VersionID | uri %];ShowVersions=[% Data.ShowVersions | uri %]">
                                                [% Data.Name | html %] ([% Translate(Data.DeplState) | html %])
                                            </a>
                                        </td>
                                        <td>
                                            [% Data.CreateByUserFullName | html %]
                                        </td>
                                        <td>
                                            [% Data.CreateTime | Localize("TimeLong") %]
                                        </td>
                                    </tr>
[% RenderBlockEnd("TreeItem") %]
                                </tbody>
                            </table>
                        </div>
                        <div class="Handle ui-resizable-handle ui-resizable-s">
                            <a href="#">[% Translate("Resize") | html %]</a>
                        </div>
                    </div>
                </div>
[% RenderBlockEnd("Tree") %]
            </div>
            <div id="ITSMItems">
                <div class="WidgetBox SpacingTop Expanded">
                    <div class="LightRow Header">
                        <div class="WidgetAction Toggle"><a href="#" title="[% Translate("Show or hide the content.") | html %]"><i class="fa fa-caret-right"></i><i class="fa fa-caret-down"></i></a></div>
                        <h2>[% Translate("Configuration Item Version Details") | html %]</h2>
                    </div>
                    <div class="WithPadding Content NoDoubleBorders">
                        <table class="DataTable">
                            <thead>
                                <tr>
                                    <th>[% Translate("Property") | html %]</th>
                                    <th>[% Translate("Value") | html %]</th>
                                </tr>
                            </thead>
                            <tbody>
[% RenderBlockStart("Data") %]
                                <tr>
                                    <td class="W25pc" style="padding-left: [% Translate(Data.Indentation) | html %]px;" title="[% Translate(Data.Description) | html %]">
                                        [% Translate(Data.Name) | html %]:
                                    </td>
                                    <td style="padding-left: [% Translate(Data.Indentation) | html %]px;">
                                        [% Data.Value %]
                                    </td>
                                </tr>
[% RenderBlockEnd("Data") %]
[% RenderBlockStart("Attachments") %]
                                <tr>
                                    <td class="W25pc" title="[% Translate("Attachments") | html %]">
                                        [% Translate("Attachments") | html %]
                                    </td>
                                    <td>
                                        <a href="[% Env("Baselink") %]Action=[% Env("Action") %];Subaction=DownloadAttachment;Filename=[% Data.Filename | uri %];ConfigItemID=[% Data.ConfigItemID | uri %]">
                                            [% Data.Filename | html %]
                                        </a>
                                        ([% Data.Filesize | Localize('Filesize') | html %])
                                    </td>
                                </tr>
[% RenderBlockEnd("Attachments") %]
[% RenderBlockStart("AttachmentRow") %]
                                <tr>
                                    <td class="W25pc">
                                    </td>
                                    <td>
                                        <a href="[% Env("Baselink") %]Action=[% Env("Action") %];Subaction=DownloadAttachment;Filename=[% Data.Filename | uri %];ConfigItemID=[% Data.ConfigItemID | uri %]">
                                            [% Data.Filename | html %]
                                        </a>
                                        ([% Data.Filesize | Localize('Filesize') | html %])
                                    </td>
                                </tr>
[% RenderBlockEnd("AttachmentRow") %]
                            </tbody>
                        </table>
                    </div>
                </div>
            </div>
[% RenderBlockStart("LinkTableComplex") %]
            <div class="Content">
                [% Data.LinkTableStrg %]
            </div>
[% RenderBlockEnd("LinkTableComplex") %]
        </div>
        <div class="Clear"></div>
    </div>
</div>

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::Console::Command::Admin::ITSM::Configitem::Delete;

use strict;
use warnings;

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

use parent qw(Kernel::System::Console::BaseCommand);

our @ObjectDependencies = (
    'Kernel::System::GeneralCatalog',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::DateTime',
);

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

    $Self->Description('Delete config items (all, by class (and deployment state) or by number).');
    $Self->AddOption(
        Name        => 'all',
        Description => "Delete all config items.",
        Required    => 0,
        HasValue    => 0,
    );
    $Self->AddArgument(
        Name        => 'accept',
        Description => "Accept delete all or cancel.",
        Required    => 0,
        ValueRegex  => qr/\A(y|n)\z/i,
    );
    $Self->AddOption(
        Name        => 'class',
        Description => "Delete all config items of this class.",
        Required    => 0,
        HasValue    => 1,
        ValueRegex  => qr/.+/,
    );
    $Self->AddOption(
        Name        => 'deployment-state',
        Description => "Delete all config items with this deployment state (ONLY TOGETHER with the --class parameter).",
        Required    => 0,
        HasValue    => 1,
        ValueRegex  => qr/.+/,
    );
    $Self->AddOption(
        Name        => 'configitem-number',
        Description => "Delete listed config items.",
        Required    => 0,
        HasValue    => 1,
        ValueRegex  => qr/\A\d+\z/,
        Multiple    => 1,
    );
    $Self->AddOption(
        Name        => 'all-old-versions',
        Description => "Delete all config item versions except the newest version.",
        Required    => 0,
        HasValue    => 0,
    );
    $Self->AddOption(
        Name        => 'all-but-keep-last-versions',
        Description => "Delete all config item versions but keep the given number of newest versions.",
        Required    => 0,
        HasValue    => 1,
        ValueRegex  => qr/\A\d+\z/,
    );
    $Self->AddOption(
        Name        => 'all-older-than-days-versions',
        Description => "Delete all config item versions older than given number of days.",
        Required    => 0,
        HasValue    => 1,
        ValueRegex  => qr/\A\d+\z/,
    );

    return;
}

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

    my $All               = $Self->GetOption('all');
    my $Class             = $Self->GetOption('class') // '';
    my @ConfigItemNumbers = @{ $Self->GetOption('configitem-number') // [] };
    my $DeploymentState   = $Self->GetOption('deployment-state') // '';
    my $AllOldVersions    = $Self->GetOption('all-old-versions') // '';
    my $AllButKeepLast    = $Self->GetOption('all-but-keep-last-versions') // '';
    my $AllOlderThanDays  = $Self->GetOption('all-older-than-days-versions') // '';

    if (
        !$All
        && !$Class
        && !@ConfigItemNumbers
        && !$DeploymentState
        && !$AllOldVersions
        && !$AllButKeepLast
        && !$AllOlderThanDays
        )
    {
        die
            "Please provide option --all, --class, --configitem-number, --all-old-versions, --all-but-keep-last-versions or --all-older-than-days-versions."
            . " For more details use --help\n";
    }

    my $AllOptionTypeCount;
    for my $Value ( $All, $AllOldVersions, $AllButKeepLast, $AllOlderThanDays ) {
        if ($Value) {
            $AllOptionTypeCount++;
        }
    }
    if ( $AllOptionTypeCount > 1 ) {
        die
            "The options --all, --all-old-versions, --all-but-keep-last-versions and --all-older-than-days-versions can not be mixed. \nFor more details use --help\n";
    }
    if ( $AllOptionTypeCount && ( $Class || @ConfigItemNumbers || $DeploymentState ) ) {
        die
            "The options --all, --all-old-versions, --all-but-keep-last-versions and --all-older-than-days-versions can not used together with any other option. \nFor more details use --help\n";
    }

    if ( $DeploymentState && !$Class ) {
        die
            "Deleting all config items with this deployment state is posible ONLY TOGETHER with the --class parameter. \nFor more details use --help\n";
    }

    if ( @ConfigItemNumbers && ( $Class || $DeploymentState ) ) {
        die
            "The option --configitem-number can not be used together with the --class or the --deployment-state parameter. \nFor more details use --help\n";
    }

    return;
}

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

    $Self->Print("<yellow>Deleting config items...</yellow>\n\n");

    my $All               = $Self->GetOption('all');
    my $Class             = $Self->GetOption('class') // '';
    my @ConfigItemNumbers = @{ $Self->GetOption('configitem-number') // [] };
    my $DeploymentState   = $Self->GetOption('deployment-state') // '';
    my $AllOldVersions    = $Self->GetOption('all-old-versions') // '';
    my $AllButKeepLast    = $Self->GetOption('all-but-keep-last-versions') // '';
    my $AllOlderThanDays  = $Self->GetOption('all-older-than-days-versions') // '';

    # delete all config items
    if ($All) {

        # get all config items ids
        my @ConfigItemIDs = @{ $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemSearch() };

        # get number of config items
        my $CICount = scalar @ConfigItemIDs;

        # if there are any CI to delete
        if ($CICount) {

            $Self->Print("<yellow>Are you sure that you want to delete ALL $CICount config items?</yellow>\n");
            $Self->Print("<yellow>This is irrevocable. [y/n] </yellow>\n");
            my $Confirmation = $Self->GetArgument('accept');
            chomp( $Confirmation = lc <STDIN> ) if !defined $Confirmation;

            # if the user confirms the deletion
            if ( $Confirmation eq 'y' ) {

                # delete config items
                $Self->Print("<green>Deleting all config items...</green>\n");
                $Self->DeleteConfigItems( ConfigItemIDs => \@ConfigItemIDs );
            }
            else {
                $Self->Print("<yellow>Command delete was canceled</yellow>\n");
                return $Self->ExitCodeOk();
            }
        }
        else {
            $Self->Print("<yellow>There are NO config items to delete.</yellow>\n");
        }
    }

    # delete listed config items
    elsif ( IsArrayRefWithData( \@ConfigItemNumbers ) ) {

        my @ConfigItemIDs;

        for my $ConfigItemNumber (@ConfigItemNumbers) {

            # checks the validity of the config item id
            my $ID = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemLookup(
                ConfigItemNumber => $ConfigItemNumber,
            );

            if ($ID) {
                push @ConfigItemIDs, $ID;
            }
            else {
                $Self->Print("<yellow>Unable to find config item $ConfigItemNumber.</yellow>\n");
            }
        }

        # delete config items (if any valid number was given)
        if (@ConfigItemIDs) {
            $Self->Print("<yellow>Deleting specified config items...</yellow>\n");
            $Self->DeleteConfigItems( ConfigItemIDs => \@ConfigItemIDs );
        }
    }

    # delete config items that belong to the class
    elsif ($Class) {

        my @ConfigItemIDs;

        # get class list
        my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
            Class => 'ITSM::ConfigItem::Class',
            Valid => 0,
        );

        # invert the hash to have the classes names as keys
        my %ClassName2ID = reverse %{$ClassList};

        if ( $ClassName2ID{$Class} ) {
            my $ID = $ClassName2ID{$Class};

            # define the search param for the class search
            my %SearchParam = (
                ClassIDs => [$ID],
            );

            # also a deployment state is given
            if ($DeploymentState) {

                # get deployment state list
                my $DeploymentStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
                    Class => 'ITSM::ConfigItem::DeploymentState',
                );

                # invert the hash to have the deployment state names as keys
                my %DeploymentState2ID = reverse %{$DeploymentStateList};

                # if the deployment state is valid
                if ( $DeploymentState2ID{$DeploymentState} ) {

                    # get the deployment state id
                    my $ID = $DeploymentState2ID{$DeploymentState};

                    # add search parameter
                    $SearchParam{DeplStateIDs} = [$ID];
                }
                else {
                    $Self->PrintError("Unable to find deployment state $DeploymentState.");
                    return $Self->ExitCodeError();
                }
            }

            # get ids of this class (and maybe deployment state) config items
            @ConfigItemIDs = @{
                $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemSearch(%SearchParam)
            };
        }
        else {
            $Self->PrintError("Unable to find class name $Class.");
            return $Self->ExitCodeError();
        }

        # delete config items (if any valid number was given)
        if (@ConfigItemIDs) {
            $Self->Print("<yellow>Deleting config items that belong to the class $Class...</yellow>\n");
            $Self->DeleteConfigItems( ConfigItemIDs => \@ConfigItemIDs );
        }
        else {
            $Self->Print("<yellow>There are no config items that belong to the class $Class...</yellow>\n");
        }
    }

    # delete versions older than xx days from all config items
    elsif ($AllOlderThanDays) {

        my $OlderDateDTObject = $Kernel::OM->Create('Kernel::System::DateTime');
        $OlderDateDTObject->Subtract(
            Days => $AllOlderThanDays,
        );

        my $VersionsOlderDate = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionListAll(
            OlderDate => $OlderDateDTObject->ToString(),
        );

        # We need to get all versions to make sure that at least one version remains
        # -> if one version of a configitem is younger than the amount of days,
        #    we can delete all Versions received by the "OlderDate" query
        # -> if no version is younger than the amount of days
        #    we have to keep one version of the "OlderDate" query result
        my $VersionsAll = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionListAll();

        my @VersionsToDelete;

        CONFIGITEMID:
        for my $ConfigItemID ( sort keys %{$VersionsOlderDate} ) {

            # number of found older versions of this CI
            my $NumberOfOlderVersions = scalar keys %{ $VersionsOlderDate->{$ConfigItemID} };

            # number of all versions of this CI
            my $NumberOfAllVersions = scalar keys %{ $VersionsAll->{$ConfigItemID} };

            # next if there are no older versions
            next CONFIGITEMID if !$NumberOfOlderVersions;

            # next if there is only one or zero of all versions
            next CONFIGITEMID if $NumberOfAllVersions <= 1;

            # if the amount of Versions we have to delete
            # is exactly the same as the amount of AllVersions
            # we have to keep the last one
            # in order to keep the system working
            #
            # -> so let's start counting at "1" instead of "0"
            # in order to stop deleting before we reach the newest version
            my $Count = 0;
            if ( $NumberOfOlderVersions == $NumberOfAllVersions ) {
                $Count = 1;
            }

            # make sure that the versions are numerically sorted
            for my $Version ( sort { $a <=> $b } keys %{ $VersionsOlderDate->{$ConfigItemID} } ) {

                if ( $Count < $NumberOfOlderVersions ) {
                    push @VersionsToDelete, $Version;
                }
                $Count++;
            }
        }

        $Self->DeleteConfigItemVersions(
            VersionIDs => \@VersionsToDelete,
            UserID     => 1,
        );
    }

    # delete all config item versions except the newest version
    elsif ($AllOldVersions) {

        my $VersionsAll = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionListAll();

        my @VersionsToDelete;
        if ( IsHashRefWithData($VersionsAll) ) {

            CONFIGITEMID:
            for my $ConfigItemID ( sort keys %{$VersionsAll} ) {

                next CONFIGITEMID if !IsHashRefWithData( $VersionsAll->{$ConfigItemID} );

                # make sure that the versions are numerically sorted
                my @ReducedVersions = sort { $a <=> $b } keys %{ $VersionsAll->{$ConfigItemID} };

                # remove the last (newest) version
                pop @ReducedVersions;

                push @VersionsToDelete, @ReducedVersions;
            }
        }

        $Self->DeleteConfigItemVersions(
            VersionIDs => \@VersionsToDelete,
            UserID     => 1,
        );
    }

    # delete all config item versions but keep the last XX versions
    elsif ($AllButKeepLast) {

        my $VersionsAll = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionListAll();

        my @VersionsToDelete;

        if ( IsHashRefWithData($VersionsAll) ) {

            CONFIGITEMID:
            for my $ConfigItemID ( sort keys %{$VersionsAll} ) {

                next CONFIGITEMID if !IsHashRefWithData( $VersionsAll->{$ConfigItemID} );

                # make sure that the versions are numerically reverse sorted
                my @ReducedVersions = reverse sort { $a <=> $b } keys %{ $VersionsAll->{$ConfigItemID} };

                my $Count = 0;
                @ReducedVersions = grep { $Count++; $Count > $AllButKeepLast } @ReducedVersions;
                push @VersionsToDelete, @ReducedVersions;
            }
        }

        $Self->DeleteConfigItemVersions(
            VersionIDs => \@VersionsToDelete,
            UserID     => 1,
        );
    }
    else {
        $Self->PrintError("No config item for delete.");
    }

    # show successfull output
    $Self->Print("<green>Done.</green>\n");
    return $Self->ExitCodeOk();
}

sub DeleteConfigItems {

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

    my $DeletedCI;

    # delete specified config items
    for my $ConfigItemID ( @{ $Param{ConfigItemIDs} } ) {
        my $True = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemDelete(
            ConfigItemID => $ConfigItemID,
            UserID       => 1,
        );
        if ( !$True ) {
            $Self->PrintError("Unable to delete config item with id $ConfigItemID.");
        }
        else {
            $DeletedCI++;
        }
    }

    $Self->Print("<green>Deleted $DeletedCI config item(s).</green>\n\n");

    return 1;
}

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

    # check needed stuff
    return if !IsArrayRefWithData( $Param{VersionIDs} );

    $Self->Print("<green>Deleting config item versions.</green>\n\n");

    $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionDelete(
        VersionIDs => $Param{VersionIDs},
        UserID     => 1,
    );

    return 1;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::Console::Command::Admin::ITSM::Configitem::ListDuplicates;

use strict;
use warnings;

use Kernel::System::VariableCheck qw(IsArrayRefWithData);

use parent qw(Kernel::System::Console::BaseCommand);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::ITSMConfigItem',
);

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

    $Self->Description('List config items which have a non-unique name.');
    $Self->AddOption(
        Name        => 'class',
        Description => "Check only config items of this class.",
        Required    => 0,
        HasValue    => 1,
        ValueRegex  => qr/.+/,
    );
    $Self->AddOption(
        Name        => 'scope',
        Description => "Define the scope for the uniqueness check (global|class)",
        Required    => 0,
        HasValue    => 1,
        ValueRegex  => qr/\A(global|class)\z/,
    );
    $Self->AddOption(
        Name        => 'all-states',
        Description => "Also check config items in non-productive states.",
        Required    => 0,
        HasValue    => 0,
    );

    return;
}

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

    # get class argument
    my $Class = $Self->GetOption('class') // '';

    if ($Class) {

        # get class list
        my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
            Class => 'ITSM::ConfigItem::Class',
        );

        # invert the hash to have the classes' names as keys
        my %ClassName2ID = reverse %{$ClassList};

        # check, whether this class exists
        if ( $ClassName2ID{$Class} ) {
            my $ID = $ClassName2ID{$Class};

            # get ids of this class' config items
            $Self->{SearchCriteria}->{ClassIDs} = [$ID];
        }
        else {
            die "Class $Class does not exist...\n";
        }
    }

    return;
}

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

    $Self->Print("<yellow>Listing duplicates of config items...</yellow>\n\n");

    if ( !$Self->GetOption('all-states') ) {

        my $StateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
            Class       => 'ITSM::ConfigItem::DeploymentState',
            Preferences => {
                Functionality => [ 'preproductive', 'productive' ],
            },
        );

        my $DeploymentStateIDs = [ keys %{$StateList} ];

        $Self->{SearchCriteria}->{DeplStateIDs} = [ keys %{$StateList} ];

    }

    # get ITSMConfigitem object
    my $ITSMConfigitemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    # get all config items ids
    my @ConfigItemIDs = @{ $ITSMConfigitemObject->ConfigItemSearch( %{ $Self->{SearchCriteria} } ) };

    # get number of config items
    my $CICount = scalar @ConfigItemIDs;

    # get class argument
    my $Class = $Self->GetOption('class') // '';

    # if there are any CI to check
    if ($CICount) {

        # if the scope was explicitely defined, set it, otherwise this script will fall back to the
        # value set in SysConfig
        my $Scope = $Self->GetOption('scope');
        if ($Scope) {
            $Kernel::OM->Get('Kernel::Config')->Set(
                Key   => 'UniqueCIName::UniquenessCheckScope',
                Value => $Scope,
            );
        }

        if ($Class) {
            $Self->Print("<yellow>Checking config items of class $Class...</yellow>\n");
        }
        else {
            $Self->Print("<yellow>Checking all config items...\n</yellow>\n");
        }

        $Self->Print( "<green>" . ( '=' x 69 ) . "</green>\n" );

        my $DuplicatesFound = 0;

        # check config items
        CONFIGITEMID:
        for my $ConfigItemID (@ConfigItemIDs) {

            # get the attributes of this config item
            my $ConfigItem = $ITSMConfigitemObject->ConfigItemGet(
                ConfigItemID => $ConfigItemID,
            );

            next CONFIGITEMID if !$ConfigItem->{LastVersionID};

            # get the latest version of this config item
            my $Version = $ITSMConfigitemObject->VersionGet(
                VersionID  => $ConfigItem->{LastVersionID},
                XMLDataGet => 1,
            );

            next CONFIGITEMID if !$Version;

            if ( !$Version->{Name} ) {
                $Self->Print("<red>Skipping ConfigItem $ConfigItemID as it doesn't have a name.\n</red>\n");
                next CONFIGITEMID;
            }

            my $Duplicates = $ITSMConfigitemObject->UniqueNameCheck(
                ConfigItemID => $ConfigItemID,
                ClassID      => $ConfigItem->{ClassID},
                Name         => $Version->{Name}
            );

            if ( IsArrayRefWithData($Duplicates) ) {

                $DuplicatesFound = 1;

                my @DuplicateData;

                for my $DuplicateID ( @{$Duplicates} ) {

                    # get the # of the duplicate
                    my $DuplicateConfigItem = $ITSMConfigitemObject->ConfigItemGet(
                        ConfigItemID => $DuplicateID,
                    );

                    my $DuplicateVersion = $ITSMConfigitemObject->VersionGet(
                        VersionID => $DuplicateConfigItem->{LastVersionID},
                    );

                    push @DuplicateData, $DuplicateVersion;
                }

                $Self->Print(
                    "<yellow>ConfigItem $Version->{Number} (Name: $Version->{Name}, ConfigItemID: "
                        . "$Version->{ConfigItemID}) has the following duplicates:</yellow>\n"
                );

                # list all the details of the duplicates
                for my $DuplicateVersion (@DuplicateData) {
                    print "\n";
                    $Self->Print(
                        "<green>\t * $DuplicateVersion->{Number} (ConfigItemID: "
                            . "$DuplicateVersion->{ConfigItemID})</green>\n"
                    );
                }

                $Self->Print( "<green>" . ( '-' x 69 ) . "</green>\n" );
            }
        }

        if ($DuplicatesFound) {
            $Self->Print("<green>Finished checking for duplicate names.\n</green>\n");
        }
        else {
            $Self->Print("<yellow>No duplicate names found.\n</yellow>\n");
        }
    }
    else {
        $Self->Print("<yellow>There are NO config items to check.\n</yellow>\n");
    }

    $Self->Print("<green>Done.</green>\n");
    return $Self->ExitCodeOk();

}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6Q29uc29sZTo6Q29tbWFuZDo6QWRtaW46OklUU006OkluY2lkZW50U3RhdGU6OlJlY2FsY3VsYXRlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHBhcmVudCBxdyhLZXJuZWw6OlN5c3RlbTo6Q29uc29sZTo6QmFzZUNvbW1hbmQpOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06OkdlbmVyYWxDYXRhbG9nJywKICAgICdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpTZXJ2aWNlJywKKTsKCnN1YiBDb25maWd1cmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAkU2VsZi0+RGVzY3JpcHRpb24oJ1JlY2FsY3VsYXRlcyB0aGUgaW5jaWRlbnQgc3RhdGUgb2YgY29uZmlnIGl0ZW1zLicpOwoKICAgIHJldHVybjsKfQoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICRTZWxmLT5QcmludCgiPHllbGxvdz5SZWNhbGN1bGF0aW5nIHRoZSBpbmNpZGVudCBzdGF0ZSBvZiBjb25maWcgaXRlbXMuLi48L3llbGxvdz5cblxuIik7CgogICAgIyBnZXQgY2xhc3MgbGlzdAogICAgbXkgJENsYXNzTGlzdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtTGlzdCgKICAgICAgICBDbGFzcyA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgKTsKCiAgICAjIGdldCB0aGUgdmFsaWQgY2xhc3MgaWRzCiAgICBteSBAVmFsaWRDbGFzc0lEcyA9IHNvcnQga2V5cyAleyRDbGFzc0xpc3R9OwoKICAgICMgZ2V0IGFsbCBjb25maWcgaXRlbXMgaWRzIGZvcm0gYWxsIHZhbGlkIGNsYXNzZXMKICAgIG15ICRDb25maWdJdGVtc0lEc1JlZiA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbScpLT5Db25maWdJdGVtU2VhcmNoKAogICAgICAgIENsYXNzSURzID0+IFxAVmFsaWRDbGFzc0lEcywKICAgICk7CgogICAgIyBnZXQgbnVtYmVyIG9mIGNvbmZpZyBpdGVtcwogICAgbXkgJENJQ291bnQgPSBzY2FsYXIgQHskQ29uZmlnSXRlbXNJRHNSZWZ9OwoKICAgICRTZWxmLT5QcmludCgiPHllbGxvdz5SZWNhbGN1bGF0aW5nIGluY2lkZW50IHN0YXRlIGZvciAkQ0lDb3VudCBjb25maWcgaXRlbXMuPC95ZWxsb3c+XG4iKTsKCiAgICAjIFJlbWVtYmVyIGNvbmZpZyBpdGVtIHJlc3VsdHMgdGhyb3VnaCBtdWx0aXBsZSBydW5zIG9mIEN1ckluY2lTdGF0ZVJlY2FsYygpLgogICAgbXkgJU5ld0NvbmZpZ0l0ZW1JbmNpZGVudFN0YXRlOwogICAgbXkgJVNjYW5uZWRDb25maWdJdGVtSURzOwoKICAgIG15ICRDb3VudCA9IDA7CiAgICBDT05GSUdJVEVNOgogICAgZm9yIG15ICRDb25maWdJdGVtSUQgKCBAeyRDb25maWdJdGVtc0lEc1JlZn0gKSB7CgogICAgICAgIG15ICRTdWNjZXNzID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtJyktPkN1ckluY2lTdGF0ZVJlY2FsYygKICAgICAgICAgICAgQ29uZmlnSXRlbUlEICAgICAgICAgICAgICAgPT4gJENvbmZpZ0l0ZW1JRCwKICAgICAgICAgICAgTmV3Q29uZmlnSXRlbUluY2lkZW50U3RhdGUgPT4gXCVOZXdDb25maWdJdGVtSW5jaWRlbnRTdGF0ZSwKICAgICAgICAgICAgU2Nhbm5lZENvbmZpZ0l0ZW1JRHMgICAgICAgPT4gXCVTY2FubmVkQ29uZmlnSXRlbUlEcywKICAgICAgICApOwoKICAgICAgICBpZiAoICEkU3VjY2VzcyApIHsKICAgICAgICAgICAgJFNlbGYtPlByaW50KCI8cmVkPi4uLiBjb3VsZCBub3QgcmVjYWxjdWxhdGUgaW5jaWRlbnQgc3RhdGUgZm9yIGNvbmZpZyBpdGVtIGlkICckQ29uZmlnSXRlbUlEJyE8L3JlZD5cbiIpOwogICAgICAgICAgICBuZXh0IENPTkZJR0lURU07CiAgICAgICAgfQoKICAgICAgICAkQ291bnQrKzsKCiAgICAgICAgaWYgKCAkQ291bnQgJSAxMDAgPT0gMCApIHsKICAgICAgICAgICAgJFNlbGYtPlByaW50KCI8Z3JlZW4+Li4uICRDb3VudCBjb25maWcgaXRlbXMgcmVjYWxjdWxhdGVkLjwvZ3JlZW4+XG4iKTsKICAgICAgICB9CiAgICB9CgogICAgJFNlbGYtPlByaW50KCJcbjxncmVlbj5SZWFkeS4gUmVjYWxjdWxhdGVkICRDb3VudCBjb25maWcgaXRlbXMuPC9ncmVlbj5cblxuIik7CgogICAgIyBnZXQgc2VydmljZSBvYmplY3QKICAgIG15ICRTZXJ2aWNlT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlNlcnZpY2UnKTsKCiAgICAjIGdldCBsaXN0IG9mIGFsbCBzZXJ2aWNlcyAodmFsaWQgYW5kIGludmFsaWQpCiAgICBteSAlU2VydmljZUxpc3QgPSAkU2VydmljZU9iamVjdC0+U2VydmljZUxpc3QoCiAgICAgICAgVmFsaWQgID0+IDAsCiAgICAgICAgVXNlcklEID0+IDEsCiAgICApOwoKICAgIG15ICROdW1iZXJPZlNlcnZpY2VzID0gc2NhbGFyIGtleXMgJVNlcnZpY2VMaXN0OwoKICAgICRTZWxmLT5QcmludCgKICAgICAgICAiPGdyZWVuPlJlc2V0dGluZyBTZXJ2aWNlUHJlZmVyZW5jZXMgJ0N1ckluY2lTdGF0ZVR5cGVGcm9tQ0lzJyBmb3IgJE51bWJlck9mU2VydmljZXMgc2VydmljZXMuLi48L2dyZWVuPlxuIgogICAgKTsKCiAgICBmb3IgbXkgJFNlcnZpY2VJRCAoIHNvcnQga2V5cyAlU2VydmljZUxpc3QgKSB7CgogICAgICAgICMgdXBkYXRlIHRoZSBjdXJyZW50IGluY2lkZW50IHN0YXRlIHR5cGUgZnJvbSBDSXMgb2YgdGhlIHNlcnZpY2Ugd2l0aCBhbiBlbXB0eSB2YWx1ZQogICAgICAgICMgdGhpcyBpcyBuZWNlc3NhcnkgdG8gZm9yY2UgYSByZWNhbGN1bGF0aW9uIG9uIGEgU2VydmljZUdldCgpCiAgICAgICAgJFNlcnZpY2VPYmplY3QtPlNlcnZpY2VQcmVmZXJlbmNlc1NldCgKICAgICAgICAgICAgU2VydmljZUlEID0+ICRTZXJ2aWNlSUQsCiAgICAgICAgICAgIEtleSAgICAgICA9PiAnQ3VySW5jaVN0YXRlVHlwZUZyb21DSXMnLAogICAgICAgICAgICBWYWx1ZSAgICAgPT4gJycsCiAgICAgICAgICAgIFVzZXJJRCAgICA9PiAxLAogICAgICAgICk7CiAgICB9CgogICAgJFNlbGYtPlByaW50KCI8Z3JlZW4+UmVhZHkuPC9ncmVlbj5cbiIpOwogICAgcmV0dXJuICRTZWxmLT5FeGl0Q29kZU9rKCk7Cgp9CgoxOwoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIHNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoTDxodHRwczovL290cnMub3JnLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQ+LgoKPWN1dAo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6Q29uc29sZTo6Q29tbWFuZDo6TWFpbnQ6OklUU006OkNvbmZpZ2l0ZW06OkRlZmluaXRpb25QZXJsMllBTUw7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgS2VybmVsOjpMYW5ndWFnZSBxdyhUcmFuc2xhdGFibGUpOwp1c2UgS2VybmVsOjpTeXN0ZW06OlZhcmlhYmxlQ2hlY2sgcXcoOmFsbCk7Cgp1c2UgcGFyZW50IHF3KEtlcm5lbDo6U3lzdGVtOjpDb25zb2xlOjpCYXNlQ29tbWFuZCk7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6REInLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycsCiAgICAnS2VybmVsOjpTeXN0ZW06OllBTUwnLAopOwoKc3ViIENvbmZpZ3VyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICRTZWxmLT5EZXNjcmlwdGlvbignQ29udmVydCBjb25maWcgaXRlbXMgZGVmaW5pdGlvbnMgZnJvbSBsZWdhY3kgUGVybCBjb2RlIHRvIGEgWUFNTCBzdHJ1Y3R1cmUuJyk7CgogICAgcmV0dXJuOwp9CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgJFNlbGYtPlByaW50KCI8eWVsbG93PkNvbnZlcnRpbmcgY29uZmlnIGl0ZW0gZGVmaW5pdGlvbnMuLi48L3llbGxvdz5cblxuIik7CgogICAgbXkgJENsYXNzTGlzdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtTGlzdCgKICAgICAgICBDbGFzcyA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgICAgIFZhbGlkID0+IDAsCgogICAgICAgICNQcmVmZXJlbmNlcyAgID0+IHsgfSwKICAgICk7CgogICAgbXkgJERCT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkRCJyk7CgogICAgcmV0dXJuIGlmICEkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6REInKS0+UHJlcGFyZSgKICAgICAgICBTUUwgPT4gIgogICAgICAgICAgICBTRUxFQ1QgaWQsIGNsYXNzX2lkLCBjb25maWdpdGVtX2RlZmluaXRpb24sIHZlcnNpb24KICAgICAgICAgICAgRlJPTSBjb25maWdpdGVtX2RlZmluaXRpb24KICAgICAgICAgICAgV0hFUkUgY29uZmlnaXRlbV9kZWZpbml0aW9uIE5PVCBMSUtFICctLS0lJwogICAgICAgICAgICBPUkRFUiBCWSBjbGFzc19pZCBBU0MsIHZlcnNpb24gQVNDIiwKICAgICk7CgogICAgbXkgQERlZmluaXRpb25MaXN0OwogICAgd2hpbGUgKCBteSBAUm93ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkRCJyktPkZldGNocm93QXJyYXkoKSApIHsKICAgICAgICBteSAlRGVmaW5pdGlvbjsKICAgICAgICAkRGVmaW5pdGlvbntEZWZpbml0aW9uSUR9ID0gJFJvd1swXTsKICAgICAgICAkRGVmaW5pdGlvbntDbGFzc0lEfSAgICAgID0gJFJvd1sxXTsKICAgICAgICAkRGVmaW5pdGlvbntEZWZpbml0aW9ufSAgID0gJFJvd1syXTsKICAgICAgICAkRGVmaW5pdGlvbntWZXJzaW9ufSAgICAgID0gJFJvd1szXTsKCiAgICAgICAgcHVzaCBARGVmaW5pdGlvbkxpc3QsIFwlRGVmaW5pdGlvbjsKICAgIH0KCiAgICBpZiAoICFzY2FsYXIgQERlZmluaXRpb25MaXN0ICkgewogICAgICAgICRTZWxmLT5QcmludCgiICBObyBjb25maWcgaXRlbXMgdXNpbmcgbGVnYWN5IFBlcmwgY29kZSBmb3VuZFxuXG4iKTsKCiAgICAgICAgJFNlbGYtPlByaW50KCI8Z3JlZW4+RG9uZS48L2dyZWVuPlxuIik7CiAgICAgICAgcmV0dXJuICRTZWxmLT5FeGl0Q29kZU9rKCk7CiAgICB9CgogICAgbXkgJFN1Y2Nlc3MgICAgPSAxOwogICAgbXkgJFlBTUxPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6WUFNTCcpOwoKICAgIERFRklOSVRJT046CiAgICBmb3IgbXkgJERlZmluaXRpb24gKEBEZWZpbml0aW9uTGlzdCkgewoKICAgICAgICBteSAkQ3VycmVudFJlY29yZAogICAgICAgICAgICA9ICIgICBQcm9jZXNzaW5nIGRlZmluaXRpb24gZm9yIGNsYXNzOiA8eWVsbG93PiRDbGFzc0xpc3QtPnsgJERlZmluaXRpb24tPntDbGFzc0lEfSB9PC95ZWxsb3c+IHZlcnNpb246IDx5ZWxsb3c+JERlZmluaXRpb24tPntWZXJzaW9ufTwveWVsbG93Pi4uLiI7CgogICAgICAgICMgVHJ5IHRvIGV2YWx1YXRlIHBlcmwgc3RydWN0dXJlLgogICAgICAgIG15ICRQZXJsU3RydWN0dXJlID0gZXZhbCAkRGVmaW5pdGlvbi0+e0RlZmluaXRpb259OyAgICAjIyBubyBjcml0aWMKICAgICAgICBpZiAoJEApIHsKICAgICAgICAgICAgbXkgJEVycm9yID0gJEA7CiAgICAgICAgICAgICRTZWxmLT5QcmludCgiJEN1cnJlbnRSZWNvcmQgPHJlZD5GYWlsZWQgZXZhbHVhdGUgcGVybCBzdHJ1Y3R1cmU6ICRFcnJvcjwvcmVkPlxuIik7CiAgICAgICAgICAgICRTdWNjZXNzID0gMDsKICAgICAgICAgICAgbmV4dCBERUZJTklUSU9OOwogICAgICAgIH0KCiAgICAgICAgIyBUcnkgdG8gY29udmVydCBmcm9tIFBlcmwgdG8gWUFNTC4KICAgICAgICBteSAkWUFNTFN0ciA9ICRZQU1MT2JqZWN0LT5EdW1wKAogICAgICAgICAgICBEYXRhID0+ICRQZXJsU3RydWN0dXJlLAogICAgICAgICk7CiAgICAgICAgaWYgKCAhJFlBTUxTdHIgKSB7CiAgICAgICAgICAgICRTZWxmLT5QcmludCgiJEN1cnJlbnRSZWNvcmQgPHJlZD5GYWlsZWQgdG8gY29udmVydCB0byBZQU1MPC9yZWQ+XG4iKTsKICAgICAgICAgICAgJFN1Y2Nlc3MgPSAwOwogICAgICAgICAgICBuZXh0IERFRklOSVRJT047CiAgICAgICAgfQoKICAgICAgICByZXR1cm4gaWYgISREQk9iamVjdC0+RG8oCiAgICAgICAgICAgIFNRTCA9PiAnCiAgICAgICAgICAgICAgICBVUERBVEUgY29uZmlnaXRlbV9kZWZpbml0aW9uCiAgICAgICAgICAgICAgICBTRVQgY29uZmlnaXRlbV9kZWZpbml0aW9uID0gPwogICAgICAgICAgICAgICAgV0hFUkUgaWQgPSA/JywKICAgICAgICAgICAgQmluZCA9PiBbIFwkWUFNTFN0ciwgXCREZWZpbml0aW9uLT57RGVmaW5pdGlvbklEfSBdLAogICAgICAgICk7CgogICAgICAgICRTZWxmLT5QcmludCgiJEN1cnJlbnRSZWNvcmQgPGdyZWVuPk9LPC9ncmVlbj5cbiIpOwoKICAgIH0KCiAgICBpZiAoICEkU3VjY2VzcyApIHsKICAgICAgICAkU2VsZi0+UHJpbnRFcnJvcigiXG5VbmFibGUgdG8gY29udmVydCBhbGwgY29uZmlnIGl0ZW0gZGVmaW5pdGlvbnMuIik7CiAgICAgICAgcmV0dXJuICRTZWxmLT5FeGl0Q29kZUVycm9yKCk7CiAgICB9CgogICAgJFNlbGYtPlByaW50KCJcbjxncmVlbj5Eb25lLjwvZ3JlZW4+XG4iKTsKICAgIHJldHVybiAkU2VsZi0+RXhpdENvZGVPaygpOwp9CgoxOwo=
# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::System::DynamicField::ConfigItem;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::System::DateTime',
    'Kernel::System::DynamicField',
    'Kernel::System::DynamicField::Backend',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::Log',
);

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

=head1 NAME

Kernel::System::DynamicField::ConfigItem - Dynamic field ConfigItem lib

=head1 PUBLIC INTERFACE

=head2 new()

    Don't use the constructor directly, use the ObjectManager instead:

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

=cut

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

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

    my @ConfigItemStandardFields
        = qw(ConfigItemID Name ClassID Class DefinitionID DeplStateID DeplState InciStateID InciState);
    my %ConfigItemStandardFields = map { $_ => 1 } @ConfigItemStandardFields;
    $Self->{ConfigItemStandardFields} = \%ConfigItemStandardFields;

    return $Self;
}

# TODO: we should use the same naming like DF-LDAP (AutoFill) or standardize these

=head2 GetAdditionalDFStorageData()

    Returns data for dynamic fields that should be filled based on the source dynamic field
    configuration and the selected config items.

    my $AdditionalDFStorageData = $DynamicFieldConfigItemObject->GetAdditionalDFStorageData(
        SourceDynamicFieldName => 'MyConfigItem',
        SelectedConfigItemIDs  => [ 45, 13, ],
        StorageType            => 'Frontend', # or 'Backend'
        UserID                 => 74,
    );

    Returns:

    {
        'MyDynamicField1' => $Value, # Can be anything. If the dynamic field is a multiselect, this is an array.
    },

=cut

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

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

    NEEDED:
    for my $Needed (qw(SourceDynamicFieldName SelectedConfigItemIDs StorageType UserID)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $DynamicFieldConfig = $DynamicFieldObject->DynamicFieldGet(
        Name => $Param{SourceDynamicFieldName},
    );
    return if !IsHashRefWithData($DynamicFieldConfig);

    return if $DynamicFieldConfig->{FieldType} ne 'ConfigItemDropdown'
        && $DynamicFieldConfig->{FieldType} ne 'ConfigItemMultiselect';

    # Only ticket dynamic fields are supported.
    return if $DynamicFieldConfig->{ObjectType} ne 'Ticket';

    # Evaluate selected config item IDs.
    $Param{SelectedConfigItemIDs} //= [];
    my @SelectedConfigItemIDs =
        grep {
        defined $_
            && $_ =~ m{\A[1-9]\d*\z}
        }
        @{ $Param{SelectedConfigItemIDs} };

    # Evaluate additional dynamic field storage configs.
    my $AdditionalDFStorageConfigs = $DynamicFieldConfig->{Config}->{AdditionalDFStorage};
    return if !IsArrayRefWithData($AdditionalDFStorageConfigs);

    # Limit to those storage configs matching the given storage type.
    # Regex for storage type because the config value can also be 'FrontendBackend'.
    my @AdditionalDFStorageConfigs = grep { $_->{Type} =~ m{$Param{StorageType}} } @{$AdditionalDFStorageConfigs};

    #
    # Assemble data of config items.
    #
    my @ConfigItemData;

    CONFIGITEMID:
    for my $ConfigItemID (@SelectedConfigItemIDs) {
        my $ConfigItemData = $Self->_GetConfigItemData(
            ConfigItemID => $ConfigItemID,
            UserID       => $Param{UserID},
        );
        next CONFIGITEMID if !IsHashRefWithData($ConfigItemData);

        push @ConfigItemData, $ConfigItemData;
    }

    # Note: @ConfigItemData can be empty at this point.
    # In this case, additional dynamic fields will be set empty.
    # See issue #12: https://git.znuny.com/Znuny/Private/Znuny-DynamicFieldConfigItem/-/issues/12

    my $AdditionalDFStorageData = $Self->_GetAdditionalDFStorageData(
        ConfigItemData             => \@ConfigItemData,
        AdditionalDFStorageConfigs => \@AdditionalDFStorageConfigs,
    );

    return $AdditionalDFStorageData;
}

=head2 StoreDynamicFieldValues()

    Stores dynamic field values with given additional dynamic field storage data.

    my $StoredDynamicFields = $DynamicFieldConfigItemObject->StoreDynamicFieldValues(
        TicketID                => 56,
        AdditionalDFStorageData => {
            # hash returned by GetAdditionalDFStorageData()
        },
        UserID => 213,
    );

    Returns array ref with names of dynamic fields that have been updated successfully.

=cut

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

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

    NEEDED:
    for my $Needed (qw(TicketID AdditionalDFStorageData UserID)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $AdditionalDFStorageData = $Param{AdditionalDFStorageData};
    return if !IsHashRefWithData($AdditionalDFStorageData);

    my @StoredDynamicFields;

    DYNAMICFIELDNAME:
    for my $DynamicFieldName ( sort keys %{$AdditionalDFStorageData} ) {
        next DYNAMICFIELDNAME if !exists $Param{AdditionalDFStorageData}->{$DynamicFieldName};

        my $DynamicFieldConfig = $DynamicFieldObject->DynamicFieldGet(
            Name => $DynamicFieldName,
        );
        next DYNAMICFIELDNAME if !IsHashRefWithData($DynamicFieldConfig);

        # Additional dynamic field storage is only supported for ticket dynamic fields.
        next DYNAMICFIELDNAME if $DynamicFieldConfig->{ObjectType} ne 'Ticket';

        my $ValueStored;
        if ( defined $Param{AdditionalDFStorageData}->{$DynamicFieldName} ) {
            $ValueStored = $DynamicFieldBackendObject->ValueSet(
                DynamicFieldConfig => $DynamicFieldConfig,
                ObjectID           => $Param{TicketID},
                Value              => $Param{AdditionalDFStorageData}->{$DynamicFieldName},
                UserID             => $Param{UserID},
            );
        }
        else {
            $ValueStored = $DynamicFieldBackendObject->ValueSet(
                DynamicFieldConfig => $DynamicFieldConfig,
                ObjectID           => $Param{TicketID},
                UserID             => $Param{UserID},
            );
        }

        next DYNAMICFIELDNAME if !$ValueStored;

        push @StoredDynamicFields, $DynamicFieldName;
    }

    return \@StoredDynamicFields;
}

=head2 _GetConfigItemData()

    Returns data of the last version in a sane form for the given config item ID.

    my $ConfigItemData = $DynamicFieldConfigItemObject->_GetConfigItemData(
        ConfigItemID => 52,
        UserID       => 54,
    );

    Returns a hash reference.

=cut

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

    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $LogObject        = $Kernel::OM->Get('Kernel::System::Log');

    NEEDED:
    for my $Needed (qw(ConfigItemID UserID)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $ConfigItemVersion = $ConfigItemObject->VersionGet(
        ConfigItemID => $Param{ConfigItemID},
        XMLDataGet   => 1,
    );
    return if !IsHashRefWithData($ConfigItemVersion);

    my %ConfigItemData;
    $ConfigItemData{XMLData} = {};
    if ( IsHashRefWithData( $ConfigItemVersion->{XMLData}->[1]->{Version}->[1] ) ) {
        $Self->_ParseXML2Data(
            %Param,
            Result          => $ConfigItemData{XMLData},
            Data            => $ConfigItemVersion->{XMLData}->[1]->{Version}->[1],
            XMLDataMultiple => 1,
        );
    }

    for my $Field ( sort keys %{ $Self->{ConfigItemStandardFields} } ) {
        $ConfigItemData{$Field} = $ConfigItemVersion->{$Field};
    }
    $ConfigItemData{XMLDefinition} = $ConfigItemVersion->{XMLDefinition};

    return \%ConfigItemData;
}

=head2 _GetAdditionalDFStorageData()

    Returns data which should be stored in additional dynamic fields for the given
    config item data.

    my $AdditionalDFStorageData = $DynamicFieldConfigItemObject->_GetAdditionalDFStorageData(
        ConfigItemData => [ # can be empty/undef, additional dynamic fields will be set empty in this case.
            {
                # Data returned by _GetConfigItemData()
            },
            {
                # Data returned by _GetConfigItemData()
            },
            # ...
        ],
        AdditionalDFStorageConfigs => [
            {
                DynamicFieldName => 'MyDynamicField',
                ConfigItemKey    => 'CPU::1',
                Type             => 'Frontend',
            },
            # ...
        ],
    );

    Returns hash with data to be stored in dynamic fields.
    Dynamic field name => value(s)

=cut

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

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

    NEEDED:
    for my $Needed (qw(AdditionalDFStorageConfigs)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my %DynamicFieldValues;

    ADDITIONALDFSTORAGECONFIG:
    for my $AdditionalDFStorageConfig ( @{ $Param{AdditionalDFStorageConfigs} } ) {
        my $DynamicFieldConfig = $DynamicFieldObject->DynamicFieldGet(
            Name => $AdditionalDFStorageConfig->{DynamicField},
        );
        next ADDITIONALDFSTORAGECONFIG if !IsHashRefWithData($DynamicFieldConfig);

        # Only ticket dynamic fields are supported
        next ADDITIONALDFSTORAGECONFIG if $DynamicFieldConfig->{ObjectType} ne 'Ticket';

        # Initialize every configured additional DF storage field with empty string
        # so that fields which are empty in the selected config item(s) will be cleared
        # and no data will be left from previously selected config item(s).
        # Also this is the default value for the dynamic field if it will be cleared
        # because no config item is selected (anymore).
        $DynamicFieldValues{ $AdditionalDFStorageConfig->{DynamicField} } = '';

        my @ConfigItemFieldRawValues;

        my $XMLValueLookup;
        my $ConfigItemKey = $AdditionalDFStorageConfig->{ConfigItemKey};
        my $AttributePath;

        # check if _Value postfix is set
        if ( $ConfigItemKey =~ m{_Value\z} ) {
            $ConfigItemKey =~ s{_Value\z}{}ms;
            $AttributePath = $ConfigItemKey;
            $AttributePath =~ s{::\d+}{}ms;
            $XMLValueLookup = 1;
        }

        CONFIGITEMDATA:
        for my $ConfigItemData ( @{ $Param{ConfigItemData} } ) {
            my $StringifiedXMLData = $Self->_XMLData2StringifiedXMLData(
                XMLData => $ConfigItemData->{XMLData},
            );

            my $ConfigItemFieldRawValue;
            my $Item;

            if ($XMLValueLookup) {
                $Item = $ConfigItemObject->DefinitionAttributeInfo(
                    AttributePath => $AttributePath,                     # 'HardDisk::Capacity',
                    Definition    => $ConfigItemData->{XMLDefinition},
                );
            }

            # standard field
            if ( $Self->{ConfigItemStandardFields}->{$ConfigItemKey} ) {
                if ($XMLValueLookup) {
                    $ConfigItemFieldRawValue = $ConfigItemObject->XMLValueLookup(
                        Item  => $Item,
                        Value => $ConfigItemData->{$ConfigItemKey},
                    );
                }
                else {
                    $ConfigItemFieldRawValue = $ConfigItemData->{$ConfigItemKey};
                }
            }

            # directly found stringified data
            elsif (
                IsHashRefWithData($StringifiedXMLData)
                && $StringifiedXMLData->{$ConfigItemKey}
                )
            {
                if ($XMLValueLookup) {
                    $ConfigItemFieldRawValue = $ConfigItemObject->XMLValueLookup(
                        Item  => $Item,
                        Value => $StringifiedXMLData->{$ConfigItemKey},
                    );
                }
                else {
                    $ConfigItemFieldRawValue = $StringifiedXMLData->{$ConfigItemKey};
                }
            }

            # check multiple values
            elsif (
                IsHashRefWithData($StringifiedXMLData)
                && $ConfigItemKey !~ m{\:\:\d+}gmx
                )
            {
                my @ConfigItemKeys = grep { $_ =~ m{^$ConfigItemKey} } sort keys %{$StringifiedXMLData};

                for my $ConfigItemKey (@ConfigItemKeys) {
                    if ($XMLValueLookup) {
                        push @{$ConfigItemFieldRawValue}, $ConfigItemObject->XMLValueLookup(
                            Item  => $Item,
                            Value => $StringifiedXMLData->{$ConfigItemKey},
                        );
                    }
                    else {
                        push @{$ConfigItemFieldRawValue}, $StringifiedXMLData->{$ConfigItemKey};
                    }
                }
            }

            if ( IsArrayRefWithData($ConfigItemFieldRawValue) ) {
                push @ConfigItemFieldRawValues, @{$ConfigItemFieldRawValue};
            }
            else {
                push @ConfigItemFieldRawValues, $ConfigItemFieldRawValue;
            }
        }

        my $DynamicFieldValue;
        if (@ConfigItemFieldRawValues) {
            $DynamicFieldValue = $Self->_ConvertConfigItemFieldRawValuesToDynamicFieldValue(
                DynamicFieldType         => $DynamicFieldConfig->{FieldType},
                ConfigItemFieldRawValues => \@ConfigItemFieldRawValues,
            );

            $DynamicFieldValues{ $AdditionalDFStorageConfig->{DynamicField} } = $DynamicFieldValue;
        }
    }

    return \%DynamicFieldValues;
}

=head2 _XMLData2StringifiedXMLData()

Converts XML data structure used by Kernel::System::ZnunyHelper::_ITSMConfigItemVersionAdd
to stringified XMLData 'CPU::1'. This function should be integrated into Znuny::ITSM.

    my $StringifiedXMLData = $DynamicFieldConfigItemObject->_XMLData2StringifiedXMLData(
        Prefix  => $Prefix,
        XMLData => {
            'NIC' => [
                {
                    'Content'    => 'NicName1',
                    'IPoverDHCP' => [
                        {
                            'Content' => 'Yes',
                        }
                    ],
                    'IPAddress' => [
                        {
                            'Content' => '123',
                        },
                    ],
                },
                {
                    'Content'   => 'NicName2',
                    'IPoverDHCP' => [
                        {
                            'Content' => 'No',
                        }
                    ],
                    'IPAddress' => [
                        {
                            'Content' => '456',
                        }
                    ],
                }
            ],
        }
    );

Returns:

    my $StringifiedXMLData = {
        NIC::1                => 'NicName1',
        NIC::1::IPAddress::1  => '123',
        NIC::1::IPoverDHCP::1 => 'Yes',
        NIC::2                => 'NicName2',
        NIC::2::IPAddress::1  => '456',
        NIC::2::IPoverDHCP::1 => 'No',
    };

=cut

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

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    if ( !$Param{XMLData} ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => "Needed parameter 'XMLData'.",
        );
        return;
    }

    my %XMLData = %{ Storable::dclone( $Param{XMLData} ) };
    my $Data;

    ITEM:
    for my $Item ( sort keys %XMLData ) {
        next ITEM if !IsArrayRefWithData( $XMLData{$Item} );

        my $CounterInsert = 1;
        for my $SubItem ( @{ $XMLData{$Item} } ) {
            my $Key = $Item . '::' . $CounterInsert;
            if ( $Param{Prefix} ) {
                $Key = $Param{Prefix} . '::' . $Key;
            }

            if ( $SubItem->{Content} ) {
                $Data->{$Key} = $SubItem->{Content};
                delete $SubItem->{Content};
            }

            my $SubXMLData = $Self->_XMLData2StringifiedXMLData(
                XMLData => $SubItem,
                Prefix  => $Key,
            );

            if ($SubXMLData) {
                %{$Data} = (
                    %{$Data},
                    %{$SubXMLData},
                );
            }
            $CounterInsert++;
        }
    }

    return $Data;
}

=head2 _ConvertConfigItemFieldRawValuesToDynamicFieldValue()

    Converts given raw values (which is always an array) to a value that can be given
    to the dynamic field method ValueSet.

    my $DynamicFieldValue = $DynamicFieldConfigItemObject->_ConvertConfigItemFieldRawValuesToDynamicFieldValue(
        DynamicFieldType         => 'Multiselect',
        ConfigItemFieldRawValues => [
            # ...
        ],
    );

    Returns value for dynamic field method ValueSet.
    Returns nothing if value could not be converted.

=cut

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

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    NEEDED:
    for my $Needed (qw(DynamicFieldType ConfigItemFieldRawValues)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $ConfigItemFieldRawValues = $Param{ConfigItemFieldRawValues} // [];
    return if !IsArrayRefWithData($ConfigItemFieldRawValues);

    my $DynamicFieldType = $Param{DynamicFieldType};

    # Leave arrays as they are for multiselect. Don't check possible values of dynamic field because frontend
    # could have changed them anyways.
    return $ConfigItemFieldRawValues if $DynamicFieldType eq 'Multiselect';

    # Use first value for dropdown.
    return $ConfigItemFieldRawValues->[0] if $DynamicFieldType eq 'Dropdown';

    # Concatenate for text(area).
    if ( $DynamicFieldType =~ m{\AText(Area)?\z} ) {

        # map call: Avoid warnings about undefined values in array.
        my $DynamicFieldValue = join ', ', map { $_ // '' } @{$ConfigItemFieldRawValues};
        return $DynamicFieldValue;
    }

    # Add/remove time for Date(Time).
    # Only consider first value from config items.
    if ( $DynamicFieldType =~ m{\ADate(Time)?\z} ) {
        my $ConfigItemFieldRawValue = $ConfigItemFieldRawValues->[0];

        my $DateTimeObject = $Kernel::OM->Create(
            'Kernel::System::DateTime',
            ObjectParams => {
                String => $ConfigItemFieldRawValue,
            }
        );
        return if !$DateTimeObject;

        return $DateTimeObject->Format( Format => '%Y-%m-%d %H:%M:%S' ) if $DynamicFieldType eq 'DateTime';
        return $DateTimeObject->Format( Format => '%Y-%m-%d' );
    }

    return;
}

#
# Taken from Znuny-Repo: Kernel::System::ZnunyHelper (8e336b82cc4e030fe48778e04c011c12e052397f)
#

=head2 _ParseXML2Data()

this is a internal function for _ITSMVersionGet to parse the additional data
stored in XMLData.

    my $Success = $ZnunyHelperObject->_ParseXML2Data(
        Parent          => $Identifier,          # optional: contains the field name of the parent xml
        Result          => $Result,              # contains the reference to the result hash
        Data            => $Data{$Field}->[1],   # contains the xml hash we want to parse
        XMLDataMultiple => 1,                    # default: 0, This option will return a more complex XMLData structure with multiple element data! Makes sense if you are using CountMin, CountMax etc..
    );

Returns:

    my $Success = 1;

=cut

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

    my $Result          = $Param{Result};
    my $XMLDataMultiple = $Param{XMLDataMultiple};
    my $Parent          = $Param{Parent} || '';
    my %Data            = %{ $Param{Data} || {} };

    FIELD:
    for my $Field ( sort keys %Data ) {
        next FIELD if !IsArrayRefWithData( $Data{$Field} );

        if ($XMLDataMultiple) {
            $Result->{$Field} = [];

            for my $Index ( 1 .. $#{ $Data{$Field} } ) {
                my $Value = $Data{$Field}->[$Index]->{Content};

                my $CurrentResult = {};

                $Self->_ParseXML2Data(
                    %Param,
                    Parent => $Field,
                    Result => $CurrentResult,
                    Data   => $Data{$Field}->[$Index],
                );

                if ( defined $Value ) {
                    $CurrentResult->{Content} = $Value;

                    if ( keys %{$CurrentResult} ) {
                        push @{ $Result->{$Field} }, $CurrentResult;
                    }
                }
            }
        }
        else {
            my $Value = $Data{$Field}->[1]->{Content};

            next FIELD if !defined $Value;

            $Result->{$Field} = $Value;
        }
    }

    return 1;
}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::System::DynamicField::Driver::ConfigItem;

use strict;
use warnings;

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

use mro 'c3';

our @ObjectDependencies = (
    'Kernel::Output::HTML::Layout',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::ITSMConfigItem',
);

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

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

    # Initialize storage of additional dynamic fields in frontend, if configured.
    # Only dynamic fields of object type 'Ticket' are supported for this.
    if ( $Param{DynamicFieldConfig}->{ObjectType} eq 'Ticket' ) {
        my $AdditionalDFStorageConfig = $Param{DynamicFieldConfig}->{Config}->{AdditionalDFStorage};
        if ( IsArrayRefWithData($AdditionalDFStorageConfig) ) {
            my @AdditionalDFStorageConfigForFrontend = grep { $_->{Type} ne 'Backend' }
                @{$AdditionalDFStorageConfig};

            if (@AdditionalDFStorageConfigForFrontend) {
                my $DynamicFieldName = $Param{DynamicFieldConfig}->{Name};

                $LayoutObject->AddJSOnDocumentComplete(
                    Code => "Znuny.DynamicField.ConfigItem.InitAdditionalDFStorage('$DynamicFieldName');",
                );
            }
        }
    }

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    $Param{DynamicFieldConfig}->{Config}->{PossibleValues} = $Self->PossibleValuesGet(%Param);

    return $Self->next::method(%Param);
}

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

    my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

    # Add empty value, none of the config item fields can have a default or pre-selected value
    # because the selectable values may change depending on deployment state.
    my %PossibleValues = (
        '' => '-',
    );

    my $Class = $Param{DynamicFieldConfig}->{Config}->{ConfigItemClass};
    return \%PossibleValues if !$Class;

    # check all CI classes
    my $HashRef = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );
    my %Classes = reverse %{ $HashRef || {} };
    return \%PossibleValues if !$Classes{$Class};

    my $ConfigItemListRef = [];

    # limit selectable config items to those which have one of the configured
    # deployment states
    my $DeplStates = $Param{DynamicFieldConfig}->{Config}->{DeplStates} // [];
    if ( IsArrayRefWithData($DeplStates) ) {
        my $DeplStateNameByID = $GeneralCatalogObject->ItemList(
            Class => 'ITSM::ConfigItem::DeploymentState',
        );
        my %DeplStateIDByName = reverse %{$DeplStateNameByID};

        my %DeplStates   = map  { $_ => 1 } @{$DeplStates};
        my @DeplStateIDs = grep { defined $_ }
            map { $DeplStateIDByName{$_} }
            keys %DeplStates;

        my $ConfigItemIDs = $ConfigItemObject->ConfigItemSearch(
            ClassIDs     => [ $Classes{$Class}, ],
            DeplStateIDs => \@DeplStateIDs,
        );

        for my $ConfigItemID ( @{$ConfigItemIDs} ) {
            my $LastVersion = $ConfigItemObject->VersionGet(
                ConfigItemID => $ConfigItemID,
                XMLDataGet   => 0,
            );

            push @{$ConfigItemListRef}, $LastVersion;
        }
    }
    else {
        # get all config items of that class
        $ConfigItemListRef = $ConfigItemObject->ConfigItemResultList(
            ClassID => $Classes{$Class},
            Start   => 0,
            Limit   => 1_000_000,
        );
    }

    CONFIGITEM:
    for my $ConfigItem ( @{ $ConfigItemListRef || {} } ) {
        $PossibleValues{ $ConfigItem->{ConfigItemID} } = $ConfigItem->{Name};
    }

    return \%PossibleValues;
}

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OkR5bmFtaWNGaWVsZDo6RHJpdmVyOjpDb25maWdJdGVtRHJvcGRvd247Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgS2VybmVsOjpTeXN0ZW06OlZhcmlhYmxlQ2hlY2sgcXcoOmFsbCk7Cgp1c2UgcGFyZW50IHF3KAogICAgS2VybmVsOjpTeXN0ZW06OkR5bmFtaWNGaWVsZDo6RHJpdmVyOjpDb25maWdJdGVtCiAgICBLZXJuZWw6OlN5c3RlbTo6RHluYW1pY0ZpZWxkOjpEcml2ZXI6OkRyb3Bkb3duCik7CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OkR5bmFtaWNGaWVsZDo6RHJpdmVyOjpDb25maWdJdGVtTXVsdGlzZWxlY3Q7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgS2VybmVsOjpTeXN0ZW06OlZhcmlhYmxlQ2hlY2sgcXcoOmFsbCk7Cgp1c2UgcGFyZW50IHF3KAogICAgS2VybmVsOjpTeXN0ZW06OkR5bmFtaWNGaWVsZDo6RHJpdmVyOjpDb25maWdJdGVtCiAgICBLZXJuZWw6OlN5c3RlbTo6RHluYW1pY0ZpZWxkOjpEcml2ZXI6Ok11bHRpc2VsZWN0Cik7CgoxOwo=
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::ITSMConfigItem;

use strict;
use warnings;

use Kernel::System::EventHandler;
use Kernel::System::ITSMConfigItem::Definition;
use Kernel::System::ITSMConfigItem::History;
use Kernel::System::ITSMConfigItem::Number;
use Kernel::System::ITSMConfigItem::Permission;
use Kernel::System::ITSMConfigItem::Version;
use Kernel::System::ITSMConfigItem::XML;
use Kernel::System::VariableCheck qw(:all);

use Storable;

use vars qw(@ISA);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::Cache',
    'Kernel::System::DB',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::LinkObject',
    'Kernel::System::Log',
    'Kernel::System::Service',
    'Kernel::System::VirtualFS',
);

=head1 NAME

Kernel::System::ITSMConfigItem - config item lib

=head1 DESCRIPTION

All config item functions.

=head1 PUBLIC INTERFACE

=head2 new()

create an object

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

=cut

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

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

    $Self->{CacheType} = 'ITSMConfigurationManagement';
    $Self->{CacheTTL}  = 60 * 60 * 24 * 20;

    @ISA = qw(
        Kernel::System::ITSMConfigItem::Definition
        Kernel::System::ITSMConfigItem::History
        Kernel::System::ITSMConfigItem::Number
        Kernel::System::ITSMConfigItem::Permission
        Kernel::System::ITSMConfigItem::Version
        Kernel::System::ITSMConfigItem::XML
        Kernel::System::EventHandler
    );

    # init of event handler
    $Self->EventHandlerInit(
        Config => 'ITSMConfigItem::EventModulePost',
    );

    return $Self;
}

=head2 ConfigItemCount()

count all records of a config item class

    my $Count = $ConfigItemObject->ConfigItemCount(
        ClassID => 123,
    );

=cut

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

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

    # get state list
    my $StateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class       => 'ITSM::ConfigItem::DeploymentState',
        Preferences => {
            Functionality => [ 'preproductive', 'productive' ],
        },
    );

    return 0 if !%{$StateList};

    # create state string
    my $DeplStateString = join q{, }, keys %{$StateList};

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => "SELECT COUNT(id) FROM configitem WHERE class_id = ? AND "
            . "cur_depl_state_id IN ( $DeplStateString )",
        Bind  => [ \$Param{ClassID} ],
        Limit => 1,
    );

    # fetch the result
    my $Count = 0;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $Count = $Row[0];
    }

    return $Count;
}

=head2 ConfigItemResultList()

return a config item list as array hash reference

    my $ConfigItemListRef = $ConfigItemObject->ConfigItemResultList(
        ClassID => 123,
        Start   => 100,
        Limit   => 50,
    );

=cut

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

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

    # get state list
    my $StateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class       => 'ITSM::ConfigItem::DeploymentState',
        Preferences => {
            Functionality => [ 'preproductive', 'productive' ],
        },
    );

    # create state string
    my $DeplStateString = join q{, }, keys %{$StateList};

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => "SELECT id FROM configitem "
            . "WHERE class_id = ? AND cur_depl_state_id IN ( $DeplStateString ) "
            . "ORDER BY change_time DESC",
        Bind  => [ \$Param{ClassID} ],
        Start => $Param{Start},
        Limit => $Param{Limit},
    );

    # fetch the result
    my @ConfigItemIDList;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        push @ConfigItemIDList, $Row[0];
    }

    # get last versions data
    my @ConfigItemList;
    for my $ConfigItemID (@ConfigItemIDList) {

        # get version data
        my $LastVersion = $Self->VersionGet(
            ConfigItemID => $ConfigItemID,
            XMLDataGet   => 0,
        );

        push @ConfigItemList, $LastVersion;
    }

    return \@ConfigItemList;
}

=head2 ConfigItemGet()

return a config item as hash reference

    my $ConfigItem = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => 123,
        Cache        => 0,    # (optional) default 1 (0|1)
    );

A hashref with the following keys is returned:

    $ConfigItem{ConfigItemID}
    $ConfigItem{Number}
    $ConfigItem{ClassID}
    $ConfigItem{Class}
    $ConfigItem{LastVersionID}
    $ConfigItem{CurDeplStateID}
    $ConfigItem{CurDeplState}
    $ConfigItem{CurDeplStateType}
    $ConfigItem{CurInciStateID}
    $ConfigItem{CurInciState}
    $ConfigItem{CurInciStateType}
    $ConfigItem{CreateTime}
    $ConfigItem{CreateBy}
    $ConfigItem{ChangeTime}
    $ConfigItem{ChangeBy}

=cut

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

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

    # enable cache per default
    if ( !defined $Param{Cache} ) {
        $Param{Cache} = 1;
    }

    # check if result is already cached
    my $CacheKey    = 'ConfigItemGet::ConfigItemID::' . $Param{ConfigItemID};
    my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');
    my $Cache       = $CacheObject->Get(
        Type => $Self->{CacheType},
        Key  => $CacheKey,
    );
    return Storable::dclone($Cache) if $Cache;

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT id, configitem_number, class_id, last_version_id, '
            . 'cur_depl_state_id, cur_inci_state_id, '
            . 'create_time, create_by, change_time, change_by '
            . 'FROM configitem WHERE id = ?',
        Bind  => [ \$Param{ConfigItemID} ],
        Limit => 1,
    );

    # fetch the result
    my %ConfigItem;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $ConfigItem{ConfigItemID}   = $Row[0];
        $ConfigItem{Number}         = $Row[1];
        $ConfigItem{ClassID}        = $Row[2];
        $ConfigItem{LastVersionID}  = $Row[3];
        $ConfigItem{CurDeplStateID} = $Row[4];
        $ConfigItem{CurInciStateID} = $Row[5];
        $ConfigItem{CreateTime}     = $Row[6];
        $ConfigItem{CreateBy}       = $Row[7];
        $ConfigItem{ChangeTime}     = $Row[8];
        $ConfigItem{ChangeBy}       = $Row[9];
    }

    # check config item
    if ( !$ConfigItem{ConfigItemID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No such ConfigItemID ($Param{ConfigItemID})!",
        );
        return;
    }

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    $ConfigItem{Class} = $ClassList->{ $ConfigItem{ClassID} };

    return \%ConfigItem if !$ConfigItem{CurDeplStateID} || !$ConfigItem{CurInciStateID};

    # get deployment state functionality
    my $DeplState = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemGet(
        ItemID => $ConfigItem{CurDeplStateID},
    );

    $ConfigItem{CurDeplState}     = $DeplState->{Name};
    $ConfigItem{CurDeplStateType} = $DeplState->{Functionality};

    # get incident state functionality
    my $InciState = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemGet(
        ItemID => $ConfigItem{CurInciStateID},
    );

    $ConfigItem{CurInciState}     = $InciState->{Name};
    $ConfigItem{CurInciStateType} = $InciState->{Functionality};

    # cache the result
    $CacheObject->Set(
        Type  => $Self->{CacheType},
        TTL   => $Self->{CacheTTL},
        Key   => $CacheKey,
        Value => Storable::dclone( \%ConfigItem ),
    );

    return \%ConfigItem;
}

=head2 ConfigItemAdd()

add a new config item

    my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        Number  => '111',  # (optional)
        ClassID => 123,
        UserID  => 1,
    );

=cut

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

    NEEDED:
    for my $Needed (qw(ClassID UserID)) {

        next NEEDED if defined $Param{$Needed};

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need $Needed!",
        );
        return;
    }

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    return if !$ClassList;
    return if ref $ClassList ne 'HASH';

    # check the class id
    if ( !$ClassList->{ $Param{ClassID} } ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'No valid class id given!',
        );
        return;
    }

    # create config item number
    if ( $Param{Number} ) {

        # find existing config item number
        my $Exists = $Self->ConfigItemNumberLookup(
            ConfigItemNumber => $Param{Number},
        );

        if ($Exists) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => 'Config item number already exists!',
            );
            return;
        }
    }
    else {

        # create config item number
        $Param{Number} = $Self->ConfigItemNumberCreate(
            Type    => $Kernel::OM->Get('Kernel::Config')->Get('ITSMConfigItem::NumberGenerator'),
            ClassID => $Param{ClassID},
        );
    }

    # insert new config item
    my $Success = $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL => 'INSERT INTO configitem '
            . '(configitem_number, class_id, create_time, create_by, change_time, change_by) '
            . 'VALUES (?, ?, current_timestamp, ?, current_timestamp, ?)',
        Bind => [ \$Param{Number}, \$Param{ClassID}, \$Param{UserID}, \$Param{UserID} ],
    );

    return if !$Success;

    # find id of new item
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT id FROM configitem WHERE '
            . 'configitem_number = ? AND class_id = ? ORDER BY id DESC',
        Bind  => [ \$Param{Number}, \$Param{ClassID} ],
        Limit => 1,
    );

    # fetch the result
    my $ConfigItemID;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $ConfigItemID = $Row[0];
    }

    # trigger ConfigItemCreate
    $Self->EventHandler(
        Event => 'ConfigItemCreate',
        Data  => {
            ConfigItemID => $ConfigItemID,
            Comment      => $ConfigItemID . '%%' . $Param{Number},
        },
        UserID => $Param{UserID},
    );

    return $ConfigItemID;
}

=head2 ConfigItemDelete()

delete an existing config item

    my $True = $ConfigItemObject->ConfigItemDelete(
        ConfigItemID => 123,
        UserID       => 1,
    );

=cut

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

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

    # remember config item data before delete
    my $ConfigItemData = $Self->ConfigItemGet(
        ConfigItemID => $Param{ConfigItemID},
    );

    # delete all links to this config item first, before deleting the versions
    return if !$Kernel::OM->Get('Kernel::System::LinkObject')->LinkDeleteAll(
        Object => 'ITSMConfigItem',
        Key    => $Param{ConfigItemID},
        UserID => $Param{UserID},
    );

    # delete existing versions
    $Self->VersionDelete(
        ConfigItemID => $Param{ConfigItemID},
        UserID       => $Param{UserID},
    );

    # get a list of all attachments
    my @ExistingAttachments = $Self->ConfigItemAttachmentList(
        ConfigItemID => $Param{ConfigItemID},
    );

    # delete all attachments of this config item
    FILENAME:
    for my $Filename (@ExistingAttachments) {

        # delete the attachment
        my $DeletionSuccess = $Self->ConfigItemAttachmentDelete(
            ConfigItemID => $Param{ConfigItemID},
            Filename     => $Filename,
            UserID       => $Param{UserID},
        );

        if ( !$DeletionSuccess ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Unknown problem when deleting attachment $Filename of ConfigItem "
                    . "$Param{ConfigItemID}. Please check the VirtualFS backend for stale "
                    . "files!",
            );
        }
    }

    # trigger ConfigItemDelete event
    # this must be done before deleting the config item from the database,
    # because of a foreign key constraint in the configitem_history table
    $Self->EventHandler(
        Event => 'ConfigItemDelete',
        Data  => {
            ConfigItemID => $Param{ConfigItemID},
            Comment      => $Param{ConfigItemID},
            Number       => $ConfigItemData->{Number},
            Class        => $ConfigItemData->{Class},
        },
        UserID => $Param{UserID},
    );

    # delete config item
    my $Success = $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL  => 'DELETE FROM configitem WHERE id = ?',
        Bind => [ \$Param{ConfigItemID} ],
    );

    # delete the cache
    my $CacheKey = 'ConfigItemGet::ConfigItemID::' . $Param{ConfigItemID};
    $Kernel::OM->Get('Kernel::System::Cache')->Delete(
        Type => $Self->{CacheType},
        Key  => $CacheKey,
    );

    return $Success;
}

=head2 ConfigItemAttachmentAdd()

adds an attachment to a config item

    my $Success = $ConfigItemObject->ConfigItemAttachmentAdd(
        ConfigItemID    => 1,
        Filename        => 'filename',
        Content         => 'content',
        ContentType     => 'text/plain',
        UserID          => 1,
    );

=cut

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

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

            return;
        }
    }

    # write to virtual fs
    my $Success = $Kernel::OM->Get('Kernel::System::VirtualFS')->Write(
        Filename    => "ConfigItem/$Param{ConfigItemID}/$Param{Filename}",
        Mode        => 'binary',
        Content     => \$Param{Content},
        Preferences => {
            ContentID    => $Param{ContentID},
            ContentType  => $Param{ContentType},
            ConfigItemID => $Param{ConfigItemID},
            UserID       => $Param{UserID},
        },
    );

    # check for error
    if ($Success) {

        # trigger AttachmentAdd-Event
        $Self->EventHandler(
            Event => 'AttachmentAddPost',
            Data  => {
                %Param,
                ConfigItemID => $Param{ConfigItemID},
                Comment      => $Param{Filename},
                HistoryType  => 'AttachmentAdd',
            },
            UserID => $Param{UserID},
        );
    }
    else {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Cannot add attachment for config item $Param{ConfigItemID}",
        );

        return;
    }

    return 1;
}

=head2 ConfigItemAttachmentDelete()

Delete the given file from the virtual filesystem.

    my $Success = $ConfigItemObject->ConfigItemAttachmentDelete(
        ConfigItemID => 123,               # used in event handling, e.g. for logging the history
        Filename     => 'Projectplan.pdf', # identifies the attachment (together with the ConfigItemID)
        UserID       => 1,
    );

=cut

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

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

            return;
        }
    }

    # add prefix
    my $Filename = 'ConfigItem/' . $Param{ConfigItemID} . '/' . $Param{Filename};

    # delete file
    my $Success = $Kernel::OM->Get('Kernel::System::VirtualFS')->Delete(
        Filename => $Filename,
    );

    # check for error
    if ($Success) {

        # trigger AttachmentDeletePost-Event
        $Self->EventHandler(
            Event => 'AttachmentDeletePost',
            Data  => {
                %Param,
                ConfigItemID => $Param{ConfigItemID},
                Comment      => $Param{Filename},
                HistoryType  => 'AttachmentDelete',
            },
            UserID => $Param{UserID},
        );
    }
    else {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Cannot delete attachment $Filename!",
        );

        return;
    }

    return $Success;
}

=head2 ConfigItemAttachmentGet()

This method returns information about one specific attachment.

    my $Attachment = $ConfigItemObject->ConfigItemAttachmentGet(
        ConfigItemID => 4,
        Filename     => 'test.txt',
    );

returns

    {
        Preferences => {
            AllPreferences => 'test',
        },
        Filename    => 'test.txt',
        Content     => 'content',
        ContentType => 'text/plain',
        Filesize    => 12348409,
        Type        => 'attachment',
    }

=cut

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

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

    # add prefix
    my $Filename = 'ConfigItem/' . $Param{ConfigItemID} . '/' . $Param{Filename};

    # find all attachments of this config item
    my @Attachments = $Kernel::OM->Get('Kernel::System::VirtualFS')->Find(
        Filename    => $Filename,
        Preferences => {
            ConfigItemID => $Param{ConfigItemID},
        },
    );

    # return error if file does not exist
    if ( !@Attachments ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Message  => "No such attachment ($Filename)!",
            Priority => 'error',
        );
        return;
    }

    # get data for attachment
    my %AttachmentData = $Kernel::OM->Get('Kernel::System::VirtualFS')->Read(
        Filename => $Filename,
        Mode     => 'binary',
    );

    my $AttachmentInfo = {
        %AttachmentData,
        Filename    => $Param{Filename},
        Content     => ${ $AttachmentData{Content} },
        ContentType => $AttachmentData{Preferences}->{ContentType},
        Type        => 'attachment',
        Filesize    => $AttachmentData{Preferences}->{FilesizeRaw},
    };

    return $AttachmentInfo;
}

=head2 ConfigItemAttachmentList()

Returns an array with all attachments of the given config item.

    my @Attachments = $ConfigItemObject->ConfigItemAttachmentList(
        ConfigItemID => 123,
    );

returns

    @Attachments = (
        'filename.txt',
        'other_file.pdf',
    );

=cut

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

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

        return;
    }

    # find all attachments of this config item
    my @Attachments = $Kernel::OM->Get('Kernel::System::VirtualFS')->Find(
        Preferences => {
            ConfigItemID => $Param{ConfigItemID},
        },
    );

    for my $Filename (@Attachments) {

        # remove extra information from filename
        $Filename =~ s{ \A ConfigItem / \d+ / }{}xms;
    }

    return @Attachments;
}

=head2 ConfigItemAttachmentExists()

Checks if a file with a given filename exists.

    my $Exists = $ConfigItemObject->ConfigItemAttachmentExists(
        Filename => 'test.txt',
        ConfigItemID => 123,
        UserID   => 1,
    );

=cut

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

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

            return;
        }
    }

    return if !$Kernel::OM->Get('Kernel::System::VirtualFS')->Find(
        Filename => 'ConfigItem/' . $Param{ConfigItemID} . '/' . $Param{Filename},
    );

    return 1;
}

=head2 ConfigItemSearchExtended()

return a config item list as an array reference

    my $ConfigItemIDs = $ConfigItemObject->ConfigItemSearchExtended(
        Number       => 'The ConfigItem Number',  # (optional)
        Name         => 'The Name',               # (optional)
        ClassIDs     => [9, 8, 7, 6],             # (optional)
        DeplStateIDs => [1, 2, 3, 4],             # (optional)
        InciStateIDs => [1, 2, 3, 4],             # (optional)

        # config items with created time after ...
        ConfigItemCreateTimeNewerDate => '2006-01-09 00:00:01',  # (optional)
        # config items with created time before then ....
        ConfigItemCreateTimeOlderDate => '2006-01-19 23:59:59',  # (optional)

        # config items with changed time after ...
        ConfigItemChangeTimeNewerDate => '2006-01-09 00:00:01',  # (optional)
        # config items with changed time before then ....
        ConfigItemChangeTimeOlderDate => '2006-01-19 23:59:59',  # (optional)

        What => [                                                # (optional)
            # each array element is a and condition
            {
                # or condition in hash
                "[%]{'ElementA'}[%]{'ElementB'}[%]{'Content'}" => '%contentA%',
                "[%]{'ElementA'}[%]{'ElementC'}[%]{'Content'}" => '%contentA%',
            },
            {
                "[%]{'ElementA'}[%]{'ElementB'}[%]{'Content'}" => '%contentB%',
                "[%]{'ElementA'}[%]{'ElementC'}[%]{'Content'}" => '%contentB%',
            },
            {
                # use array reference if different content with same key was searched
                "[%]{'ElementA'}[%]{'ElementB'}[%]{'Content'}" => ['%contentC%', '%contentD%', '%contentE%'],
                "[%]{'ElementA'}[%]{'ElementC'}[%]{'Content'}" => ['%contentC%', '%contentD%', '%contentE%'],
            },
        ],

        PreviousVersionSearch => 1,  # (optional) default 0 (0|1)

        OrderBy => [ 'ConfigItemID', 'Number' ],                  # (optional)
        # default: [ 'ConfigItemID' ]
        # (ConfigItemID, Number, Name, ClassID, DeplStateID, InciStateID,
        # CreateTime, CreateBy, ChangeTime, ChangeBy)

        # 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)
        # default: [ 'Down' ]
        # (Down | Up)

        Limit          => 122,  # (optional)
        UsingWildcards => 0,    # (optional) default 1
    );

=cut

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

    # set limit
    my $Limit = $Param{Limit};
    $Param{Limit} = undef;

    # config item search is required if one of these params is given
    my @ConfigItemSearchParams = (
        'ConfigItemCreateTimeNewerDate',
        'ConfigItemCreateTimeOlderDate',
        'ConfigItemChangeTimeNewerDate',
        'ConfigItemChangeTimeOlderDate',
    );

    # check, if config item search is required
    my %RequiredSearch;
    CONFIGITEMPARAM:
    for my $ConfigItemParam (@ConfigItemSearchParams) {
        next CONFIGITEMPARAM if !$Param{$ConfigItemParam};

        $RequiredSearch{ConfigItem} = 1;
        last CONFIGITEMPARAM;
    }

    # special handling for config item number
    # number 0 is allowed but not the empty string
    if ( defined $Param{Number} && $Param{Number} ne '' ) {
        $RequiredSearch{ConfigItem} = 1;
    }

    # version search is required if Name, What or PreviousVersionSearch is given
    if (
        IsStringWithData( $Param{Name} )
        || IsArrayRefWithData( $Param{What} )
        || $Param{PreviousVersionSearch}
        )
    {
        $RequiredSearch{Version} = 1;
    }

    # version search is also required if sorting by name (fix for bug #7072)
    ORDERBY:
    for my $OrderBy ( @{ $Param{OrderBy} } ) {
        if ( $OrderBy eq 'Name' ) {
            $RequiredSearch{Version} = 1;
            last ORDERBY;
        }
    }

    # xml version search is required if What is given
    if ( IsArrayRefWithData( $Param{What} ) ) {
        $RequiredSearch{XMLVersion} = 1;
    }

    # use config item search as fallback
    if ( !%RequiredSearch ) {
        $RequiredSearch{ConfigItem} = 1;
    }

    # start config item search
    my %ConfigItemLists;
    if ( $RequiredSearch{ConfigItem} ) {

        # search config items
        $ConfigItemLists{ConfigItem} = $Self->ConfigItemSearch(%Param);

        return    if !$ConfigItemLists{ConfigItem};
        return    if ref $ConfigItemLists{ConfigItem} ne 'ARRAY';
        return [] if !@{ $ConfigItemLists{ConfigItem} };
    }

    # start version search
    if ( $RequiredSearch{Version} ) {

        # search versions
        $ConfigItemLists{Version} = $Self->VersionSearch(%Param);

        return    if !$ConfigItemLists{Version};
        return    if ref $ConfigItemLists{Version} ne 'ARRAY';
        return [] if !@{ $ConfigItemLists{Version} };
    }

    # start xml version search
    if ( $RequiredSearch{XMLVersion} ) {

        # search xml versions
        my $XMLVersionList = $Self->_XMLVersionSearch(%Param);

        return    if !$XMLVersionList;
        return    if ref $XMLVersionList ne 'HASH';
        return [] if !%{$XMLVersionList};

        # get config item ids
        my %ConfigItemListTmp;
        VERSIONID:
        for my $VersionID ( sort keys %{$XMLVersionList} ) {
            my $ConfigItemID = $Self->VersionConfigItemIDGet(
                VersionID => $VersionID,
            );

            next VERSIONID if !$ConfigItemID;

            $ConfigItemListTmp{$ConfigItemID} = 1;
        }

        # add ids to config item list
        $ConfigItemLists{XMLVersion} = \%ConfigItemListTmp;
    }

    # create the result list
    my @ResultList;
    if ( $RequiredSearch{ConfigItem} && $RequiredSearch{Version} ) {

        # build a lookup hash of all found configitem ids in $ConfigItemLists{ConfigItem}
        my %ConfigItemSeen = map { $_ => 1 } @{ $ConfigItemLists{ConfigItem} };

        # check all config item ids, we need to keep the sorting
        CONFIGITEMID:
        for my $ConfigItemID ( @{ $ConfigItemLists{Version} } ) {
            next CONFIGITEMID if !$ConfigItemSeen{$ConfigItemID};
            push @ResultList, $ConfigItemID;
        }
    }
    elsif ( $RequiredSearch{ConfigItem} ) {
        @ResultList = @{ $ConfigItemLists{ConfigItem} };
    }
    elsif ( $RequiredSearch{Version} ) {
        @ResultList = @{ $ConfigItemLists{Version} };
    }

    # consider the XML result
    if ( $RequiredSearch{XMLVersion} ) {
        @ResultList = grep { $ConfigItemLists{XMLVersion}->{$_} } @ResultList;
    }

    # consider limit
    if ( $Limit && $Limit < scalar @ResultList ) {

        # extract the limited ids
        $Limit--;
        @ResultList = @ResultList[ 0 .. $Limit ];
    }

    return \@ResultList;
}

=head2 ConfigItemSearch()

return a config item list as an array reference

    my $ConfigItemIDs = $ConfigItemObject->ConfigItemSearch(
        Number       => 'The ConfigItem Number',  # (optional)
        ClassIDs     => [9, 8, 7, 6],             # (optional)
        DeplStateIDs => [1, 2, 3, 4],             # (optional)
        InciStateIDs => [1, 2, 3, 4],             # (optional)
        CreateBy     => [1, 2, 3],                # (optional)
        ChangeBy     => [3, 2, 1],                # (optional)

        # config items with created time after ...
        ConfigItemCreateTimeNewerDate => '2006-01-09 00:00:01',  # (optional)
        # config items with created time before then ....
        ConfigItemCreateTimeOlderDate => '2006-01-19 23:59:59',  # (optional)

        # config items with changed time after ...
        ConfigItemChangeTimeNewerDate => '2006-01-09 00:00:01',  # (optional)
        # config items with changed time before then ....
        ConfigItemChangeTimeOlderDate => '2006-01-19 23:59:59',  # (optional)

        OrderBy => [ 'ConfigItemID', 'Number' ],                  # (optional)
        # default: [ 'ConfigItemID' ]
        # (ConfigItemID, Number, ClassID, DeplStateID, InciStateID,
        # CreateTime, CreateBy, ChangeTime, ChangeBy)

        # 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)
        # default: [ 'Down' ]
        # (Down | Up)

        Limit          => 122,  # (optional)
        UsingWildcards => 0,    # (optional) default 1
    );

=cut

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

    # verify that all passed array parameters contain an arrayref
    ARGUMENT:
    for my $Argument (
        qw(
        OrderBy
        OrderByDirection
        )
        )
    {
        if ( !defined $Param{$Argument} ) {
            $Param{$Argument} ||= [];

            next ARGUMENT;
        }

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

    # define order table
    my %OrderByTable = (
        ConfigItemID => 'id',
        Number       => 'configitem_number',
        ClassID      => 'class_id',
        DeplStateID  => 'cur_depl_state_id',
        InciStateID  => 'cur_inci_state_id',
        CreateTime   => 'create_time',
        CreateBy     => 'create_by',
        ChangeTime   => 'change_time',
        ChangeBy     => 'change_by',
    );

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

        next ORDERBY if $OrderBy eq 'Name';

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

            # found an error
            $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;
    }

    # set default values
    if ( !defined $Param{UsingWildcards} ) {
        $Param{UsingWildcards} = 1;
    }

    # get like escape string needed for some databases (e.g. oracle)
    my $LikeEscapeString = $Kernel::OM->Get('Kernel::System::DB')->GetDatabaseFunction('LikeEscapeString');

    # assemble the ORDER BY clause
    my @SQLOrderBy;
    my $Count = 0;
    ORDERBY:
    for my $OrderBy ( @{ $Param{OrderBy} } ) {

        next ORDERBY if $OrderBy eq 'Name';

        # 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';
            }
        }

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

    }
    continue {
        $Count++;
    }

    # if there is a possibility that the ordering is not determined
    # we add an ascending ordering by id
    if ( !grep { $_ eq 'ConfigItemID' } ( @{ $Param{OrderBy} } ) ) {
        push @SQLOrderBy, "$OrderByTable{ConfigItemID} ASC";
    }

    # add number to sql where array
    my @SQLWhere;
    if ( defined $Param{Number} && $Param{Number} ne '' && ref $Param{Number} ne 'ARRAY' ) {

        # quote
        $Param{Number} = $Kernel::OM->Get('Kernel::System::DB')->Quote( $Param{Number} );

        if ( $Param{UsingWildcards} ) {

            # prepare like string
            $Self->_PrepareLikeString( \$Param{Number} );

            push @SQLWhere,
                "LOWER(configitem_number) LIKE LOWER('$Param{Number}') $LikeEscapeString";
        }
        else {
            push @SQLWhere, "LOWER(configitem_number) = LOWER('$Param{Number}')";
        }
    }
    elsif ( defined $Param{Number} && $Param{Number} ne '' && ref $Param{Number} eq 'ARRAY' ) {

        # Create string.
        my $InString = join q{, }, @{ $Param{Number} };

        push @SQLWhere, "LOWER(configitem_number) IN ($InString)";
    }

    # set array params
    my %ArrayParams = (
        ConfigItemIDs => 'id',
        ClassIDs      => 'class_id',
        DeplStateIDs  => 'cur_depl_state_id',
        InciStateIDs  => 'cur_inci_state_id',
        CreateBy      => 'create_by',
        ChangeBy      => 'change_by',
    );

    ARRAYPARAM:
    for my $ArrayParam ( sort keys %ArrayParams ) {

        next ARRAYPARAM if !$Param{$ArrayParam};

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

        next ARRAYPARAM if !@{ $Param{$ArrayParam} };

        # quote as integer
        for my $OneParam ( @{ $Param{$ArrayParam} } ) {
            $OneParam = $Kernel::OM->Get('Kernel::System::DB')->Quote( $OneParam, 'Integer' );
        }

        # create string
        my $InString = join q{, }, @{ $Param{$ArrayParam} };

        next ARRAYPARAM if !$InString;

        push @SQLWhere, "$ArrayParams{ $ArrayParam } IN ($InString)";
    }

    # set time params
    my %TimeParams = (
        ConfigItemCreateTimeNewerDate => 'create_time >=',
        ConfigItemCreateTimeOlderDate => 'create_time <=',
        ConfigItemChangeTimeNewerDate => 'change_time >=',
        ConfigItemChangeTimeOlderDate => 'change_time <=',
    );

    TIMEPARAM:
    for my $TimeParam ( sort keys %TimeParams ) {

        next TIMEPARAM if !$Param{$TimeParam};

        if ( $Param{$TimeParam} !~ m{ \A \d\d\d\d-\d\d-\d\d \s \d\d:\d\d:\d\d \z }xms ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Invalid date format found!",
            );
            return;
        }

        # quote
        $Param{$TimeParam} = $Kernel::OM->Get('Kernel::System::DB')->Quote( $Param{$TimeParam} );

        push @SQLWhere, "$TimeParams{ $TimeParam } '$Param{ $TimeParam }'";
    }

    # create where string
    my $WhereString = @SQLWhere ? ' WHERE ' . join q{ AND }, @SQLWhere : '';

    # set limit
    if ( $Param{Limit} ) {
        $Param{Limit} = $Kernel::OM->Get('Kernel::System::DB')->Quote( $Param{Limit}, 'Integer' );
    }

    my $SQL = "SELECT id FROM configitem $WhereString ";

    # add the ORDER BY clause
    if (@SQLOrderBy) {
        $SQL .= 'ORDER BY ';
        $SQL .= join ', ', @SQLOrderBy;
        $SQL .= ' ';
    }

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL   => $SQL,
        Limit => $Param{Limit},
    );

    # fetch the result
    my @ConfigItemList;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        push @ConfigItemList, $Row[0];
    }

    return \@ConfigItemList;
}

=head2 ConfigItemLookup()

This method does a lookup for a config-item. If a config-item id is given,
it returns the number of the config-item. If a config-item number is given,
the appropriate id is returned.

    my $Number = $ConfigItemObject->ConfigItemLookup(
        ConfigItemID => 1234,
    );

    my $ID = $ConfigItemObject->ConfigItemLookup(
        ConfigItemNumber => 1000001,
    );

=cut

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

    my ($Key) = grep { $Param{$_} } qw(ConfigItemID ConfigItemNumber);

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

    # if result is cached return that result
    return $Self->{Cache}->{ConfigItemLookup}->{$Key}->{ $Param{$Key} }
        if $Self->{Cache}->{ConfigItemLookup}->{$Key}->{ $Param{$Key} };

    # set the appropriate SQL statement
    my $SQL = 'SELECT configitem_number FROM configitem WHERE id = ?';

    if ( $Key eq 'ConfigItemNumber' ) {
        $SQL = 'SELECT id FROM configitem WHERE configitem_number = ?';
    }

    # fetch the requested value
    return if !$Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL   => $SQL,
        Bind  => [ \$Param{$Key} ],
        Limit => 1,
    );

    my $Value;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $Value = $Row[0];
    }

    $Self->{Cache}->{ConfigItemLookup}->{$Key}->{ $Param{$Key} } = $Value;

    return $Value;
}

=head2 UniqueNameCheck()

This method checks all already existing config items, whether the given name does already exist
within the same config item class or among all classes, depending on the SysConfig value of
UniqueCIName::UniquenessCheckScope (Class or Global).

This method requires 3 parameters: ConfigItemID, Name and Class
"ConfigItemID"  is the ID of the ConfigItem, which is to be checked for uniqueness
"Name"          is the config item name to be checked for uniqueness
"ClassID"       is the ID of the config item's class

All parameters are mandatory.

my $DuplicateNames = $ConfigItemObject->UniqueNameCheck(
    ConfigItemID => '73'
    Name         => 'PC#005',
    ClassID      => '32',
);

The given name is not unique
my $NameDuplicates = [ 5, 35, 48, ];    # IDs of ConfigItems with the same name

The given name is unique
my $NameDuplicates = [];

=cut

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

    # check for needed stuff
    for my $Needed (qw(ConfigItemID Name ClassID)) {
        if ( !$Param{$Needed} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Missing parameter $Needed!",
            );
            return;
        }
    }

    # check ConfigItemID param for valid format
    if (
        !IsInteger( $Param{ConfigItemID} )
        && ( IsStringWithData( $Param{ConfigItemID} ) && $Param{ConfigItemID} ne 'NEW' )
        )
    {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "The ConfigItemID parameter needs to be an integer or 'NEW'",
        );
        return;
    }

    # check Name param for valid format
    if ( !IsStringWithData( $Param{Name} ) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "The Name parameter needs to be a string!",
        );
        return;
    }

    # check ClassID param for valid format
    if ( !IsInteger( $Param{ClassID} ) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "The ClassID parameter needs to be an integer",
        );
        return;
    }

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # check class list for validity
    if ( !IsHashRefWithData($ClassList) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Unable to retrieve a valid class list!",
        );
        return;
    }

    # get the class name from the class list
    my $Class = $ClassList->{ $Param{ClassID} };

    # check class for validity
    if ( !IsStringWithData($Class) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Unable to determine a config item class using the given ClassID!",
        );
        return;
    }
    elsif ( $Kernel::OM->Get('Kernel::Config')->{Debug} > 0 ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'debug',
            Message  => "Resolved ClassID $Param{ClassID} to class $Class",
        );
    }

    # get the uniqueness scope from SysConfig
    my $Scope = $Kernel::OM->Get('Kernel::Config')->Get('UniqueCIName::UniquenessCheckScope');

    # check scope for validity
    if ( !IsStringWithData($Scope) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "The configuration of UniqueCIName::UniquenessCheckScope is invalid!",
        );
        return;
    }

    if ( $Scope ne 'global' && $Scope ne 'class' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "UniqueCIName::UniquenessCheckScope is $Scope, but must be either "
                . "'global' or 'class'!",
        );
        return;
    }

    if ( $Kernel::OM->Get('Kernel::Config')->{Debug} > 0 ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'debug',
            Message  => "The scope for checking the uniqueness is $Scope",
        );
    }

    my %SearchCriteria;

    # add the config item class to the search criteria if the uniqueness scope is not global
    if ( $Scope ne 'global' ) {
        $SearchCriteria{ClassIDs} = [ $Param{ClassID} ];
    }

    $SearchCriteria{Name} = $Param{Name};

    # search for a config item matching the given name
    my $ConfigItem = $Self->ConfigItemSearchExtended(%SearchCriteria);

    # remove the provided ConfigItemID from the results, otherwise the duplicate check would fail
    # because the ConfigItem itself is found as duplicate
    my @Duplicates = map {$_} grep { $_ ne $Param{ConfigItemID} } @{$ConfigItem};

    # if a config item was found, the given name is not unique
    # if no config item was found, the given name is unique

    # return the result of the config item search for duplicates
    return \@Duplicates;
}

=head2 CurInciStateRecalc()

recalculates the current incident state of this config item and all linked config items

    my $Success = $ConfigItemObject->CurInciStateRecalc(
        ConfigItemID               => 123,
        NewConfigItemIncidentState => $NewConfigItemIncidentState,  # optional, incident states of already checked CIs
        ScannedConfigItemIDs       => $ScannedConfigItemIDs,        # optional, IDs of already checked CIs
    );

=cut

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

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

    # get incident link types and directions from config
    my $IncidentLinkTypeDirection = $Kernel::OM->Get('Kernel::Config')->Get('ITSM::Core::IncidentLinkTypeDirection');

    # to store the new incident state for CIs
    # calculated from all incident link types
    # Incorporate data from previous run(s) and remember known data.
    $Param{NewConfigItemIncidentState} //= {};
    my $KnownNewConfigItemIncidentState = Storable::dclone( $Param{NewConfigItemIncidentState} );

    # to store the relation between services and linked CIs
    my %ServiceCIRelation;

    # remember the scanned config items
    # Incorporate data from previous run(s) and remember known data.
    $Param{ScannedConfigItemIDs} //= {};
    my $KnownScannedConfigItemIDs = Storable::dclone( $Param{ScannedConfigItemIDs} );

    # find all config items with an incident state
    $Self->_FindInciConfigItems(
        ConfigItemID              => $Param{ConfigItemID},
        IncidentLinkTypeDirection => $IncidentLinkTypeDirection,
        ScannedConfigItemIDs      => $Param{ScannedConfigItemIDs},
    );

    # calculate the new CI incident state for each configured linktype
    LINKTYPE:
    for my $LinkType ( sort keys %{$IncidentLinkTypeDirection} ) {

        # get the direction
        my $LinkDirection = $IncidentLinkTypeDirection->{$LinkType};

        # investigate all config items with a warning state
        CONFIGITEMID:
        for my $ConfigItemID ( sort keys %{ $Param{ScannedConfigItemIDs} } ) {

            # Skip config items known from previous execution(s).
            if (
                IsStringWithData( $KnownScannedConfigItemIDs->{$ConfigItemID}->{Type} )
                && $KnownScannedConfigItemIDs->{$ConfigItemID}->{Type} eq
                $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type}
                )
            {
                next CONFIGITEMID;
            }

            # investigate only config items with an incident state
            next CONFIGITEMID if $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type} ne 'incident';

            $Self->_FindWarnConfigItems(
                ConfigItemID         => $ConfigItemID,
                LinkType             => $LinkType,
                Direction            => $LinkDirection,
                NumberOfLinkTypes    => scalar keys %{$IncidentLinkTypeDirection},
                ScannedConfigItemIDs => $Param{ScannedConfigItemIDs},
            );
        }

        CONFIGITEMID:
        for my $ConfigItemID ( sort keys %{ $Param{ScannedConfigItemIDs} } ) {

            # Skip config items known from previous execution(s).
            if (
                IsStringWithData( $KnownScannedConfigItemIDs->{$ConfigItemID}->{Type} )
                && $KnownScannedConfigItemIDs->{$ConfigItemID}->{Type} eq
                $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type}
                )
            {
                next CONFIGITEMID;
            }

            # extract incident state type
            my $InciStateType = $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type};

            # find all linked services of this CI
            my %LinkedServiceIDs = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkKeyList(
                Object1   => 'ITSMConfigItem',
                Key1      => $ConfigItemID,
                Object2   => 'Service',
                State     => 'Valid',
                Type      => $LinkType,
                Direction => $LinkDirection,
                UserID    => 1,
            );

            SERVICEID:
            for my $ServiceID ( sort keys %LinkedServiceIDs ) {

                # remember the CIs that are linked with this service
                push @{ $ServiceCIRelation{$ServiceID} }, $ConfigItemID;
            }

            next CONFIGITEMID if $InciStateType eq 'incident';

            $Param{NewConfigItemIncidentState}->{$ConfigItemID} = $InciStateType;
        }
    }

    # get the incident state list of warnings
    my $WarnStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class       => 'ITSM::Core::IncidentState',
        Preferences => {
            Functionality => 'warning',
        },
    );

    if ( !defined $WarnStateList ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "ITSM::Core::IncidentState Warning cannot be invalid.",
        );
    }

    my %ReverseWarnStateList = reverse %{$WarnStateList};
    my @SortedWarnList       = sort keys %ReverseWarnStateList;
    my $WarningStateID       = $ReverseWarnStateList{Warning} || $ReverseWarnStateList{ $SortedWarnList[0] };
    my $CacheObject          = $Kernel::OM->Get('Kernel::System::Cache');

    # set the new current incident state for CIs
    CONFIGITEMID:
    for my $ConfigItemID ( sort keys %{ $Param{NewConfigItemIncidentState} } ) {

        # Skip config items known from previous execution(s).
        if (
            IsStringWithData( $KnownNewConfigItemIncidentState->{$ConfigItemID} )
            && $KnownNewConfigItemIncidentState->{$ConfigItemID} eq $Param{NewConfigItemIncidentState}->{$ConfigItemID}
            )
        {
            next CONFIGITEMID;
        }

        # get new incident state type (can only be 'operational' or 'warning')
        my $InciStateType = $Param{NewConfigItemIncidentState}->{$ConfigItemID};

        # get last version
        my $LastVersion = $Self->VersionGet(
            ConfigItemID => $ConfigItemID,
            XMLDataGet   => 0,
        );

        my $CurInciStateID;
        if ( $InciStateType eq 'warning' ) {

            # check the current incident state type is in 'incident'
            # then we do not want to change it to warning
            next CONFIGITEMID if $LastVersion->{InciStateType} eq 'incident';

            $CurInciStateID = $WarningStateID;
        }
        elsif ( $InciStateType eq 'operational' ) {
            $CurInciStateID = $LastVersion->{InciStateID};
        }

        # No update necessary if incident state id of version and config item match.
        next CONFIGITEMID if $LastVersion->{CurInciStateID} eq $CurInciStateID;

        # update current incident state
        $Kernel::OM->Get('Kernel::System::DB')->Do(
            SQL  => 'UPDATE configitem SET cur_inci_state_id = ? WHERE id = ?',
            Bind => [ \$CurInciStateID, \$ConfigItemID ],
        );

        # delete the cache
        my $CacheKey = 'ConfigItemGet::ConfigItemID::' . $ConfigItemID;
        $CacheObject->Delete(
            Type => $Self->{CacheType},
            Key  => $CacheKey,
        );

        # delete affected caches for ConfigItemID
        $CacheKey = 'VersionGet::ConfigItemID::' . $ConfigItemID . '::XMLData::';
        for my $XMLData (qw(0 1)) {
            $CacheObject->Delete(
                Type => $Self->{CacheType},
                Key  => $CacheKey . $XMLData,
            );
        }
        $CacheObject->Delete(
            Type => $Self->{CacheType},
            Key  => 'VersionNameGet::ConfigItemID::' . $ConfigItemID,
        );

        # delete affected caches for last version
        my $VersionList = $Self->VersionList(
            ConfigItemID => $ConfigItemID,
        );
        my $VersionID = $VersionList->[-1];
        $CacheKey = 'VersionGet::VersionID::' . $VersionID . '::XMLData::';
        for my $XMLData (qw(0 1)) {
            $CacheObject->Delete(
                Type => $Self->{CacheType},
                Key  => $CacheKey . $XMLData,
            );
        }
        $CacheObject->Delete(
            Type => $Self->{CacheType},
            Key  => 'VersionNameGet::VersionID::' . $VersionID,
        );
    }

    # set the current incident state type for each service (influenced by linked CIs)
    SERVICEID:
    for my $ServiceID ( sort keys %ServiceCIRelation ) {

        # set default incident state type
        my $CurInciStateTypeFromCIs = 'operational';

        # get the unique config item ids which are direcly linked to this service
        my %UniqueConfigItemIDs = map { $_ => 1 } @{ $ServiceCIRelation{$ServiceID} };

        # investigate the current incident state of each config item
        CONFIGITEMID:
        for my $ConfigItemID ( sort keys %UniqueConfigItemIDs ) {

            # get config item data
            my $ConfigItemData = $Self->ConfigItemGet(
                ConfigItemID => $ConfigItemID,
                Cache        => 0,
            );

            next CONFIGITEMID if $ConfigItemData->{CurDeplStateType} ne 'productive';
            next CONFIGITEMID if $ConfigItemData->{CurInciStateType} eq 'operational';

            # check if service must be set to 'warning'
            if ( $ConfigItemData->{CurInciStateType} eq 'warning' ) {
                $CurInciStateTypeFromCIs = 'warning';
                next CONFIGITEMID;
            }

            # check if service must be set to 'incident'
            if ( $ConfigItemData->{CurInciStateType} eq 'incident' ) {
                $CurInciStateTypeFromCIs = 'incident';
                last CONFIGITEMID;
            }
        }

        # update the current incident state type from CIs of the service
        $Kernel::OM->Get('Kernel::System::Service')->ServicePreferencesSet(
            ServiceID => $ServiceID,
            Key       => 'CurInciStateTypeFromCIs',
            Value     => $CurInciStateTypeFromCIs,
            UserID    => 1,
        );
    }

    return 1;
}

=head1 INTERNAL INTERFACE

=head2 _FindInciConfigItems()

find all config items with an incident state

    $ConfigItemObject->_FindInciConfigItems(
        ConfigItemID              => $ConfigItemID,
        IncidentLinkTypeDirection => $IncidentLinkTypeDirection,
        ScannedConfigItemIDs     => \%ScannedConfigItemIDs,
    );

=cut

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

    # check needed stuff
    return if !$Param{ConfigItemID};

    # ignore already scanned ids (infinite loop protection)
    return if $Param{ScannedConfigItemIDs}->{ $Param{ConfigItemID} };

    $Param{ScannedConfigItemIDs}->{ $Param{ConfigItemID} }->{Type} = 'operational';

    # add own config item id to list of linked config items
    my %ConfigItemIDs = (
        $Param{ConfigItemID} => 1,
    );

    LINKTYPE:
    for my $LinkType ( sort keys %{ $Param{IncidentLinkTypeDirection} } ) {

        # find all linked config items (childs)
        my %LinkedConfigItemIDs = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkKeyList(
            Object1 => 'ITSMConfigItem',
            Key1    => $Param{ConfigItemID},
            Object2 => 'ITSMConfigItem',
            State   => 'Valid',
            Type    => $LinkType,

            # Direction must ALWAYS be 'Both' here as we need to include
            # all linked CIs that could influence this one!
            Direction => 'Both',

            UserID => 1,
        );

        # remember the config item ids
        %ConfigItemIDs = ( %ConfigItemIDs, %LinkedConfigItemIDs );
    }

    CONFIGITEMID:
    for my $ConfigItemID ( sort keys %ConfigItemIDs ) {

        # get config item data
        my $ConfigItem = $Self->ConfigItemGet(
            ConfigItemID => $ConfigItemID,
            Cache        => 0,
        );

        # set incident state
        if ( $ConfigItem->{CurInciStateType} eq 'incident' ) {
            $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type} = 'incident';
            next CONFIGITEMID;
        }

        # start recursion
        $Self->_FindInciConfigItems(
            ConfigItemID              => $ConfigItemID,
            IncidentLinkTypeDirection => $Param{IncidentLinkTypeDirection},
            ScannedConfigItemIDs      => $Param{ScannedConfigItemIDs},
        );
    }

    return 1;
}

=head2 _FindWarnConfigItems()

find all config items with a warning

    $ConfigItemObject->_FindWarnConfigItems(
        ConfigItemID         => $ConfigItemID,
        LinkType             => $LinkType,
        Direction            => $LinkDirection,
        NumberOfLinkTypes    => 2,
        ScannedConfigItemIDs => $ScannedConfigItemIDs,
    );

=cut

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

    # check needed stuff
    return if !$Param{ConfigItemID};

    my $IncidentCount = 0;
    for my $ConfigItemID ( sort keys %{ $Param{ScannedConfigItemIDs} } ) {
        if (
            $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type}
            && $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type} eq 'incident'
            )
        {
            $IncidentCount++;
        }
    }

# ignore already scanned ids (infinite loop protection)
# it is ok that a config item is investigated as many times as there are configured link types * number of incident config iteems
    if (
        $Param{ScannedConfigItemIDs}->{ $Param{ConfigItemID} }->{FindWarn}
        && $Param{ScannedConfigItemIDs}->{ $Param{ConfigItemID} }->{FindWarn}
        >= ( $Param{NumberOfLinkTypes} * $IncidentCount )
        )
    {
        return;
    }

    # increase the visit counter
    $Param{ScannedConfigItemIDs}->{ $Param{ConfigItemID} }->{FindWarn}++;

    # find all linked config items
    my %LinkedConfigItemIDs = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkKeyList(
        Object1   => 'ITSMConfigItem',
        Key1      => $Param{ConfigItemID},
        Object2   => 'ITSMConfigItem',
        State     => 'Valid',
        Type      => $Param{LinkType},
        Direction => $Param{Direction},
        UserID    => 1,
    );

    CONFIGITEMID:
    for my $ConfigItemID ( sort keys %LinkedConfigItemIDs ) {

        # start recursion
        $Self->_FindWarnConfigItems(
            ConfigItemID         => $ConfigItemID,
            LinkType             => $Param{LinkType},
            Direction            => $Param{Direction},
            NumberOfLinkTypes    => $Param{NumberOfLinkTypes},
            ScannedConfigItemIDs => $Param{ScannedConfigItemIDs},
        );

        next CONFIGITEMID
            if $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type}
            && $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type} eq 'incident';

        # set warning state
        $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type} = 'warning';
    }

    return 1;
}

=head2 _PrepareLikeString()

internal function to prepare like strings

    $ConfigItemObject->_PrepareLikeString( $StringRef );

=cut

sub _PrepareLikeString {
    my ( $Self, $Value ) = @_;

    return if !$Value;
    return if ref $Value ne 'SCALAR';

    # Quote
    ${$Value} = $Kernel::OM->Get('Kernel::System::DB')->Quote( ${$Value}, 'Like' );

    # replace * with %
    ${$Value} =~ s{ \*+ }{%}xmsg;

    return;
}

1;

=head1 ITSM Config Item events:

ConfigItemCreate, VersionCreate, DeploymentStateUpdate, IncidentStateUpdate,
ConfigItemDelete, LinkAdd, LinkDelete, DefinitionUpdate, NameUpdate, ValueUpdate
DefinitionCreate, VersionDelete

=cut

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::ITSMConfigItem;

use strict;
use warnings;

use Kernel::System::EventHandler;
use Kernel::System::ITSMConfigItem::Definition;
use Kernel::System::ITSMConfigItem::History;
use Kernel::System::ITSMConfigItem::Number;
use Kernel::System::ITSMConfigItem::Permission;
use Kernel::System::ITSMConfigItem::Version;
use Kernel::System::ITSMConfigItem::XML;
use Kernel::System::VariableCheck qw(:all);

use Storable;

use vars qw(@ISA);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::Cache',
    'Kernel::System::DB',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::LinkObject',
    'Kernel::System::Log',
    'Kernel::System::Service',
    'Kernel::System::VirtualFS',
);

=head1 NAME

Kernel::System::ITSMConfigItem - config item lib

=head1 DESCRIPTION

All config item functions.

=head1 PUBLIC INTERFACE

=head2 new()

create an object

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

=cut

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

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

    $Self->{CacheType} = 'ITSMConfigurationManagement';
    $Self->{CacheTTL}  = 60 * 60 * 24 * 20;

    @ISA = qw(
        Kernel::System::ITSMConfigItem::Definition
        Kernel::System::ITSMConfigItem::History
        Kernel::System::ITSMConfigItem::Number
        Kernel::System::ITSMConfigItem::Permission
        Kernel::System::ITSMConfigItem::Version
        Kernel::System::ITSMConfigItem::XML
        Kernel::System::EventHandler
    );

    # init of event handler
    $Self->EventHandlerInit(
        Config => 'ITSMConfigItem::EventModulePost',
    );

    return $Self;
}

=head2 ConfigItemCount()

count all records of a config item class

    my $Count = $ConfigItemObject->ConfigItemCount(
        ClassID => 123,
    );

=cut

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

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

    # get state list
    my $StateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class       => 'ITSM::ConfigItem::DeploymentState',
        Preferences => {
            Functionality => [ 'preproductive', 'productive' ],
        },
    );

    return 0 if !%{$StateList};

    # create state string
    my $DeplStateString = join q{, }, keys %{$StateList};

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => "SELECT COUNT(id) FROM configitem WHERE class_id = ? AND "
            . "cur_depl_state_id IN ( $DeplStateString )",
        Bind  => [ \$Param{ClassID} ],
        Limit => 1,
    );

    # fetch the result
    my $Count = 0;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $Count = $Row[0];
    }

    return $Count;
}

=head2 ConfigItemResultList()

return a config item list as array hash reference

    my $ConfigItemListRef = $ConfigItemObject->ConfigItemResultList(
        ClassID => 123,
        Start   => 100,
        Limit   => 50,
    );

=cut

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

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

    # get state list
    my $StateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class       => 'ITSM::ConfigItem::DeploymentState',
        Preferences => {
            Functionality => [ 'preproductive', 'productive' ],
        },
    );

    # create state string
    my $DeplStateString = join q{, }, keys %{$StateList};

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => "SELECT id FROM configitem "
            . "WHERE class_id = ? AND cur_depl_state_id IN ( $DeplStateString ) "
            . "ORDER BY change_time DESC",
        Bind  => [ \$Param{ClassID} ],
        Start => $Param{Start},
        Limit => $Param{Limit},
    );

    # fetch the result
    my @ConfigItemIDList;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        push @ConfigItemIDList, $Row[0];
    }

    # get last versions data
    my @ConfigItemList;
    for my $ConfigItemID (@ConfigItemIDList) {

        # get version data
        my $LastVersion = $Self->VersionGet(
            ConfigItemID => $ConfigItemID,
            XMLDataGet   => 0,
        );

        push @ConfigItemList, $LastVersion;
    }

    return \@ConfigItemList;
}

=head2 ConfigItemGet()

return a config item as hash reference

    my $ConfigItem = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => 123,
        Cache        => 0,    # (optional) default 1 (0|1)
    );

A hashref with the following keys is returned:

    $ConfigItem{ConfigItemID}
    $ConfigItem{Number}
    $ConfigItem{ClassID}
    $ConfigItem{Class}
    $ConfigItem{LastVersionID}
    $ConfigItem{CurDeplStateID}
    $ConfigItem{CurDeplState}
    $ConfigItem{CurDeplStateType}
    $ConfigItem{CurInciStateID}
    $ConfigItem{CurInciState}
    $ConfigItem{CurInciStateType}
    $ConfigItem{CreateTime}
    $ConfigItem{CreateBy}
    $ConfigItem{ChangeTime}
    $ConfigItem{ChangeBy}

=cut

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

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

    # enable cache per default
    if ( !defined $Param{Cache} ) {
        $Param{Cache} = 1;
    }

    # check if result is already cached
    my $CacheKey    = 'ConfigItemGet::ConfigItemID::' . $Param{ConfigItemID};
    my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');
    my $Cache       = $CacheObject->Get(
        Type => $Self->{CacheType},
        Key  => $CacheKey,
    );
    return Storable::dclone($Cache) if $Cache;

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT id, configitem_number, class_id, last_version_id, '
            . 'cur_depl_state_id, cur_inci_state_id, '
            . 'create_time, create_by, change_time, change_by '
            . 'FROM configitem WHERE id = ?',
        Bind  => [ \$Param{ConfigItemID} ],
        Limit => 1,
    );

    # fetch the result
    my %ConfigItem;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $ConfigItem{ConfigItemID}   = $Row[0];
        $ConfigItem{Number}         = $Row[1];
        $ConfigItem{ClassID}        = $Row[2];
        $ConfigItem{LastVersionID}  = $Row[3];
        $ConfigItem{CurDeplStateID} = $Row[4];
        $ConfigItem{CurInciStateID} = $Row[5];
        $ConfigItem{CreateTime}     = $Row[6];
        $ConfigItem{CreateBy}       = $Row[7];
        $ConfigItem{ChangeTime}     = $Row[8];
        $ConfigItem{ChangeBy}       = $Row[9];
    }

    # check config item
    if ( !$ConfigItem{ConfigItemID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No such ConfigItemID ($Param{ConfigItemID})!",
        );
        return;
    }

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    $ConfigItem{Class} = $ClassList->{ $ConfigItem{ClassID} };

    return \%ConfigItem if !$ConfigItem{CurDeplStateID} || !$ConfigItem{CurInciStateID};

    # get deployment state functionality
    my $DeplState = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemGet(
        ItemID => $ConfigItem{CurDeplStateID},
    );

    $ConfigItem{CurDeplState}     = $DeplState->{Name};
    $ConfigItem{CurDeplStateType} = $DeplState->{Functionality};

    # get incident state functionality
    my $InciState = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemGet(
        ItemID => $ConfigItem{CurInciStateID},
    );

    $ConfigItem{CurInciState}     = $InciState->{Name};
    $ConfigItem{CurInciStateType} = $InciState->{Functionality};

    # cache the result
    $CacheObject->Set(
        Type  => $Self->{CacheType},
        TTL   => $Self->{CacheTTL},
        Key   => $CacheKey,
        Value => Storable::dclone( \%ConfigItem ),
    );

    return \%ConfigItem;
}

=head2 ConfigItemAdd()

add a new config item

    my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        Number  => '111',  # (optional)
        ClassID => 123,
        UserID  => 1,
    );

=cut

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

    NEEDED:
    for my $Needed (qw(ClassID UserID)) {

        next NEEDED if defined $Param{$Needed};

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need $Needed!",
        );
        return;
    }

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    return if !$ClassList;
    return if ref $ClassList ne 'HASH';

    # check the class id
    if ( !$ClassList->{ $Param{ClassID} } ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'No valid class id given!',
        );
        return;
    }

    # create config item number
    if ( $Param{Number} ) {

        # find existing config item number
        my $Exists = $Self->ConfigItemNumberLookup(
            ConfigItemNumber => $Param{Number},
        );

        if ($Exists) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => 'Config item number already exists!',
            );
            return;
        }
    }
    else {

        # create config item number
        $Param{Number} = $Self->ConfigItemNumberCreate(
            Type    => $Kernel::OM->Get('Kernel::Config')->Get('ITSMConfigItem::NumberGenerator'),
            ClassID => $Param{ClassID},
        );
    }

    # insert new config item
    my $Success = $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL => 'INSERT INTO configitem '
            . '(configitem_number, class_id, create_time, create_by, change_time, change_by) '
            . 'VALUES (?, ?, current_timestamp, ?, current_timestamp, ?)',
        Bind => [ \$Param{Number}, \$Param{ClassID}, \$Param{UserID}, \$Param{UserID} ],
    );

    return if !$Success;

    # find id of new item
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT id FROM configitem WHERE '
            . 'configitem_number = ? AND class_id = ? ORDER BY id DESC',
        Bind  => [ \$Param{Number}, \$Param{ClassID} ],
        Limit => 1,
    );

    # fetch the result
    my $ConfigItemID;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $ConfigItemID = $Row[0];
    }

    # trigger ConfigItemCreate
    $Self->EventHandler(
        Event => 'ConfigItemCreate',
        Data  => {
            ConfigItemID => $ConfigItemID,
            Comment      => $ConfigItemID . '%%' . $Param{Number},
        },
        UserID => $Param{UserID},
    );

    return $ConfigItemID;
}

=head2 ConfigItemDelete()

delete an existing config item

    my $True = $ConfigItemObject->ConfigItemDelete(
        ConfigItemID => 123,
        UserID       => 1,
    );

=cut

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

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

    # remember config item data before delete
    my $ConfigItemData = $Self->ConfigItemGet(
        ConfigItemID => $Param{ConfigItemID},
    );

    # delete all links to this config item first, before deleting the versions
    return if !$Kernel::OM->Get('Kernel::System::LinkObject')->LinkDeleteAll(
        Object => 'ITSMConfigItem',
        Key    => $Param{ConfigItemID},
        UserID => $Param{UserID},
    );

    # delete existing versions
    $Self->VersionDelete(
        ConfigItemID => $Param{ConfigItemID},
        UserID       => $Param{UserID},
    );

    # get a list of all attachments
    my @ExistingAttachments = $Self->ConfigItemAttachmentList(
        ConfigItemID => $Param{ConfigItemID},
    );

    # delete all attachments of this config item
    FILENAME:
    for my $Filename (@ExistingAttachments) {

        # delete the attachment
        my $DeletionSuccess = $Self->ConfigItemAttachmentDelete(
            ConfigItemID => $Param{ConfigItemID},
            Filename     => $Filename,
            UserID       => $Param{UserID},
        );

        if ( !$DeletionSuccess ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Unknown problem when deleting attachment $Filename of ConfigItem "
                    . "$Param{ConfigItemID}. Please check the VirtualFS backend for stale "
                    . "files!",
            );
        }
    }

    # trigger ConfigItemDelete event
    # this must be done before deleting the config item from the database,
    # because of a foreign key constraint in the configitem_history table
    $Self->EventHandler(
        Event => 'ConfigItemDelete',
        Data  => {
            ConfigItemID => $Param{ConfigItemID},
            Comment      => $Param{ConfigItemID},
            Number       => $ConfigItemData->{Number},
            Class        => $ConfigItemData->{Class},
        },
        UserID => $Param{UserID},
    );

    # delete config item
    my $Success = $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL  => 'DELETE FROM configitem WHERE id = ?',
        Bind => [ \$Param{ConfigItemID} ],
    );

    # delete the cache
    my $CacheKey = 'ConfigItemGet::ConfigItemID::' . $Param{ConfigItemID};
    $Kernel::OM->Get('Kernel::System::Cache')->Delete(
        Type => $Self->{CacheType},
        Key  => $CacheKey,
    );

    return $Success;
}

=head2 ConfigItemAttachmentAdd()

adds an attachment to a config item

    my $Success = $ConfigItemObject->ConfigItemAttachmentAdd(
        ConfigItemID    => 1,
        Filename        => 'filename',
        Content         => 'content',
        ContentType     => 'text/plain',
        UserID          => 1,
    );

=cut

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

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

            return;
        }
    }

    # write to virtual fs
    my $Success = $Kernel::OM->Get('Kernel::System::VirtualFS')->Write(
        Filename    => "ConfigItem/$Param{ConfigItemID}/$Param{Filename}",
        Mode        => 'binary',
        Content     => \$Param{Content},
        Preferences => {
            ContentID    => $Param{ContentID},
            ContentType  => $Param{ContentType},
            ConfigItemID => $Param{ConfigItemID},
            UserID       => $Param{UserID},
        },
    );

    # check for error
    if ($Success) {

        # trigger AttachmentAdd-Event
        $Self->EventHandler(
            Event => 'AttachmentAddPost',
            Data  => {
                %Param,
                ConfigItemID => $Param{ConfigItemID},
                Comment      => $Param{Filename},
                HistoryType  => 'AttachmentAdd',
            },
            UserID => $Param{UserID},
        );
    }
    else {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Cannot add attachment for config item $Param{ConfigItemID}",
        );

        return;
    }

    return 1;
}

=head2 ConfigItemAttachmentDelete()

Delete the given file from the virtual filesystem.

    my $Success = $ConfigItemObject->ConfigItemAttachmentDelete(
        ConfigItemID => 123,               # used in event handling, e.g. for logging the history
        Filename     => 'Projectplan.pdf', # identifies the attachment (together with the ConfigItemID)
        UserID       => 1,
    );

=cut

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

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

            return;
        }
    }

    # add prefix
    my $Filename = 'ConfigItem/' . $Param{ConfigItemID} . '/' . $Param{Filename};

    # delete file
    my $Success = $Kernel::OM->Get('Kernel::System::VirtualFS')->Delete(
        Filename => $Filename,
    );

    # check for error
    if ($Success) {

        # trigger AttachmentDeletePost-Event
        $Self->EventHandler(
            Event => 'AttachmentDeletePost',
            Data  => {
                %Param,
                ConfigItemID => $Param{ConfigItemID},
                Comment      => $Param{Filename},
                HistoryType  => 'AttachmentDelete',
            },
            UserID => $Param{UserID},
        );
    }
    else {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Cannot delete attachment $Filename!",
        );

        return;
    }

    return $Success;
}

=head2 ConfigItemAttachmentGet()

This method returns information about one specific attachment.

    my $Attachment = $ConfigItemObject->ConfigItemAttachmentGet(
        ConfigItemID => 4,
        Filename     => 'test.txt',
    );

returns

    {
        Preferences => {
            AllPreferences => 'test',
        },
        Filename    => 'test.txt',
        Content     => 'content',
        ContentType => 'text/plain',
        Filesize    => 12348409,
        Type        => 'attachment',
    }

=cut

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

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

    # add prefix
    my $Filename = 'ConfigItem/' . $Param{ConfigItemID} . '/' . $Param{Filename};

    # find all attachments of this config item
    my @Attachments = $Kernel::OM->Get('Kernel::System::VirtualFS')->Find(
        Filename    => $Filename,
        Preferences => {
            ConfigItemID => $Param{ConfigItemID},
        },
    );

    # return error if file does not exist
    if ( !@Attachments ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Message  => "No such attachment ($Filename)!",
            Priority => 'error',
        );
        return;
    }

    # get data for attachment
    my %AttachmentData = $Kernel::OM->Get('Kernel::System::VirtualFS')->Read(
        Filename => $Filename,
        Mode     => 'binary',
    );

    my $AttachmentInfo = {
        %AttachmentData,
        Filename    => $Param{Filename},
        Content     => ${ $AttachmentData{Content} },
        ContentType => $AttachmentData{Preferences}->{ContentType},
        Type        => 'attachment',
        Filesize    => $AttachmentData{Preferences}->{FilesizeRaw},
    };

    return $AttachmentInfo;
}

=head2 ConfigItemAttachmentList()

Returns an array with all attachments of the given config item.

    my @Attachments = $ConfigItemObject->ConfigItemAttachmentList(
        ConfigItemID => 123,
    );

returns

    @Attachments = (
        'filename.txt',
        'other_file.pdf',
    );

=cut

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

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

        return;
    }

    # find all attachments of this config item
    my @Attachments = $Kernel::OM->Get('Kernel::System::VirtualFS')->Find(
        Preferences => {
            ConfigItemID => $Param{ConfigItemID},
        },
    );

    for my $Filename (@Attachments) {

        # remove extra information from filename
        $Filename =~ s{ \A ConfigItem / \d+ / }{}xms;
    }

    return @Attachments;
}

=head2 ConfigItemAttachmentExists()

Checks if a file with a given filename exists.

    my $Exists = $ConfigItemObject->ConfigItemAttachmentExists(
        Filename => 'test.txt',
        ConfigItemID => 123,
        UserID   => 1,
    );

=cut

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

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

            return;
        }
    }

    return if !$Kernel::OM->Get('Kernel::System::VirtualFS')->Find(
        Filename => 'ConfigItem/' . $Param{ConfigItemID} . '/' . $Param{Filename},
    );

    return 1;
}

=head2 ConfigItemSearchExtended()

return a config item list as an array reference

    my $ConfigItemIDs = $ConfigItemObject->ConfigItemSearchExtended(
        Number       => 'The ConfigItem Number',  # (optional)
        Name         => 'The Name',               # (optional)
        ClassIDs     => [9, 8, 7, 6],             # (optional)
        DeplStateIDs => [1, 2, 3, 4],             # (optional)
        InciStateIDs => [1, 2, 3, 4],             # (optional)

        # config items with created time after ...
        ConfigItemCreateTimeNewerDate => '2006-01-09 00:00:01',  # (optional)
        # config items with created time before then ....
        ConfigItemCreateTimeOlderDate => '2006-01-19 23:59:59',  # (optional)

        # config items with changed time after ...
        ConfigItemChangeTimeNewerDate => '2006-01-09 00:00:01',  # (optional)
        # config items with changed time before then ....
        ConfigItemChangeTimeOlderDate => '2006-01-19 23:59:59',  # (optional)

        What => [                                                # (optional)
            # each array element is a and condition
            {
                # or condition in hash
                "[%]{'ElementA'}[%]{'ElementB'}[%]{'Content'}" => '%contentA%',
                "[%]{'ElementA'}[%]{'ElementC'}[%]{'Content'}" => '%contentA%',
            },
            {
                "[%]{'ElementA'}[%]{'ElementB'}[%]{'Content'}" => '%contentB%',
                "[%]{'ElementA'}[%]{'ElementC'}[%]{'Content'}" => '%contentB%',
            },
            {
                # use array reference if different content with same key was searched
                "[%]{'ElementA'}[%]{'ElementB'}[%]{'Content'}" => ['%contentC%', '%contentD%', '%contentE%'],
                "[%]{'ElementA'}[%]{'ElementC'}[%]{'Content'}" => ['%contentC%', '%contentD%', '%contentE%'],
            },
        ],

        PreviousVersionSearch => 1,  # (optional) default 0 (0|1)

        OrderBy => [ 'ConfigItemID', 'Number' ],                  # (optional)
        # default: [ 'ConfigItemID' ]
        # (ConfigItemID, Number, Name, ClassID, DeplStateID, InciStateID,
        # CreateTime, CreateBy, ChangeTime, ChangeBy)

        # 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)
        # default: [ 'Down' ]
        # (Down | Up)

        Limit          => 122,  # (optional)
        UsingWildcards => 0,    # (optional) default 1
    );

=cut

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

    # set limit
    my $Limit = $Param{Limit};
    $Param{Limit} = undef;

    # config item search is required if one of these params is given
    my @ConfigItemSearchParams = (
        'ConfigItemCreateTimeNewerDate',
        'ConfigItemCreateTimeOlderDate',
        'ConfigItemChangeTimeNewerDate',
        'ConfigItemChangeTimeOlderDate',
    );

    # check, if config item search is required
    my %RequiredSearch;
    CONFIGITEMPARAM:
    for my $ConfigItemParam (@ConfigItemSearchParams) {
        next CONFIGITEMPARAM if !$Param{$ConfigItemParam};

        $RequiredSearch{ConfigItem} = 1;
        last CONFIGITEMPARAM;
    }

    # special handling for config item number
    # number 0 is allowed but not the empty string
    if ( defined $Param{Number} && $Param{Number} ne '' ) {
        $RequiredSearch{ConfigItem} = 1;
    }

    # version search is required if Name, What or PreviousVersionSearch is given
    if (
        IsStringWithData( $Param{Name} )
        || IsArrayRefWithData( $Param{What} )
        || $Param{PreviousVersionSearch}
        )
    {
        $RequiredSearch{Version} = 1;
    }

    # version search is also required if sorting by name (fix for bug #7072)
    ORDERBY:
    for my $OrderBy ( @{ $Param{OrderBy} } ) {
        if ( $OrderBy eq 'Name' ) {
            $RequiredSearch{Version} = 1;
            last ORDERBY;
        }
    }

    # xml version search is required if What is given
    if ( IsArrayRefWithData( $Param{What} ) ) {
        $RequiredSearch{XMLVersion} = 1;
    }

    # use config item search as fallback
    if ( !%RequiredSearch ) {
        $RequiredSearch{ConfigItem} = 1;
    }

    # start config item search
    my %ConfigItemLists;
    if ( $RequiredSearch{ConfigItem} ) {

        # search config items
        $ConfigItemLists{ConfigItem} = $Self->ConfigItemSearch(%Param);

        return    if !$ConfigItemLists{ConfigItem};
        return    if ref $ConfigItemLists{ConfigItem} ne 'ARRAY';
        return [] if !@{ $ConfigItemLists{ConfigItem} };
    }

    # start version search
    if ( $RequiredSearch{Version} ) {

        # search versions
        $ConfigItemLists{Version} = $Self->VersionSearch(%Param);

        return    if !$ConfigItemLists{Version};
        return    if ref $ConfigItemLists{Version} ne 'ARRAY';
        return [] if !@{ $ConfigItemLists{Version} };
    }

    # start xml version search
    if ( $RequiredSearch{XMLVersion} ) {

        # search xml versions
        my $XMLVersionList = $Self->_XMLVersionSearch(%Param);

        return    if !$XMLVersionList;
        return    if ref $XMLVersionList ne 'HASH';
        return [] if !%{$XMLVersionList};

        # get config item ids
        my %ConfigItemListTmp;
        VERSIONID:
        for my $VersionID ( sort keys %{$XMLVersionList} ) {
            my $ConfigItemID = $Self->VersionConfigItemIDGet(
                VersionID => $VersionID,
            );

            next VERSIONID if !$ConfigItemID;

            $ConfigItemListTmp{$ConfigItemID} = 1;
        }

        # add ids to config item list
        $ConfigItemLists{XMLVersion} = \%ConfigItemListTmp;
    }

    # create the result list
    my @ResultList;
    if ( $RequiredSearch{ConfigItem} && $RequiredSearch{Version} ) {

        # build a lookup hash of all found configitem ids in $ConfigItemLists{ConfigItem}
        my %ConfigItemSeen = map { $_ => 1 } @{ $ConfigItemLists{ConfigItem} };

        # check all config item ids, we need to keep the sorting
        CONFIGITEMID:
        for my $ConfigItemID ( @{ $ConfigItemLists{Version} } ) {
            next CONFIGITEMID if !$ConfigItemSeen{$ConfigItemID};
            push @ResultList, $ConfigItemID;
        }
    }
    elsif ( $RequiredSearch{ConfigItem} ) {
        @ResultList = @{ $ConfigItemLists{ConfigItem} };
    }
    elsif ( $RequiredSearch{Version} ) {
        @ResultList = @{ $ConfigItemLists{Version} };
    }

    # consider the XML result
    if ( $RequiredSearch{XMLVersion} ) {
        @ResultList = grep { $ConfigItemLists{XMLVersion}->{$_} } @ResultList;
    }

    # consider limit
    if ( $Limit && $Limit < scalar @ResultList ) {

        # extract the limited ids
        $Limit--;
        @ResultList = @ResultList[ 0 .. $Limit ];
    }

    return \@ResultList;
}

=head2 ConfigItemSearch()

return a config item list as an array reference

    my $ConfigItemIDs = $ConfigItemObject->ConfigItemSearch(
        Number       => 'The ConfigItem Number',  # (optional)
        ClassIDs     => [9, 8, 7, 6],             # (optional)
        DeplStateIDs => [1, 2, 3, 4],             # (optional)
        InciStateIDs => [1, 2, 3, 4],             # (optional)
        CreateBy     => [1, 2, 3],                # (optional)
        ChangeBy     => [3, 2, 1],                # (optional)

        # config items with created time after ...
        ConfigItemCreateTimeNewerDate => '2006-01-09 00:00:01',  # (optional)
        # config items with created time before then ....
        ConfigItemCreateTimeOlderDate => '2006-01-19 23:59:59',  # (optional)

        # config items with changed time after ...
        ConfigItemChangeTimeNewerDate => '2006-01-09 00:00:01',  # (optional)
        # config items with changed time before then ....
        ConfigItemChangeTimeOlderDate => '2006-01-19 23:59:59',  # (optional)

        OrderBy => [ 'ConfigItemID', 'Number' ],                  # (optional)
        # default: [ 'ConfigItemID' ]
        # (ConfigItemID, Number, ClassID, DeplStateID, InciStateID,
        # CreateTime, CreateBy, ChangeTime, ChangeBy)

        # 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)
        # default: [ 'Down' ]
        # (Down | Up)

        Limit          => 122,  # (optional)
        UsingWildcards => 0,    # (optional) default 1
    );

=cut

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

    # verify that all passed array parameters contain an arrayref
    ARGUMENT:
    for my $Argument (
        qw(
        OrderBy
        OrderByDirection
        )
        )
    {
        if ( !defined $Param{$Argument} ) {
            $Param{$Argument} ||= [];

            next ARGUMENT;
        }

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

    # define order table
    my %OrderByTable = (
        ConfigItemID => 'id',
        Number       => 'configitem_number',
        ClassID      => 'class_id',
        DeplStateID  => 'cur_depl_state_id',
        InciStateID  => 'cur_inci_state_id',
        CreateTime   => 'create_time',
        CreateBy     => 'create_by',
        ChangeTime   => 'change_time',
        ChangeBy     => 'change_by',
    );

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

        next ORDERBY if $OrderBy eq 'Name';

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

            # found an error
            $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;
    }

    # set default values
    if ( !defined $Param{UsingWildcards} ) {
        $Param{UsingWildcards} = 1;
    }

    # get like escape string needed for some databases (e.g. oracle)
    my $LikeEscapeString = $Kernel::OM->Get('Kernel::System::DB')->GetDatabaseFunction('LikeEscapeString');

    # assemble the ORDER BY clause
    my @SQLOrderBy;
    my $Count = 0;
    ORDERBY:
    for my $OrderBy ( @{ $Param{OrderBy} } ) {

        next ORDERBY if $OrderBy eq 'Name';

        # 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';
            }
        }

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

    }
    continue {
        $Count++;
    }

    # if there is a possibility that the ordering is not determined
    # we add an ascending ordering by id
    if ( !grep { $_ eq 'ConfigItemID' } ( @{ $Param{OrderBy} } ) ) {
        push @SQLOrderBy, "$OrderByTable{ConfigItemID} ASC";
    }

    # add number to sql where array
    my @SQLWhere;
    if ( defined $Param{Number} && $Param{Number} ne '' && ref $Param{Number} ne 'ARRAY' ) {

        # quote
        $Param{Number} = $Kernel::OM->Get('Kernel::System::DB')->Quote( $Param{Number} );

        if ( $Param{UsingWildcards} ) {

            # prepare like string
            $Self->_PrepareLikeString( \$Param{Number} );

            push @SQLWhere,
                "LOWER(configitem_number) LIKE LOWER('$Param{Number}') $LikeEscapeString";
        }
        else {
            push @SQLWhere, "LOWER(configitem_number) = LOWER('$Param{Number}')";
        }
    }
    elsif ( defined $Param{Number} && $Param{Number} ne '' && ref $Param{Number} eq 'ARRAY' ) {

        # Create string.
        my $InString = join q{, }, @{ $Param{Number} };

        push @SQLWhere, "LOWER(configitem_number) IN ($InString)";
    }

    # set array params
    my %ArrayParams = (
        ConfigItemIDs => 'id',
        ClassIDs      => 'class_id',
        DeplStateIDs  => 'cur_depl_state_id',
        InciStateIDs  => 'cur_inci_state_id',
        CreateBy      => 'create_by',
        ChangeBy      => 'change_by',
    );

    ARRAYPARAM:
    for my $ArrayParam ( sort keys %ArrayParams ) {

        next ARRAYPARAM if !$Param{$ArrayParam};

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

        next ARRAYPARAM if !@{ $Param{$ArrayParam} };

        # quote as integer
        for my $OneParam ( @{ $Param{$ArrayParam} } ) {
            $OneParam = $Kernel::OM->Get('Kernel::System::DB')->Quote( $OneParam, 'Integer' );
        }

        # create string
        my $InString = join q{, }, @{ $Param{$ArrayParam} };

        next ARRAYPARAM if !$InString;

        push @SQLWhere, "$ArrayParams{ $ArrayParam } IN ($InString)";
    }

    # set time params
    my %TimeParams = (
        ConfigItemCreateTimeNewerDate => 'create_time >=',
        ConfigItemCreateTimeOlderDate => 'create_time <=',
        ConfigItemChangeTimeNewerDate => 'change_time >=',
        ConfigItemChangeTimeOlderDate => 'change_time <=',
    );

    TIMEPARAM:
    for my $TimeParam ( sort keys %TimeParams ) {

        next TIMEPARAM if !$Param{$TimeParam};

        if ( $Param{$TimeParam} !~ m{ \A \d\d\d\d-\d\d-\d\d \s \d\d:\d\d:\d\d \z }xms ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Invalid date format found!",
            );
            return;
        }

        # quote
        $Param{$TimeParam} = $Kernel::OM->Get('Kernel::System::DB')->Quote( $Param{$TimeParam} );

        push @SQLWhere, "$TimeParams{ $TimeParam } '$Param{ $TimeParam }'";
    }

    # create where string
    my $WhereString = @SQLWhere ? ' WHERE ' . join q{ AND }, @SQLWhere : '';

    # set limit
    if ( $Param{Limit} ) {
        $Param{Limit} = $Kernel::OM->Get('Kernel::System::DB')->Quote( $Param{Limit}, 'Integer' );
    }

    my $SQL = "SELECT id FROM configitem $WhereString ";

    # add the ORDER BY clause
    if (@SQLOrderBy) {
        $SQL .= 'ORDER BY ';
        $SQL .= join ', ', @SQLOrderBy;
        $SQL .= ' ';
    }

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL   => $SQL,
        Limit => $Param{Limit},
    );

    # fetch the result
    my @ConfigItemList;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        push @ConfigItemList, $Row[0];
    }

    return \@ConfigItemList;
}

=head2 ConfigItemLookup()

This method does a lookup for a config-item. If a config-item id is given,
it returns the number of the config-item. If a config-item number is given,
the appropriate id is returned.

    my $Number = $ConfigItemObject->ConfigItemLookup(
        ConfigItemID => 1234,
    );

    my $ID = $ConfigItemObject->ConfigItemLookup(
        ConfigItemNumber => 1000001,
    );

=cut

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

    my ($Key) = grep { $Param{$_} } qw(ConfigItemID ConfigItemNumber);

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

    # if result is cached return that result
    return $Self->{Cache}->{ConfigItemLookup}->{$Key}->{ $Param{$Key} }
        if $Self->{Cache}->{ConfigItemLookup}->{$Key}->{ $Param{$Key} };

    # set the appropriate SQL statement
    my $SQL = 'SELECT configitem_number FROM configitem WHERE id = ?';

    if ( $Key eq 'ConfigItemNumber' ) {
        $SQL = 'SELECT id FROM configitem WHERE configitem_number = ?';
    }

    # fetch the requested value
    return if !$Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL   => $SQL,
        Bind  => [ \$Param{$Key} ],
        Limit => 1,
    );

    my $Value;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $Value = $Row[0];
    }

    $Self->{Cache}->{ConfigItemLookup}->{$Key}->{ $Param{$Key} } = $Value;

    return $Value;
}

=head2 UniqueNameCheck()

This method checks all already existing config items, whether the given name does already exist
within the same config item class or among all classes, depending on the SysConfig value of
UniqueCIName::UniquenessCheckScope (Class or Global).

This method requires 3 parameters: ConfigItemID, Name and Class
"ConfigItemID"  is the ID of the ConfigItem, which is to be checked for uniqueness
"Name"          is the config item name to be checked for uniqueness
"ClassID"       is the ID of the config item's class

All parameters are mandatory.

my $DuplicateNames = $ConfigItemObject->UniqueNameCheck(
    ConfigItemID => '73'
    Name         => 'PC#005',
    ClassID      => '32',
);

The given name is not unique
my $NameDuplicates = [ 5, 35, 48, ];    # IDs of ConfigItems with the same name

The given name is unique
my $NameDuplicates = [];

=cut

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

    # check for needed stuff
    for my $Needed (qw(ConfigItemID Name ClassID)) {
        if ( !$Param{$Needed} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Missing parameter $Needed!",
            );
            return;
        }
    }

    # check ConfigItemID param for valid format
    if (
        !IsInteger( $Param{ConfigItemID} )
        && ( IsStringWithData( $Param{ConfigItemID} ) && $Param{ConfigItemID} ne 'NEW' )
        )
    {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "The ConfigItemID parameter needs to be an integer or 'NEW'",
        );
        return;
    }

    # check Name param for valid format
    if ( !IsStringWithData( $Param{Name} ) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "The Name parameter needs to be a string!",
        );
        return;
    }

    # check ClassID param for valid format
    if ( !IsInteger( $Param{ClassID} ) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "The ClassID parameter needs to be an integer",
        );
        return;
    }

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # check class list for validity
    if ( !IsHashRefWithData($ClassList) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Unable to retrieve a valid class list!",
        );
        return;
    }

    # get the class name from the class list
    my $Class = $ClassList->{ $Param{ClassID} };

    # check class for validity
    if ( !IsStringWithData($Class) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Unable to determine a config item class using the given ClassID!",
        );
        return;
    }
    elsif ( $Kernel::OM->Get('Kernel::Config')->{Debug} > 0 ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'debug',
            Message  => "Resolved ClassID $Param{ClassID} to class $Class",
        );
    }

    # get the uniqueness scope from SysConfig
    my $Scope = $Kernel::OM->Get('Kernel::Config')->Get('UniqueCIName::UniquenessCheckScope');

    # check scope for validity
    if ( !IsStringWithData($Scope) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "The configuration of UniqueCIName::UniquenessCheckScope is invalid!",
        );
        return;
    }

    if ( $Scope ne 'global' && $Scope ne 'class' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "UniqueCIName::UniquenessCheckScope is $Scope, but must be either "
                . "'global' or 'class'!",
        );
        return;
    }

    if ( $Kernel::OM->Get('Kernel::Config')->{Debug} > 0 ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'debug',
            Message  => "The scope for checking the uniqueness is $Scope",
        );
    }

    my %SearchCriteria;

    # add the config item class to the search criteria if the uniqueness scope is not global
    if ( $Scope ne 'global' ) {
        $SearchCriteria{ClassIDs} = [ $Param{ClassID} ];
    }

    $SearchCriteria{Name} = $Param{Name};

    # search for a config item matching the given name
    my $ConfigItem = $Self->ConfigItemSearchExtended(%SearchCriteria);

    # remove the provided ConfigItemID from the results, otherwise the duplicate check would fail
    # because the ConfigItem itself is found as duplicate
    my @Duplicates = map {$_} grep { $_ ne $Param{ConfigItemID} } @{$ConfigItem};

    # if a config item was found, the given name is not unique
    # if no config item was found, the given name is unique

    # return the result of the config item search for duplicates
    return \@Duplicates;
}

=head2 CurInciStateRecalc()

recalculates the current incident state of this config item and all linked config items

    my $Success = $ConfigItemObject->CurInciStateRecalc(
        ConfigItemID               => 123,
        NewConfigItemIncidentState => $NewConfigItemIncidentState,  # optional, incident states of already checked CIs
        ScannedConfigItemIDs       => $ScannedConfigItemIDs,        # optional, IDs of already checked CIs
    );

=cut

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

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

    # get incident link types and directions from config
    my $IncidentLinkTypeDirection = $Kernel::OM->Get('Kernel::Config')->Get('ITSM::Core::IncidentLinkTypeDirection');

    # to store the new incident state for CIs
    # calculated from all incident link types
    # Incorporate data from previous run(s) and remember known data.
    $Param{NewConfigItemIncidentState} //= {};
    my $KnownNewConfigItemIncidentState = Storable::dclone( $Param{NewConfigItemIncidentState} );

    # to store the relation between services and linked CIs
    my %ServiceCIRelation;

    # remember the scanned config items
    # Incorporate data from previous run(s) and remember known data.
    $Param{ScannedConfigItemIDs} //= {};
    my $KnownScannedConfigItemIDs = Storable::dclone( $Param{ScannedConfigItemIDs} );

    # find all config items with an incident state
    $Self->_FindInciConfigItems(
        ConfigItemID              => $Param{ConfigItemID},
        IncidentLinkTypeDirection => $IncidentLinkTypeDirection,
        ScannedConfigItemIDs      => $Param{ScannedConfigItemIDs},
    );

    # calculate the new CI incident state for each configured linktype
    LINKTYPE:
    for my $LinkType ( sort keys %{$IncidentLinkTypeDirection} ) {

        # get the direction
        my $LinkDirection = $IncidentLinkTypeDirection->{$LinkType};

        # investigate all config items with a warning state
        CONFIGITEMID:
        for my $ConfigItemID ( sort keys %{ $Param{ScannedConfigItemIDs} } ) {

            # Skip config items known from previous execution(s).
            if (
                IsStringWithData( $KnownScannedConfigItemIDs->{$ConfigItemID}->{Type} )
                && $KnownScannedConfigItemIDs->{$ConfigItemID}->{Type} eq
                $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type}
                )
            {
                next CONFIGITEMID;
            }

            # investigate only config items with an incident state
            next CONFIGITEMID if $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type} ne 'incident';

            $Self->_FindWarnConfigItems(
                ConfigItemID         => $ConfigItemID,
                LinkType             => $LinkType,
                Direction            => $LinkDirection,
                NumberOfLinkTypes    => scalar keys %{$IncidentLinkTypeDirection},
                ScannedConfigItemIDs => $Param{ScannedConfigItemIDs},
            );
        }

        CONFIGITEMID:
        for my $ConfigItemID ( sort keys %{ $Param{ScannedConfigItemIDs} } ) {

            # Skip config items known from previous execution(s).
            if (
                IsStringWithData( $KnownScannedConfigItemIDs->{$ConfigItemID}->{Type} )
                && $KnownScannedConfigItemIDs->{$ConfigItemID}->{Type} eq
                $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type}
                )
            {
                next CONFIGITEMID;
            }

            # extract incident state type
            my $InciStateType = $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type};

            # find all linked services of this CI
            my %LinkedServiceIDs = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkKeyList(
                Object1   => 'ITSMConfigItem',
                Key1      => $ConfigItemID,
                Object2   => 'Service',
                State     => 'Valid',
                Type      => $LinkType,
                Direction => $LinkDirection,
                UserID    => 1,
            );

            SERVICEID:
            for my $ServiceID ( sort keys %LinkedServiceIDs ) {

                # remember the CIs that are linked with this service
                push @{ $ServiceCIRelation{$ServiceID} }, $ConfigItemID;
            }

            next CONFIGITEMID if $InciStateType eq 'incident';

            $Param{NewConfigItemIncidentState}->{$ConfigItemID} = $InciStateType;
        }
    }

    # get the incident state list of warnings
    my $WarnStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class       => 'ITSM::Core::IncidentState',
        Preferences => {
            Functionality => 'warning',
        },
    );

    if ( !defined $WarnStateList ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "ITSM::Core::IncidentState Warning cannot be invalid.",
        );
    }

    my %ReverseWarnStateList = reverse %{$WarnStateList};
    my @SortedWarnList       = sort keys %ReverseWarnStateList;
    my $WarningStateID       = $ReverseWarnStateList{Warning} || $ReverseWarnStateList{ $SortedWarnList[0] };
    my $CacheObject          = $Kernel::OM->Get('Kernel::System::Cache');

    # set the new current incident state for CIs
    CONFIGITEMID:
    for my $ConfigItemID ( sort keys %{ $Param{NewConfigItemIncidentState} } ) {

        # Skip config items known from previous execution(s).
        if (
            IsStringWithData( $KnownNewConfigItemIncidentState->{$ConfigItemID} )
            && $KnownNewConfigItemIncidentState->{$ConfigItemID} eq $Param{NewConfigItemIncidentState}->{$ConfigItemID}
            )
        {
            next CONFIGITEMID;
        }

        # get new incident state type (can only be 'operational' or 'warning')
        my $InciStateType = $Param{NewConfigItemIncidentState}->{$ConfigItemID};

        # get last version
        my $LastVersion = $Self->VersionGet(
            ConfigItemID => $ConfigItemID,
            XMLDataGet   => 0,
        );

        my $CurInciStateID;
        if ( $InciStateType eq 'warning' ) {

            # check the current incident state type is in 'incident'
            # then we do not want to change it to warning
            next CONFIGITEMID if $LastVersion->{InciStateType} eq 'incident';

            $CurInciStateID = $WarningStateID;
        }
        elsif ( $InciStateType eq 'operational' ) {
            $CurInciStateID = $LastVersion->{InciStateID};
        }

        # No update necessary if incident state id of version and config item match.
        next CONFIGITEMID if $LastVersion->{CurInciStateID} eq $CurInciStateID;

        # update current incident state
        $Kernel::OM->Get('Kernel::System::DB')->Do(
            SQL  => 'UPDATE configitem SET cur_inci_state_id = ? WHERE id = ?',
            Bind => [ \$CurInciStateID, \$ConfigItemID ],
        );

        # delete the cache
        my $CacheKey = 'ConfigItemGet::ConfigItemID::' . $ConfigItemID;
        $CacheObject->Delete(
            Type => $Self->{CacheType},
            Key  => $CacheKey,
        );

        # delete affected caches for ConfigItemID
        $CacheKey = 'VersionGet::ConfigItemID::' . $ConfigItemID . '::XMLData::';
        for my $XMLData (qw(0 1)) {
            $CacheObject->Delete(
                Type => $Self->{CacheType},
                Key  => $CacheKey . $XMLData,
            );
        }
        $CacheObject->Delete(
            Type => $Self->{CacheType},
            Key  => 'VersionNameGet::ConfigItemID::' . $ConfigItemID,
        );

        # delete affected caches for last version
        my $VersionList = $Self->VersionList(
            ConfigItemID => $ConfigItemID,
        );
        my $VersionID = $VersionList->[-1];
        $CacheKey = 'VersionGet::VersionID::' . $VersionID . '::XMLData::';
        for my $XMLData (qw(0 1)) {
            $CacheObject->Delete(
                Type => $Self->{CacheType},
                Key  => $CacheKey . $XMLData,
            );
        }
        $CacheObject->Delete(
            Type => $Self->{CacheType},
            Key  => 'VersionNameGet::VersionID::' . $VersionID,
        );
    }

    # set the current incident state type for each service (influenced by linked CIs)
    SERVICEID:
    for my $ServiceID ( sort keys %ServiceCIRelation ) {

        # set default incident state type
        my $CurInciStateTypeFromCIs = 'operational';

        # get the unique config item ids which are direcly linked to this service
        my %UniqueConfigItemIDs = map { $_ => 1 } @{ $ServiceCIRelation{$ServiceID} };

        # investigate the current incident state of each config item
        CONFIGITEMID:
        for my $ConfigItemID ( sort keys %UniqueConfigItemIDs ) {

            # get config item data
            my $ConfigItemData = $Self->ConfigItemGet(
                ConfigItemID => $ConfigItemID,
                Cache        => 0,
            );

            next CONFIGITEMID if $ConfigItemData->{CurDeplStateType} ne 'productive';
            next CONFIGITEMID if $ConfigItemData->{CurInciStateType} eq 'operational';

            # check if service must be set to 'warning'
            if ( $ConfigItemData->{CurInciStateType} eq 'warning' ) {
                $CurInciStateTypeFromCIs = 'warning';
                next CONFIGITEMID;
            }

            # check if service must be set to 'incident'
            if ( $ConfigItemData->{CurInciStateType} eq 'incident' ) {
                $CurInciStateTypeFromCIs = 'incident';
                last CONFIGITEMID;
            }
        }

        # update the current incident state type from CIs of the service
        $Kernel::OM->Get('Kernel::System::Service')->ServicePreferencesSet(
            ServiceID => $ServiceID,
            Key       => 'CurInciStateTypeFromCIs',
            Value     => $CurInciStateTypeFromCIs,
            UserID    => 1,
        );
    }

    return 1;
}

=head1 INTERNAL INTERFACE

=head2 _FindInciConfigItems()

find all config items with an incident state

    $ConfigItemObject->_FindInciConfigItems(
        ConfigItemID              => $ConfigItemID,
        IncidentLinkTypeDirection => $IncidentLinkTypeDirection,
        ScannedConfigItemIDs     => \%ScannedConfigItemIDs,
    );

=cut

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

    # check needed stuff
    return if !$Param{ConfigItemID};

    # ignore already scanned ids (infinite loop protection)
    return if $Param{ScannedConfigItemIDs}->{ $Param{ConfigItemID} };

    $Param{ScannedConfigItemIDs}->{ $Param{ConfigItemID} }->{Type} = 'operational';

    # add own config item id to list of linked config items
    my %ConfigItemIDs = (
        $Param{ConfigItemID} => 1,
    );

    LINKTYPE:
    for my $LinkType ( sort keys %{ $Param{IncidentLinkTypeDirection} } ) {

        # find all linked config items (childs)
        my %LinkedConfigItemIDs = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkKeyList(
            Object1 => 'ITSMConfigItem',
            Key1    => $Param{ConfigItemID},
            Object2 => 'ITSMConfigItem',
            State   => 'Valid',
            Type    => $LinkType,

            # Direction must ALWAYS be 'Both' here as we need to include
            # all linked CIs that could influence this one!
            Direction => 'Both',

            UserID => 1,
        );

        # remember the config item ids
        %ConfigItemIDs = ( %ConfigItemIDs, %LinkedConfigItemIDs );
    }

    CONFIGITEMID:
    for my $ConfigItemID ( sort keys %ConfigItemIDs ) {

        # get config item data
        my $ConfigItem = $Self->ConfigItemGet(
            ConfigItemID => $ConfigItemID,
            Cache        => 0,
        );

        # set incident state
        if ( $ConfigItem->{CurInciStateType} eq 'incident' ) {
            $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type} = 'incident';
            next CONFIGITEMID;
        }

        # start recursion
        $Self->_FindInciConfigItems(
            ConfigItemID              => $ConfigItemID,
            IncidentLinkTypeDirection => $Param{IncidentLinkTypeDirection},
            ScannedConfigItemIDs      => $Param{ScannedConfigItemIDs},
        );
    }

    return 1;
}

=head2 _FindWarnConfigItems()

find all config items with a warning

    $ConfigItemObject->_FindWarnConfigItems(
        ConfigItemID         => $ConfigItemID,
        LinkType             => $LinkType,
        Direction            => $LinkDirection,
        NumberOfLinkTypes    => 2,
        ScannedConfigItemIDs => $ScannedConfigItemIDs,
    );

=cut

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

    # check needed stuff
    return if !$Param{ConfigItemID};

    my $IncidentCount = 0;
    for my $ConfigItemID ( sort keys %{ $Param{ScannedConfigItemIDs} } ) {
        if (
            $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type}
            && $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type} eq 'incident'
            )
        {
            $IncidentCount++;
        }
    }

# ignore already scanned ids (infinite loop protection)
# it is ok that a config item is investigated as many times as there are configured link types * number of incident config iteems
    if (
        $Param{ScannedConfigItemIDs}->{ $Param{ConfigItemID} }->{FindWarn}
        && $Param{ScannedConfigItemIDs}->{ $Param{ConfigItemID} }->{FindWarn}
        >= ( $Param{NumberOfLinkTypes} * $IncidentCount )
        )
    {
        return;
    }

    # increase the visit counter
    $Param{ScannedConfigItemIDs}->{ $Param{ConfigItemID} }->{FindWarn}++;

    # find all linked config items
    my %LinkedConfigItemIDs = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkKeyList(
        Object1   => 'ITSMConfigItem',
        Key1      => $Param{ConfigItemID},
        Object2   => 'ITSMConfigItem',
        State     => 'Valid',
        Type      => $Param{LinkType},
        Direction => $Param{Direction},
        UserID    => 1,
    );

    CONFIGITEMID:
    for my $ConfigItemID ( sort keys %LinkedConfigItemIDs ) {

        # start recursion
        $Self->_FindWarnConfigItems(
            ConfigItemID         => $ConfigItemID,
            LinkType             => $Param{LinkType},
            Direction            => $Param{Direction},
            NumberOfLinkTypes    => $Param{NumberOfLinkTypes},
            ScannedConfigItemIDs => $Param{ScannedConfigItemIDs},
        );

        next CONFIGITEMID
            if $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type}
            && $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type} eq 'incident';

        # set warning state
        $Param{ScannedConfigItemIDs}->{$ConfigItemID}->{Type} = 'warning';
    }

    return 1;
}

=head2 _PrepareLikeString()

internal function to prepare like strings

    $ConfigItemObject->_PrepareLikeString( $StringRef );

=cut

sub _PrepareLikeString {
    my ( $Self, $Value ) = @_;

    return if !$Value;
    return if ref $Value ne 'SCALAR';

    # Quote
    ${$Value} = $Kernel::OM->Get('Kernel::System::DB')->Quote( ${$Value}, 'Like' );

    # replace * with %
    ${$Value} =~ s{ \*+ }{%}xmsg;

    return;
}

1;

=head1 ITSM Config Item events:

ConfigItemCreate, VersionCreate, DeploymentStateUpdate, IncidentStateUpdate,
ConfigItemDelete, LinkAdd, LinkDelete, DefinitionUpdate, NameUpdate, ValueUpdate
DefinitionCreate, VersionDelete

=cut

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --
## nofilter(TidyAll::Plugin::Znuny::Perl::LayoutObject)

package Kernel::System::ITSMConfigItem::CIAttachment;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::Output::HTML::Layout',
    'Kernel::System::Log',
    'Kernel::System::VirtualFS',
);

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

=head1 NAME

Kernel::System::ITSMConfigItem::CIAttachment - CIAttachment lib

=head1 SYNOPSIS

All CIAttachment functions

=head1 PUBLIC INTERFACE

=head2 new()

    Don't use the constructor directly, use the ObjectManager instead:

    my $CIAttachmentObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem::CIAttachment');

=cut

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

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

    $Self->{CacheType} = 'CIAttachment';
    $Self->{CacheTTL}  = 60 * 60 * 24 * 20;

    return $Self;
}

=head2 CIAttachmentAdd()

adds an config item attachment to a config item

    my $Success = $CIAttachmentObject->CIAttachmentAdd(
        AttachmentID    => 1,
        ConfigItemField => 'Picture',
        ConfigItemID    => 123
        VersionID       => 123,
        Filename        => 'filename',
        Content         => 'content',
        ContentType     => 'text/plain',
    );

Returns:

    my $Success = 1;

=cut

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

    my $VirtualFSObject = $Kernel::OM->Get('Kernel::System::VirtualFS');
    my $LogObject       = $Kernel::OM->Get('Kernel::System::Log');

    # check needed stuff
    NEEDED:
    for my $Needed (qw(AttachmentID ConfigItemField ConfigItemID VersionID Filename Content ContentType)) {

        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    $Param{Mode} ||= 'binary';

    my $Success = $VirtualFSObject->Write(
        Filename    => "CIAttachment/$Param{AttachmentID}",
        Mode        => $Param{Mode},
        Content     => \$Param{Content},
        Preferences => {
            CIAttachmentAttachmentID    => $Param{AttachmentID},
            CIAttachmentFilename        => $Param{Filename},
            CIAttachmentContentID       => $Param{ContentID},
            CIAttachmentContentType     => $Param{ContentType},
            CIAttachmentConfigItemField => $Param{ConfigItemField},
            CIAttachmentConfigItemID    => $Param{ConfigItemID},
            CIAttachmentVersionID       => $Param{VersionID},
        },
    );
    return if !$Success;

    return $Param{Filename};
}

=head2 CIAttachmentGet()

This function returns a CIAttachment.

    my %Attachment = $CIAttachmentObject->CIAttachmentGet(
        AttachmentID => '123',
    );

Returns:

    my %Attachment = (
        Content     => 'content',
        ContentType => 'text/plain',
        Filename    => 'blub.txt',
        FilesizeRaw => '13',
        Filesize    => '13 B',
        Type        => 'attachment',
    )

=cut

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

    my $VirtualFSObject = $Kernel::OM->Get('Kernel::System::VirtualFS');
    my $LogObject       = $Kernel::OM->Get('Kernel::System::Log');
    my $LayoutObject    = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    # check needed stuff
    NEEDED:
    for my $Needed (qw(AttachmentID)) {

        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    $Param{Mode} ||= 'binary';

    my %AttachmentData = $VirtualFSObject->Read(
        Filename        => "CIAttachment/$Param{AttachmentID}",
        Mode            => $Param{Mode},
        DisableWarnings => 1,
    );
    if ( $AttachmentData{Preferences}->{FilesizeRaw} ) {
        $AttachmentData{Preferences}->{Filesize}
            = $LayoutObject->HumanReadableDataSize( Size => $AttachmentData{Preferences}->{FilesizeRaw} );
    }

    return if !%AttachmentData;
    return if !$AttachmentData{Content};

    my %Attachment = (
        Content     => ${ $AttachmentData{Content} },
        Filename    => $AttachmentData{Preferences}->{CIAttachmentFilename},
        ContentType => $AttachmentData{Preferences}->{CIAttachmentContentType},
        Filesize    => $AttachmentData{Preferences}->{Filesize},
        FilesizeRaw => $AttachmentData{Preferences}->{FilesizeRaw},
        Type        => 'attachment',
    );

    return %Attachment;
}

=head2 CIAttachmentList()

Returns an array with all config item attachments of the given config item.

    my @CIAttachmentIDs = $CIAttachmentObject->CIAttachmentList(
        ConfigItemField => 'Picture',  # optional
        ConfigItemID    => 123,        # optional
        VersionID       => 123,        # optional
    );

returns

    @CIAttachmentIDs = (
        'fewz544t4334f34sfrg',
        'dfsdfs94f0w4f48f43f',
    );

=cut

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

    my $LogObject       = $Kernel::OM->Get('Kernel::System::Log');
    my $VirtualFSObject = $Kernel::OM->Get('Kernel::System::VirtualFS');

    my %Preferences;
    if ( $Param{ConfigItemField} ) {
        $Preferences{CIAttachmentConfigItemField} = $Param{ConfigItemField};
    }
    if ( $Param{ConfigItemID} ) {
        $Preferences{CIAttachmentConfigItemID} = $Param{ConfigItemID};
    }
    if ( $Param{VersionID} ) {
        $Preferences{CIAttachmentVersionID} = $Param{VersionID};
    }

    if ( !%Preferences ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => "You need to set ConfigItemField, ConfigItemID or VersionID!",
        );
        return;
    }

    # find all attachments of this config item
    my @Attachments = $VirtualFSObject->Find(
        Filename    => $Param{Filename} || '',
        Preferences => {
            %Preferences,
        },
    );

    for my $Attachment (@Attachments) {
        $Attachment =~ s{CIAttachment \/}{}xmsi;
    }

    return @Attachments;
}

=head2 CIAttachmentDelete()

Delete the given file from the virtual filesystem.

    my $Success = $CIAttachmentObject->CIAttachmentDelete(
        AttachmentID => 123,
    );

=cut

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

    my $LogObject       = $Kernel::OM->Get('Kernel::System::Log');
    my $VirtualFSObject = $Kernel::OM->Get('Kernel::System::VirtualFS');

    # check needed stuff
    for my $Needed (qw(AttachmentID)) {
        if ( !$Param{$Needed} ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );

            return;
        }
    }

    # delete file
    my $Success = $VirtualFSObject->Delete(
        Filename => "CIAttachment/$Param{AttachmentID}",
    );

    return $Success;
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::ITSMConfigItem::Definition;

use strict;
use warnings;

use Kernel::Language qw(Translatable);

our $ObjectManagerDisabled = 1;

=head1 NAME

Kernel::System::ITSMConfigItem::Definition - sub module of Kernel::System::ITSMConfigItem

=head1 PUBLIC INTERFACE

=head1 DESCRIPTION

All definition functions.

=head2 DefinitionList()

return a config item definition list as array-hash reference

    my $DefinitionListRef = $ConfigItemObject->DefinitionList(
        ClassID => 123,
    );

returns

    my $DefinitionListRef = [
          {
            'Version'      => '1',
            'CreateTime'   => '2012-06-12 14:09:43',
            'DefinitionID' => '1',
            'CreateBy'     => '123',
            'Definition'   => '---
- Key: Vendor
  Name: Vendor
  Searchable: 1
  Input:
  - Type: Text
    Size: 50
    MaxLength: 50

- Key: Description
  Name: Description
  Searchable: 1
  Input:
  - Type: TextArea

- Key: Type
  Name: Type
  Searchable: 1
  Input:
  - Type: TextArea
  - Type: GeneralCatalog\
    Class: ITSM::ConfigItem::Computer::Type
    Translation: 1
# ... etc ...
',
          },
    ];

=cut

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

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

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT id, configitem_definition, version, create_time, create_by '
            . 'FROM configitem_definition WHERE class_id = ? ORDER BY version',
        Bind => [ \$Param{ClassID} ],
    );

    my @DefinitionList;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        my %Definition;
        $Definition{DefinitionID} = $Row[0];
        $Definition{Definition}   = $Row[1] || "--- []";
        $Definition{Version}      = $Row[2];
        $Definition{CreateTime}   = $Row[3];
        $Definition{CreateBy}     = $Row[4];

        # Check if definition code is not a YAML string.
        if ( substr( $Definition{Definition}, 0, 3 ) ne '---' ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'notice',
                Message  => "DefinitionID: $Definition{DefinitionID}"
                    . " ClassID: $Param{ClassID}"
                    . " found in legacy Perl code format, can not continue",
            );

            $Definition{Definition} = "--- []";
        }

        push @DefinitionList, \%Definition;
    }

    return \@DefinitionList;
}

=head2 DefinitionGet()

return a config item definition as hash reference

Return
    $Definition->{DefinitionID}
    $Definition->{ClassID}
    $Definition->{Class}
    $Definition->{Definition}
    $Definition->{DefinitionRef}
    $Definition->{Version}
    $Definition->{CreateTime}
    $Definition->{CreateBy}

    my $DefinitionRef = $ConfigItemObject->DefinitionGet(
        DefinitionID => 123,
    );

    or

    my $DefinitionRef = $ConfigItemObject->DefinitionGet(
        ClassID => 123,
    );

=cut

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

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

    if ( $Param{DefinitionID} ) {

        # check if result is already cached
        return $Self->{Cache}->{DefinitionGet}->{ $Param{DefinitionID} }
            if $Self->{Cache}->{DefinitionGet}->{ $Param{DefinitionID} };

        # ask database
        $Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL => 'SELECT id, class_id, configitem_definition, version, create_time, create_by '
                . 'FROM configitem_definition WHERE id = ?',
            Bind  => [ \$Param{DefinitionID} ],
            Limit => 1,
        );
    }
    else {

        # ask database
        $Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL => 'SELECT id, class_id, configitem_definition, version, create_time, create_by '
                . 'FROM configitem_definition '
                . 'WHERE class_id = ? ORDER BY version DESC',
            Bind  => [ \$Param{ClassID} ],
            Limit => 1,
        );
    }

    # fetch the result
    my %Definition;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {

        $Definition{DefinitionID} = $Row[0];
        $Definition{ClassID}      = $Row[1];
        $Definition{Definition}   = $Row[2] || "--- []";
        $Definition{Version}      = $Row[3];
        $Definition{CreateTime}   = $Row[4];
        $Definition{CreateBy}     = $Row[5];

        # Check if definition code is not a YAML string.
        if ( substr( $Definition{Definition}, 0, 3 ) ne '---' ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'notice',
                Message  => "DefinitionID: $Definition{DefinitionID}"
                    . " ClassID: $Definition{ClassID}"
                    . " found in legacy Perl code format, can not continue",
            );

            $Definition{Definition} = "--- []";
        }

        $Definition{DefinitionRef} = $Kernel::OM->Get('Kernel::System::YAML')->Load(
            Data => $Definition{Definition},
        );
    }

    return {} if !$Definition{DefinitionID};

    # prepare definition
    if ( $Definition{DefinitionRef} && ref $Definition{DefinitionRef} eq 'ARRAY' ) {
        $Self->_DefinitionPrepare(
            DefinitionRef => $Definition{DefinitionRef},
        );
    }
    else {
        $Definition{DefinitionRef} = '';
    }

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # add class
    $Definition{Class} = $ClassList->{ $Definition{ClassID} };

    # cache the result
    $Self->{Cache}->{DefinitionGet}->{ $Definition{DefinitionID} } = \%Definition;

    return \%Definition;
}

=head2 DefinitionAdd()

add a new definition

    my $DefinitionID = $ConfigItemObject->DefinitionAdd(
        ClassID    => 123,
        Definition => 'the definition code',
        UserID     => 1,
    );

=cut

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

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

    # check definition
    my $Check = $Self->DefinitionCheck(
        Definition => $Param{Definition},
    );

    return if !$Check;

    # get last definition
    my $LastDefinition = $Self->DefinitionGet(
        ClassID => $Param{ClassID},
    );

    # stop add, if definition was not changed
    if ( $LastDefinition->{DefinitionID} && $LastDefinition->{Definition} eq $Param{Definition} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Can't add new definition! The definition was not changed.",
        );
        return;
    }

    # set version
    my $Version = 1;
    if ( $LastDefinition->{Version} ) {
        $Version = $LastDefinition->{Version};
        $Version++;
    }

    # insert new definition
    my $Success = $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL => 'INSERT INTO configitem_definition '
            . '(class_id, configitem_definition, version, create_time, create_by) VALUES '
            . '(?, ?, ?, current_timestamp, ?)',
        Bind => [ \$Param{ClassID}, \$Param{Definition}, \$Version, \$Param{UserID} ],
    );

    return if !$Success;

    # get id of new definition
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT id FROM configitem_definition WHERE '
            . 'class_id = ? AND version = ? '
            . 'ORDER BY version DESC',
        Bind  => [ \$Param{ClassID}, \$Version ],
        Limit => 1,
    );

    # fetch the result
    my $DefinitionID;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $DefinitionID = $Row[0];
    }

    # trigger DefinitionCreate event
    $Self->EventHandler(
        Event => 'DefinitionCreate',
        Data  => {
            Comment => $DefinitionID,
        },
        UserID => $Param{UserID},
    );

    return $DefinitionID;
}

=head2 DefinitionCheck()

check the syntax of a new definition

    my $True = $ConfigItemObject->DefinitionCheck(
        Definition      => 'the definition code',
        CheckSubElement => 1,                 # (optional, default 0, to check sub elements recursively)
    );

=cut

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

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

    # if check sub elements is enabled, we must not YAML load it
    # because this has been done in an earlier recursion step already
    my $Definition;
    if ( $Param{CheckSubElement} ) {
        $Definition = $Param{Definition};
    }
    else {
        if ( substr( $Param{Definition}, 0, 3 ) ne '---' ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Definition must be a YAML string",
            );
            return;
        }

        $Definition = $Kernel::OM->Get('Kernel::System::YAML')->Load(
            Data => $Param{Definition},
        );
    }

    # check if definition exists at all
    if ( !$Definition ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Invalid Definition! You have an syntax error in the definition.',
        );
        return;
    }

    # definition must be an array
    if ( ref $Definition ne 'ARRAY' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Invalid Definition! Definition is not an array reference.',
        );
        return;
    }

    # check each definition attribute
    for my $Attribute ( @{$Definition} ) {

        # each definition attribute must be a hash reference with data
        if ( !$Attribute || ref $Attribute ne 'HASH' || !%{$Attribute} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => 'Invalid Definition! At least one definition attribute is not a hash reference.',
            );
            return;
        }

        # check if the key contains no spaces
        if ( $Attribute->{Key} && $Attribute->{Key} =~ m{ \s }xms ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Invalid Definition! Key '$Attribute->{Key}' must not contain whitespace!",
            );
            return;
        }

        # check if the key contains non-ascii characters
        if ( $Attribute->{Key} && $Attribute->{Key} =~ m{ ([^\x{00}-\x{7f}]) }xms ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Invalid Definition! Key '$Attribute->{Key}' must not contain non ASCII characters '$1'!",
            );
            return;
        }

        # recursion check for Sub-Elements
        for my $Key ( sort keys %{$Attribute} ) {

            my $Value = $Attribute->{$Key};

            if ( $Key eq 'Sub' && ref $Value eq 'ARRAY' ) {

                # check the sub array
                my $Check = $Self->DefinitionCheck(
                    Definition      => $Value,
                    CheckSubElement => 1,
                );

                if ( !$Check ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message =>
                            "Invalid Sub-Definition of element with the key '$Attribute->{Key}'.",
                    );
                    return;
                }
            }
        }
    }

    return 1;
}

=head2 DefinitionAttributeInfo()

Return attribute information from the definition for a given attribute path name

    my $AttributeInfo = $ConfigItemObject->DefinitionAttributeInfo(
        AttributePath => 'HardDisk::Capacity',
        Definition    => '
            [
                # ...

                {
                    Key => 'HardDisk',
                    Name => 'Hard Disk',
                    Input => {
                        Type => 'Text',
                        Size => 50,
                        MaxLength => 100,
                    },
                    CountMax => 10,
                    Sub => [
                        {
                            Key => 'Capacity',
                            Name => 'Capacity',
                            Input => {
                                Type => 'Text',
                                Size => 20,
                                MaxLength => 10,
                            },
                        },
                    ],
                },

                # ...
            ];
        ',
    );

    Returns:

        my $AttributeInfo =  {
            CountDefault => 1,
            CountMax     => 1,
            Input => {
                Size      => 20,
                Type      => 'Text',
                MaxLength => 10
            },
            Name     => 'Capacity',
            CountMin => 1,
            Key      => 'Capacity'
        };

=cut

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

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

    my $Subtree = $Param{Definition};
    my $Info;

    PART:
    for my $Part ( split /::/, $Param{AttributePath} ) {
        my ($Found) = grep { $_->{Key} eq $Part } @{$Subtree};

        last PART if !$Found;

        $Subtree = $Found->{Sub};
        $Info    = $Found;
    }

    return $Info;
}

=begin Internal:

=head2 _DefinitionPrepare()

Prepare the syntax of a new definition

    my $True = $ConfigItemObject->_DefinitionPrepare(
        DefinitionRef => $ArrayRef,
    );

=cut

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

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

    for my $Item ( @{ $Param{DefinitionRef} } ) {

        # set CountMin
        if ( !defined $Item->{CountMin} ) {
            $Item->{CountMin} = 1;
        }

        # set CountMax
        $Item->{CountMax} ||= 1;

        # set CountMin
        if ( $Item->{CountMin} > $Item->{CountMax} ) {
            $Item->{CountMin} = $Item->{CountMax};
        }

        # set CountDefault
        if ( !defined $Item->{CountDefault} ) {
            $Item->{CountDefault} = 1;
        }
        if ( $Item->{CountDefault} < $Item->{CountMin} ) {
            $Item->{CountDefault} = $Item->{CountMin};
        }
        if ( $Item->{CountDefault} > $Item->{CountMax} ) {
            $Item->{CountDefault} = $Item->{CountMax};
        }

        # start recursion, if "Sub" is defined.
        if ( $Item->{Sub} && ref $Item->{Sub} eq 'ARRAY' ) {
            $Self->_DefinitionPrepare(
                DefinitionRef => $Item->{Sub},
            );
        }
        else {
            delete $Item->{Sub};
        }
    }

    return 1;
}

1;

=end Internal:

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::ITSMConfigItem::Definition;

use strict;
use warnings;

use Kernel::Language qw(Translatable);

our $ObjectManagerDisabled = 1;

=head1 NAME

Kernel::System::ITSMConfigItem::Definition - sub module of Kernel::System::ITSMConfigItem

=head1 PUBLIC INTERFACE

=head1 DESCRIPTION

All definition functions.

=head2 DefinitionList()

return a config item definition list as array-hash reference

    my $DefinitionListRef = $ConfigItemObject->DefinitionList(
        ClassID => 123,
    );

returns

    my $DefinitionListRef = [
          {
            'Version'      => '1',
            'CreateTime'   => '2012-06-12 14:09:43',
            'DefinitionID' => '1',
            'CreateBy'     => '123',
            'Definition'   => '---
- Key: Vendor
  Name: Vendor
  Searchable: 1
  Input:
  - Type: Text
    Size: 50
    MaxLength: 50

- Key: Description
  Name: Description
  Searchable: 1
  Input:
  - Type: TextArea

- Key: Type
  Name: Type
  Searchable: 1
  Input:
  - Type: TextArea
  - Type: GeneralCatalog\
    Class: ITSM::ConfigItem::Computer::Type
    Translation: 1
# ... etc ...
',
          },
    ];

=cut

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

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

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT id, configitem_definition, version, create_time, create_by '
            . 'FROM configitem_definition WHERE class_id = ? ORDER BY version',
        Bind => [ \$Param{ClassID} ],
    );

    my @DefinitionList;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        my %Definition;
        $Definition{DefinitionID} = $Row[0];
        $Definition{Definition}   = $Row[1] || "--- []";
        $Definition{Version}      = $Row[2];
        $Definition{CreateTime}   = $Row[3];
        $Definition{CreateBy}     = $Row[4];

        # Check if definition code is not a YAML string.
        if ( substr( $Definition{Definition}, 0, 3 ) ne '---' ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'notice',
                Message  => "DefinitionID: $Definition{DefinitionID}"
                    . " ClassID: $Param{ClassID}"
                    . " found in legacy Perl code format, can not continue",
            );

            $Definition{Definition} = "--- []";
        }

        push @DefinitionList, \%Definition;
    }

    return \@DefinitionList;
}

=head2 DefinitionGet()

return a config item definition as hash reference

Return
    $Definition->{DefinitionID}
    $Definition->{ClassID}
    $Definition->{Class}
    $Definition->{Definition}
    $Definition->{DefinitionRef}
    $Definition->{Version}
    $Definition->{CreateTime}
    $Definition->{CreateBy}

    my $DefinitionRef = $ConfigItemObject->DefinitionGet(
        DefinitionID => 123,
    );

    or

    my $DefinitionRef = $ConfigItemObject->DefinitionGet(
        ClassID => 123,
    );

=cut

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

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

    if ( $Param{DefinitionID} ) {

        # check if result is already cached
        return $Self->{Cache}->{DefinitionGet}->{ $Param{DefinitionID} }
            if $Self->{Cache}->{DefinitionGet}->{ $Param{DefinitionID} };

        # ask database
        $Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL => 'SELECT id, class_id, configitem_definition, version, create_time, create_by '
                . 'FROM configitem_definition WHERE id = ?',
            Bind  => [ \$Param{DefinitionID} ],
            Limit => 1,
        );
    }
    else {

        # ask database
        $Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL => 'SELECT id, class_id, configitem_definition, version, create_time, create_by '
                . 'FROM configitem_definition '
                . 'WHERE class_id = ? ORDER BY version DESC',
            Bind  => [ \$Param{ClassID} ],
            Limit => 1,
        );
    }

    # fetch the result
    my %Definition;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {

        $Definition{DefinitionID} = $Row[0];
        $Definition{ClassID}      = $Row[1];
        $Definition{Definition}   = $Row[2] || "--- []";
        $Definition{Version}      = $Row[3];
        $Definition{CreateTime}   = $Row[4];
        $Definition{CreateBy}     = $Row[5];

        # Check if definition code is not a YAML string.
        if ( substr( $Definition{Definition}, 0, 3 ) ne '---' ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'notice',
                Message  => "DefinitionID: $Definition{DefinitionID}"
                    . " ClassID: $Definition{ClassID}"
                    . " found in legacy Perl code format, can not continue",
            );

            $Definition{Definition} = "--- []";
        }

        $Definition{DefinitionRef} = $Kernel::OM->Get('Kernel::System::YAML')->Load(
            Data => $Definition{Definition},
        );
    }

    return {} if !$Definition{DefinitionID};

    # prepare definition
    if ( $Definition{DefinitionRef} && ref $Definition{DefinitionRef} eq 'ARRAY' ) {
        $Self->_DefinitionPrepare(
            DefinitionRef => $Definition{DefinitionRef},
        );
    }
    else {
        $Definition{DefinitionRef} = '';
    }

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # add class
    $Definition{Class} = $ClassList->{ $Definition{ClassID} };

    # cache the result
    $Self->{Cache}->{DefinitionGet}->{ $Definition{DefinitionID} } = \%Definition;

    return \%Definition;
}

=head2 DefinitionAdd()

add a new definition

    my $DefinitionID = $ConfigItemObject->DefinitionAdd(
        ClassID    => 123,
        Definition => 'the definition code',
        UserID     => 1,
    );

=cut

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

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

    # check definition
    my $Check = $Self->DefinitionCheck(
        Definition => $Param{Definition},
    );

    return if !$Check;

    # get last definition
    my $LastDefinition = $Self->DefinitionGet(
        ClassID => $Param{ClassID},
    );

    # stop add, if definition was not changed
    if ( $LastDefinition->{DefinitionID} && $LastDefinition->{Definition} eq $Param{Definition} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Can't add new definition! The definition was not changed.",
        );
        return;
    }

    # set version
    my $Version = 1;
    if ( $LastDefinition->{Version} ) {
        $Version = $LastDefinition->{Version};
        $Version++;
    }

    # insert new definition
    my $Success = $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL => 'INSERT INTO configitem_definition '
            . '(class_id, configitem_definition, version, create_time, create_by) VALUES '
            . '(?, ?, ?, current_timestamp, ?)',
        Bind => [ \$Param{ClassID}, \$Param{Definition}, \$Version, \$Param{UserID} ],
    );

    return if !$Success;

    # get id of new definition
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT id FROM configitem_definition WHERE '
            . 'class_id = ? AND version = ? '
            . 'ORDER BY version DESC',
        Bind  => [ \$Param{ClassID}, \$Version ],
        Limit => 1,
    );

    # fetch the result
    my $DefinitionID;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $DefinitionID = $Row[0];
    }

    # trigger DefinitionCreate event
    $Self->EventHandler(
        Event => 'DefinitionCreate',
        Data  => {
            Comment => $DefinitionID,
        },
        UserID => $Param{UserID},
    );

    return $DefinitionID;
}

=head2 DefinitionCheck()

check the syntax of a new definition

    my $True = $ConfigItemObject->DefinitionCheck(
        Definition      => 'the definition code',
        CheckSubElement => 1,                 # (optional, default 0, to check sub elements recursively)
    );

=cut

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

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

    # if check sub elements is enabled, we must not YAML load it
    # because this has been done in an earlier recursion step already
    my $Definition;
    if ( $Param{CheckSubElement} ) {
        $Definition = $Param{Definition};
    }
    else {
        if ( substr( $Param{Definition}, 0, 3 ) ne '---' ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Definition must be a YAML string",
            );
            return;
        }

        $Definition = $Kernel::OM->Get('Kernel::System::YAML')->Load(
            Data => $Param{Definition},
        );
    }

    # check if definition exists at all
    if ( !$Definition ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Invalid Definition! You have an syntax error in the definition.',
        );
        return;
    }

    # definition must be an array
    if ( ref $Definition ne 'ARRAY' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Invalid Definition! Definition is not an array reference.',
        );
        return;
    }

    # check each definition attribute
    for my $Attribute ( @{$Definition} ) {

        # each definition attribute must be a hash reference with data
        if ( !$Attribute || ref $Attribute ne 'HASH' || !%{$Attribute} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => 'Invalid Definition! At least one definition attribute is not a hash reference.',
            );
            return;
        }

        # check if the key contains no spaces
        if ( $Attribute->{Key} && $Attribute->{Key} =~ m{ \s }xms ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Invalid Definition! Key '$Attribute->{Key}' must not contain whitespace!",
            );
            return;
        }

        # check if the key contains non-ascii characters
        if ( $Attribute->{Key} && $Attribute->{Key} =~ m{ ([^\x{00}-\x{7f}]) }xms ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Invalid Definition! Key '$Attribute->{Key}' must not contain non ASCII characters '$1'!",
            );
            return;
        }

        # recursion check for Sub-Elements
        for my $Key ( sort keys %{$Attribute} ) {

            my $Value = $Attribute->{$Key};

            if ( $Key eq 'Sub' && ref $Value eq 'ARRAY' ) {

                # check the sub array
                my $Check = $Self->DefinitionCheck(
                    Definition      => $Value,
                    CheckSubElement => 1,
                );

                if ( !$Check ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message =>
                            "Invalid Sub-Definition of element with the key '$Attribute->{Key}'.",
                    );
                    return;
                }
            }
        }
    }

    return 1;
}

=head2 DefinitionAttributeInfo()

Return attribute information from the definition for a given attribute path name

    my $AttributeInfo = $ConfigItemObject->DefinitionAttributeInfo(
        AttributePath => 'HardDisk::Capacity',
        Definition    => '
            [
                # ...

                {
                    Key => 'HardDisk',
                    Name => 'Hard Disk',
                    Input => {
                        Type => 'Text',
                        Size => 50,
                        MaxLength => 100,
                    },
                    CountMax => 10,
                    Sub => [
                        {
                            Key => 'Capacity',
                            Name => 'Capacity',
                            Input => {
                                Type => 'Text',
                                Size => 20,
                                MaxLength => 10,
                            },
                        },
                    ],
                },

                # ...
            ];
        ',
    );

    Returns:

        my $AttributeInfo =  {
            CountDefault => 1,
            CountMax     => 1,
            Input => {
                Size      => 20,
                Type      => 'Text',
                MaxLength => 10
            },
            Name     => 'Capacity',
            CountMin => 1,
            Key      => 'Capacity'
        };

=cut

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

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

    my $Subtree = $Param{Definition};
    my $Info;

    PART:
    for my $Part ( split /::/, $Param{AttributePath} ) {
        my ($Found) = grep { $_->{Key} eq $Part } @{$Subtree};

        last PART if !$Found;

        $Subtree = $Found->{Sub};
        $Info    = $Found;
    }

    return $Info;
}

=begin Internal:

=head2 _DefinitionPrepare()

Prepare the syntax of a new definition

    my $True = $ConfigItemObject->_DefinitionPrepare(
        DefinitionRef => $ArrayRef,
    );

=cut

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

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

    for my $Item ( @{ $Param{DefinitionRef} } ) {

        # set CountMin
        if ( !defined $Item->{CountMin} ) {
            $Item->{CountMin} = 1;
        }

        # set CountMax
        $Item->{CountMax} ||= 1;

        # set CountMin
        if ( $Item->{CountMin} > $Item->{CountMax} ) {
            $Item->{CountMin} = $Item->{CountMax};
        }

        # set CountDefault
        if ( !defined $Item->{CountDefault} ) {
            $Item->{CountDefault} = 1;
        }
        if ( $Item->{CountDefault} < $Item->{CountMin} ) {
            $Item->{CountDefault} = $Item->{CountMin};
        }
        if ( $Item->{CountDefault} > $Item->{CountMax} ) {
            $Item->{CountDefault} = $Item->{CountMax};
        }

        # start recursion, if "Sub" is defined.
        if ( $Item->{Sub} && ref $Item->{Sub} eq 'ARRAY' ) {
            $Self->_DefinitionPrepare(
                DefinitionRef => $Item->{Sub},
            );
        }
        else {
            delete $Item->{Sub};
        }
    }

    return 1;
}

1;

=end Internal:

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::ITSMConfigItem::Definition;

use strict;
use warnings;

use Kernel::Language qw(Translatable);

our $ObjectManagerDisabled = 1;

=head1 NAME

Kernel::System::ITSMConfigItem::Definition - sub module of Kernel::System::ITSMConfigItem

=head1 PUBLIC INTERFACE

=head1 DESCRIPTION

All definition functions.

=head2 DefinitionList()

return a config item definition list as array-hash reference

    my $DefinitionListRef = $ConfigItemObject->DefinitionList(
        ClassID => 123,
    );

returns

    my $DefinitionListRef = [
          {
            'Version'      => '1',
            'CreateTime'   => '2012-06-12 14:09:43',
            'DefinitionID' => '1',
            'CreateBy'     => '123',
            'Definition'   => '---
- Key: Vendor
  Name: Vendor
  Searchable: 1
  Input:
  - Type: Text
    Size: 50
    MaxLength: 50

- Key: Description
  Name: Description
  Searchable: 1
  Input:
  - Type: TextArea

- Key: Type
  Name: Type
  Searchable: 1
  Input:
  - Type: TextArea
  - Type: GeneralCatalog\
    Class: ITSM::ConfigItem::Computer::Type
    Translation: 1
# ... etc ...
',
          },
    ];

=cut

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

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

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT id, configitem_definition, version, create_time, create_by '
            . 'FROM configitem_definition WHERE class_id = ? ORDER BY version',
        Bind => [ \$Param{ClassID} ],
    );

    my @DefinitionList;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        my %Definition;
        $Definition{DefinitionID} = $Row[0];
        $Definition{Definition}   = $Row[1] || "--- []";
        $Definition{Version}      = $Row[2];
        $Definition{CreateTime}   = $Row[3];
        $Definition{CreateBy}     = $Row[4];

        # Check if definition code is not a YAML string.
        if ( substr( $Definition{Definition}, 0, 3 ) ne '---' ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'notice',
                Message  => "DefinitionID: $Definition{DefinitionID}"
                    . " ClassID: $Param{ClassID}"
                    . " found in legacy Perl code format, can not continue",
            );

            $Definition{Definition} = "--- []";
        }

        push @DefinitionList, \%Definition;
    }

    return \@DefinitionList;
}

=head2 DefinitionGet()

return a config item definition as hash reference

Return
    $Definition->{DefinitionID}
    $Definition->{ClassID}
    $Definition->{Class}
    $Definition->{Definition}
    $Definition->{DefinitionRef}
    $Definition->{Version}
    $Definition->{CreateTime}
    $Definition->{CreateBy}

    my $DefinitionRef = $ConfigItemObject->DefinitionGet(
        DefinitionID => 123,
    );

    or

    my $DefinitionRef = $ConfigItemObject->DefinitionGet(
        ClassID => 123,
    );

=cut

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

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

    if ( $Param{DefinitionID} ) {

        # check if result is already cached
        return $Self->{Cache}->{DefinitionGet}->{ $Param{DefinitionID} }
            if $Self->{Cache}->{DefinitionGet}->{ $Param{DefinitionID} };

        # ask database
        $Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL => 'SELECT id, class_id, configitem_definition, version, create_time, create_by '
                . 'FROM configitem_definition WHERE id = ?',
            Bind  => [ \$Param{DefinitionID} ],
            Limit => 1,
        );
    }
    else {

        # ask database
        $Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL => 'SELECT id, class_id, configitem_definition, version, create_time, create_by '
                . 'FROM configitem_definition '
                . 'WHERE class_id = ? ORDER BY version DESC',
            Bind  => [ \$Param{ClassID} ],
            Limit => 1,
        );
    }

    # fetch the result
    my %Definition;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {

        $Definition{DefinitionID} = $Row[0];
        $Definition{ClassID}      = $Row[1];
        $Definition{Definition}   = $Row[2] || "--- []";
        $Definition{Version}      = $Row[3];
        $Definition{CreateTime}   = $Row[4];
        $Definition{CreateBy}     = $Row[5];

        # Check if definition code is not a YAML string.
        if ( substr( $Definition{Definition}, 0, 3 ) ne '---' ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'notice',
                Message  => "DefinitionID: $Definition{DefinitionID}"
                    . " ClassID: $Definition{ClassID}"
                    . " found in legacy Perl code format, can not continue",
            );

            $Definition{Definition} = "--- []";
        }

        $Definition{DefinitionRef} = $Kernel::OM->Get('Kernel::System::YAML')->Load(
            Data => $Definition{Definition},
        );
    }

    return {} if !$Definition{DefinitionID};

    # prepare definition
    if ( $Definition{DefinitionRef} && ref $Definition{DefinitionRef} eq 'ARRAY' ) {
        $Self->_DefinitionPrepare(
            DefinitionRef => $Definition{DefinitionRef},
        );
    }
    else {
        $Definition{DefinitionRef} = '';
    }

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # add class
    $Definition{Class} = $ClassList->{ $Definition{ClassID} };

    # cache the result
    $Self->{Cache}->{DefinitionGet}->{ $Definition{DefinitionID} } = \%Definition;

    return \%Definition;
}

=head2 DefinitionAdd()

add a new definition

    my $DefinitionID = $ConfigItemObject->DefinitionAdd(
        ClassID    => 123,
        Definition => 'the definition code',
        UserID     => 1,
    );

=cut

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

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

    # check definition
    my $Check = $Self->DefinitionCheck(
        Definition => $Param{Definition},
    );

    return if !$Check;

    # get last definition
    my $LastDefinition = $Self->DefinitionGet(
        ClassID => $Param{ClassID},
    );

    # stop add, if definition was not changed
    if ( $LastDefinition->{DefinitionID} && $LastDefinition->{Definition} eq $Param{Definition} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Can't add new definition! The definition was not changed.",
        );
        return;
    }

    # set version
    my $Version = 1;
    if ( $LastDefinition->{Version} ) {
        $Version = $LastDefinition->{Version};
        $Version++;
    }

    # insert new definition
    my $Success = $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL => 'INSERT INTO configitem_definition '
            . '(class_id, configitem_definition, version, create_time, create_by) VALUES '
            . '(?, ?, ?, current_timestamp, ?)',
        Bind => [ \$Param{ClassID}, \$Param{Definition}, \$Version, \$Param{UserID} ],
    );

    return if !$Success;

    # get id of new definition
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT id FROM configitem_definition WHERE '
            . 'class_id = ? AND version = ? '
            . 'ORDER BY version DESC',
        Bind  => [ \$Param{ClassID}, \$Version ],
        Limit => 1,
    );

    # fetch the result
    my $DefinitionID;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $DefinitionID = $Row[0];
    }

    # trigger DefinitionCreate event
    $Self->EventHandler(
        Event => 'DefinitionCreate',
        Data  => {
            Comment => $DefinitionID,
        },
        UserID => $Param{UserID},
    );

    return $DefinitionID;
}

=head2 DefinitionCheck()

check the syntax of a new definition

    my $True = $ConfigItemObject->DefinitionCheck(
        Definition      => 'the definition code',
        CheckSubElement => 1,                 # (optional, default 0, to check sub elements recursively)
    );

=cut

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

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

    # if check sub elements is enabled, we must not YAML load it
    # because this has been done in an earlier recursion step already
    my $Definition;
    if ( $Param{CheckSubElement} ) {
        $Definition = $Param{Definition};
    }
    else {
        if ( substr( $Param{Definition}, 0, 3 ) ne '---' ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Definition must be a YAML string",
            );
            return;
        }

        $Definition = $Kernel::OM->Get('Kernel::System::YAML')->Load(
            Data => $Param{Definition},
        );
    }

    # check if definition exists at all
    if ( !$Definition ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Invalid Definition! You have an syntax error in the definition.',
        );
        return;
    }

    # definition must be an array
    if ( ref $Definition ne 'ARRAY' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Invalid Definition! Definition is not an array reference.',
        );
        return;
    }

    # check each definition attribute
    for my $Attribute ( @{$Definition} ) {

        # each definition attribute must be a hash reference with data
        if ( !$Attribute || ref $Attribute ne 'HASH' || !%{$Attribute} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => 'Invalid Definition! At least one definition attribute is not a hash reference.',
            );
            return;
        }

        # check if the key contains no spaces
        if ( $Attribute->{Key} && $Attribute->{Key} =~ m{ \s }xms ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Invalid Definition! Key '$Attribute->{Key}' must not contain whitespace!",
            );
            return;
        }

        # check if the key contains non-ascii characters
        if ( $Attribute->{Key} && $Attribute->{Key} =~ m{ ([^\x{00}-\x{7f}]) }xms ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Invalid Definition! Key '$Attribute->{Key}' must not contain non ASCII characters '$1'!",
            );
            return;
        }

        # recursion check for Sub-Elements
        for my $Key ( sort keys %{$Attribute} ) {

            my $Value = $Attribute->{$Key};

            if ( $Key eq 'Sub' && ref $Value eq 'ARRAY' ) {

                # check the sub array
                my $Check = $Self->DefinitionCheck(
                    Definition      => $Value,
                    CheckSubElement => 1,
                );

                if ( !$Check ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message =>
                            "Invalid Sub-Definition of element with the key '$Attribute->{Key}'.",
                    );
                    return;
                }
            }
        }
    }

    return 1;
}

=head2 DefinitionAttributeInfo()

Return attribute information from the definition for a given attribute path name

    my $AttributeInfo = $ConfigItemObject->DefinitionAttributeInfo(
        AttributePath => 'HardDisk::Capacity',
        Definition    => '
            [
                # ...

                {
                    Key => 'HardDisk',
                    Name => 'Hard Disk',
                    Input => {
                        Type => 'Text',
                        Size => 50,
                        MaxLength => 100,
                    },
                    CountMax => 10,
                    Sub => [
                        {
                            Key => 'Capacity',
                            Name => 'Capacity',
                            Input => {
                                Type => 'Text',
                                Size => 20,
                                MaxLength => 10,
                            },
                        },
                    ],
                },

                # ...
            ];
        ',
    );

    Returns:

        my $AttributeInfo =  {
            CountDefault => 1,
            CountMax     => 1,
            Input => {
                Size      => 20,
                Type      => 'Text',
                MaxLength => 10
            },
            Name     => 'Capacity',
            CountMin => 1,
            Key      => 'Capacity'
        };

=cut

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

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

    my $Subtree = $Param{Definition};
    my $Info;

    PART:
    for my $Part ( split /::/, $Param{AttributePath} ) {
        my ($Found) = grep { $_->{Key} eq $Part } @{$Subtree};

        last PART if !$Found;

        $Subtree = $Found->{Sub};
        $Info    = $Found;
    }

    return $Info;
}

=begin Internal:

=head2 _DefinitionPrepare()

Prepare the syntax of a new definition

    my $True = $ConfigItemObject->_DefinitionPrepare(
        DefinitionRef => $ArrayRef,
    );

=cut

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

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

    for my $Item ( @{ $Param{DefinitionRef} } ) {

        # set CountMin
        if ( !defined $Item->{CountMin} ) {
            $Item->{CountMin} = 1;
        }

        # set CountMax
        $Item->{CountMax} ||= 1;

        # set CountMin
        if ( $Item->{CountMin} > $Item->{CountMax} ) {
            $Item->{CountMin} = $Item->{CountMax};
        }

        # set CountDefault
        if ( !defined $Item->{CountDefault} ) {
            $Item->{CountDefault} = 1;
        }
        if ( $Item->{CountDefault} < $Item->{CountMin} ) {
            $Item->{CountDefault} = $Item->{CountMin};
        }
        if ( $Item->{CountDefault} > $Item->{CountMax} ) {
            $Item->{CountDefault} = $Item->{CountMax};
        }

        # start recursion, if "Sub" is defined.
        if ( $Item->{Sub} && ref $Item->{Sub} eq 'ARRAY' ) {
            $Self->_DefinitionPrepare(
                DefinitionRef => $Item->{Sub},
            );
        }
        else {
            delete $Item->{Sub};
        }
    }

    return 1;
}

1;

=end Internal:

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KIyMgbm9maWx0ZXIoVGlkeUFsbDo6UGx1Z2luOjpabnVueTo6UGVybDo6UGFyYW1PYmplY3QpCgpwYWNrYWdlIEtlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6RXZlbnQ6OkNJQXR0YWNobWVudDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6VmFyaWFibGVDaGVjayBxdyg6YWxsKTsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAogICAgJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpJVFNNQ29uZmlnSXRlbTo6TGF5b3V0Q0lBdHRhY2htZW50JywKICAgICdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OkNJQXR0YWNobWVudCcsCiAgICAnS2VybmVsOjpTeXN0ZW06OkxvZycsCiAgICAnS2VybmVsOjpTeXN0ZW06OldlYjo6UmVxdWVzdCcsCik7CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHt9OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgIHJldHVybiAkU2VsZjsKfQoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRMb2dPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyk7CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIE5FRURFRDoKICAgIGZvciBteSAkTmVlZGVkIChxdyggRGF0YSBFdmVudCBDb25maWcgVXNlcklEICkpIHsKCiAgICAgICAgbmV4dCBORUVERUQgaWYgJFBhcmFteyROZWVkZWR9OwoKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJOZWVkICROZWVkZWQhIgogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQogICAgTkVFREVEOgogICAgZm9yIG15ICROZWVkZWQgKHF3KCBDb25maWdJdGVtSUQgKSkgewoKICAgICAgICBuZXh0IE5FRURFRCBpZiAkUGFyYW17RGF0YX0tPnskTmVlZGVkfTsKCiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkTmVlZGVkIGluIERhdGEhIgogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICRTZWxmLT5WZXJzaW9uQ3JlYXRlKCVQYXJhbSk7CiAgICAkU2VsZi0+VmVyc2lvbkRlbGV0ZSglUGFyYW0pOwoKICAgIHJldHVybiAxOwp9Cgo9aGVhZDIgVmVyc2lvbkNyZWF0ZSgpCgpUaGlzIGZ1bmN0aW9uIHdpbGwgc2F2ZSB0aGUgYXR0YWNobWVudHMgdG8gdGhlIGZpbGUgc3lzdGVtIGlmIHRoZSBjb25maWcgaXRlbSBpcyBjcmVhdGVkLgoKICAgIG15ICRTdWNjZXNzID0gJEV2ZW50T2JqZWN0LT5WZXJzaW9uQ3JlYXRlKCVQYXJhbSk7CgpSZXR1cm5zOgoKICAgIG15ICRTdWNjZXNzID0gMTsKCj1jdXQKCnN1YiBWZXJzaW9uQ3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJENJQXR0YWNobWVudE9iamVjdCAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6Q0lBdHRhY2htZW50Jyk7CiAgICBteSAkTGF5b3V0Q0lBdHRhY2htZW50T2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpPdXRwdXQ6OkhUTUw6OklUU01Db25maWdJdGVtOjpMYXlvdXRDSUF0dGFjaG1lbnQnKTsKICAgIG15ICRMb2dPYmplY3QgICAgICAgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyk7CiAgICBteSAkUGFyYW1PYmplY3QgICAgICAgICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OldlYjo6UmVxdWVzdCcpOwoKICAgIHJldHVybiBpZiAkUGFyYW17RXZlbnR9IG5lICdWZXJzaW9uQ3JlYXRlJzsKCiAgICBteSBAUGFyYW1OYW1lcyA9ICRQYXJhbU9iamVjdC0+R2V0UGFyYW1OYW1lcygpOwoKICAgIG15ICRDb25maWdJdGVtSUQgPSAkUGFyYW17RGF0YX0tPntDb25maWdJdGVtSUR9OwogICAgbXkgJFZlcnNpb25JRCAgICA9ICRQYXJhbXtEYXRhfS0+e0NvbW1lbnR9OwoKICAgIFBBUkFNOgogICAgZm9yIG15ICRQYXJhbU5hbWUgKEBQYXJhbU5hbWVzKSB7CgogICAgICAgICMgb25seSBjaGVjayBwYXJhbXMgb2YgdGhlIENJQXR0YWNobWVudCBiYWNrZW5kCiAgICAgICAgbmV4dCBQQVJBTSBpZiAkUGFyYW1OYW1lICF+IG17OjpDSUF0dGFjaG1lbnRcen14bXNpOwoKICAgICAgICAjIHJlbW92ZSBzdWZmaXgKICAgICAgICAkUGFyYW1OYW1lID1+IHN7OjpDSUF0dGFjaG1lbnRcen17fXhtc2k7CgogICAgICAgICMgZ2V0IGZpZWxkIGJhc2Uga2V5CiAgICAgICAgbXkgJEF0dGFjaG1lbnRLZXkgPSAkTGF5b3V0Q0lBdHRhY2htZW50T2JqZWN0LT5BdHRhY2htZW50S2V5R2V0KAogICAgICAgICAgICBLZXkgPT4gJFBhcmFtTmFtZSwKICAgICAgICApOwogICAgICAgIG5leHQgUEFSQU0gaWYgIWRlZmluZWQgJEF0dGFjaG1lbnRLZXk7CgogICAgICAgICMgZ2V0IGF0dGFjaG1lbnQgaW5kZXggb2YgdGhlIHRlbXAgY2FjaGUKICAgICAgICBteSAkQXR0YWNobWVudEluZGV4ID0gJExheW91dENJQXR0YWNobWVudE9iamVjdC0+QXR0YWNobWVudEluZGV4R2V0KAogICAgICAgICAgICBLZXkgPT4gJFBhcmFtTmFtZSwKICAgICAgICApOwogICAgICAgIG5leHQgUEFSQU0gaWYgIWRlZmluZWQgJEF0dGFjaG1lbnRJbmRleDsKCiAgICAgICAgIyBnZXQgZm9ybSBpZAogICAgICAgIG15ICRGb3JtSUQgPSAkTGF5b3V0Q0lBdHRhY2htZW50T2JqZWN0LT5Gb3JtSURHZXQoCiAgICAgICAgICAgIEF0dGFjaG1lbnRLZXkgPT4gJEF0dGFjaG1lbnRLZXksCiAgICAgICAgKTsKICAgICAgICBuZXh0IFBBUkFNIGlmICFkZWZpbmVkICRGb3JtSUQ7CgogICAgICAgICMgZ2V0IGN1cnJlbnQgdmFsdWUgb2YgdGhlIGF0dGFjaG1lbnQKICAgICAgICBteSAkQXR0YWNobWVudElEID0gJFBhcmFtT2JqZWN0LT5HZXRQYXJhbSggUGFyYW0gPT4gJFBhcmFtTmFtZSApOwogICAgICAgIG5leHQgUEFSQU0gaWYgIWRlZmluZWQgJEF0dGFjaG1lbnRJRDsKCiAgICAgICAgIyBnZXQgYXR0YWNobWVudCBiYXNlZCBvbiB0aGUgY2FjaGVkIGRhdGEKICAgICAgICBteSAlQXR0YWNobWVudCA9ICRMYXlvdXRDSUF0dGFjaG1lbnRPYmplY3QtPkF0dGFjaG1lbnRDYWNoZUdldCgKICAgICAgICAgICAgSW5kZXggID0+ICRBdHRhY2htZW50SW5kZXgsCiAgICAgICAgICAgIEZvcm1JRCA9PiAkRm9ybUlELAogICAgICAgICk7CiAgICAgICAgbmV4dCBQQVJBTSBpZiAhJUF0dGFjaG1lbnQ7CgogICAgICAgICMgc2F2ZSBkYXRhIHRvIGZpbGUgc3lzdGVtCiAgICAgICAgbXkgJENvbnRlbnRJRCA9ICRDSUF0dGFjaG1lbnRPYmplY3QtPkNJQXR0YWNobWVudEFkZCgKICAgICAgICAgICAgQXR0YWNobWVudElEICAgID0+ICRBdHRhY2htZW50SUQsCiAgICAgICAgICAgIENvbmZpZ0l0ZW1GaWVsZCA9PiAkQXR0YWNobWVudEtleSwKICAgICAgICAgICAgQ29uZmlnSXRlbUlEICAgID0+ICRDb25maWdJdGVtSUQsCiAgICAgICAgICAgIFZlcnNpb25JRCAgICAgICA9PiAkVmVyc2lvbklELAogICAgICAgICAgICBGaWxlbmFtZSAgICAgICAgPT4gJEF0dGFjaG1lbnR7RmlsZW5hbWV9LAogICAgICAgICAgICBDb250ZW50ICAgICAgICAgPT4gJEF0dGFjaG1lbnR7Q29udGVudH0sCiAgICAgICAgICAgIENvbnRlbnRUeXBlICAgICA9PiAkQXR0YWNobWVudHtDb250ZW50VHlwZX0sCiAgICAgICAgKTsKCiAgICAgICAgaWYgKCAhJENvbnRlbnRJRCApIHsKICAgICAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJGYWlsZWQgdG8gd3JpdGUgYXR0YWNobWVudCBmb3IgJyRQYXJhbU5hbWUnIiwKICAgICAgICAgICAgKTsKICAgICAgICAgICAgcmV0dXJuOwogICAgICAgIH0KICAgIH0KCiAgICByZXR1cm4gMTsKfQoKPWhlYWQyIFZlcnNpb25EZWxldGUoKQoKVGhpcyBmdW5jdGlvbiB3aWxsIHNhdmUgdGhlIGF0dGFjaG1lbnRzIHRvIHRoZSBmaWxlIHN5c3RlbSBpZiB0aGUgY29uZmlnIGl0ZW0gaXMgY3JlYXRlZC4KCiAgICBteSAkU3VjY2VzcyA9ICRFdmVudE9iamVjdC0+VmVyc2lvbkRlbGV0ZSglUGFyYW0pOwoKUmV0dXJuczoKCiAgICBteSAkU3VjY2VzcyA9IDE7Cgo9Y3V0CgpzdWIgVmVyc2lvbkRlbGV0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRDSUF0dGFjaG1lbnRPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OkNJQXR0YWNobWVudCcpOwoKICAgIHJldHVybiBpZiAkUGFyYW17RXZlbnR9IG5lICdWZXJzaW9uRGVsZXRlJzsKCiAgICBteSAkQ29uZmlnSXRlbUlEID0gJFBhcmFte0RhdGF9LT57Q29uZmlnSXRlbUlEfTsKICAgIG15ICRWZXJzaW9uSUQgICAgPSAkUGFyYW17RGF0YX0tPntDb21tZW50fTsKCiAgICBteSBAQXR0YWNobWVudHMgPSAkQ0lBdHRhY2htZW50T2JqZWN0LT5DSUF0dGFjaG1lbnRMaXN0KAogICAgICAgIENvbmZpZ0l0ZW1JRCA9PiAkQ29uZmlnSXRlbUlELAogICAgICAgIFZlcnNpb25JRCAgICA9PiAkVmVyc2lvbklELAogICAgKTsKCiAgICBmb3IgbXkgJEF0dGFjaG1lbnRJRCAoQEF0dGFjaG1lbnRzKSB7CiAgICAgICAgJENJQXR0YWNobWVudE9iamVjdC0+Q0lBdHRhY2htZW50RGVsZXRlKAogICAgICAgICAgICBBdHRhY2htZW50SUQgPT4gJEF0dGFjaG1lbnRJRCwKICAgICAgICApOwogICAgfQoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OkV2ZW50OjpEb0hpc3Rvcnk7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpMb2cnLAopOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6RXZlbnQ6OkRvSGlzdG9yeSAtIEV2ZW50IGhhbmRsZXIgdGhhdCBkb2VzIHRoZSBoaXN0b3J5Cgo9aGVhZDEgUFVCTElDIElOVEVSRkFDRQoKPWhlYWQyIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgdXNlIEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyOwogICAgbG9jYWwgJEtlcm5lbDo6T00gPSBLZXJuZWw6OlN5c3RlbTo6T2JqZWN0TWFuYWdlci0+bmV3KCk7CiAgICBteSAkRG9IaXN0b3J5T2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpFdmVudDo6RG9IaXN0b3J5Jyk7Cgo9Y3V0CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHt9OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgIHJldHVybiAkU2VsZjsKfQoKPWhlYWQyIFJ1bigpCgpUaGlzIG1ldGhvZCBoYW5kbGVzIHRoZSBldmVudC4KCiAgICAkRG9IaXN0b3J5T2JqZWN0LT5SdW4oCiAgICAgICAgRXZlbnQgPT4gJ0NvbmZpZ0l0ZW1DcmVhdGUnLAogICAgICAgIERhdGEgID0+IHsKICAgICAgICAgICAgQ29tbWVudCAgICAgID0+ICduZXcgdmFsdWU6IDEnLAogICAgICAgICAgICBDb25maWdJdGVtSUQgPT4gMTIzLAogICAgICAgIH0sCiAgICAgICAgVXNlcklEID0+IDEsCiAgICApOwoKPWN1dAoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYXMgRGVmaW5pdGlvbkNyZWF0ZSBkb2VzIG5vdCBiZWxvbmcgdG8gYW4gaXRlbSwgd2UgZG9uJ3QgY3JlYXRlCiAgICAjIGEgaGlzdG9yeSBlbnRyeQogICAgaWYgKCAkUGFyYW17RXZlbnR9ICYmICRQYXJhbXtFdmVudH0gZXEgJ0RlZmluaXRpb25DcmVhdGUnICkgewogICAgICAgIHJldHVybjsKICAgIH0KCiAgICBORUVERUQ6CiAgICBmb3IgbXkgJE5lZWRlZCAocXcoRGF0YSBFdmVudCBVc2VySUQpKSB7CgogICAgICAgIG5leHQgTkVFREVEIGlmIGRlZmluZWQgJFBhcmFteyROZWVkZWR9OwoKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gIk5lZWQgJE5lZWRlZCEiLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICMgZHVlIHRvIGNvbnNpc3RlbmN5IHdpdGggdGlja2V0IGhpc3RvcnksIHdlIG5lZWQgSGlzdG9yeVR5cGUKICAgICRQYXJhbXtIaXN0b3J5VHlwZX0gPSAkUGFyYW17RXZlbnR9OwoKICAgICMgZGlzcGF0Y2ggdGFibGUgZm9yIGFsbCBldmVudHMKICAgIG15ICVEaXNwYXRjaGVyID0gKAogICAgICAgIENvbmZpZ0l0ZW1DcmVhdGUgICAgICA9PiBcJl9IaXN0b3J5QWRkLAogICAgICAgIENvbmZpZ0l0ZW1EZWxldGUgICAgICA9PiBcJl9Db25maWdJdGVtRGVsZXRlLAogICAgICAgIExpbmtBZGQgICAgICAgICAgICAgICA9PiBcJl9IaXN0b3J5QWRkLAogICAgICAgIExpbmtEZWxldGUgICAgICAgICAgICA9PiBcJl9IaXN0b3J5QWRkLAogICAgICAgIE5hbWVVcGRhdGUgICAgICAgICAgICA9PiBcJl9IaXN0b3J5QWRkLAogICAgICAgIEluY2lkZW50U3RhdGVVcGRhdGUgICA9PiBcJl9IaXN0b3J5QWRkLAogICAgICAgIERlcGxveW1lbnRTdGF0ZVVwZGF0ZSA9PiBcJl9IaXN0b3J5QWRkLAogICAgICAgIERlZmluaXRpb25VcGRhdGUgICAgICA9PiBcJl9IaXN0b3J5QWRkLAogICAgICAgIFZlcnNpb25DcmVhdGUgICAgICAgICA9PiBcJl9IaXN0b3J5QWRkLAogICAgICAgIFZhbHVlVXBkYXRlICAgICAgICAgICA9PiBcJl9IaXN0b3J5QWRkLAogICAgICAgIERlZmluaXRpb25DcmVhdGUgICAgICA9PiBcJl9IaXN0b3J5QWRkLAogICAgICAgIFZlcnNpb25EZWxldGUgICAgICAgICA9PiBcJl9IaXN0b3J5QWRkLAogICAgICAgIEF0dGFjaG1lbnRBZGRQb3N0ICAgICA9PiBcJl9IaXN0b3J5QWRkLAogICAgICAgIEF0dGFjaG1lbnREZWxldGVQb3N0ICA9PiBcJl9IaXN0b3J5QWRkLAogICAgKTsKCiAgICAjIGVycm9yIGhhbmRsaW5nCiAgICBpZiAoICFleGlzdHMgJERpc3BhdGNoZXJ7ICRQYXJhbXtFdmVudH0gfSApIHsKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ25vbiBleGlzdGFudCBoaXN0b3J5IHR5cGU6ICcgLiAkUGFyYW17RXZlbnR9LAogICAgICAgICk7CgogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAjIGNhbGwgY2FsbGJhY2sKICAgIG15ICRTdWIgPSAkRGlzcGF0Y2hlcnsgJFBhcmFte0V2ZW50fSB9OwogICAgJFNlbGYtPiRTdWIoCiAgICAgICAgJVBhcmFtLAogICAgICAgICV7ICRQYXJhbXtEYXRhfSB9LAogICAgKTsKCiAgICByZXR1cm4gMTsKfQoKPWhlYWQxIElOVEVSTkFMIElOVEVSRkFDRQoKPWhlYWQyIF9Db25maWdJdGVtRGVsZXRlKCkKCmhpc3RvcnkncyBldmVudCBoYW5kbGVyIGZvciBDb25maWdJdGVtRGVsZXRlCgo9Y3V0CgpzdWIgX0NvbmZpZ0l0ZW1EZWxldGUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGRlbGV0ZSBoaXN0b3J5CiAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nKS0+SGlzdG9yeURlbGV0ZSgKICAgICAgICBDb25maWdJdGVtSUQgPT4gJFBhcmFte0NvbmZpZ0l0ZW1JRH0sCiAgICApOwoKICAgIHJldHVybiAxOwp9Cgo9aGVhZDIgX0hpc3RvcnlBZGQoKQoKaGlzdG9yeSdzIGRlZmF1bHQgZXZlbnQgaGFuZGxlci4KCj1jdXQKCnN1YiBfSGlzdG9yeUFkZCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWRkIGhpc3RvcnkgZW50cnkKICAgICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbScpLT5IaXN0b3J5QWRkKAogICAgICAgICVQYXJhbSwKICAgICk7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKClRoaXMgc29mdHdhcmUgaXMgcGFydCBvZiB0aGUgT1RSUyBwcm9qZWN0IChMPGh0dHBzOi8vb3Rycy5vcmcvPikuCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIEw8aHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dD4uCgo9Y3V0Cg==
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::ITSMConfigItem::History;

use strict;
use warnings;

our $ObjectManagerDisabled = 1;

=head1 NAME

Kernel::System::ITSMConfigItem::History - module for ITSMConfigItem.pm with history functions

=head1 PUBLIC INTERFACE

=head1 DESCRIPTION

All history functions.

=head2 HistoryGet()

Returns an array reference with all history entries for the given config item.
Each array element is a hash reference representing one history entry.

These hash references contain information about:

    $Info{HistoryEntryID}
    $Info{ConfigItemID}
    $Info{HistoryType}
    $Info{HistoryTypeID}
    $Info{Comment}
    $Info{CreatedBy}
    $Info{CreateTime}
    $Info{UserID}
    $Info{UserLogin}
    $Info{UserLastname}
    $Info{UserFirstname}
    $Info{UserFullname}

    my $Info = $ConfigItemObject->HistoryGet(
        ConfigItemID => 1234,
    );

=cut

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

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

    # if cached result exists, return that result
    return $Self->{Cache}->{CIVersions}->{ $Param{ConfigItemID} }
        if $Self->{Cache}->{CIVersions}->{ $Param{ConfigItemID} };

    # fetch some data from history for given config item
    return if !$Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT ch.id, ch.configitem_id, ch.content, ch.type_id, '
            . 'ch.create_by, ch.create_time, cht.name '
            . 'FROM configitem_history ch, configitem_history_type cht '
            . 'WHERE ch.type_id = cht.id AND ch.configitem_id = ? '
            . 'ORDER BY ch.id',
        Bind => [ \$Param{ConfigItemID} ],
    );

    # save data from history in array
    my @Entries;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        my %Tmp = (
            HistoryEntryID => $Row[0],
            ConfigItemID   => $Row[1],
            Comment        => $Row[2],
            HistoryTypeID  => $Row[3],
            CreateBy       => $Row[4],
            CreateTime     => $Row[5],
            HistoryType    => $Row[6],
        );

        push @Entries, \%Tmp;
    }

    # get more information about user who created history entries
    for my $Entry (@Entries) {

        # get user information
        my %UserInfo = $Kernel::OM->Get('Kernel::System::User')->GetUserData(
            UserID => $Entry->{CreateBy},
            Cached => 1,
        );

        # save additional information for history entry
        $Entry->{UserID}        = $UserInfo{UserID};
        $Entry->{UserLogin}     = $UserInfo{UserLogin};
        $Entry->{UserFirstname} = $UserInfo{UserFirstname};
        $Entry->{UserLastname}  = $UserInfo{UserLastname};
        $Entry->{UserFullname}  = $UserInfo{UserFullname};
    }

    # save result in cache
    $Self->{Cache}->{CIVersions}->{ $Param{ConfigItemID} } = \@Entries;

    return \@Entries;
}

=head2 HistoryEntryGet()

Returns a hash reference with information about a single history entry.
The hash reference contain information about:

    $Info{HistoryEntryID}
    $Info{ConfigItemID}
    $Info{HistoryType}
    $Info{HistoryTypeID}
    $Info{Comment}
    $Info{CreateBy}
    $Info{CreateTime}
    $Info{UserID}
    $Info{UserLogin}
    $Info{UserLastname}
    $Info{UserFirstname}
    $Info{UserFullname}

    my $Info = $ConfigItemObject->HistoryEntryGet(
        HistoryEntryID => 1234,
    );

=cut

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

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

    # if cached result exists, return that result
    if ( $Self->{Cache}->{Versions}->{ $Param{HistoryEntryID} } ) {
        my ($ConfigItemID) = keys %{ $Self->{Cache}->{Versions}->{ $Param{HistoryEntryID} } };
        return $Self->{Cache}->{Versions}->{ $Param{HistoryEntryID} }->{$ConfigItemID};
    }

    # fetch a single entry from history
    return if !$Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT ch.id, ch.configitem_id, ch.content, ch.type_id, '
            . 'ch.create_by, ch.create_time, cht.name '
            . 'FROM configitem_history ch, configitem_history_type cht '
            . 'WHERE ch.type_id = cht.id AND ch.id = ?',
        Bind  => [ \$Param{HistoryEntryID} ],
        Limit => 1,
    );

    my %Entry;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {

        %Entry = (
            HistoryEntryID => $Row[0],
            ConfigItemID   => $Row[1],
            Comment        => $Row[2],
            HistoryTypeID  => $Row[3],
            CreateBy       => $Row[4],
            CreateTime     => $Row[5],
            HistoryType    => $Row[6],
        );
    }

    # get user data for this entry
    my %UserInfo = $Kernel::OM->Get('Kernel::System::User')->GetUserData(
        UserID => $Entry{CreateBy},
        Cached => 1,
    );

    $Entry{UserID}        = $UserInfo{UserID};
    $Entry{UserLogin}     = $UserInfo{UserLogin};
    $Entry{UserFirstname} = $UserInfo{UserFirstname};
    $Entry{UserLastname}  = $UserInfo{UserLastname};
    $Entry{UserFullname}  = $UserInfo{UserFullname};

    $Self->{Cache}->{Versions}->{ $Param{HistoryEntryID} }->{ $Entry{ConfigItemID} } = \%Entry;

    return \%Entry;
}

=head2 HistoryAdd()

Adds a single history entry to the history.

    $ConfigItemObject->HistoryAdd(
        ConfigItemID  => 1234,
        HistoryType   => 'NewConfigItem', # either HistoryType or HistoryTypeID is needed
        HistoryTypeID => 1,
        UserID        => 1,
        Comment       => 'Any useful information',
    );

=cut

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

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

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

    # get history type id from history type if history type is given.
    if ( $Param{HistoryType} ) {
        my $Id = $Self->HistoryTypeLookup( HistoryType => $Param{HistoryType} );

        if ( !$Id ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => 'Invalid history type given!',
            );
            return;
        }

        $Param{HistoryTypeID} = $Id;
    }

    # if history type is given
    elsif ( $Param{HistoryTypeID} ) {
        my $Name = $Self->HistoryTypeLookup( HistoryTypeID => $Param{HistoryTypeID} );

        if ( !$Name ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => 'Invalid history type id given!',
            );
            return;
        }
    }

    # check if given config item id points to an existing config item number
    if ( $Param{ConfigItemID} ) {

        my $Number = $Self->ConfigItemLookup(
            ConfigItemID => $Param{ConfigItemID},
        );

        if ( !$Number ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => 'Invalid config item id given!',
            );
            return;
        }
    }

    # delete cached results
    delete $Self->{Cache}->{CIVersions}->{ $Param{ConfigItemID} };

    # shorten the comment if it is bigger than max length
    if ( length( $Param{Comment} ) > 255 ) {

        my ( $Field, $Old, $New ) = split '%%', $Param{Comment}, 3;

        my $Length = int( ( 255 - length($Field) - 4 ) / 2 );

        if ( length($Old) > $Length ) {
            my $Index = int( $Length / 2 );
            $Old = substr( $Old, 0, $Index - 2 ) . '...' . substr( $Old, length($Old) - $Index + 2 );
        }
        if ( length($New) > $Length ) {
            my $Index = int( $Length / 2 );
            $New = substr( $New, 0, $Index - 2 ) . '...' . substr( $New, length($New) - $Index + 2 );
        }
        my $NewComment = $Field . '%%' . $Old . '%%' . $New;

        $Param{Comment} = $NewComment;
    }

    # insert history entry
    return $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL => 'INSERT INTO configitem_history ( configitem_id, content, create_by, '
            . 'create_time, type_id ) VALUES ( ?, ?, ?, current_timestamp, ? )',
        Bind => [
            \$Param{ConfigItemID},
            \$Param{Comment},
            \$Param{UserID},
            \$Param{HistoryTypeID},
        ],
    );
}

=head2 HistoryDelete()

Deletes complete history for a given config item

    $ConfigItemObject->HistoryDelete(
        ConfigItemID => 123,
    );

=cut

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

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

    # delete cached results
    delete $Self->{Cache}->{CIVersion}->{ $Param{ConfigItemID} };
    for my $VersionNr ( sort keys %{ $Self->{Cache}->{Versions} } ) {
        my ($CacheConfigItem) = keys %{ $Self->{Cache}->{Versions}->{$VersionNr} };
        delete $Self->{Cache}->{Versions}->{$VersionNr} if $CacheConfigItem eq $Param{ConfigItemID};
    }

    # delete history for given config item
    return $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL  => 'DELETE FROM configitem_history WHERE configitem_id = ?',
        Bind => [ \$Param{ConfigItemID} ],
    );
}

=head2 HistoryEntryDelete()

Deletes a single history entry.

    $ConfigItemObject->HistoryEntryDelete(
        HistoryEntryID => 123,
    );

=cut

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

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

    # delete single entry
    return $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL  => 'DELETE FROM configitem_history WHERE id = ?',
        Bind => [ \$Param{HistoryEntryID} ],
    );
}

=head2 HistoryTypeLookup()

This method does a lookup for a history type. If a history type id is given,
it returns the name of the history type. If a history type is given, the appropriate
id is returned.

    my $Name = $ConfigItemObject->HistoryTypeLookup(
        HistoryTypeID => 1234,
    );

    my $Id = $ConfigItemObject->HistoryTypeLookup(
        HistoryType => 'ConfigItemCreate',
    );

=cut

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

    my ($Key) = grep { $Param{$_} } qw(HistoryTypeID HistoryType);

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

    # if result is cached return that result
    return $Self->{Cache}->{HistoryTypeLookup}->{ $Param{$Key} }
        if $Self->{Cache}->{HistoryTypeLookup}->{ $Param{$Key} };

    # set the appropriate SQL statement
    my $SQL = 'SELECT name FROM configitem_history_type WHERE id = ?';

    if ( $Key eq 'HistoryType' ) {
        $SQL = 'SELECT id FROM configitem_history_type WHERE name = ?';
    }

    # fetch the requested value
    return if !$Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL   => $SQL,
        Bind  => [ \$Param{$Key} ],
        Limit => 1,
    );

    my $Value;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $Value = $Row[0];
    }

    # save value in cache
    $Self->{Cache}->{HistoryTypeLookup}->{ $Param{$Key} } = $Value;

    return $Value;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::ITSMConfigItem::Number;

use strict;
use warnings;

our $ObjectManagerDisabled = 1;

=head1 NAME

Kernel::System::ITSMConfigItem::Number - sub module of Kernel::System::ITSMConfigItem

=head1 DESCRIPTION

All config item number functions.

=head1 PUBLIC INTERFACE

=head2 ConfigItemNumberLookup()

return config item id or config item number

    my $ConfigItemNumber = $ConfigItemObject->ConfigItemNumberLookup(
        ConfigItemID => 123,
    );

    or

    my $ConfigItemID = $ConfigItemObject->ConfigItemNumberLookup(
        ConfigItemNumber => '123454321',
    );

=cut

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

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

    if ( $Param{ConfigItemID} ) {

        # check if result is already cached
        return $Self->{Cache}->{ConfigItemNumberLookup}->{ID}->{ $Param{ConfigItemID} }
            if $Self->{Cache}->{ConfigItemNumberLookup}->{ID}->{ $Param{ConfigItemID} };

        # ask database
        $Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL   => 'SELECT configitem_number FROM configitem WHERE id = ?',
            Bind  => [ \$Param{ConfigItemID} ],
            Limit => 1,
        );

        # fetch the result
        my $ConfigItemNumber;
        while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
            $ConfigItemNumber = $Row[0];
        }

        # cache the result
        $Self->{Cache}->{ConfigItemNumberLookup}->{ID}->{ $Param{ConfigItemID} } = $ConfigItemNumber;

        return $ConfigItemNumber;
    }

    # check if result is already cached
    return $Self->{Cache}->{ConfigItemNumberLookup}->{Number}->{ $Param{ConfigItemNumber} }
        if $Self->{Cache}->{ConfigItemNumberLookup}->{Number}->{ $Param{ConfigItemNumber} };

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL   => 'SELECT id FROM configitem WHERE configitem_number = ?',
        Bind  => [ \$Param{ConfigItemNumber} ],
        Limit => 1,
    );

    # fetch the result
    my $ConfigItemID;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $ConfigItemID = $Row[0];
    }

    # cache the result
    $Self->{Cache}->{ConfigItemNumberLookup}->{Number}->{ $Param{ConfigItemNumber} } = $ConfigItemID;

    return $ConfigItemID;
}

=head2 ConfigItemNumberCreate()

create a new config item number

    my $Number = $ConfigItemObject->ConfigItemNumberCreate(
        Type    => 'AutoIncrement',
        ClassID => 123,
    );

=cut

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

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

    # load backend
    if ( !$Kernel::OM->Get('Kernel::System::Main')->Require( $Param{Type} ) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Can't load config item number generator backend module $Param{Type}! $@",
        );
        return;
    }

    # load backend
    return if !$Kernel::OM->Get('Kernel::System::Main')->RequireBaseClass( $Param{Type} );

    # create number
    my $Number = $Self->_ConfigItemNumberCreate(%Param);

    return $Number;
}

=head2 CurrentCounterGet()

return the current counter of a class

    my $Counter = $ConfigItemObject->CurrentCounterGet(
        ClassID => 123,
        Type    => 'AutoIncrement',
    );

=cut

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

    NEEDED:
    for my $Needed (qw(ClassID Type)) {

        next NEEDED if defined $Param{$Needed};

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need $Needed!",
        );
        return;
    }

    # ask the database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT counter FROM configitem_counter WHERE '
            . 'class_id = ? AND counter_type = ?',
        Bind  => [ \$Param{ClassID}, \$Param{Type} ],
        Limit => 1,
    );

    # fetch the result
    my $Number;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $Number = $Row[0];
    }

    return $Number;
}

=head2 CurrentCounterSet()

set the current counter of a class

    my $True = $ConfigItemObject->CurrentCounterSet(
        ClassID => 123,
        Type    => 'AutoIncrement',
        Counter => '12',
    );

=cut

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

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

    # delete old counter
    $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL  => 'DELETE FROM configitem_counter WHERE class_id = ?',
        Bind => [ \$Param{ClassID} ],
    );

    # set new counter
    $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL => 'INSERT INTO configitem_counter '
            . '(class_id, counter_type, counter) VALUES (?, ?, ?)',
        Bind => [ \$Param{ClassID}, \$Param{Type}, \$Param{Counter} ],
    );

    return 1;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06Ok51bWJlcjo6QXV0b0luY3JlbWVudDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCm91ciAkT2JqZWN0TWFuYWdlckRpc2FibGVkID0gMTsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06Ok51bWJlcjo6QXV0b0luY3JlbWVudCAtIGNvbmZpZyBpdGVtIG51bWJlciBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIERFU0NSSVBUSU9OCgpBbGwgYXV0byBpbmNyZW1lbnQgY29uZmlnIGl0ZW0gbnVtYmVyIGZ1bmN0aW9ucwoKPWhlYWQyIF9Db25maWdJdGVtTnVtYmVyQ3JlYXRlKCkKCmNyZWF0ZSBhIG5ldyBjb25maWcgaXRlbSBudW1iZXIKCiAgICBteSAkTnVtYmVyID0gJEJhY2tlbmRPYmplY3QtPl9Db25maWdJdGVtTnVtYmVyQ3JlYXRlKAogICAgICAgIENsYXNzSUQgPT4gMTIzLAogICAgKTsKCj1jdXQKCnN1YiBfQ29uZmlnSXRlbU51bWJlckNyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17Q2xhc3NJRH0gKSB7CiAgICAgICAgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpLT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIENsYXNzSUQhJywKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAjIGdldCBzeXN0ZW0gaWQKICAgIG15ICRTeXN0ZW1JRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyktPkdldCgnU3lzdGVtSUQnKTsKCiAgICAjIGdldCBjdXJyZW50IGNvdW50ZXIKICAgIG15ICRDdXJyZW50Q291bnRlciA9ICRTZWxmLT5DdXJyZW50Q291bnRlckdldCgKICAgICAgICBDbGFzc0lEID0+ICRQYXJhbXtDbGFzc0lEfSwKICAgICAgICBUeXBlICAgID0+ICdBdXRvSW5jcmVtZW50JywKICAgICkgfHwgMDsKCiAgICBDSVBIRVI6CiAgICBmb3IgbXkgJENpcGhlciAoIDEgLi4gMV8wMDBfMDAwXzAwMCApIHsKCiAgICAgICAgIyBjcmVhdGUgbmV3IG51bWJlcgogICAgICAgIG15ICROdW1iZXIgPSAkU3lzdGVtSUQgLiAkUGFyYW17Q2xhc3NJRH0gLiBzcHJpbnRmKCAiJTA2ZCIsICggJEN1cnJlbnRDb3VudGVyICsgJENpcGhlciApICk7CgogICAgICAgICMgZmluZCBleGlzdGluZyBudW1iZXIKICAgICAgICBteSAkRHVwbGljYXRlID0gJFNlbGYtPkNvbmZpZ0l0ZW1OdW1iZXJMb29rdXAoCiAgICAgICAgICAgIENvbmZpZ0l0ZW1OdW1iZXIgPT4gJE51bWJlciwKICAgICAgICApOwoKICAgICAgICBuZXh0IENJUEhFUiBpZiAkRHVwbGljYXRlOwoKICAgICAgICAjIHNldCBjb3VudGVyCiAgICAgICAgJFNlbGYtPkN1cnJlbnRDb3VudGVyU2V0KAogICAgICAgICAgICBDbGFzc0lEID0+ICRQYXJhbXtDbGFzc0lEfSwKICAgICAgICAgICAgVHlwZSAgICA9PiAnQXV0b0luY3JlbWVudCcsCiAgICAgICAgICAgIENvdW50ZXIgPT4gKCAkQ3VycmVudENvdW50ZXIgKyAkQ2lwaGVyICksCiAgICAgICAgKTsKCiAgICAgICAgcmV0dXJuICROdW1iZXI7CiAgICB9CgogICAgcmV0dXJuOwp9CgoxOwoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIHNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoTDxodHRwczovL290cnMub3JnLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQ+LgoKPWN1dAo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlBlcm1pc3Npb247Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgJE9iamVjdE1hbmFnZXJEaXNhYmxlZCA9IDE7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpQZXJtaXNzaW9uIC0gbW9kdWxlIGZvciBJVFNNQ29uZmlnSXRlbS5wbSB3aXRoIFBlcm1pc3Npb24gZnVuY3Rpb25zCgo9aGVhZDEgREVTQ1JJUFRJT04KCkFsbCBQZXJtaXNzaW9uIGZ1bmN0aW9ucy4KCj1oZWFkMSBQVUJMSUMgSU5URVJGQUNFCgo9aGVhZDIgUGVybWlzc2lvbigpCgpyZXR1cm5zIHdoZXRoZXIgdGhlIHVzZXIgaGFzIHBlcm1pc3Npb25zIG9yIG5vdAoKICAgIG15ICRBY2Nlc3MgPSAkQ29uZmlnSXRlbU9iamVjdC0+UGVybWlzc2lvbigKICAgICAgICBUeXBlICAgICA9PiAncm8nLAogICAgICAgIFNjb3BlICAgID0+ICdDbGFzcycsICMgQ2xhc3MgfHwgSXRlbQogICAgICAgIENsYXNzSUQgID0+IDEyMywgICAgICMgaWYgU2NvcGUgaXMgJ0NsYXNzJwogICAgICAgIEl0ZW1JRCAgID0+IDEyMywgICAgICMgaWYgU2NvcGUgaXMgJ0l0ZW0nCiAgICAgICAgVXNlcklEICAgPT4gMTIzLAogICAgKTsKCm9yIHdpdGhvdXQgbG9nZ2luZywgZm9yIGV4YW1wbGUgZm9yIHRvIGNoZWNrIGlmIGEgbGluay9hY3Rpb24gc2hvdWxkIGJlIHNob3duCgogICAgbXkgJEFjY2VzcyA9ICRDb25maWdJdGVtT2JqZWN0LT5QZXJtaXNzaW9uKAogICAgICAgIFR5cGUgICAgID0+ICdybycsCiAgICAgICAgU2NvcGUgICAgPT4gJ0NsYXNzJywgIyBDbGFzcyB8fCBJdGVtCiAgICAgICAgQ2xhc3NJRCAgPT4gMTIzLCAgICAgIyBpZiBTY29wZSBpcyAnQ2xhc3MnCiAgICAgICAgSXRlbUlEICAgPT4gMTIzLCAgICAgIyBpZiBTY29wZSBpcyAnSXRlbScKICAgICAgICBMb2dObyAgICA9PiAxLAogICAgICAgIFVzZXJJRCAgID0+IDEyMywKICAgICk7Cgo9Y3V0CgpzdWIgUGVybWlzc2lvbiB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBmb3IgbXkgJE5lZWRlZCAocXcoVHlwZSBTY29wZSBVc2VySUQpKSB7CiAgICAgICAgaWYgKCAhJFBhcmFteyROZWVkZWR9ICkgewogICAgICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkTmVlZGVkISIsCiAgICAgICAgICAgICk7CiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICB9CgogICAgIyBjaGVjayBmb3IgZXhpc3RlbmNlIG9mIEl0ZW1JRCBvciBDbGFzc0lEIGRlcGVuZGVudAogICAgIyBvbiB0aGUgU2NvcGUKICAgIGlmICgKICAgICAgICAoICRQYXJhbXtTY29wZX0gZXEgJ0NsYXNzJyAmJiAhJFBhcmFte0NsYXNzSUR9ICkKICAgICAgICB8fCAoICRQYXJhbXtTY29wZX0gZXEgJ0l0ZW0nICYmICEkUGFyYW17SXRlbUlEfSApCiAgICAgICAgKQogICAgewogICAgICAgICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCBDbGFzc0lEIGlmIFNjb3BlIGlzICdDbGFzcycgb3IgSXRlbUlEIGlmIFNjb3BlIGlzICdJdGVtJyEiLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICMgcnVuIGFsbCBJVFNNQ29uZmlnSXRlbSBQZXJtaXNzaW9uIG1vZHVsZXMKICAgIGlmICgKICAgICAgICByZWYgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpDb25maWcnKS0+R2V0KCAnSVRTTUNvbmZpZ0l0ZW06OlBlcm1pc3Npb246OicgLiAkUGFyYW17U2NvcGV9ICkgZXEgJ0hBU0gnCiAgICAgICAgKQogICAgewogICAgICAgIG15ICVNb2R1bGVzID0gJXsKICAgICAgICAgICAgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpDb25maWcnKS0+R2V0KCAnSVRTTUNvbmZpZ0l0ZW06OlBlcm1pc3Npb246OicgLiAkUGFyYW17U2NvcGV9ICkKICAgICAgICB9OwogICAgICAgIE1PRFVMRToKICAgICAgICBmb3IgbXkgJE1vZHVsZSAoIHNvcnQga2V5cyAlTW9kdWxlcyApIHsKCiAgICAgICAgICAgICMgbG9hZCBtb2R1bGUKICAgICAgICAgICAgbmV4dCBNT0RVTEUKICAgICAgICAgICAgICAgIGlmICEkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TWFpbicpLT5SZXF1aXJlKCAkTW9kdWxlc3skTW9kdWxlfS0+e01vZHVsZX0gKTsKCiAgICAgICAgICAgICMgY3JlYXRlIG9iamVjdAogICAgICAgICAgICBteSAkTW9kdWxlT2JqZWN0ID0gJE1vZHVsZXN7JE1vZHVsZX0tPntNb2R1bGV9LT5uZXcoKTsKCiAgICAgICAgICAgICMgZXhlY3V0ZSBSdW4oKQogICAgICAgICAgICBteSAkQWNjZXNzT2sgPSAkTW9kdWxlT2JqZWN0LT5SdW4oJVBhcmFtKTsKCiAgICAgICAgICAgICMgY2hlY2sgZ3JhbnRlZCBvcHRpb24gKHNob3VsZCBJIHNheSBvaykKICAgICAgICAgICAgaWYgKCAkQWNjZXNzT2sgJiYgJE1vZHVsZXN7JE1vZHVsZX0tPntHcmFudGVkfSApIHsKCiAgICAgICAgICAgICAgICAjIGFjY2VzcyBvawogICAgICAgICAgICAgICAgcmV0dXJuIDE7CiAgICAgICAgICAgIH0KCiAgICAgICAgICAgICMgcmV0dXJuIGJlY2F1c2UgYWNjZXNzIGlzIGZhbHNlIGJ1dCBpdCdzIHJlcXVpcmVkCiAgICAgICAgICAgIGlmICggISRBY2Nlc3NPayAmJiAkTW9kdWxlc3skTW9kdWxlfS0+e1JlcXVpcmVkfSApIHsKICAgICAgICAgICAgICAgIGlmICggISRQYXJhbXtMb2dOb30gKSB7CiAgICAgICAgICAgICAgICAgICAgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpLT5Mb2coCiAgICAgICAgICAgICAgICAgICAgICAgIFByaW9yaXR5ID0+ICdub3RpY2UnLAogICAgICAgICAgICAgICAgICAgICAgICBNZXNzYWdlICA9PiAiUGVybWlzc2lvbiBkZW5pZWQgYmVjYXVzZSBtb2R1bGUgIgogICAgICAgICAgICAgICAgICAgICAgICAgICAgLiAiKCRNb2R1bGVzeyRNb2R1bGV9LT57TW9kdWxlfSkgaXMgcmVxdWlyZWQgIgogICAgICAgICAgICAgICAgICAgICAgICAgICAgLiAiKFVzZXJJRDogJFBhcmFte1VzZXJJRH0gJyRQYXJhbXtUeXBlfScgIgogICAgICAgICAgICAgICAgICAgICAgICAgICAgLiAib24gJFBhcmFte1Njb3BlfTogIiAuICRQYXJhbXsgJFBhcmFte1Njb3BlfSAuICdJRCcgfSAuICIpISIsCiAgICAgICAgICAgICAgICAgICAgKTsKICAgICAgICAgICAgICAgIH0KCiAgICAgICAgICAgICAgICAjIGFjY2VzcyBub3Qgb2sKICAgICAgICAgICAgICAgIHJldHVybjsKICAgICAgICAgICAgfQogICAgICAgIH0KICAgIH0KCiAgICAjIGRvbid0IGdyYW50IGFjY2VzcwogICAgaWYgKCAhJFBhcmFte0xvZ05vfSApIHsKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ25vdGljZScsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJQZXJtaXNzaW9uIGRlbmllZCAoVXNlcklEOiAkUGFyYW17VXNlcklEfSAnJFBhcmFte1R5cGV9JyAiCiAgICAgICAgICAgICAgICAuICJvbiAkUGFyYW17U2NvcGV9OiAiIC4gJFBhcmFteyAkUGFyYW17U2NvcGV9IC4gJ0lEJyB9IC4gIikhIiwKICAgICAgICApOwogICAgfQoKICAgIHJldHVybjsKfQoKMTsKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKEw8aHR0cHM6Ly9vdHJzLm9yZy8+KS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0Pi4KCj1jdXQK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlBlcm1pc3Npb246OkNsYXNzR3JvdXBDaGVjazsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycsCiAgICAnS2VybmVsOjpTeXN0ZW06Okdyb3VwJywKICAgICdLZXJuZWw6OlN5c3RlbTo6TG9nJywKKTsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlBlcm1pc3Npb246OkNsYXNzR3JvdXBDaGVjayAtIGNoZWNrIGlmIGEgdXNlciBiZWxvbmdzIHRvIGEgZ3JvdXAKCj1oZWFkMSBQVUJMSUMgSU5URVJGQUNFCgo9aGVhZDIgbmV3KCkKCmNyZWF0ZSBhbiBvYmplY3QKCiAgICB1c2UgS2VybmVsOjpTeXN0ZW06Ok9iamVjdE1hbmFnZXI7CiAgICBsb2NhbCAkS2VybmVsOjpPTSA9IEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyLT5uZXcoKTsKICAgIG15ICRDaGVja09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6UGVybWlzc2lvbjo6Q2xhc3NHcm91cENoZWNrJyk7Cgo9Y3V0CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHt9OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgIHJldHVybiAkU2VsZjsKfQoKPWhlYWQyIFJ1bigpCgp0aGlzIG1ldGhvZCBkb2VzIHRoZSBjaGVjayBpZiB0aGUgdXNlIGJlbG9uZ3MgdG8gYSBnaXZlbiBncm91cAoKICAgIG15ICRIYXNBY2Nlc3MgPSAkQ2hlY2tPYmplY3QtPlJ1bigKICAgICAgICBVc2VySUQgID0+IDEyMywKICAgICAgICBUeXBlICAgID0+ICdybycsCiAgICAgICAgQ2xhc3NJRCA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3M6OkNvbXB1dGVyJywKICAgICk7Cgo9Y3V0CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGZvciBteSAkTmVlZGVkIChxdyhVc2VySUQgVHlwZSBDbGFzc0lEKSkgewogICAgICAgIGlmICggISRQYXJhbXskTmVlZGVkfSApIHsKICAgICAgICAgICAgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpLT5Mb2coCiAgICAgICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICAgICAgTWVzc2FnZSAgPT4gIk5lZWQgJE5lZWRlZCEiLAogICAgICAgICAgICApOwogICAgICAgICAgICByZXR1cm47CiAgICAgICAgfQogICAgfQoKICAgICMgZ2V0IENsYXNzIGRhdGEKICAgIG15ICRDbGFzc0l0ZW0gPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6R2VuZXJhbENhdGFsb2cnKS0+SXRlbUdldCgKICAgICAgICBJdGVtSUQgPT4gJFBhcmFte0NsYXNzSUR9LAogICAgKTsKCiAgICAjIGdldCB1c2VyIGdyb3VwcwogICAgbXkgQEdyb3VwSURzID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06Okdyb3VwJyktPkdyb3VwTWVtYmVyTGlzdCgKICAgICAgICBVc2VySUQgPT4gJFBhcmFte1VzZXJJRH0sCiAgICAgICAgVHlwZSAgID0+ICRQYXJhbXtUeXBlfSwKICAgICAgICBSZXN1bHQgPT4gJ0lEJywKICAgICAgICBDYWNoZWQgPT4gMSwKICAgICk7CgogICAgIyBsb29raW5nIGZvciBncm91cCBpZCwgcmV0dXJuIGFjY2VzcyBpZiB1c2VyIGlzIGluIGdyb3VwCiAgICBmb3IgbXkgJEdyb3VwSUQgKEBHcm91cElEcykgewogICAgICAgIHJldHVybiAxIGlmICRDbGFzc0l0ZW0tPntQZXJtaXNzaW9ufSAmJiAkR3JvdXBJRCBlcSAkQ2xhc3NJdGVtLT57UGVybWlzc2lvbn07CiAgICB9CgogICAgIyByZXR1cm4gbm8gYWNjZXNzCiAgICByZXR1cm47Cn0KCjE7Cgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKClRoaXMgU29mdHdhcmUgaXMgcGFydCBvZiB0aGUgT1RSUyBwcm9qZWN0IChMPGh0dHA6Ly9vdHJzLm9yZy8+KS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0Pi4KCj1jdXQK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlBlcm1pc3Npb246Okl0ZW1DbGFzc0dyb3VwQ2hlY2s7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6R2VuZXJhbENhdGFsb2cnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpHcm91cCcsCiAgICAnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtJywKICAgICdLZXJuZWw6OlN5c3RlbTo6TG9nJywKKTsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlBlcm1pc3Npb246Okl0ZW1DbGFzc0dyb3VwQ2hlY2sgLSBjaGVjayBpZiBhIHVzZXIgY2FuIGFjY2VzcyBhbiBpdGVtCgo9aGVhZDEgREVTQ1JJUFRJT04KCkFsbCBjb25maWcgaXRlbSBmdW5jdGlvbnMuCgo9aGVhZDEgUFVCTElDIElOVEVSRkFDRQoKPWhlYWQyIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgdXNlIEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyOwogICAgbG9jYWwgJEtlcm5lbDo6T00gPSBLZXJuZWw6OlN5c3RlbTo6T2JqZWN0TWFuYWdlci0+bmV3KCk7CiAgICBteSAkQ2hlY2tPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlBlcm1pc3Npb246Okl0ZW1DbGFzc0dyb3VwQ2hlY2snKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aGVhZDIgUnVuKCkKCnRoaXMgbWV0aG9kIGRvZXMgdGhlIGNoZWNrIGlmIHRoZSB1c2VyIGNhbiBhY2Nlc3MgYW4gaXRlbQoKICAgIG15ICRIYXNBY2Nlc3MgPSAkQ2hlY2tPYmplY3QtPlJ1bigKICAgICAgICBVc2VySUQgPT4gMTIzLAogICAgICAgIFR5cGUgICA9PiAncm8nLAogICAgICAgIEl0ZW1JRCA9PiAzNDUsCiAgICApOwoKPWN1dAoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBmb3IgbXkgJE5lZWRlZCAocXcoVXNlcklEIFR5cGUgSXRlbUlEKSkgewogICAgICAgIGlmICggISRQYXJhbXskTmVlZGVkfSApIHsKICAgICAgICAgICAgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpLT5Mb2coCiAgICAgICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICAgICAgTWVzc2FnZSAgPT4gIk5lZWQgJE5lZWRlZCEiLAogICAgICAgICAgICApOwogICAgICAgICAgICByZXR1cm47CiAgICAgICAgfQogICAgfQoKICAgICMgZ2V0IGNvbmZpZyBpdGVtIGRhdGEKICAgIG15ICRDb25maWdJdGVtID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtJyktPkNvbmZpZ0l0ZW1HZXQoCiAgICAgICAgQ29uZmlnSXRlbUlEID0+ICRQYXJhbXtJdGVtSUR9LAogICAgKTsKCiAgICAjIGdldCBDbGFzcyBkYXRhCiAgICBteSAkQ2xhc3NJdGVtID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkdlbmVyYWxDYXRhbG9nJyktPkl0ZW1HZXQoCiAgICAgICAgSXRlbUlEID0+ICRDb25maWdJdGVtLT57Q2xhc3NJRH0sCiAgICApOwoKICAgICMgZ2V0IHVzZXIgZ3JvdXBzCiAgICBteSBAR3JvdXBJRHMgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6R3JvdXAnKS0+R3JvdXBNZW1iZXJMaXN0KAogICAgICAgIFVzZXJJRCA9PiAkUGFyYW17VXNlcklEfSwKICAgICAgICBUeXBlICAgPT4gJFBhcmFte1R5cGV9LAogICAgICAgIFJlc3VsdCA9PiAnSUQnLAogICAgICAgIENhY2hlZCA9PiAxLAogICAgKTsKCiAgICAjIGxvb2tpbmcgZm9yIGdyb3VwIGlkLCByZXR1cm4gYWNjZXNzIGlmIHVzZXIgaXMgaW4gZ3JvdXAKICAgIGZvciBteSAkR3JvdXBJRCAoQEdyb3VwSURzKSB7CiAgICAgICAgcmV0dXJuIDEgaWYgJENsYXNzSXRlbS0+e1Blcm1pc3Npb259ICYmICRHcm91cElEIGVxICRDbGFzc0l0ZW0tPntQZXJtaXNzaW9ufTsKICAgIH0KCiAgICAjIHJldHVybiBubyBhY2Nlc3MKICAgIHJldHVybjsKfQoKMTsKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBTb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKEw8aHR0cDovL290cnMub3JnLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQ+LgoKPWN1dAo=
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::ITSMConfigItem::Version;

use strict;
use warnings;

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

use Storable;

our $ObjectManagerDisabled = 1;

=head1 NAME

Kernel::System::ITSMConfigItem::Version - sub module of Kernel::System::ITSMConfigItem

=head1 DESCRIPTION

All version functions.

=head1 PUBLIC INTERFACE

=head2 VersionZoomList()

return a config item version list as array-hash reference

    my $VersionListRef = $ConfigItemObject->VersionZoomList(
        ConfigItemID => 123,
    );

=cut

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

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

    # get config item
    my $ConfigItem = $Self->ConfigItemGet(
        ConfigItemID => $Param{ConfigItemID},
    );

    # get version zoom list
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT id, name, depl_state_id, inci_state_id, create_time, create_by '
            . 'FROM configitem_version WHERE configitem_id = ? ORDER BY id',
        Bind => [ \$Param{ConfigItemID} ],
    );

    # fetch the result
    my @VersionList;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        my %Version;
        $Version{VersionID}   = $Row[0];
        $Version{Name}        = $Row[1];
        $Version{DeplStateID} = $Row[2];
        $Version{InciStateID} = $Row[3];
        $Version{CreateTime}  = $Row[4];
        $Version{CreateBy}    = $Row[5];

        push @VersionList, \%Version;
    }

    for my $Version (@VersionList) {

        # get deployment state functionality
        my $DeplState = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemGet(
            ItemID => $Version->{DeplStateID},
        );

        $Version->{DeplState}     = $DeplState->{Name};
        $Version->{DeplStateType} = $DeplState->{Functionality};

        # get incident state functionality
        my $InciState = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemGet(
            ItemID => $Version->{InciStateID},
        );

        $Version->{InciState}     = $InciState->{Name};
        $Version->{InciStateType} = $InciState->{Functionality};

        # add config item data
        $Version->{ClassID}          = $ConfigItem->{ClassID};
        $Version->{Class}            = $ConfigItem->{Class};
        $Version->{Number}           = $ConfigItem->{Number};
        $Version->{CurDeplStateID}   = $ConfigItem->{CurDeplStateID};
        $Version->{CurDeplState}     = $ConfigItem->{CurDeplState};
        $Version->{CurDeplStateType} = $ConfigItem->{CurDeplStateType};
        $Version->{CurInciStateID}   = $ConfigItem->{CurInciStateID};
        $Version->{CurInciState}     = $ConfigItem->{CurInciState};
        $Version->{CurInciStateType} = $ConfigItem->{CurInciStateType};
    }

    return \@VersionList;
}

=head2 VersionListAll()

Returns a two-dimensional hash reference with the config item ids as keys of the first level
and the corresponding version ids as keys of the second level,
then followed by the version data as hash reference.

    my $VersionListRef = $ConfigItemObject->VersionListAll(
        ConfigItemIDs => [ 1, 2, 3, 4, ...],    # optional

        OlderDate     => '2014-12-22 23:59:59', # optional
                                                # finds versions older than the given date
                                                # Format MUST be
                                                # YYYY-MM-DD HH:MM:SS
                                                # fill missing values with 0 first
                                                # Example: 2014-04-01 07:03:04
                                                # otherwise it won't be taken as
                                                # valid search parameter

        Limit         => 1000000,               # optional
    );

Returns:

$VersionListRef = {

    # ConfigItemID
    1 => {

        # VersionID
        100 => {
            VersionID    => 100,
            ConfigItemID => 1,
            Name         => 'ConfigItem1',
            DefinitionID => 5,
            DeplStateID  => 3,
            InciStateID  => 2,
            CreateTime   => '2016-03-22 17:58:00',
            CreateBy     => 1,
        },

        # VersionID
        101 => {
            VersionID    => 101,
            ConfigItemID => 1,
            Name         => 'ConfigItem2',
            DefinitionID => 5,
            DeplStateID  => 3,
            InciStateID  => 2,
            CreateTime   => '2016-03-22 17:58:00',
            CreateBy     => 1,
        },
    },

    # ConfigItemID
    2 => {

        # VersionID
        150 => {
            VersionID    => 150,
            ConfigItemID => 2,
            Name         => 'ConfigItem1',
            DefinitionID => 5,
            DeplStateID  => 3,
            InciStateID  => 2,
            CreateTime   => '2016-03-22 17:58:00',
            CreateBy     => 1,
        },

        # VersionID
        151 => {
            VersionID    => 151,
            ConfigItemID => 2,
            Name         => 'ConfigItem1',
            DefinitionID => 5,
            DeplStateID  => 3,
            InciStateID  => 2,
            CreateTime   => '2016-03-22 17:58:00',
            CreateBy     => 1,
        },
    },
};

=cut

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

    # build sql
    my $SQL = 'SELECT id, configitem_id, name, definition_id,
        depl_state_id, inci_state_id, create_time, create_by
        FROM configitem_version WHERE 1=1';

    # if we got ConfigItemIDs make sure we just have numeric ids,
    # extract those and use it for the query
    if ( IsArrayRefWithData( $Param{ConfigItemIDs} ) ) {
        my @ConfigItemIDs = grep { $_ =~ /^\d+$/ } @{ $Param{ConfigItemIDs} };
        $SQL .= ' AND configitem_id IN (' . join ', ', @ConfigItemIDs . ')';
    }

    my @BindParameter;
    if ( $Param{OlderDate} && $Param{OlderDate} =~ /^\d{4}\-\d{2}\-\d{2}\ \d{2}\:\d{2}:\d{2}$/ ) {
        $SQL .= ' AND create_time < ?';
        push @BindParameter, \$Param{OlderDate};
    }

    # set limit
    if ( $Param{Limit} ) {
        $Param{Limit} = $Kernel::OM->Get('Kernel::System::DB')->Quote( $Param{Limit}, 'Integer' );
    }

    if (@BindParameter) {
        $Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL   => $SQL,
            Bind  => \@BindParameter,
            Limit => $Param{Limit},
        );
    }
    else {
        $Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL   => $SQL,
            Limit => $Param{Limit},
        );
    }

    my %Results;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $Results{ $Row[1] }->{ $Row[0] } = {
            VersionID    => $Row[0],
            ConfigItemID => $Row[1],
            Name         => $Row[2] || '',
            DefinitionID => $Row[3] || '',
            DeplStateID  => $Row[4] || '',
            InciStateID  => $Row[5] || '',
            CreateTime   => $Row[6] || '',
            CreateBy     => $Row[7] || '',
        };
    }

    return \%Results;
}

=head2 VersionList()

return a config item version list as array reference

    my $VersionListRef = $ConfigItemObject->VersionList(
        ConfigItemID => 123,
    );

=cut

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

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

    # get version list
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL  => 'SELECT id FROM configitem_version WHERE configitem_id = ? ORDER BY id',
        Bind => [ \$Param{ConfigItemID} ],
    );

    # fetch the result
    my @VersionList;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        push @VersionList, $Row[0];
    }

    return \@VersionList;
}

=head2 VersionGet()

returns a version of a config item as hash reference.
The returned hash contains following attributes.

    $Version{VersionID}
    $Version{ConfigItemID}
    $Version{Number}
    $Version{ClassID}
    $Version{Class}
    $Version{LastVersionID}
    $Version{Name}
    $Version{DefinitionID}
    $Version{DeplStateID}
    $Version{DeplState}
    $Version{DeplStateType}
    $Version{CurDeplStateID}
    $Version{CurDeplState}
    $Version{CurDeplStateType}
    $Version{InciStateID}
    $Version{InciState}
    $Version{InciStateType}
    $Version{CurInciStateID}
    $Version{CurInciState}
    $Version{CurInciStateType}
    $Version{XMLDefinition}
    $Version{XMLData}
    $Version{CreateTime}
    $Version{CreateBy}

    my $VersionRef = $ConfigItemObject->VersionGet(
        VersionID  => 123,
        XMLDataGet => 1,    # (optional) default 1 (0|1)
    );

    or

    my $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => 123,
    );

When the data from the XML storage is not needed then fetching the XML data can be
explicitly turned off by passing XMLDataGet => 0.

    my $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => 123,
        XMLDataGet   => 0,
    );

=cut

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

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

    if ( !defined $Param{XMLDataGet} ) {
        $Param{XMLDataGet} = 1;
    }

    my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');

    if ( $Param{VersionID} ) {

        # check if result is already cached
        my $CacheKey = 'VersionGet::VersionID::' . $Param{VersionID} . '::XMLData::' . $Param{XMLDataGet};
        my $Cache    = $CacheObject->Get(
            Type => $Self->{CacheType},
            Key  => $CacheKey,
        );
        return Storable::dclone($Cache) if $Cache;

        # get version
        $Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL => 'SELECT id, configitem_id, name, definition_id, '
                . 'depl_state_id, inci_state_id, create_time, create_by '
                . 'FROM configitem_version WHERE id = ?',
            Bind  => [ \$Param{VersionID} ],
            Limit => 1,
        );
    }
    else {

        # check if result is already cached
        my $CacheKey = 'VersionGet::ConfigItemID::' . $Param{ConfigItemID} . '::XMLData::' . $Param{XMLDataGet};
        my $Cache    = $CacheObject->Get(
            Type => $Self->{CacheType},
            Key  => $CacheKey,
        );
        return Storable::dclone($Cache) if $Cache;

        # get version
        $Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL => 'SELECT id, configitem_id, name, definition_id, '
                . 'depl_state_id, inci_state_id, create_time, create_by '
                . 'FROM configitem_version '
                . 'WHERE configitem_id = ? ORDER BY id DESC',
            Bind  => [ \$Param{ConfigItemID} ],
            Limit => 1,
        );
    }

    # fetch the result
    my %Version;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $Version{VersionID}    = $Row[0];
        $Version{ConfigItemID} = $Row[1];
        $Version{Name}         = $Row[2];
        $Version{DefinitionID} = $Row[3];
        $Version{DeplStateID}  = $Row[4];
        $Version{InciStateID}  = $Row[5];
        $Version{CreateTime}   = $Row[6];
        $Version{CreateBy}     = $Row[7];
    }

    # check version
    if ( !$Version{VersionID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'No such config item version!',
        );
        return;
    }

    # get deployment state functionality
    my $DeplState = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemGet(
        ItemID => $Version{DeplStateID},
    );

    $Version{DeplState}     = $DeplState->{Name};
    $Version{DeplStateType} = $DeplState->{Functionality};

    # get incident state functionality
    my $InciState = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemGet(
        ItemID => $Version{InciStateID},
    );

    $Version{InciState}     = $InciState->{Name};
    $Version{InciStateType} = $InciState->{Functionality};

    # get config item
    my $ConfigItem = $Self->ConfigItemGet(
        ConfigItemID => $Version{ConfigItemID},
        Cache        => 0,
    );

    # check config item data
    if ( !$ConfigItem || ref $ConfigItem ne 'HASH' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Can't get config item $Version{ConfigItemID}!",
        );
        return;
    }

    $Version{ClassID}          = $ConfigItem->{ClassID};
    $Version{Class}            = $ConfigItem->{Class};
    $Version{LastVersionID}    = $ConfigItem->{LastVersionID};
    $Version{Number}           = $ConfigItem->{Number};
    $Version{CurDeplStateID}   = $ConfigItem->{CurDeplStateID};
    $Version{CurDeplState}     = $ConfigItem->{CurDeplState};
    $Version{CurDeplStateType} = $ConfigItem->{CurDeplStateType};
    $Version{CurInciStateID}   = $ConfigItem->{CurInciStateID};
    $Version{CurInciState}     = $ConfigItem->{CurInciState};
    $Version{CurInciStateType} = $ConfigItem->{CurInciStateType};

    # set cache for VersionID without xml data (always)
    my $CacheKey = 'VersionGet::VersionID::' . $Version{VersionID} . '::XMLData::0';
    $CacheObject->Set(
        Type  => $Self->{CacheType},
        TTL   => $Self->{CacheTTL},
        Key   => $CacheKey,
        Value => Storable::dclone( \%Version ),
    );

    # set cache for ConfigItemID without xml data (only if called with ConfigItemID)
    if ( $Param{ConfigItemID} ) {
        $CacheKey = 'VersionGet::ConfigItemID::' . $Version{ConfigItemID} . '::XMLData::0';
        $CacheObject->Set(
            Type  => $Self->{CacheType},
            TTL   => $Self->{CacheTTL},
            Key   => $CacheKey,
            Value => Storable::dclone( \%Version ),
        );
    }

    # done if we don't need xml data
    return \%Version if !$Param{XMLDataGet};

    # get xml definition
    my $Definition = $Self->DefinitionGet(
        DefinitionID => $Version{DefinitionID},
    );
    $Version{XMLDefinition} = $Definition->{DefinitionRef};

    # get xml data
    $Version{XMLData} = $Self->_XMLVersionGet(
        ClassID   => $ConfigItem->{ClassID},
        VersionID => $Version{VersionID},
    );

    # set cache for VersionID (always)
    $CacheKey = 'VersionGet::VersionID::' . $Version{VersionID} . '::XMLData::1';
    $CacheObject->Set(
        Type  => $Self->{CacheType},
        TTL   => $Self->{CacheTTL},
        Key   => $CacheKey,
        Value => Storable::dclone( \%Version ),
    );

    # set cache for ConfigItemID (only if called with ConfigItemID)
    if ( $Param{ConfigItemID} ) {
        $CacheKey = 'VersionGet::ConfigItemID::' . $Version{ConfigItemID} . '::XMLData::1';
        $CacheObject->Set(
            Type  => $Self->{CacheType},
            TTL   => $Self->{CacheTTL},
            Key   => $CacheKey,
            Value => Storable::dclone( \%Version ),
        );
    }

    return \%Version;
}

=head2 VersionNameGet()

returns the name of a version of a config item.

    my $VersionName = $ConfigItemObject->VersionNameGet(
        VersionID => 123,
    );

    or

    my $VersionName = $ConfigItemObject->VersionNameGet(
        ConfigItemID => 123,
    );

=cut

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

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

    my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');

    if ( $Param{VersionID} ) {

        # check if result is already cached
        my $CacheKey = 'VersionNameGet::VersionID::' . $Param{VersionID};
        my $Cache    = $CacheObject->Get(
            Type => $Self->{CacheType},
            Key  => $CacheKey,
        );
        return ${$Cache} if $Cache;

        # get version
        $Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL => 'SELECT id, name '
                . 'FROM configitem_version WHERE id = ?',
            Bind  => [ \$Param{VersionID} ],
            Limit => 1,
        );
    }
    else {

        # check if result is already cached
        my $CacheKey = 'VersionNameGet::ConfigItemID::' . $Param{ConfigItemID};
        my $Cache    = $CacheObject->Get(
            Type => $Self->{CacheType},
            Key  => $CacheKey,
        );
        return ${$Cache} if $Cache;

        # get version
        $Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL => 'SELECT id, name '
                . 'FROM configitem_version '
                . 'WHERE configitem_id = ? ORDER BY id DESC',
            Bind  => [ \$Param{ConfigItemID} ],
            Limit => 1,
        );
    }

    # fetch the result
    my %Version;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $Version{VersionID} = $Row[0];
        $Version{Name}      = $Row[1];
    }

    # check version
    if ( !$Version{VersionID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'No such config item version!',
        );
        return;
    }

    # set cache for VersionID (always)
    my $CacheKey = 'VersionNameGet::VersionID::' . $Version{VersionID};
    $CacheObject->Set(
        Type  => $Self->{CacheType},
        TTL   => $Self->{CacheTTL},
        Key   => $CacheKey,
        Value => \$Version{Name},
    );

    # set cache for ConfigItemID (only if called with ConfigItemID)
    if ( $Param{ConfigItemID} ) {
        $CacheKey = 'VersionNameGet::ConfigItemID::' . $Param{ConfigItemID};
        $CacheObject->Set(
            Type  => $Self->{CacheType},
            TTL   => $Self->{CacheTTL},
            Key   => $CacheKey,
            Value => \$Version{Name},
        );
    }

    return $Version{Name};
}

=head2 VersionConfigItemIDGet()

return the config item id of a version

    my $ConfigItemID = $ConfigItemObject->VersionConfigItemIDGet(
        VersionID => 123,
    );

=cut

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

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

    # check if result is already cached
    return $Self->{Cache}->{VersionConfigItemIDGet}->{ $Param{VersionID} }
        if $Self->{Cache}->{VersionConfigItemIDGet}->{ $Param{VersionID} };

    # get config item id
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL   => 'SELECT configitem_id FROM configitem_version WHERE id = ?',
        Bind  => [ \$Param{VersionID} ],
        Limit => 1,
    );

    # fetch the result
    my $ConfigItemID;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $ConfigItemID = $Row[0];
    }

    # cache the result
    $Self->{Cache}->{VersionConfigItemIDGet}->{ $Param{VersionID} } = $ConfigItemID;

    return $ConfigItemID;
}

=head2 VersionAdd()

add a new version

    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => 123,
        Name         => 'The Name',
        DefinitionID => 1212,
        DeplStateID  => 8,
        InciStateID  => 4,
        XMLData      => $ArrayHashRef,  # (optional)
        UserID       => 1,
    );

=cut

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

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

    my $DBObject    = $Kernel::OM->Get('Kernel::System::DB');
    my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');

    # get deployment state list
    my $DeplStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    return if !$DeplStateList;
    return if ref $DeplStateList ne 'HASH';

    # check the deployment state id
    if ( !$DeplStateList->{ $Param{DeplStateID} } ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'No valid deployment state id given!',
        );
        return;
    }

    # get incident state list
    my $InciStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::Core::IncidentState',
    );

    return if !$InciStateList;
    return if ref $InciStateList ne 'HASH';

    # check the incident state id
    if ( !$InciStateList->{ $Param{InciStateID} } ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'No valid incident state id given!',
        );
        return;
    }

    # get VersionList
    my $VersionList = $Self->VersionList(
        ConfigItemID => $Param{ConfigItemID},
    );

    my $ConfigItemInfo = {};

    if ( @{$VersionList} ) {

        # get old version info for comparisons with current version
        # this is needed to trigger some events
        $ConfigItemInfo = $Self->VersionGet(
            ConfigItemID => $Param{ConfigItemID},
            XMLDataGet   => 0,
        );
    }
    else {

        # get config item
        $ConfigItemInfo = $Self->ConfigItemGet(
            ConfigItemID => $Param{ConfigItemID},
        );
    }

    return if !$ConfigItemInfo;
    return if ref $ConfigItemInfo ne 'HASH';

    # check, whether the feature to check for a unique name is enabled
    if ( $Kernel::OM->Get('Kernel::Config')->Get('UniqueCIName::EnableUniquenessCheck') ) {

        my $NameDuplicates = $Self->UniqueNameCheck(
            ConfigItemID => $Param{ConfigItemID},
            ClassID      => $ConfigItemInfo->{ClassID},
            Name         => $Param{Name},
        );

        # stop processing if the name is not unique
        if ( IsArrayRefWithData($NameDuplicates) ) {

            # build a string of all duplicate IDs
            my $Duplicates = join ', ', @{$NameDuplicates};

            # write an error log message containing all the duplicate IDs
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "The name $Param{Name} is already in use (ConfigItemIDs: $Duplicates)!",
            );
            return;
        }
    }

    my $Events = $Self->_GetEvents(
        Param          => \%Param,
        ConfigItemInfo => $ConfigItemInfo,
    );

    my $ReturnVersionID = scalar @{$VersionList} ? $VersionList->[-1] : 0;
    return $ReturnVersionID if !( $Events && keys %{$Events} );

    # Special case that only XML attributes have been changed that should not create a new version
    if ( $Events->{UpdateLastVersion} && $Events->{UpdateXMLData} && ref $Events->{UpdateXMLData} eq 'ARRAY' ) {

        my %ValueHash2D = $Kernel::OM->Get('Kernel::System::XML')->XMLHash2D(
            XMLHash => $Events->{UpdateXMLData},
        );

        UPDATEVALUE:
        for my $UpdateValue ( sort keys %{ $Events->{ValueUpdate} } ) {

            my $ContentString = $UpdateValue . '{\'Content\'}';

            # Check if the key exists.
            next UPDATEVALUE if !exists $ValueHash2D{$ContentString};

            my $Type = 'ITSM::ConfigItem::' . $ConfigItemInfo->{ClassID};

            # Delete old entries from the database.
            return if !$DBObject->Do(
                SQL => '
                    DELETE FROM xml_storage
                    WHERE xml_type = ?
                    AND xml_key = ?
                    AND xml_content_key = ?
                ',
                Bind => [
                    \$Type,
                    \$ReturnVersionID,
                    \$ContentString,
                ],
            );

            # Add updated entry to the database.
            return if !$DBObject->Do(
                SQL => '
                    INSERT INTO xml_storage
                    (xml_type, xml_key, xml_content_key, xml_content_value)
                    VALUES (?, ?, ?, ?)
                ',
                Bind => [
                    \$Type,
                    \$ReturnVersionID,
                    \$ContentString,
                    \$ValueHash2D{$ContentString},
                ],
            );

            # Delete cache.
            $CacheObject->Delete(
                Type => 'XML',
                Key  => "$Type-$ReturnVersionID",
            );
        }

        # delete affected caches
        my $CacheKey = 'VersionGet::ConfigItemID::' . $Param{ConfigItemID} . '::XMLData::';
        for my $XMLData (qw(0 1)) {
            $CacheObject->Delete(
                Type => $Self->{CacheType},
                Key  => $CacheKey . $XMLData,
            );
        }

        # delete affected caches
        $CacheKey = 'VersionGet::VersionID::' . $ReturnVersionID . '::XMLData::';
        for my $XMLData (qw(0 1)) {
            $CacheObject->Delete(
                Type => $Self->{CacheType},
                Key  => $CacheKey . $XMLData,
            );
        }

        # update change_time and change_by of the config item
        return if !$DBObject->Do(
            SQL => '
                UPDATE configitem
                SET change_time = current_timestamp,
                change_by = ?
                WHERE id = ?',
            Bind => [
                \$Param{UserID},
                \$Param{ConfigItemID},
            ],
        );

        # delete config item cache
        $CacheKey = 'ConfigItemGet::ConfigItemID::' . $Param{ConfigItemID};
        $CacheObject->Delete(
            Type => $Self->{CacheType},
            Key  => $CacheKey,
        );

        # Trigger ValueUpdate event.
        $Self->_EventHandlerForChangedXMLValues(
            ConfigItemID => $Param{ConfigItemID},
            UpdateValues => $Events->{ValueUpdate},
            UserID       => $Param{UserID},
        );

        return $ReturnVersionID;
    }

    # insert new version
    my $Success = $DBObject->Do(
        SQL => 'INSERT INTO configitem_version '
            . '(configitem_id, name, definition_id, '
            . 'depl_state_id, inci_state_id, create_time, create_by) VALUES '
            . '(?, ?, ?, ?, ?, current_timestamp, ?)',
        Bind => [
            \$Param{ConfigItemID},
            \$Param{Name},
            \$Param{DefinitionID},
            \$Param{DeplStateID},
            \$Param{InciStateID},
            \$Param{UserID},
        ],
    );

    return if !$Success;

    # delete affected caches
    my $CacheKey = 'VersionGet::ConfigItemID::' . $Param{ConfigItemID} . '::XMLData::';
    for my $XMLData (qw(0 1)) {
        $CacheObject->Delete(
            Type => $Self->{CacheType},
            Key  => $CacheKey . $XMLData,
        );
    }
    $CacheObject->Delete(
        Type => $Self->{CacheType},
        Key  => 'VersionNameGet::ConfigItemID::' . $Param{ConfigItemID},
    );

    # get id of new version
    $DBObject->Prepare(
        SQL => 'SELECT id, create_time FROM configitem_version WHERE '
            . 'configitem_id = ? ORDER BY id DESC',
        Bind  => [ \$Param{ConfigItemID} ],
        Limit => 1,
    );

    # fetch the result
    my $VersionID;
    my $CreateTime;
    while ( my @Row = $DBObject->FetchrowArray() ) {
        $VersionID  = $Row[0];
        $CreateTime = $Row[1];
    }

    # check version id
    if ( !$VersionID ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Can't get the new version id!",
        );
        return;
    }

    # add xml data
    if ( $Param{XMLData} && ref $Param{XMLData} eq 'ARRAY' ) {
        $Self->_XMLVersionAdd(
            ClassID      => $ConfigItemInfo->{ClassID},
            ConfigItemID => $Param{ConfigItemID},
            VersionID    => $VersionID,
            XMLData      => $Param{XMLData},
        );
    }

    # update last_version_id, cur_depl_state_id, cur_inci_state_id, change_time and change_by
    $DBObject->Do(
        SQL => 'UPDATE configitem SET last_version_id = ?, '
            . 'cur_depl_state_id = ?, cur_inci_state_id = ?, '
            . 'change_time = ?, change_by = ? '
            . 'WHERE id = ?',
        Bind => [
            \$VersionID,
            \$Param{DeplStateID},
            \$Param{InciStateID},
            \$CreateTime,
            \$Param{UserID},
            \$Param{ConfigItemID},
        ],
    );

    # delete config item cache
    $CacheKey = 'ConfigItemGet::ConfigItemID::' . $Param{ConfigItemID};
    $CacheObject->Delete(
        Type => $Self->{CacheType},
        Key  => $CacheKey,
    );

    # trigger VersionCreate event
    $Self->EventHandler(
        Event => 'VersionCreate',
        Data  => {
            ConfigItemID => $Param{ConfigItemID},
            Comment      => $VersionID,
        },
        UserID => $Param{UserID},
    );

    # compare current and old values
    if ( $Events->{ValueUpdate} ) {
        $Self->_EventHandlerForChangedXMLValues(
            ConfigItemID => $Param{ConfigItemID},
            UpdateValues => $Events->{ValueUpdate},
            UserID       => $Param{UserID},
        );
    }

    # trigger definition update event
    if ( $Events->{DefinitionUpdate} ) {
        $Self->EventHandler(
            Event => 'DefinitionUpdate',
            Data  => {
                ConfigItemID => $Param{ConfigItemID},
                Comment      => $Events->{DefinitionUpdate},
            },
            UserID => $Param{UserID},
        );
    }

    # check old and new name
    if ( $Events->{NameUpdate} ) {
        $Self->EventHandler(
            Event => 'NameUpdate',
            Data  => {
                ConfigItemID => $Param{ConfigItemID},
                Comment      => $Events->{NameUpdate},
            },
            UserID => $Param{UserID},
        );
    }

    # trigger incident state update event
    if ( $Events->{IncidentStateUpdate} ) {
        $Self->EventHandler(
            Event => 'IncidentStateUpdate',
            Data  => {
                ConfigItemID => $Param{ConfigItemID},
                Comment      => $Events->{IncidentStateUpdate},
            },
            UserID => $Param{UserID},
        );
    }

    # trigger deployment state update event
    if ( $Events->{DeploymentStateUpdate} ) {
        $Self->EventHandler(
            Event => 'DeploymentStateUpdate',
            Data  => {
                ConfigItemID => $Param{ConfigItemID},
                Comment      => $Events->{DeploymentStateUpdate},
            },
            UserID => $Param{UserID},
        );
    }

    # recalculate the current incident state of all linked config items
    $Self->CurInciStateRecalc(
        ConfigItemID => $Param{ConfigItemID},
    );

    return $VersionID;
}

=head2 VersionDelete()

delete an existing version or versions

    my $True = $ConfigItemObject->VersionDelete(
        VersionID => 123,
        UserID    => 1,
    );

    or

    my $True = $ConfigItemObject->VersionDelete(
        ConfigItemID => 321,
        UserID       => 1,
    );

    my $True = $ConfigItemObject->VersionDelete(
        VersionIDs => [ 1, 2, 3, 4 ],
        UserID     => 1,
    );

=cut

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

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

    if ( !$Param{VersionID} && !$Param{ConfigItemID} && !IsArrayRefWithData( $Param{VersionIDs} ) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need VersionID, ConfigItemID or VersionIDs!',
        );
        return;
    }

    my $VersionList = [];
    if ( $Param{VersionID} ) {
        push @{$VersionList}, $Param{VersionID};
    }
    elsif ( $Param{VersionIDs} ) {
        push @{$VersionList}, @{ $Param{VersionIDs} };
    }
    else {
        # get version list
        $VersionList = $Self->VersionList(
            ConfigItemID => $Param{ConfigItemID},
        );
    }

    return 1 if !scalar @{$VersionList};

    # get config item id for version (needed for event handling)
    my $ConfigItemID = $Param{ConfigItemID};

    my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');

    # to store unique config item ids if there are versions from different config items
    my %ConfigItemIDs;

    my $Success;
    for my $VersionID ( @{$VersionList} ) {

        # get configitem id if neccessary
        if ( $Param{VersionID} || $Param{VersionIDs} ) {
            $ConfigItemID = $Self->VersionConfigItemIDGet(
                VersionID => $VersionID,
            );
        }

        # remember the unique config item ids
        $ConfigItemIDs{$ConfigItemID} = 1;

        # delete the xml version data
        $Self->_XMLVersionDelete(
            VersionID => $VersionID,
            UserID    => $Param{UserID},
        );

        # delete version
        $Success = $Kernel::OM->Get('Kernel::System::DB')->Do(
            SQL  => "DELETE FROM configitem_version WHERE id = ?",
            Bind => [ \$VersionID ],
        );

        # trigger VersionDelete event when deletion was successful
        if ($Success) {

            $Self->EventHandler(
                Event => 'VersionDelete',
                Data  => {
                    ConfigItemID => $ConfigItemID,
                    Comment      => $VersionID,
                },
                UserID => $Param{UserID},
            );

            # delete affected caches
            my $CacheKey = 'VersionGet::VersionID::' . $VersionID . '::XMLData::';
            for my $XMLData (qw(0 1)) {
                $CacheObject->Delete(
                    Type => $Self->{CacheType},
                    Key  => $CacheKey . $XMLData,
                );
            }
            $CacheObject->Delete(
                Type => $Self->{CacheType},
                Key  => 'VersionNameGet::VersionID::' . $VersionID,
            );

            delete $Self->{Cache}->{VersionConfigItemIDGet}->{$VersionID};
        }
    }

    for my $ConfigItemID ( sort keys %ConfigItemIDs ) {

        # delete affected caches for ConfigItemID (most recent version might have been removed)
        my $CacheKey = 'VersionGet::ConfigItemID::' . $ConfigItemID . '::XMLData::';
        for my $XMLData (qw(0 1)) {
            $CacheObject->Delete(
                Type => $Self->{CacheType},
                Key  => $CacheKey . $XMLData,
            );
        }
        $CacheObject->Delete(
            Type => $Self->{CacheType},
            Key  => 'VersionNameGet::ConfigItemID::' . $ConfigItemID,
        );
    }

    return $Success;
}

=head2 VersionSearch()

return a config item list as an array reference

    my $ConfigItemIDs = $ConfigItemObject->VersionSearch(
        Name         => 'The Name',      # (optional)
        ClassIDs     => [ 9, 8, 7, 6 ],  # (optional)
        DeplStateIDs => [ 321, 123 ],    # (optional)
        InciStateIDs => [ 321, 123 ],    # (optional)

        PreviousVersionSearch => 1,  # (optional) default 0 (0|1)

        OrderBy => [ 'ConfigItemID', 'Number' ],                  # (optional)
        # default: [ 'ConfigItemID' ]
        # (ConfigItemID, Name, Number, ClassID, DeplStateID, InciStateID
        # CreateTime, CreateBy, ChangeTime, ChangeBy)

        # Additional information for OrderBy:
        # The OrderByDirection can be specified for each OrderBy attribute.
        # The pairing is made by the array indices.

        OrderByDirection => [ 'Up', 'Down' ],                    # (optional)
        # default: [ 'Up' ]
        # (Down | Up)

        Limit          => 122,  # (optional)
        UsingWildcards => 0,    # (optional) default 1
    );

=cut

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

    # set default values
    if ( !defined $Param{UsingWildcards} ) {
        $Param{UsingWildcards} = 1;
    }

    # verify that all passed array parameters contain an arrayref
    ARGUMENT:
    for my $Argument (
        qw(
        OrderBy
        OrderByDirection
        )
        )
    {
        if ( !defined $Param{$Argument} ) {
            $Param{$Argument} ||= [];

            next ARGUMENT;
        }

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

    # set default order and order direction
    if ( !@{ $Param{OrderBy} } ) {
        $Param{OrderBy} = ['ConfigItemID'];
    }
    if ( !@{ $Param{OrderByDirection} } ) {
        $Param{OrderByDirection} = ['Up'];
    }

    # define order table
    my %OrderByTable = (
        ConfigItemID => 'vr.configitem_id',
        Name         => 'vr.name',
        Number       => 'ci.configitem_number',
        ClassID      => 'ci.class_id',
        DeplStateID  => 'vr.depl_state_id',
        InciStateID  => 'vr.inci_state_id',
        CreateTime   => 'ci.create_time',
        CreateBy     => 'ci.create_by',

        # the change time of the CI is the same as the create time of the version!
        ChangeTime => 'vr.create_time',

        ChangeBy => 'ci.change_by',
    );

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

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

            # found an error
            $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;
    }

    # assemble the ORDER BY clause
    my @SQLOrderBy;
    my $Count = 0;
    my @OrderBySelectColumns;
    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';
            }
        }

        # add SQL
        push @SQLOrderBy,           "$OrderByTable{$OrderBy} $Direction";
        push @OrderBySelectColumns, $OrderByTable{$OrderBy};

    }
    continue {
        $Count++;
    }

    # get like escape string needed for some databases (e.g. oracle)
    my $LikeEscapeString = $Kernel::OM->Get('Kernel::System::DB')->GetDatabaseFunction('LikeEscapeString');

    # add name to sql where array
    my @SQLWhere;
    if ( defined $Param{Name} && $Param{Name} ne '' ) {

        # duplicate the name
        my $Name = $Param{Name};

        # quote
        $Name = $Kernel::OM->Get('Kernel::System::DB')->Quote($Name);

        if ( $Param{UsingWildcards} ) {

            # prepare like string
            $Self->_PrepareLikeString( \$Name );

            push @SQLWhere, "LOWER(vr.name) LIKE LOWER('$Name') $LikeEscapeString";
        }
        else {
            push @SQLWhere, "LOWER(vr.name) = LOWER('$Name')";
        }
    }

    # set array params
    my %ArrayParams = (
        ClassIDs     => 'ci.id = vr.configitem_id AND ci.class_id',
        DeplStateIDs => 'vr.depl_state_id',
        InciStateIDs => 'vr.inci_state_id',
    );

    ARRAYPARAM:
    for my $ArrayParam ( sort keys %ArrayParams ) {

        next ARRAYPARAM if !$Param{$ArrayParam};

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

        next ARRAYPARAM if !@{ $Param{$ArrayParam} };

        # quote as integer
        for my $OneParam ( @{ $Param{$ArrayParam} } ) {
            $OneParam = $Kernel::OM->Get('Kernel::System::DB')->Quote( $OneParam, 'Integer' );
        }

        # create string
        my $InString = join q{, }, @{ $Param{$ArrayParam} };

        push @SQLWhere, "$ArrayParams{ $ArrayParam } IN ($InString)";
    }

    # add previous version param
    if ( !$Param{PreviousVersionSearch} ) {
        push @SQLWhere, 'ci.last_version_id = vr.id';
    }

    # create where string
    my $WhereString = @SQLWhere ? ' WHERE ' . join q{ AND }, @SQLWhere : '';

    # set limit, quote as integer
    if ( $Param{Limit} ) {
        $Param{Limit} = $Kernel::OM->Get('Kernel::System::DB')->Quote( $Param{Limit}, 'Integer' );
    }

    # add the order by columns also to the selected columns
    my $OrderBySelectString = '';
    if (@OrderBySelectColumns) {
        $OrderBySelectString = join ', ', @OrderBySelectColumns;
        $OrderBySelectString = ', ' . $OrderBySelectString;
    }

    # build SQL
    my $SQL = "SELECT DISTINCT vr.configitem_id $OrderBySelectString "
        . 'FROM configitem ci, configitem_version vr '
        . $WhereString;

    # add the ORDER BY clause
    if (@SQLOrderBy) {
        $SQL .= ' ORDER BY ';
        $SQL .= join ', ', @SQLOrderBy;
        $SQL .= ' ';
    }

    # ask the database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL   => $SQL,
        Limit => $Param{Limit},
    );

    # fetch the result
    my @ConfigItemList;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        push @ConfigItemList, $Row[0];
    }

    return \@ConfigItemList;
}

=head1 INTERNAL INTERFACE

=head2 _GetEvents()

This method checks what values were changed and what events have to be triggered.
It returns a hash reference with all event names as keys that should be triggered.

    my $Events = $CIObject->_GetEvents(
        Param => {
            DeplStateID => 123,
        },
        ConfigItemInfo => {
            DeplStateID => 234,
        },
    );

    print keys %{$Events}; # prints "DeploymentStateUpdate"

=cut

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

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

    my $Events = {};

    # check old and new name
    my $OldName = $Param{ConfigItemInfo}->{Name} || '';
    my $NewName = $Param{Param}->{Name}          || '';

    if ( $OldName ne $NewName ) {
        $Events->{NameUpdate} = $NewName . '%%' . $OldName;
    }

    # if depl_state is updated
    my $LastDeplStateID = $Param{ConfigItemInfo}->{DeplStateID} || '';
    my $CurDeplStateID  = $Param{Param}->{DeplStateID}          || '';

    if ( $LastDeplStateID ne $CurDeplStateID ) {
        $Events->{DeploymentStateUpdate} = $CurDeplStateID . '%%' . $LastDeplStateID;
    }

    # if incistate is updated
    my $LastInciStateID = $Param{ConfigItemInfo}->{InciStateID} || '';
    my $CurInciStateID  = $Param{Param}->{InciStateID}          || '';

    if ( $LastInciStateID ne $CurInciStateID ) {
        $Events->{IncidentStateUpdate} = $CurInciStateID . '%%' . $LastInciStateID;
    }

    # check old and new definition_id
    my $OldDefinitionID = $Param{ConfigItemInfo}->{DefinitionID} || '';
    my $NewDefinitionID = $Param{Param}->{DefinitionID}          || '';

    if ( $OldDefinitionID ne $NewDefinitionID ) {
        $Events->{DefinitionUpdate} = $NewDefinitionID;
    }

    # store if anything like Name, DeploymentState, IncidentState, Definition has been changed.
    my $NothingElseChanged;
    if ( !%{$Events} ) {
        $NothingElseChanged = 1;
    }

    # check for changes in XML data
    if ( $Param{Param}->{XMLData} && ref $Param{Param}->{XMLData} eq 'ARRAY' ) {
        my %UpdateValues = $Self->_FindChangedXMLValues(
            ConfigItemID       => $Param{Param}->{ConfigItemID},
            NewXMLData         => $Param{Param}->{XMLData},
            DefinitionID       => $NewDefinitionID,
            NothingElseChanged => $NothingElseChanged,
        );

        if (%UpdateValues) {

            # if only attributes have changed that should not create a new version
            if ( $UpdateValues{UpdateLastVersion} && $NothingElseChanged ) {

                $Events->{UpdateLastVersion} = 1;

                # Clone the data structure, so we can delete the origial.
                $Events->{UpdateXMLData} = $Kernel::OM->Get('Kernel::System::Storable')->Clone(
                    Data => $UpdateValues{UpdateXMLData},
                );

                # We need to delete the flag and data again, to not cause any problems later
                #   when processing the values.
                delete $UpdateValues{UpdateLastVersion};
                delete $UpdateValues{UpdateXMLData};
            }

            $Events->{ValueUpdate} = \%UpdateValues;
        }
    }

    return $Events;
}

=head2 _EventHandlerForChangedXMLValues()

This method calls the event handler for each changed value of the config item.
The changed values are passed in C<UpdateValues> as an hashref with tag-keys as keys.
Please note that this only handles values inside the XML structure, not general
attributes like C<CurInciState>.

    my $Success = $ConfigItemObject->_EventHandlerForChangedXMLValues(
        ConfigItemID => 123,
        UpdateValues =>
            {
               "[1]{'Version'}[1]{'Vendor'}[1]" => 'OldVendor%%NewVendor',
               "[1]{'Version'}[1]{'Type'}[1]"   => '127%%128',
            }
        UserID       => 1,
    );

=cut

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

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

    # trigger ValueUpdate event for each changed value
    for my $Key ( sort keys %{ $Param{UpdateValues} } ) {
        $Self->EventHandler(
            Event => 'ValueUpdate',
            Data  => {
                ConfigItemID => $Param{ConfigItemID},
                Comment      => $Key . '%%' . $Param{UpdateValues}->{$Key},
            },
            UserID => $Param{UserID},
        );
    }

    return 1;
}

=head2 _FindChangedXMLValues()

compares the new xml data C<NewXMLData> with the xml data of the latest version
of the config item C<ConfigItemID>. Note that the new XML data does not contain tag keys.
All values of the two data sets are compared.
When a changed value is encountered, the tag key and the old and the new value are stored in a hash.
The hash with the updated values is returned.

    my %UpdateValues = $ConfigItemObject->_FindChangedXMLValues(
        ConfigItemID => 123,
        NewXMLData   =>
            [
                undef,
                {
                    'Version' =>
                        [
                            undef,
                            {
                                'Owner' =>
                                    [
                                       undef,
                                       {
                                           'Content' => ''
                                       },
                                    ],
                            },
                        ],
                },
            ],
    );

The returned hash looks like:

    %UpdateValues = (
       "[1]{'Version'}[1]{'Vendor'}[1]" => 'OldVendor%%NewVendor',
       "[1]{'Version'}[1]{'Type'}[1]"   => '127%%128',
    );

The key is a tag key. The values contains the old and the new XML value.

=cut

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

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

    # get a list with all versionnumbers that exist for the
    # given config item
    my $VersionList = $Self->VersionList(
        ConfigItemID => $Param{ConfigItemID},
    );

    # skip the check if this is the first version of the item
    return if !@{$VersionList};

    # get old version
    my $OldVersion = $Self->VersionGet(
        VersionID => $VersionList->[-1],
    );

    # The short names for new and old xml data are used in the 'eval' below,
    #    and we do not want to modify the original parameter data NewXMLData
    #    due to side effects of Perls autovivification feature used later here
    #    in another eval statement.
    my $NewXMLData = Storable::dclone( $Param{NewXMLData} );
    my $OldXMLData = $OldVersion->{XMLData};

    # get xml definition
    my $Definition = $Self->DefinitionGet(
        DefinitionID => $Param{DefinitionID},
    );

    # get all tagkeys in new and old XML data
    # use a side effect of XMLHash2D(), which adds the tag keys to the passed in data structure
    $Kernel::OM->Get('Kernel::System::XML')->XMLHash2D( XMLHash => $NewXMLData );
    my @TagKeys = $Self->_GrabTagKeys( Data => [ $OldXMLData, $NewXMLData ] );

    # get an unique list of all tag keys
    my %UniqueTagKeys = map { $_ => 1 } @TagKeys;

    my %UpdateValues;
    my %SuppressVersionAdd;

    for my $TagKey ( sort keys %UniqueTagKeys ) {
        my $NewContent = eval '$NewXMLData->' . $TagKey . '->{Content}' || '';    ## no critic
        my $OldContent = eval '$OldXMLData->' . $TagKey . '->{Content}' || '';    ## no critic

        if ( $NewContent ne $OldContent ) {

            # extract attribute path name
            my $AttributePath = $TagKey;
            $AttributePath =~ s{ \A \[.*?\] \{'Version'\} \[.*?\] \{' }{}xms;
            $AttributePath =~ s{ '\} \[.*?\] \{' }{::}xmsg;
            $AttributePath =~ s{ '\} \[.*?\] \z }{}xms;

            # get definition info about attribute
            my $AttributeInfo = $Self->DefinitionAttributeInfo(
                Definition    => $Definition->{DefinitionRef},
                AttributePath => $AttributePath,
            );

            # check if this attribute should be suppressed and nothing else of
            #    Name, DeploymentState, IncidentState, Definition has been changed.
            if ( $AttributeInfo->{SuppressVersionAdd} && $Param{NothingElseChanged} ) {

                if ( $AttributeInfo->{SuppressVersionAdd} eq 'UpdateLastVersion' ) {
                    $SuppressVersionAdd{UpdateLastVersion}->{$TagKey} = join '%%', $OldContent, $NewContent;

                    # Build a new data structire only with the update values.
                    eval '$SuppressVersionAdd{UpdateXMLData}->' . $TagKey . '->{Content} = $NewContent';    ## no critic
                }
                elsif ( $AttributeInfo->{SuppressVersionAdd} eq 'Ignore' ) {
                    $SuppressVersionAdd{Ignore}->{$TagKey} = join '%%', $OldContent, $NewContent;
                }
            }

            # change was found
            $UpdateValues{$TagKey} = join '%%', $OldContent, $NewContent;
        }
    }

    # count how many of each SuppressVersionAdd types we have.
    my $UpdateValuesCount      = scalar keys %UpdateValues;
    my $IgnoreCount            = scalar keys %{ $SuppressVersionAdd{Ignore} };
    my $UpdateLastVersionCount = scalar keys %{ $SuppressVersionAdd{UpdateLastVersion} };

    if (%UpdateValues) {

        # Update values contains only values that should be ignored.
        if ( $UpdateValuesCount == $IgnoreCount ) {
            %UpdateValues = ();
        }

        # Update values contains only values that should update the last version or should be ignored,
        elsif (
            ( $UpdateValuesCount == $UpdateLastVersionCount )
            || ( $UpdateValuesCount == ( $UpdateLastVersionCount + $IgnoreCount ) )
            )
        {
            %UpdateValues                    = %{ $SuppressVersionAdd{UpdateLastVersion} };
            $UpdateValues{UpdateXMLData}     = $SuppressVersionAdd{UpdateXMLData};
            $UpdateValues{UpdateLastVersion} = 1;
        }
    }

    return %UpdateValues;
}

=head2 _GrabTagKeys()

recursively scans a perl data structure for the hash key 'TagKey' and returns a
list of all the values for that key.

    my @TagKeys = $ConfigItemObject->_GrabTagKeys(
        Data => $XMLHashReferenz,
    );

=cut

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

    return () if !$Param{Data};

    my @TagKeys;
    if ( ref $Param{Data} eq 'ARRAY' ) {

        ELEM:
        for my $Elem ( @{ $Param{Data} } ) {

            next ELEM if !$Elem;
            next ELEM if !ref $Elem;

            push @TagKeys, $Self->_GrabTagKeys( Data => $Elem );
        }
    }
    elsif ( ref $Param{Data} eq 'HASH' ) {

        for my $Key ( sort keys %{ $Param{Data} } ) {

            if ( $Key eq 'TagKey' ) {
                push @TagKeys, $Param{Data}->{$Key};
            }
            elsif ( ref $Param{Data}->{$Key} ) {
                push @TagKeys, $Self->_GrabTagKeys( Data => $Param{Data}->{$Key} );
            }
        }
    }

    return @TagKeys;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::ITSMConfigItem::XML;

use strict;
use warnings;

our $ObjectManagerDisabled = 1;

=head1 NAME

Kernel::System::ITSMConfigItem::XML - sub module of Kernel::System::ITSMConfigItem

=head1 DESCRIPTION

All xml functions.

=head1 PUBLIC INTERFACE

=head2 XMLValueLookup()

lookup a xml value

    my $Value = $ConfigItemObject->XMLValueLookup(
        Item  => $ItemRef,
        Value => 5,
    );

=cut

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

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

    # load backend
    my $BackendObject = $Kernel::OM->Get(
        'Kernel::System::ITSMConfigItem::XML::Type::' . $Param{Item}->{Input}->{Type}
    );

    return '' if !$BackendObject;

    # lookup item value
    my $Value = $BackendObject->ValueLookup(%Param);

    return $Value;
}

=head2 XMLStatsAttributeCreate()

create a attribute array for the stats framework

    my $Value = $ConfigItemObject->XMLStatsAttributeCreate(
        Item => $ItemRef,
    );

=cut

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

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

    # load backend
    my $BackendObject = $Kernel::OM->Get(
        'Kernel::System::ITSMConfigItem::XML::Type::' . $Param{Item}->{Input}->{Type}
    );

    return if !$BackendObject;

    # create stats attribute array
    my $Attribute = $BackendObject->StatsAttributeCreate(%Param);

    return $Attribute;
}

=head2 XMLExportSearchValuePrepare()

prepare xml search data for export

    my $ArrayRef = $ConfigItemObject->XMLExportSearchValuePrepare(
        Item  => $ItemRef,
        Value => 5,
    );

=cut

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

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

    # load backend
    my $BackendObject = $Kernel::OM->Get(
        'Kernel::System::ITSMConfigItem::XML::Type::' . $Param{Item}->{Input}->{Type}
    );

    return if !$BackendObject;

    # prepare value
    my $Array = $BackendObject->ExportSearchValuePrepare(%Param);

    return $Array;
}

=head2 XMLExportValuePrepare()

prepare xml data for export

    my $Value = $ConfigItemObject->XMLExportValuePrepare(
        Item  => $ItemRef,
        Value => 5,
    );

=cut

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

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

    # load backend
    my $BackendObject = $Kernel::OM->Get(
        'Kernel::System::ITSMConfigItem::XML::Type::' . $Param{Item}->{Input}->{Type}
    );

    return if !$BackendObject;

    # prepare value
    my $Value = $BackendObject->ExportValuePrepare(%Param);

    return $Value;
}

=head2 XMLImportSearchValuePrepare()

prepare xml search data for import

    my $ArrayRef = $ConfigItemObject->XMLImportSearchValuePrepare(
        Item  => $ItemRef,
        Value => 5,
    );

=cut

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

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

    # load backend
    my $BackendObject = $Kernel::OM->Get(
        'Kernel::System::ITSMConfigItem::XML::Type::' . $Param{Item}->{Input}->{Type}
    );

    return if !$BackendObject;

    # prepare value
    my $Array = $BackendObject->ImportSearchValuePrepare(%Param);

    return $Array;
}

=head2 XMLImportValuePrepare()

prepare xml data for import

    my $Value = $ConfigItemObject->XMLImportValuePrepare(
        Item  => $ItemRef,
        Value => 5,
    );

=cut

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

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

    # load backend
    my $BackendObject = $Kernel::OM->Get(
        'Kernel::System::ITSMConfigItem::XML::Type::' . $Param{Item}->{Input}->{Type}
    );

    return if !$BackendObject;

    # prepare value
    my $Value = $BackendObject->ImportValuePrepare(%Param);

    return $Value;
}

=head2 _XMLVersionSearch()

Search xml data of a version and return a hash reference.
The C<What> parameter is a bit like the parameter used in L<SQL::Abstract>.
The returned hash reference has C<VersionID>s as keys and C<1> as value.

    my $VersionIDs = $ConfigItemObject->_XMLVersionSearch(
        ClassIDs => [1, 2, 3],  # (optional)

        What => [
            # each array element is a and condition
            {
                # or condition in hash
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => '%contentA%',
                "[%]{'Version'}[%]{'ConfigItemAttrC'}[%]{'Content'}" => '%contentA%',
            },
            {
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => '%contentB%',
                "[%]{'Version'}[%]{'ConfigItemAttrC'}[%]{'Content'}" => '%contentB%',
            },
            {
                # use array reference if different content with same key was searched
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => [
                    '%contentC%',
                    '%contentD%',
                    '%contentE%',
                ],
                "[%]{'Version'}[%]{'ConfigItemAttrC'}[%]{'Content'}" => [
                    '%contentC%',
                    '%contentD%',
                    '%contentE%',
                ],
            },
            {
                # use hash reference for specifying comparison ops, apart from the default 'LIKE'
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => { '<'        => 'alphabetically_lower_or_equal' },
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => { '<='       => 'alphabetically_less_or_equal' },
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => { '='        => 'exact_match' },
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => { '>='       => 'alphabetically_larger_or_equal' },
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => { '>'        => 'alphabetically_larger' },
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => { '-between' => [ 'lower_bound', 'upper_bound' ] },
            },
        ],

        PreviousVersionSearch => 1,  # (optional) default 0 (0|1)
    );

=cut

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

    # check needed stuff
    if ( !$Param{What} || ref $Param{What} ne 'ARRAY' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need What as array reference!",
        );
        return;
    }

    if ( !$Param{ClassIDs} || ref $Param{ClassIDs} ne 'ARRAY' || !@{ $Param{ClassIDs} } ) {

        # get class list
        my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
            Class => 'ITSM::ConfigItem::Class',
        );

        @{ $Param{ClassIDs} } = keys %{$ClassList};
    }

    # search in active versions
    my %VersionIDs;
    for my $ClassID ( @{ $Param{ClassIDs} } ) {

        # start xml search
        my @Keys = $Self->_XMLHashSearch(
            Type => "ITSM::ConfigItem::$ClassID",
            What => $Param{What},
        );

        # add all ids to version id hash
        for my $VersionID (@Keys) {
            $VersionIDs{$VersionID} = 1;
        }
    }

    return \%VersionIDs if !$Param{PreviousVersionSearch};

    # search also in old versions (archiv)
    for my $ClassID ( @{ $Param{ClassIDs} } ) {

        # start xml search
        my @Keys = $Self->_XMLHashSearch(
            Type => "ITSM::ConfigItem::Archiv::$ClassID",
            What => $Param{What},
        );

        # add all ids to version id hash
        for my $VersionID (@Keys) {
            $VersionIDs{$VersionID} = 1;
        }
    }

    return \%VersionIDs;
}

=head2 _XMLVersionGet()

get the xml data of a version

    my $ArrayRef = $ConfigItemObject->_XMLVersionGet(
        ClassID   => 1,
        VersionID => 123,
    );

=cut

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

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

    # get version
    my @XML = $Kernel::OM->Get('Kernel::System::XML')->XMLHashGet(
        Type => "ITSM::ConfigItem::$Param{ClassID}",
        Key  => $Param{VersionID},
    );

    return \@XML if @XML;

    # get version from archiv
    @XML = $Kernel::OM->Get('Kernel::System::XML')->XMLHashGet(
        Type => "ITSM::ConfigItem::Archiv::$Param{ClassID}",
        Key  => $Param{VersionID},
    );

    return \@XML;
}

=head2 _XMLVersionAdd()

add the xml data of a new version and move old xml versions in archive class

    my $XMLID = $ConfigItemObject->_XMLVersionAdd(
        ClassID      => 1,
        ConfigItemID => 222,
        VersionID    => 123,
        XMLData      => $ArrayRef,
    );

=cut

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

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

    # add xml version
    my $XMLID = $Kernel::OM->Get('Kernel::System::XML')->XMLHashAdd(
        Type    => "ITSM::ConfigItem::$Param{ClassID}",
        Key     => $Param{VersionID},
        XMLHash => $Param{XMLData},
    );

    # get last versions
    my $OldVersionIDs = $Self->VersionList(
        ConfigItemID => $Param{ConfigItemID},
    );

    # Find Version to move
    my $MoveVersion;
    OLDVERSIONID:
    for my $OldVersionID ( @{$OldVersionIDs} ) {
        last OLDVERSIONID if $OldVersionID eq $Param{VersionID};

        $MoveVersion = $OldVersionID;
    }

    return $XMLID if !$MoveVersion;

    # move old version in archiv
    $Kernel::OM->Get('Kernel::System::XML')->XMLHashMove(
        OldType => "ITSM::ConfigItem::$Param{ClassID}",
        OldKey  => $MoveVersion,
        NewType => "ITSM::ConfigItem::Archiv::$Param{ClassID}",
        NewKey  => $MoveVersion,
    );

    return $XMLID;
}

=head2 _XMLVersionDelete()

delete the xml data of a version

    my $True = $ConfigItemObject->_XMLVersionDelete(
        VersionID => 123,
        UserID    => 1,
    );

=cut

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

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

    # get version data
    my $Version = $Self->VersionGet(
        VersionID => $Param{VersionID},
    );

    return if !$Version;

    # delete xml data from a version
    $Kernel::OM->Get('Kernel::System::XML')->XMLHashDelete(
        Type => "ITSM::ConfigItem::$Version->{ClassID}",
        Key  => $Param{VersionID},
    );

    # delete xml data from archiv
    $Kernel::OM->Get('Kernel::System::XML')->XMLHashDelete(
        Type => "ITSM::ConfigItem::Archiv::$Version->{ClassID}",
        Key  => $Param{VersionID},
    );

    return 1;
}

=head2 _XMLHashSearch()

Search a xml hash from database.
This method is based on Kernel::System::XMLHashSearch, but has support for some extra features.
A specific operator can be specified in a hash.
The syntax is based on L<SQL::Abstract>.

    my @Keys = $ConfigItemObject->_XMLHashSearch(
        Type => 'SomeType',
        What => [
            # each array element is a and condition
            {
                # or condition in hash
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => '%contentA%',
                "[%]{'Version'}[%]{'ConfigItemAttrC'}[%]{'Content'}" => '%contentA%',
            },
            {
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => '%contentB%',
                "[%]{'Version'}[%]{'ConfigItemAttrC'}[%]{'Content'}" => '%contentB%',
            },
            {
                # use array reference if different content with same key was searched
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => ['%contentC%', '%contentD%', '%contentE%'],
                "[%]{'Version'}[%]{'ConfigItemAttrC'}[%]{'Content'}" => ['%contentC%', '%contentD%', '%contentE%'],
            },
            {
                # use hash reference for specifying comparison ops, apart from the default 'LIKE'
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => { '<'         => 'alphabetically_lower_or_equal' },
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => { '<='        => 'alphabetically_less_or_equal' },
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => { '='         => 'exact_match' },
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => { '!='        => 'exact_match' },
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => { '>='        => 'alphabetically_larger_or_equal' },
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => { '>'         => 'alphabetically_larger' },
                "[%]{'Version'}[%]{'ConfigItemAttrB'}[%]{'Content'}" => { '-between'  => [ 'lower_bound', 'upper_bound' ] },
            },
        ],
    );

=cut

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

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

    # get like escape string needed for some databases (e.g. oracle)
    my $LikeEscapeString = $Kernel::OM->Get('Kernel::System::DB')->GetDatabaseFunction('LikeEscapeString');

    # get all entries if we have no restriction
    if ( !$Param{What} || ref $Param{What} ne 'ARRAY' ) {
        return if !$Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL  => 'SELECT DISTINCT(xml_key) FROM xml_storage WHERE xml_type = ?',
            Bind => [ \$Param{Type} ],
        );

        my @Keys;
        while ( my @Data = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
            push @Keys, $Data[0];
        }

        return @Keys;
    }

    # otherwise only get entries restricted by 'What' parameter

    my %OpIsSupported = map { $_ => 1 } ( '<', '<=', '=', '!=', '>=', '>', '-between' );

    # the array elements are 'and' combined
    my $Intersect = 0;
    my %Hash;
    for my $And ( @{ $Param{What} } ) {

        # the key/value pairs are 'or' combined
        my @OrConditions;
        for my $Key ( sort keys %{$And} ) {
            my $Value = $And->{$Key};

            $Self->_PrepareLikeString( \$Key );

            if ( $Value && ref $Value eq 'ARRAY' ) {

                # when an array of possible values is given,
                # we use 'LIKE'-conditions and combine them with 'OR'
                for my $Element ( @{$Value} ) {

                    $Self->_PrepareLikeString( \$Element );

                    push @OrConditions,
                        " (xml_content_key LIKE '$Key' $LikeEscapeString "
                        . "AND xml_content_value LIKE '$Element' $LikeEscapeString)";
                }
            }
            elsif ( $Value && ref $Value eq 'HASH' ) {

                # a hashref indicates a specific comparison op
                # currently only a single op, with a single value, is supported

                # Under Oracle the attribute 'xml_content_value' is a CLOB,
                # a Character Locator Object. While selection with LIKE is possible,
                # the alphabetical comparison ops are not supported.
                # See http://download.oracle.com/docs/cd/B12037_01/appdev.101/b10796/\
                # adlob_sq.htm#1006215
                # As a workaround we cast the CLOB to a VARCHAR2 with TO_CHAR().
                my $XMLContentValueColumn = 'xml_content_value';
                if (
                    $Kernel::OM->Get('Kernel::System::DB')->GetDatabaseFunction('Type') eq
                    'oracle'
                    )
                {
                    $XMLContentValueColumn = 'TO_CHAR(xml_content_value)';
                }

                my ($Op) = keys %{$Value};
                my $Element = $Value->{$Op};
                if ( $Op && $Op eq '-between' && ref $Element eq 'ARRAY' ) {
                    my $LowerBound = $Kernel::OM->Get('Kernel::System::DB')->Quote( $Element->[0] );
                    my $UpperBound = $Kernel::OM->Get('Kernel::System::DB')->Quote( $Element->[1] );
                    push @OrConditions,
                        " ( xml_content_key LIKE '$Key' $LikeEscapeString "
                        . "AND $XMLContentValueColumn >= '$LowerBound' "
                        . "AND $XMLContentValueColumn <= '$UpperBound' )";
                }
                elsif ( $Op && $OpIsSupported{$Op} && !ref $Element ) {
                    $Element = $Kernel::OM->Get('Kernel::System::DB')->Quote($Element) // '';
                    push @OrConditions,
                        " ( xml_content_key LIKE '$Key' $LikeEscapeString "
                        . "AND $XMLContentValueColumn $Op '$Element' )";
                }
                else {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => 'Got unexpected data in search!',
                    );
                    push @OrConditions, '( 1 = 1 )';
                }
            }
            else {

                # when a single  possible value is given,
                # we use a 'LIKE'-condition
                $Self->_PrepareLikeString( \$Value );

                push @OrConditions,
                    " (xml_content_key LIKE '$Key' $LikeEscapeString "
                    . "AND xml_content_value LIKE '$Value' $LikeEscapeString)";
            }
        }

        # assemble the SQL
        my $SQL = 'SELECT DISTINCT(xml_key) FROM xml_storage WHERE xml_type = ? ';
        if (@OrConditions) {
            $SQL .= 'AND ( ' . join( ' OR ', @OrConditions ) . ' )';
        }

        # execute
        $Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL  => $SQL,
            Bind => [ \$Param{Type} ],
        );

        # intersection between the current key set, and the keys from the last 'SELECT'
        # only the keys which are in all results survive
        my %HashNew;
        while ( my @Data = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
            if ( !$Intersect || $Hash{ $Data[0] } ) {
                $HashNew{ $Data[0] } = 1;
            }
        }
        %Hash = %HashNew;

        # remember to later intersect results
        ++$Intersect;
    }

    my @Keys = keys %Hash;

    return @Keys;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KIyMgbm9maWx0ZXIoVGlkeUFsbDo6UGx1Z2luOjpabnVueTo6Q29kZVN0eWxlOjpHdWFyZENsYXVzZSkKCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OkJhc2VTZWxlY3RhYmxlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06OkxvZycsCik7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OkJhc2VTZWxlY3RhYmxlIC0gQmFzZSBiYWNrZW5kIG1vZHVsZSBmb3Igc2VsZWN0LWFibGUgdmFsdWVzCgo9aGVhZDEgU1lOT1BTSVMKCkJhc2UgYmFja2VuZCBtb2R1bGUgZm9yIHNlbGVjdC1hYmxlIHZhbHVlcwoKPWhlYWQyIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgbXkgJEJhY2tlbmRPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6QmFzZVNlbGVjdGFibGUnKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aGVhZDIgVmFsdWVMb29rdXAoKQoKZ2V0IHRoZSB4bWwgZGF0YSBvZiBhIHZlcnNpb24KCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+VmFsdWVMb29rdXAoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBWYWx1ZUxvb2t1cCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiAnJzsKfQoKPWhlYWQyIFN0YXRzQXR0cmlidXRlQ3JlYXRlKCkKCmNyZWF0ZSBhIGF0dHJpYnV0ZSBhcnJheSBmb3IgdGhlIHN0YXRzIGZyYW1ld29yawoKICAgIG15ICRBdHRyaWJ1dGUgPSAkQmFja2VuZE9iamVjdC0+U3RhdHNBdHRyaWJ1dGVDcmVhdGUoCiAgICAgICAgS2V5ICA9PiAnS2V5OjpTdWJrZXknLAogICAgICAgIE5hbWUgPT4gJ05hbWUnLAogICAgICAgIEl0ZW0gPT4gJEl0ZW1SZWYsCiAgICApOwoKPWN1dAoKc3ViIFN0YXRzQXR0cmlidXRlQ3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJExvZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgQVJHVU1FTlQ6CiAgICBmb3IgbXkgJEFyZ3VtZW50IChxdyhLZXkgTmFtZSBJdGVtKSkgewogICAgICAgIG5leHQgQVJHVU1FTlQgaWYgJFBhcmFteyRBcmd1bWVudH07CgogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gIk5lZWQgJEFyZ3VtZW50ISIsCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBjcmVhdGUgYXR0cmlidXRlCiAgICBteSAkQXR0cmlidXRlID0gWwogICAgICAgIHsKICAgICAgICAgICAgTmFtZSAgICAgICAgICAgICA9PiAkUGFyYW17TmFtZX0sCiAgICAgICAgICAgIFVzZUFzWHZhbHVlICAgICAgPT4gMSwKICAgICAgICAgICAgVXNlQXNWYWx1ZVNlcmllcyA9PiAxLAogICAgICAgICAgICBVc2VBc1Jlc3RyaWN0aW9uID0+IDEsCiAgICAgICAgICAgIEVsZW1lbnQgICAgICAgICAgPT4gJFBhcmFte0tleX0sCiAgICAgICAgICAgIEJsb2NrICAgICAgICAgICAgPT4gJ0lucHV0RmllbGQnLAogICAgICAgIH0sCiAgICBdOwoKICAgIHJldHVybiAkQXR0cmlidXRlOwp9Cgo9aGVhZDIgRXhwb3J0U2VhcmNoVmFsdWVQcmVwYXJlKCkKCnByZXBhcmUgc2VhcmNoIHZhbHVlIGZvciBleHBvcnQKCiAgICBteSAkQXJyYXlSZWYgPSAkQmFja2VuZE9iamVjdC0+RXhwb3J0U2VhcmNoVmFsdWVQcmVwYXJlKAogICAgICAgIFZhbHVlID0+IDExLCAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgRXhwb3J0U2VhcmNoVmFsdWVQcmVwYXJlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIGlmICFkZWZpbmVkICRQYXJhbXtWYWx1ZX07CiAgICByZXR1cm4gJFBhcmFte1ZhbHVlfTsKfQoKPWhlYWQyIEV4cG9ydFZhbHVlUHJlcGFyZSgpCgpwcmVwYXJlIHZhbHVlIGZvciBleHBvcnQKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+RXhwb3J0VmFsdWVQcmVwYXJlKAogICAgICAgIFZhbHVlID0+IDExLCAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgRXhwb3J0VmFsdWVQcmVwYXJlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIGlmICFkZWZpbmVkICRQYXJhbXtWYWx1ZX07CiAgICByZXR1cm4gJFBhcmFte1ZhbHVlfTsKfQoKPWhlYWQyIEltcG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgpCgpwcmVwYXJlIHNlYXJjaCB2YWx1ZSBmb3IgaW1wb3J0CgogICAgbXkgJEFycmF5UmVmID0gJEJhY2tlbmRPYmplY3QtPkltcG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEltcG9ydFNlYXJjaFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBJbXBvcnRWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSB2YWx1ZSBmb3IgaW1wb3J0CgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkltcG9ydFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEltcG9ydFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCjE7Cg==
# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --
## nofilter(TidyAll::Plugin::Znuny::CodeStyle::GuardClause)

package Kernel::System::ITSMConfigItem::XML::Type::CI;

use strict;
use warnings;

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

our @ObjectDependencies = (
    'Kernel::System::GeneralCatalog',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::Log'
);

=head1 NAME

Kernel::System::ITSMConfigItem::XML::Type::CI - xml backend module for CI objects

=head1 SYNOPSIS

All xml functions of CI objects

=head2 new()

create an object

    my $BackendObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem::XML::Type::CI');

=cut

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

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

    return $Self;
}

=head2 ValueLookup()

get the xml data of a version

    my $Value = $BackendObject->ValueLookup(
        Value => 1, # (optional)
    );

=cut

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

    my $ITSMConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    return '' if !$Param{Value};

    my $CIVersionData = $ITSMConfigItemObject->VersionGet(
        ConfigItemID => $Param{Value},
        XMLDataGet   => 0,
    );
    my $CIName = $Param{Value};

    #if ( $CIVersionData && ( ref($CIVersionData) eq 'HASH' ) && $CIVersionData->{Name})
    return $CIName if !IsHashRefWithData($CIVersionData);
    return $CIName if !$CIVersionData->{Name};

    $CIName = $CIVersionData->{Name} . " (" . $CIVersionData->{Number} . ")";

    return $CIName;
}

=head2 StatsAttributeCreate()

create a attribute array for the stats framework

    my $Attribute = $BackendObject->StatsAttributeCreate(
        Key  => 'Key::Subkey',
        Name => 'Name',
        Item => $ItemRef,
    );

=cut

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

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    # check needed stuff
    PARAM:
    for my $Argument (qw(Key Name Item)) {
        next PARAM if $Param{$Argument};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Need $Argument!"
        );
        return;
    }

    # create arrtibute
    my $Attribute = [
        {
            Name             => $Param{Name},
            UseAsXvalue      => 0,
            UseAsValueSeries => 0,
            UseAsRestriction => 1,
            Element          => $Param{Key},
            Block            => 'InputField',
        },
    ];

    return $Attribute;
}

=head2 ExportSearchValuePrepare()

prepare search value for export

    my $ArrayRef = $BackendObject->ExportSearchValuePrepare(
        Value => 1, # (optional)
    );

=cut

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

    my $ITSMConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    return    if !defined $Param{Value};
    return '' if !$Param{Value};

    # lookup CI number
    my $CI = $ITSMConfigItemObject->ConfigItemGet(
        ConfigItemID => $Param{Value},
    );

    #if ( $CI && ref $CI eq 'HASH' && $CI->{Number} ) {
    if ( IsHashRefWithData($CI) && $CI->{Number} ) {
        return $CI->{Number};
    }

    return '';
}

=head2 ExportValuePrepare()

prepare value for export

    my $Value = $BackendObject->ExportValuePrepare(
        Value => 11, # (optional)
    );

=cut

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

    my $ITSMConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

    return    if !defined $Param{Value};
    return '' if !$Param{Value};

    my $SearchAttr = $Param{Item}->{Input}->{CIAttributeKey} || '';

    if ($SearchAttr) {

        my $VersionData = $ITSMConfigItemObject->VersionGet(
            ConfigItemID => $Param{Value},
        );

        #if ( $VersionData && ref $VersionData eq 'HASH' ) {
        if ( IsHashRefWithData($VersionData) ) {

            # get ConfigItem class ID
            my $CIClassID = "";

            if (
                IsHashRefWithData( $Param{Item} )
                && IsHashRefWithData( $Param{Item}->{Input} )
                && $Param{Item}->{Input}->{CIClassName}
                )
            {

                my $ItemDataRef = $GeneralCatalogObject->ItemGet(
                    Class => 'ITSM::ConfigItem::Class',
                    Name  => $Param{Item}->{Input}->{CIClassName} || '',
                );

                # if ( $ItemDataRef && ref($ItemDataRef) eq 'HASH' && $ItemDataRef->{ItemID} ) {
                if ( IsHashRefWithData($ItemDataRef) && $ItemDataRef->{ItemID} ) {
                    $CIClassID = $ItemDataRef->{ItemID} || '';
                }

                my $XMLDefinition =
                    $ITSMConfigItemObject->DefinitionGet(
                    ClassID => $CIClassID,
                    );

                my $ArrRef = $Self->{CIACUtilsObject}->GetAttributeValuesByKey(
                    KeyName       => $SearchAttr,
                    XMLData       => $VersionData->{XMLData}->[1]->{Version}->[1],
                    XMLDefinition => $XMLDefinition->{DefinitionRef},
                );

                if ( $ArrRef && $ArrRef->[0] ) {
                    return $ArrRef->[0];
                }
            }
        }
    }

    # lookup CI number for given CI ID
    my $CI = $ITSMConfigItemObject->ConfigItemGet(
        ConfigItemID => $Param{Value},
    );

    #if ( $CI && ref $CI eq 'HASH' && $CI->{Number} ) {
    if ( IsHashRefWithData($CI) && $CI->{Number} ) {
        return $CI->{Number};
    }

    return '';
}

=head2 ImportSearchValuePrepare()

prepare search value for import

    my $ArrayRef = $BackendObject->ImportSearchValuePrepare(
        Value => 11, # (optional)
    );

=cut

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

    my $ITSMConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    return    if !defined $Param{Value};
    return '' if !$Param{Value};

    # check if CI number was given
    my $CIID = $ITSMConfigItemObject->ConfigItemLookup(
        ConfigItemNumber => $Param{Value},
    );
    return $CIID if $CIID;

    # check if given value is a valid CI ID
    if ( $Param{Value} !~ /\D/ ) {
        my $CINumber = $ITSMConfigItemObject->ConfigItemLookup(
            ConfigItemID => $Param{Value},
        );
        return $Param{Value} if $CINumber;
    }

    return '';
}

=head2 ImportValuePrepare()

prepare value for import

    my $Value = $BackendObject->ImportValuePrepare(
        Value => 11, # (optional)
    );

=cut

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

    my $ITSMConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

    return    if !defined $Param{Value};
    return '' if !$Param{Value};

    my $SearchAttr = $Param{Item}->{Input}->{CIAttributeKey} || '';

    # make CI-Number out of given value
    if ($SearchAttr) {

        # get ConfigItem class ID
        my $CIClassID = "";

        if (
            IsHashRefWithData( $Param{Item} )
            && IsHashRefWithData( $Param{Item}->{Input} )
            && $Param{Item}->{Input}->{CIClassName}
            )
        {

            my $ItemDataRef = $GeneralCatalogObject->ItemGet(
                Class => 'ITSM::ConfigItem::Class',
                Name  => $Param{Item}->{Input}->{CIClassName} || '',
            );

            if ( IsHashRefWithData($ItemDataRef) && $ItemDataRef->{ItemID} ) {
                $CIClassID = $ItemDataRef->{ItemID} || '';
            }

            # prepare search params
            my %SearchParams = ();
            my %SearchData   = ();

            $SearchData{$SearchAttr} = $Param{Value};

            my $XMLDefinition =
                $ITSMConfigItemObject->DefinitionGet(
                ClassID => $CIClassID,
                );

            my @SearchParamsWhat;
            $Self->_XMLSearchDataPrepare(
                XMLDefinition => $XMLDefinition->{DefinitionRef},
                What          => \@SearchParamsWhat,
                SearchData    => \%SearchData,
            );

            if (@SearchParamsWhat) {
                $SearchParams{What} = \@SearchParamsWhat;
            }

            # search the config items
            my $ConfigItemIDs = $ITSMConfigItemObject->ConfigItemSearchExtended(
                %SearchParams,
                ClassIDs              => [$CIClassID],
                PreviousVersionSearch => 0,
            );

            # get and return CofigItem ID
            my $CIID = "";
            if ( $ConfigItemIDs && ref($ConfigItemIDs) eq 'ARRAY' ) {
                $CIID = $ConfigItemIDs->[0] || '';
            }
            return $CIID;
        }
    }

    my $CIID = $ITSMConfigItemObject->ConfigItemLookup(
        ConfigItemNumber => $Param{Value},
    );
    return $CIID if $CIID;

    # make CI-Number out of given Name...
    my $CIClassID = "";

    if (
        IsHashRefWithData( $Param{Item} )
        && $Param{Item}->{Input}
        && IsHashRefWithData( $Param{Item}->{Input} )
        && $Param{Item}->{Input}->{CIClassName}
        )
    {

        my $RefClassName = $Param{Item}->{Input}->{CIClassName};
        my $ItemDataRef  = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::Class',
            Name  => $Param{Item}->{Input}->{CIClassName} || '',
        );

        if ( IsHashRefWithData($ItemDataRef) && $ItemDataRef->{ItemID} ) {
            $CIClassID = $ItemDataRef->{ItemID} || '';
        }
        my $ConfigItemIDs = $ITSMConfigItemObject->ConfigItemSearchExtended(
            Name     => $Param{Value},
            ClassIDs => [$CIClassID],
        );
        my $CIID = "";
        if ( $ConfigItemIDs && ref($ConfigItemIDs) eq 'ARRAY' ) {
            $CIID = $ConfigItemIDs->[0] || '';
        }
        return $CIID if $CIID;
    }

    # check if given value is a valid CI ID
    if ( $Param{Value} !~ /\D/ ) {
        my $CINumber = $ITSMConfigItemObject->ConfigItemLookup(
            ConfigItemID => $Param{Value},
        );
        return $Param{Value} if $CINumber;
    }

    return '';
}

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

    # check needed stuff
    return if !$Param{XMLDefinition} || ref $Param{XMLDefinition} ne 'ARRAY';
    return if !$Param{What}          || ref $Param{What} ne 'ARRAY';
    return if !$Param{SearchData}    || ref $Param{SearchData} ne 'HASH';

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        # create key
        my $Key =
            $Param{Prefix} ? $Param{Prefix} . '::' . $Item->{Key} : $Item->{Key};

        if ( $Param{SearchData}->{$Key} ) {

            # create search key
            my $SearchKey = $Key;
            $SearchKey =~ s{ :: }{\'\}[%]\{\'}xmsg;

            # create search hash
            my $SearchHash =
                {
                '[1]{\'Version\'}[1]{\''
                    . $SearchKey
                    . '\'}[%]{\'Content\'}' => $Param{SearchData}->{$Key},
                };
            push @{ $Param{What} }, $SearchHash;
        }
        next ITEM if !$Item->{Sub};

        # start recursion, if "Sub" was found
        $Self->_XMLSearchDataPrepare(
            XMLDefinition => $Item->{Sub},
            What          => $Param{What},
            SearchData    => $Param{SearchData},
            Prefix        => $Key,
        );
    }
    return 1;
}

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OkNJQXR0YWNobWVudDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAopOwoKdXNlIHBhcmVudCAnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OkJhc2VTZWxlY3RhYmxlJzsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6Q0lBdHRhY2htZW50IC0geG1sIGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgU1lOT1BTSVMKCkFsbCB4bWwgZnVuY3Rpb25zIG9mIGF0dGFjaG1lbnQgb2JqZWN0cwoKPWhlYWQyIFZhbHVlTG9va3VwKCkKCmdldCB0aGUgeG1sIGRhdGEgb2YgYSB2ZXJzaW9uCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPlZhbHVlTG9va3VwKAogICAgICAgIEl0ZW0gID0+ICRJdGVtUmVmLAogICAgICAgIFZhbHVlID0+IDEuMS4xLjEsCiAgICApOwoKPWN1dAoKc3ViIFZhbHVlTG9va3VwIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OkNJU2ltcGxlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIEtlcm5lbDo6U3lzdGVtOjpWYXJpYWJsZUNoZWNrIHF3KDphbGwpOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXM7Cgp1c2UgcGFyZW50IHF3KCBLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6Q0kgKTsKCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6Q3VzdG9tZXI7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6Q3VzdG9tZXJVc2VyJywKICAgICdLZXJuZWw6OlN5c3RlbTo6TG9nJywKKTsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6Q3VzdG9tZXIgLSB4bWwgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBERVNDUklQVElPTgoKQWxsIHhtbCBmdW5jdGlvbnMgb2YgY3VzdG9tZXIgb2JqZWN0cwoKPWhlYWQyIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgdXNlIEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyOwogICAgbG9jYWwgJEtlcm5lbDo6T00gPSBLZXJuZWw6OlN5c3RlbTo6T2JqZWN0TWFuYWdlci0+bmV3KCk7CiAgICBteSAkWE1MVHlwZUN1c3RvbWVyQmFja2VuZE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6WE1MOjpUeXBlOjpDdXN0b21lcicpOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1oZWFkMiBWYWx1ZUxvb2t1cCgpCgpnZXQgdGhlIHhtbCBkYXRhIG9mIGEgdmVyc2lvbgoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5WYWx1ZUxvb2t1cCgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIFZhbHVlTG9va3VwIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuICcnIGlmICEkUGFyYW17VmFsdWV9OwoKICAgIG15ICVDdXN0b21lclNlYXJjaExpc3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6Q3VzdG9tZXJVc2VyJyktPkN1c3RvbWVyU2VhcmNoKAogICAgICAgIFNlYXJjaCA9PiAkUGFyYW17VmFsdWV9LAogICAgKTsKCiAgICByZXR1cm4gJEN1c3RvbWVyU2VhcmNoTGlzdHsgJFBhcmFte1ZhbHVlfSB9IHx8ICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBTdGF0c0F0dHJpYnV0ZUNyZWF0ZSgpCgpjcmVhdGUgYSBhdHRyaWJ1dGUgYXJyYXkgZm9yIHRoZSBzdGF0cyBmcmFtZXdvcmsKCiAgICBteSAkQXR0cmlidXRlID0gJEJhY2tlbmRPYmplY3QtPlN0YXRzQXR0cmlidXRlQ3JlYXRlKAogICAgICAgIEtleSA9PiAnS2V5OjpTdWJrZXknLAogICAgICAgIE5hbWUgPT4gJ05hbWUnLAogICAgICAgIEl0ZW0gPT4gJEl0ZW1SZWYsCiAgICApOwoKPWN1dAoKc3ViIFN0YXRzQXR0cmlidXRlQ3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgTkVFREVEOgogICAgZm9yIG15ICROZWVkZWQgKHF3KEtleSBOYW1lIEl0ZW0pKSB7CgogICAgICAgIG5leHQgTkVFREVEIGlmIGRlZmluZWQgJFBhcmFteyROZWVkZWR9OwoKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gIk5lZWQgJE5lZWRlZCEiLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICMgY3JlYXRlIGF0dHJpYnV0ZQogICAgbXkgJEF0dHJpYnV0ZSA9IFsKICAgICAgICB7CiAgICAgICAgICAgIE5hbWUgICAgICAgICAgICAgPT4gJFBhcmFte05hbWV9LAogICAgICAgICAgICBVc2VBc1h2YWx1ZSAgICAgID0+IDEsCiAgICAgICAgICAgIFVzZUFzVmFsdWVTZXJpZXMgPT4gMSwKICAgICAgICAgICAgVXNlQXNSZXN0cmljdGlvbiA9PiAxLAogICAgICAgICAgICBFbGVtZW50ICAgICAgICAgID0+ICRQYXJhbXtLZXl9LAogICAgICAgICAgICBCbG9jayAgICAgICAgICAgID0+ICdJbnB1dEZpZWxkJywKICAgICAgICB9LAogICAgXTsKCiAgICByZXR1cm4gJEF0dHJpYnV0ZTsKfQoKPWhlYWQyIEV4cG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgpCgpwcmVwYXJlIHNlYXJjaCB2YWx1ZSBmb3IgZXhwb3J0CgogICAgbXkgJEFycmF5UmVmID0gJEJhY2tlbmRPYmplY3QtPkV4cG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEV4cG9ydFNlYXJjaFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBFeHBvcnRWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSB2YWx1ZSBmb3IgZXhwb3J0CgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkV4cG9ydFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEV4cG9ydFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBJbXBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSBzZWFyY2ggdmFsdWUgZm9yIGltcG9ydAoKICAgIG15ICRBcnJheVJlZiA9ICRCYWNrZW5kT2JqZWN0LT5JbXBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBJbXBvcnRTZWFyY2hWYWx1ZVByZXBhcmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gaWYgIWRlZmluZWQgJFBhcmFte1ZhbHVlfTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9Owp9Cgo9aGVhZDIgSW1wb3J0VmFsdWVQcmVwYXJlKCkKCnByZXBhcmUgdmFsdWUgZm9yIGltcG9ydAoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5JbXBvcnRWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBJbXBvcnRWYWx1ZVByZXBhcmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gaWYgIWRlZmluZWQgJFBhcmFte1ZhbHVlfTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9Owp9CgoxOwoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIHNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoTDxodHRwczovL290cnMub3JnLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQ+LgoKPWN1dAo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6Q3VzdG9tZXJDb21wYW55OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06OkN1c3RvbWVyQ29tcGFueScsCiAgICAnS2VybmVsOjpTeXN0ZW06OkxvZycsCik7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OkN1c3RvbWVyQ29tcGFueSAtIHhtbCBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIERFU0NSSVBUSU9OCgpBbGwgeG1sIGZ1bmN0aW9ucyBvZiBjdXN0b21lciBjb21wYW55IG9iamVjdHMKCj1oZWFkMiBuZXcoKQoKY3JlYXRlIGFuIG9iamVjdAoKICAgIHVzZSBLZXJuZWw6OlN5c3RlbTo6T2JqZWN0TWFuYWdlcjsKICAgIGxvY2FsICRLZXJuZWw6Ok9NID0gS2VybmVsOjpTeXN0ZW06Ok9iamVjdE1hbmFnZXItPm5ldygpOwogICAgbXkgJFhNTFR5cGVDdXN0b21lckNvbXBhbnlCYWNrZW5kT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OkN1c3RvbWVyQ29tcGFueScpOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1oZWFkMiBWYWx1ZUxvb2t1cCgpCgpnZXQgdGhlIHhtbCBkYXRhIG9mIGEgdmVyc2lvbgoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5WYWx1ZUxvb2t1cCgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIFZhbHVlTG9va3VwIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuICcnIGlmICEkUGFyYW17VmFsdWV9OwoKICAgIG15ICVDdXN0b21lckNvbXBhbnkgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6Q3VzdG9tZXJDb21wYW55JyktPkN1c3RvbWVyQ29tcGFueUdldCgKICAgICAgICBDdXN0b21lcklEID0+ICRQYXJhbXtWYWx1ZX0sCiAgICApOwoKICAgIHJldHVybiAkQ3VzdG9tZXJDb21wYW55e0N1c3RvbWVyQ29tcGFueU5hbWV9IHx8ICcnOwp9Cgo9aGVhZDIgU3RhdHNBdHRyaWJ1dGVDcmVhdGUoKQoKY3JlYXRlIGEgYXR0cmlidXRlIGFycmF5IGZvciB0aGUgc3RhdHMgZnJhbWV3b3JrCgogICAgbXkgJEF0dHJpYnV0ZSA9ICRCYWNrZW5kT2JqZWN0LT5TdGF0c0F0dHJpYnV0ZUNyZWF0ZSgKICAgICAgICBLZXkgPT4gJ0tleTo6U3Via2V5JywKICAgICAgICBOYW1lID0+ICdOYW1lJywKICAgICAgICBJdGVtID0+ICRJdGVtUmVmLAogICAgKTsKCj1jdXQKCnN1YiBTdGF0c0F0dHJpYnV0ZUNyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBmb3IgbXkgJEFyZ3VtZW50IChxdyhLZXkgTmFtZSBJdGVtKSkgewogICAgICAgIGlmICggISRQYXJhbXskQXJndW1lbnR9ICkgewogICAgICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkQXJndW1lbnQhIiwKICAgICAgICAgICAgKTsKICAgICAgICAgICAgcmV0dXJuOwogICAgICAgIH0KICAgIH0KCiAgICAjIGNyZWF0ZSBhdHRyaWJ1dGUKICAgIG15ICRBdHRyaWJ1dGUgPSBbCiAgICAgICAgewogICAgICAgICAgICBOYW1lICAgICAgICAgICAgID0+ICRQYXJhbXtOYW1lfSwKICAgICAgICAgICAgVXNlQXNYdmFsdWUgICAgICA9PiAxLAogICAgICAgICAgICBVc2VBc1ZhbHVlU2VyaWVzID0+IDEsCiAgICAgICAgICAgIFVzZUFzUmVzdHJpY3Rpb24gPT4gMSwKICAgICAgICAgICAgRWxlbWVudCAgICAgICAgICA9PiAkUGFyYW17S2V5fSwKICAgICAgICAgICAgQmxvY2sgICAgICAgICAgICA9PiAnSW5wdXRGaWVsZCcsCiAgICAgICAgfSwKICAgIF07CgogICAgcmV0dXJuICRBdHRyaWJ1dGU7Cn0KCj1oZWFkMiBFeHBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSBzZWFyY2ggdmFsdWUgZm9yIGV4cG9ydAoKICAgIG15ICRBcnJheVJlZiA9ICRCYWNrZW5kT2JqZWN0LT5FeHBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBFeHBvcnRTZWFyY2hWYWx1ZVByZXBhcmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gaWYgIWRlZmluZWQgJFBhcmFte1ZhbHVlfTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9Owp9Cgo9aGVhZDIgRXhwb3J0VmFsdWVQcmVwYXJlKCkKCnByZXBhcmUgdmFsdWUgZm9yIGV4cG9ydAoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5FeHBvcnRWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBFeHBvcnRWYWx1ZVByZXBhcmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gaWYgIWRlZmluZWQgJFBhcmFte1ZhbHVlfTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9Owp9Cgo9aGVhZDIgSW1wb3J0U2VhcmNoVmFsdWVQcmVwYXJlKCkKCnByZXBhcmUgc2VhcmNoIHZhbHVlIGZvciBpbXBvcnQKCiAgICBteSAkQXJyYXlSZWYgPSAkQmFja2VuZE9iamVjdC0+SW1wb3J0U2VhcmNoVmFsdWVQcmVwYXJlKAogICAgICAgIFZhbHVlID0+IDExLCAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgSW1wb3J0U2VhcmNoVmFsdWVQcmVwYXJlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIGlmICFkZWZpbmVkICRQYXJhbXtWYWx1ZX07CiAgICByZXR1cm4gJFBhcmFte1ZhbHVlfTsKfQoKPWhlYWQyIEltcG9ydFZhbHVlUHJlcGFyZSgpCgpwcmVwYXJlIHZhbHVlIGZvciBpbXBvcnQKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+SW1wb3J0VmFsdWVQcmVwYXJlKAogICAgICAgIFZhbHVlID0+IDExLCAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgSW1wb3J0VmFsdWVQcmVwYXJlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIGlmICFkZWZpbmVkICRQYXJhbXtWYWx1ZX07CiAgICByZXR1cm4gJFBhcmFte1ZhbHVlfTsKfQoKMTsKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKEw8aHR0cHM6Ly9vdHJzLm9yZy8+KS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0Pi4KCj1jdXQK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6RGF0ZTsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpMb2cnLAopOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6WE1MOjpUeXBlOjpEYXRlIC0geG1sIGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgREVTQ1JJUFRJT04KCkFsbCB4bWwgZnVuY3Rpb25zIG9mIGRhdGUgb2JqZWN0cwoKPWhlYWQyIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgdXNlIEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyOwogICAgbG9jYWwgJEtlcm5lbDo6T00gPSBLZXJuZWw6OlN5c3RlbTo6T2JqZWN0TWFuYWdlci0+bmV3KCk7CiAgICBteSAkWE1MVHlwZURhdGVCYWNrZW5kT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OkRhdGUnKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aGVhZDIgVmFsdWVMb29rdXAoKQoKZ2V0IHRoZSBkYXRlIGRhdGEgb2YgYSB2ZXJzaW9uCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPlZhbHVlTG9va3VwKAogICAgICAgIFZhbHVlID0+ICcyMDA3LTAzLTI2JywgICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBWYWx1ZUxvb2t1cCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiAkUGFyYW17VmFsdWV9IHx8ICcnOwp9Cgo9aGVhZDIgU3RhdHNBdHRyaWJ1dGVDcmVhdGUoKQoKY3JlYXRlIGEgYXR0cmlidXRlIGFycmF5IGZvciB0aGUgc3RhdHMgZnJhbWV3b3JrCgogICAgbXkgJEF0dHJpYnV0ZSA9ICRCYWNrZW5kT2JqZWN0LT5TdGF0c0F0dHJpYnV0ZUNyZWF0ZSgpOwoKPWN1dAoKc3ViIFN0YXRzQXR0cmlidXRlQ3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGZvciBteSAkQXJndW1lbnQgKHF3KEtleSBOYW1lIEl0ZW0pKSB7CiAgICAgICAgaWYgKCAhJFBhcmFteyRBcmd1bWVudH0gKSB7CiAgICAgICAgICAgICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKS0+TG9nKAogICAgICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJOZWVkICRBcmd1bWVudCEiLAogICAgICAgICAgICApOwogICAgICAgICAgICByZXR1cm47CiAgICAgICAgfQogICAgfQoKICAgICMgY3JlYXRlIGF0dHJpYnV0ZQogICAgbXkgJEF0dHJpYnV0ZSA9IFsKICAgICAgICB7CiAgICAgICAgICAgIE5hbWUgICAgICAgICAgICAgPT4gJFBhcmFte05hbWV9LAogICAgICAgICAgICBVc2VBc1h2YWx1ZSAgICAgID0+IDEsCiAgICAgICAgICAgIFVzZUFzVmFsdWVTZXJpZXMgPT4gMSwKICAgICAgICAgICAgVXNlQXNSZXN0cmljdGlvbiA9PiAxLAogICAgICAgICAgICBFbGVtZW50ICAgICAgICAgID0+ICRQYXJhbXtLZXl9LAogICAgICAgICAgICBUaW1lUGVyaW9kRm9ybWF0ID0+ICdEYXRlSW5wdXRGb3JtYXQnLAogICAgICAgICAgICBCbG9jayAgICAgICAgICAgID0+ICdUaW1lJywKICAgICAgICAgICAgVmFsdWVzICAgICAgICAgICA9PiB7CiAgICAgICAgICAgICAgICBUaW1lU3RhcnQgPT4gJFBhcmFte0tleX0gLiAnTmV3ZXJEYXRlJywKICAgICAgICAgICAgICAgIFRpbWVTdG9wICA9PiAkUGFyYW17S2V5fSAuICdPbGRlckRhdGUnLAogICAgICAgICAgICB9LAogICAgICAgIH0sCiAgICBdOwoKICAgIHJldHVybiAkQXR0cmlidXRlOwp9Cgo9aGVhZDIgRXhwb3J0U2VhcmNoVmFsdWVQcmVwYXJlKCkKCnByZXBhcmUgc2VhcmNoIHZhbHVlIGZvciBleHBvcnQKCiAgICBteSAkQXJyYXlSZWYgPSAkQmFja2VuZE9iamVjdC0+RXhwb3J0U2VhcmNoVmFsdWVQcmVwYXJlKAogICAgICAgIFZhbHVlID0+IDExLCAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgRXhwb3J0U2VhcmNoVmFsdWVQcmVwYXJlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIGlmICFkZWZpbmVkICRQYXJhbXtWYWx1ZX07CiAgICByZXR1cm4gJFBhcmFte1ZhbHVlfTsKfQoKPWhlYWQyIEV4cG9ydFZhbHVlUHJlcGFyZSgpCgpwcmVwYXJlIHZhbHVlIGZvciBleHBvcnQKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+RXhwb3J0VmFsdWVQcmVwYXJlKAogICAgICAgIFZhbHVlID0+IDExLCAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgRXhwb3J0VmFsdWVQcmVwYXJlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIGlmICFkZWZpbmVkICRQYXJhbXtWYWx1ZX07CiAgICByZXR1cm4gJFBhcmFte1ZhbHVlfTsKfQoKPWhlYWQyIEltcG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgpCgpwcmVwYXJlIHNlYXJjaCB2YWx1ZSBmb3IgaW1wb3J0CgogICAgbXkgJEFycmF5UmVmID0gJEJhY2tlbmRPYmplY3QtPkltcG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEltcG9ydFNlYXJjaFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBJbXBvcnRWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSB2YWx1ZSBmb3IgaW1wb3J0CgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkltcG9ydFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEltcG9ydFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCjE7Cgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKClRoaXMgc29mdHdhcmUgaXMgcGFydCBvZiB0aGUgT1RSUyBwcm9qZWN0IChMPGh0dHBzOi8vb3Rycy5vcmcvPikuCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIEw8aHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dD4uCgo9Y3V0Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6RGF0ZVRpbWU7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6TG9nJywKKTsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6RGF0ZVRpbWUgLSB4bWwgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBERVNDUklQVElPTgoKQWxsIHhtbCBmdW5jdGlvbnMgb2YgZGF0ZSBvYmplY3RzCgo9aGVhZDIgbmV3KCkKCmNyZWF0ZSBhbiBvYmplY3QKCiAgICB1c2UgS2VybmVsOjpTeXN0ZW06Ok9iamVjdE1hbmFnZXI7CiAgICBsb2NhbCAkS2VybmVsOjpPTSA9IEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyLT5uZXcoKTsKICAgIG15ICRYTUxUeXBlRGF0ZVRpbWVCYWNrZW5kT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OkRhdGVUaW1lJyk7Cgo9Y3V0CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHt9OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgIHJldHVybiAkU2VsZjsKfQoKPWhlYWQyIFZhbHVlTG9va3VwKCkKCmdldCB0aGUgZGF0ZSB0aW1lIGRhdGEgb2YgYSB2ZXJzaW9uCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPlZhbHVlTG9va3VwKAogICAgICAgIFZhbHVlID0+ICcyMDA3LTAzLTI2IDIyOjAxJywgICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBWYWx1ZUxvb2t1cCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiAkUGFyYW17VmFsdWV9IHx8ICcnOwp9Cgo9aGVhZDIgU3RhdHNBdHRyaWJ1dGVDcmVhdGUoKQoKY3JlYXRlIGEgYXR0cmlidXRlIGFycmF5IGZvciB0aGUgc3RhdHMgZnJhbWV3b3JrCgogICAgbXkgJEF0dHJpYnV0ZSA9ICRCYWNrZW5kT2JqZWN0LT5TdGF0c0F0dHJpYnV0ZUNyZWF0ZSgpOwoKPWN1dAoKc3ViIFN0YXRzQXR0cmlidXRlQ3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGZvciBteSAkQXJndW1lbnQgKHF3KEtleSBOYW1lIEl0ZW0pKSB7CiAgICAgICAgaWYgKCAhJFBhcmFteyRBcmd1bWVudH0gKSB7CiAgICAgICAgICAgICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKS0+TG9nKAogICAgICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJOZWVkICRBcmd1bWVudCEiLAogICAgICAgICAgICApOwogICAgICAgICAgICByZXR1cm47CiAgICAgICAgfQogICAgfQoKICAgICMgY3JlYXRlIGF0dHJpYnV0ZQogICAgbXkgJEF0dHJpYnV0ZSA9IFsKICAgICAgICB7CiAgICAgICAgICAgIE5hbWUgICAgICAgICAgICAgPT4gJFBhcmFte05hbWV9LAogICAgICAgICAgICBVc2VBc1h2YWx1ZSAgICAgID0+IDEsCiAgICAgICAgICAgIFVzZUFzVmFsdWVTZXJpZXMgPT4gMSwKICAgICAgICAgICAgVXNlQXNSZXN0cmljdGlvbiA9PiAxLAogICAgICAgICAgICBFbGVtZW50ICAgICAgICAgID0+ICRQYXJhbXtLZXl9LAogICAgICAgICAgICBUaW1lUGVyaW9kRm9ybWF0ID0+ICdEYXRlSW5wdXRGb3JtYXRMb25nJywKICAgICAgICAgICAgQmxvY2sgICAgICAgICAgICA9PiAnVGltZScsCiAgICAgICAgICAgIFZhbHVlcyAgICAgICAgICAgPT4gewogICAgICAgICAgICAgICAgVGltZVN0YXJ0ID0+ICRQYXJhbXtLZXl9IC4gJ05ld2VyRGF0ZScsCiAgICAgICAgICAgICAgICBUaW1lU3RvcCAgPT4gJFBhcmFte0tleX0gLiAnT2xkZXJEYXRlJywKICAgICAgICAgICAgfSwKICAgICAgICB9LAogICAgXTsKCiAgICByZXR1cm4gJEF0dHJpYnV0ZTsKfQoKPWhlYWQyIEV4cG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgpCgpwcmVwYXJlIHNlYXJjaCB2YWx1ZSBmb3IgZXhwb3J0CgogICAgbXkgJEFycmF5UmVmID0gJEJhY2tlbmRPYmplY3QtPkV4cG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEV4cG9ydFNlYXJjaFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBFeHBvcnRWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSB2YWx1ZSBmb3IgZXhwb3J0CgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkV4cG9ydFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEV4cG9ydFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBJbXBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSBzZWFyY2ggdmFsdWUgZm9yIGltcG9ydAoKICAgIG15ICRBcnJheVJlZiA9ICRCYWNrZW5kT2JqZWN0LT5JbXBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBJbXBvcnRTZWFyY2hWYWx1ZVByZXBhcmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gaWYgIWRlZmluZWQgJFBhcmFte1ZhbHVlfTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9Owp9Cgo9aGVhZDIgSW1wb3J0VmFsdWVQcmVwYXJlKCkKCnByZXBhcmUgdmFsdWUgZm9yIGltcG9ydAoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5JbXBvcnRWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBJbXBvcnRWYWx1ZVByZXBhcmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gaWYgIWRlZmluZWQgJFBhcmFte1ZhbHVlfTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9Owp9CgoxOwoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIHNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoTDxodHRwczovL290cnMub3JnLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQ+LgoKPWN1dAo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6RHVtbXk7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgpOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6WE1MOjpUeXBlOjpEdW1teSAtIHhtbCBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIERFU0NSSVBUSU9OCgpBbGwgeG1sIGZ1bmN0aW9ucyBvZiBkdW1teSBvYmplY3RzCgo9aGVhZDIgbmV3KCkKCmNyZWF0ZSBhbiBvYmplY3QKCiAgICB1c2UgS2VybmVsOjpTeXN0ZW06Ok9iamVjdE1hbmFnZXI7CiAgICBsb2NhbCAkS2VybmVsOjpPTSA9IEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyLT5uZXcoKTsKICAgIG15ICRYTUxUeXBlRHVtbXlCYWNrZW5kT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OkR1bW15Jyk7Cgo9Y3V0CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHt9OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgIHJldHVybiAkU2VsZjsKfQoKPWhlYWQyIFZhbHVlTG9va3VwKCkKCmdldCB0aGUgZHVtbXkgZGF0YSBvZiBhIHZlcnNpb24KCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+VmFsdWVMb29rdXAoKTsKCj1jdXQKCnN1YiBWYWx1ZUxvb2t1cCB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgIHJldHVybiAnJzsKfQoKPWhlYWQyIFN0YXRzQXR0cmlidXRlQ3JlYXRlKCkKCmNyZWF0ZSBhIGF0dHJpYnV0ZSBhcnJheSBmb3IgdGhlIHN0YXRzIGZyYW1ld29yawoKICAgIG15ICRBdHRyaWJ1dGUgPSAkQmFja2VuZE9iamVjdC0+U3RhdHNBdHRyaWJ1dGVDcmVhdGUoKTsKCj1jdXQKCnN1YiBTdGF0c0F0dHJpYnV0ZUNyZWF0ZSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgIHJldHVybjsKfQoKPWhlYWQyIEV4cG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgpCgpwcmVwYXJlIHNlYXJjaCB2YWx1ZSBmb3IgZXhwb3J0CgogICAgbXkgJEFycmF5UmVmID0gJEJhY2tlbmRPYmplY3QtPkV4cG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEV4cG9ydFNlYXJjaFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBFeHBvcnRWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSB2YWx1ZSBmb3IgZXhwb3J0CgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkV4cG9ydFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEV4cG9ydFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBJbXBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSBzZWFyY2ggdmFsdWUgZm9yIGltcG9ydAoKICAgIG15ICRBcnJheVJlZiA9ICRCYWNrZW5kT2JqZWN0LT5JbXBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBJbXBvcnRTZWFyY2hWYWx1ZVByZXBhcmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gaWYgIWRlZmluZWQgJFBhcmFte1ZhbHVlfTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9Owp9Cgo9aGVhZDIgSW1wb3J0VmFsdWVQcmVwYXJlKCkKCnByZXBhcmUgdmFsdWUgZm9yIGltcG9ydAoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5JbXBvcnRWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBJbXBvcnRWYWx1ZVByZXBhcmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gaWYgIWRlZmluZWQgJFBhcmFte1ZhbHVlfTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9Owp9CgoxOwoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIHNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoTDxodHRwczovL290cnMub3JnLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQ+LgoKPWN1dAo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6R2VuZXJhbENhdGFsb2c7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6R2VuZXJhbENhdGFsb2cnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpMb2cnLAopOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6WE1MOjpUeXBlOjpHZW5lcmFsQ2F0YWxvZyAtIHhtbCBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIERFU0NSSVBUSU9OCgpBbGwgeG1sIGZ1bmN0aW9ucyBvZiBnZW5lcmFsIGNhdGFsb2cgb2JqZWN0cwoKPWhlYWQyIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgdXNlIEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyOwogICAgbG9jYWwgJEtlcm5lbDo6T00gPSBLZXJuZWw6OlN5c3RlbTo6T2JqZWN0TWFuYWdlci0+bmV3KCk7CiAgICBteSAkWE1MVHlwZUdlbmVyYWxDYXRhbG9nQmFja2VuZE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6WE1MOjpUeXBlOjpHZW5lcmFsQ2F0YWxvZycpOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1oZWFkMiBWYWx1ZUxvb2t1cCgpCgpnZXQgdGhlIHhtbCBkYXRhIG9mIGEgdmVyc2lvbgoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5WYWx1ZUxvb2t1cCgKICAgICAgICBJdGVtICA9PiAkSXRlbVJlZiwKICAgICAgICBWYWx1ZSA9PiAxMSwgICAgICAgICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBWYWx1ZUxvb2t1cCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpLT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIEl0ZW0hJywKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICByZXR1cm4gaWYgISRQYXJhbXtWYWx1ZX07CgogICAgIyBnZXQgaXRlbSBsaXN0CiAgICBteSAkSXRlbUxpc3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6R2VuZXJhbENhdGFsb2cnKS0+SXRlbUxpc3QoCiAgICAgICAgQ2xhc3MgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57Q2xhc3N9IHx8ICcnLAogICAgKTsKCiAgICByZXR1cm4gaWYgISRJdGVtTGlzdDsKICAgIHJldHVybiBpZiByZWYgJEl0ZW1MaXN0IG5lICdIQVNIJzsKCiAgICBteSAkVmFsdWUgPSAkSXRlbUxpc3QtPnsgJFBhcmFte1ZhbHVlfSB9OwoKICAgIHJldHVybiAkVmFsdWU7Cn0KCj1oZWFkMiBTdGF0c0F0dHJpYnV0ZUNyZWF0ZSgpCgpjcmVhdGUgYSBhdHRyaWJ1dGUgYXJyYXkgZm9yIHRoZSBzdGF0cyBmcmFtZXdvcmsKCiAgICBteSAkQXR0cmlidXRlID0gJEJhY2tlbmRPYmplY3QtPlN0YXRzQXR0cmlidXRlQ3JlYXRlKAogICAgICAgIEtleSAgPT4gJ0tleTo6U3Via2V5JywKICAgICAgICBOYW1lID0+ICdOYW1lJywKICAgICAgICBJdGVtID0+ICRJdGVtUmVmLAogICAgKTsKCj1jdXQKCnN1YiBTdGF0c0F0dHJpYnV0ZUNyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBmb3IgbXkgJEFyZ3VtZW50IChxdyhLZXkgTmFtZSBJdGVtKSkgewogICAgICAgIGlmICggISRQYXJhbXskQXJndW1lbnR9ICkgewogICAgICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkQXJndW1lbnQhIiwKICAgICAgICAgICAgKTsKICAgICAgICAgICAgcmV0dXJuOwogICAgICAgIH0KICAgIH0KCiAgICAjIGdldCBpdGVtIGxpc3QKICAgIG15ICRJdGVtTGlzdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtTGlzdCgKICAgICAgICBDbGFzcyA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntDbGFzc30gfHwgJycsCiAgICApOwoKICAgICMgY3JlYXRlIGF0dHJpYnV0ZQogICAgbXkgJEF0dHJpYnV0ZSA9IFsKICAgICAgICB7CiAgICAgICAgICAgIE5hbWUgICAgICAgICAgICAgPT4gJFBhcmFte05hbWV9LAogICAgICAgICAgICBVc2VBc1h2YWx1ZSAgICAgID0+IDEsCiAgICAgICAgICAgIFVzZUFzVmFsdWVTZXJpZXMgPT4gMSwKICAgICAgICAgICAgVXNlQXNSZXN0cmljdGlvbiA9PiAxLAogICAgICAgICAgICBFbGVtZW50ICAgICAgICAgID0+ICRQYXJhbXtLZXl9LAogICAgICAgICAgICBCbG9jayAgICAgICAgICAgID0+ICdNdWx0aVNlbGVjdEZpZWxkJywKICAgICAgICAgICAgVmFsdWVzICAgICAgICAgICA9PiAkSXRlbUxpc3QgfHwge30sCiAgICAgICAgfSwKICAgIF07CgogICAgcmV0dXJuICRBdHRyaWJ1dGU7Cn0KCj1oZWFkMiBFeHBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSBzZWFyY2ggdmFsdWUgZm9yIGV4cG9ydAoKICAgIG15ICRBcnJheVJlZiA9ICRCYWNrZW5kT2JqZWN0LT5FeHBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBFeHBvcnRTZWFyY2hWYWx1ZVByZXBhcmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gaWYgIWRlZmluZWQgJFBhcmFte1ZhbHVlfTsKCiAgICBteSBAVmFsdWVzID0gc3BsaXQgJyMjIyMjJywgJFBhcmFte1ZhbHVlfTsKICAgIEBWYWx1ZXMgPSBncmVwIHskX30gQFZhbHVlczsKCiAgICByZXR1cm4gXEBWYWx1ZXM7Cn0KCj1oZWFkMiBFeHBvcnRWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSB2YWx1ZSBmb3IgZXhwb3J0CgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkV4cG9ydFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEV4cG9ydFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwoKICAgICMgZ2V0IGl0ZW0gbGlzdAogICAgbXkgJEl0ZW1MaXN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkdlbmVyYWxDYXRhbG9nJyktPkl0ZW1MaXN0KAogICAgICAgIENsYXNzID0+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e0NsYXNzfSB8fCAnJywKICAgICk7CgogICAgcmV0dXJuICRJdGVtTGlzdC0+eyAkUGFyYW17VmFsdWV9IH0gfHwgJFBhcmFte1ZhbHVlfTsKfQoKPWhlYWQyIEltcG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgpCgpwcmVwYXJlIHNlYXJjaCB2YWx1ZSBmb3IgaW1wb3J0CgogICAgbXkgJEFycmF5UmVmID0gJEJhY2tlbmRPYmplY3QtPkltcG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwKICAgICk7Cgo9Y3V0CgpzdWIgSW1wb3J0U2VhcmNoVmFsdWVQcmVwYXJlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIGlmICFkZWZpbmVkICRQYXJhbXtWYWx1ZX07CgogICAgIyBnZXQgaXRlbSBsaXN0CiAgICBteSAkSXRlbUxpc3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6R2VuZXJhbENhdGFsb2cnKS0+SXRlbUxpc3QoCiAgICAgICAgQ2xhc3MgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57Q2xhc3N9IHx8ICcnLAogICAgKTsKCiAgICAjIHJldmVyc2UgdGhlIGxpc3QKICAgIG15ICVOYW1lMklEID0gcmV2ZXJzZSAleyRJdGVtTGlzdH07CgogICAgbXkgJEdlbmVyYWxDYXRhbG9nSUQgPSAkTmFtZTJJRHsgJFBhcmFte1ZhbHVlfSB9OwoKICAgIGlmICggISRHZW5lcmFsQ2F0YWxvZ0lEICkgewogICAgICAgICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiR2VuZXJhbCBjYXRhbG9nIGxvb2t1cCBvZickUGFyYW17VmFsdWV9JyBmYWlsZWQhIiwKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICByZXR1cm4gJEdlbmVyYWxDYXRhbG9nSUQ7Cgp9Cgo9aGVhZDIgSW1wb3J0VmFsdWVQcmVwYXJlKCkKCnByZXBhcmUgdmFsdWUgZm9yIGltcG9ydAoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5JbXBvcnRWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsCiAgICApOwoKPWN1dAoKc3ViIEltcG9ydFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwoKICAgICMgSnVzdCByZXR1cm4gZW1wdHkgdmFsdWUgaWYgaXRlbSBpcyBub3QgcmVxdWlyZWQuCiAgICBpZiAoICRQYXJhbXtWYWx1ZX0gZXEgJycgJiYgISRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1JlcXVpcmVkfSApIHsKICAgICAgICByZXR1cm4gJFBhcmFte1ZhbHVlfTsKICAgIH0KCiAgICAjIGdldCBpdGVtIGxpc3QKICAgIG15ICRJdGVtTGlzdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtTGlzdCgKICAgICAgICBDbGFzcyA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntDbGFzc30gfHwgJycsCiAgICApOwoKICAgICMgcmV2ZXJzZSB0aGUgbGlzdAogICAgbXkgJU5hbWUySUQgPSByZXZlcnNlICV7JEl0ZW1MaXN0fTsKCiAgICBteSAkR2VuZXJhbENhdGFsb2dJRCA9ICROYW1lMklEeyAkUGFyYW17VmFsdWV9IH07CgogICAgaWYgKCAhJEdlbmVyYWxDYXRhbG9nSUQgKSB7CiAgICAgICAgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpLT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJHZW5lcmFsIGNhdGFsb2cgbG9va3VwIG9mICckUGFyYW17VmFsdWV9JyBmYWlsZWQhIiwKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICByZXR1cm4gJEdlbmVyYWxDYXRhbG9nSUQ7Cn0KCjE7Cgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKClRoaXMgc29mdHdhcmUgaXMgcGFydCBvZiB0aGUgT1RSUyBwcm9qZWN0IChMPGh0dHBzOi8vb3Rycy5vcmcvPikuCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIEw8aHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dD4uCgo9Y3V0Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6SW50ZWdlcjsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpMb2cnLAopOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6WE1MOjpUeXBlOjpJbnRlZ2VyIC0geG1sIGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgREVTQ1JJUFRJT04KCkFsbCB4bWwgZnVuY3Rpb25zIG9mIGludGVnZXIgb2JqZWN0cwoKPWhlYWQyIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgdXNlIEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyOwogICAgbG9jYWwgJEtlcm5lbDo6T00gPSBLZXJuZWw6OlN5c3RlbTo6T2JqZWN0TWFuYWdlci0+bmV3KCk7CiAgICBteSAkWE1MVHlwZUludGVnZXJCYWNrZW5kT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OkludGVnZXInKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aGVhZDIgVmFsdWVMb29rdXAoKQoKZ2V0IHRoZSBpbnRlZ2VyIGRhdGEgb2YgYSB2ZXJzaW9uCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPlZhbHVlTG9va3VwKAogICAgICAgIFZhbHVlID0+IDExLCAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIFZhbHVlTG9va3VwIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBTdGF0c0F0dHJpYnV0ZUNyZWF0ZSgpCgpjcmVhdGUgYSBhdHRyaWJ1dGUgYXJyYXkgZm9yIHRoZSBzdGF0cyBmcmFtZXdvcmsKCiAgICBteSAkQXR0cmlidXRlID0gJEJhY2tlbmRPYmplY3QtPlN0YXRzQXR0cmlidXRlQ3JlYXRlKCk7Cgo9Y3V0CgpzdWIgU3RhdHNBdHRyaWJ1dGVDcmVhdGUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICBORUVERUQ6CiAgICBmb3IgbXkgJE5lZWRlZCAocXcoS2V5IE5hbWUgSXRlbSkpIHsKCiAgICAgICAgbmV4dCBORUVERUQgaWYgZGVmaW5lZCAkUGFyYW17JE5lZWRlZH07CgogICAgICAgICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkTmVlZGVkISIsCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgbXkgJFZhbHVlTWluID0gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VmFsdWVNaW59IHx8IDE7CiAgICBteSAkVmFsdWVNYXggPSAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntWYWx1ZU1heH0gfHwgMTsKCiAgICBteSAlVmFsdWVzID0gbWFwIHsgJF8gPT4gJF8gfSAoICRWYWx1ZU1pbiAuLiAkVmFsdWVNYXggKTsKCiAgICAjIGNyZWF0ZSBhdHRyaWJ1dGUKICAgIG15ICRBdHRyaWJ1dGUgPSBbCiAgICAgICAgewogICAgICAgICAgICBOYW1lICAgICAgICAgICAgID0+ICRQYXJhbXtOYW1lfSwKICAgICAgICAgICAgVXNlQXNYdmFsdWUgICAgICA9PiAxLAogICAgICAgICAgICBVc2VBc1ZhbHVlU2VyaWVzID0+IDEsCiAgICAgICAgICAgIFVzZUFzUmVzdHJpY3Rpb24gPT4gMSwKICAgICAgICAgICAgRWxlbWVudCAgICAgICAgICA9PiAkUGFyYW17S2V5fSwKICAgICAgICAgICAgQmxvY2sgICAgICAgICAgICA9PiAnTXVsdGlTZWxlY3RGaWVsZCcsCiAgICAgICAgICAgIFZhbHVlcyAgICAgICAgICAgPT4gXCVWYWx1ZXMsCiAgICAgICAgICAgIFNvcnQgICAgICAgICAgICAgPT4gJ051bWVyaWNLZXknLAogICAgICAgIH0sCiAgICBdOwoKICAgIHJldHVybiAkQXR0cmlidXRlOwp9Cgo9aGVhZDIgRXhwb3J0U2VhcmNoVmFsdWVQcmVwYXJlKCkKCnByZXBhcmUgc2VhcmNoIHZhbHVlIGZvciBleHBvcnQKCiAgICBteSAkQXJyYXlSZWYgPSAkQmFja2VuZE9iamVjdC0+RXhwb3J0U2VhcmNoVmFsdWVQcmVwYXJlKAogICAgICAgIFZhbHVlID0+IDExLCAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgRXhwb3J0U2VhcmNoVmFsdWVQcmVwYXJlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIGlmICFkZWZpbmVkICRQYXJhbXtWYWx1ZX07CiAgICByZXR1cm4gJFBhcmFte1ZhbHVlfTsKfQoKPWhlYWQyIEV4cG9ydFZhbHVlUHJlcGFyZSgpCgpwcmVwYXJlIHZhbHVlIGZvciBleHBvcnQKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+RXhwb3J0VmFsdWVQcmVwYXJlKAogICAgICAgIFZhbHVlID0+IDExLCAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgRXhwb3J0VmFsdWVQcmVwYXJlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIGlmICFkZWZpbmVkICRQYXJhbXtWYWx1ZX07CiAgICByZXR1cm4gJFBhcmFte1ZhbHVlfTsKfQoKPWhlYWQyIEltcG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgpCgpwcmVwYXJlIHNlYXJjaCB2YWx1ZSBmb3IgaW1wb3J0CgogICAgbXkgJEFycmF5UmVmID0gJEJhY2tlbmRPYmplY3QtPkltcG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEltcG9ydFNlYXJjaFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBJbXBvcnRWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSB2YWx1ZSBmb3IgaW1wb3J0CgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkltcG9ydFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEltcG9ydFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCjE7Cgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKClRoaXMgc29mdHdhcmUgaXMgcGFydCBvZiB0aGUgT1RSUyBwcm9qZWN0IChMPGh0dHBzOi8vb3Rycy5vcmcvPikuCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIEw8aHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dD4uCgo9Y3V0Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OlByaW9yaXR5OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06OlByaW9yaXR5JywKKTsKCnVzZSBwYXJlbnQgJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6WE1MOjpUeXBlOjpCYXNlU2VsZWN0YWJsZSc7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OlByaW9yaXR5IC0geG1sIGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgU1lOT1BTSVMKClhNTCBmdW5jdGlvbnMgZm9yIHByaW9yaXR5IG9iamVjdHMKCj1oZWFkMiBWYWx1ZUxvb2t1cCgpCgpnZXQgdGhlIHhtbCBkYXRhIG9mIGEgdmVyc2lvbgoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5WYWx1ZUxvb2t1cCgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIFZhbHVlTG9va3VwIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuICcnIGlmICEkUGFyYW17VmFsdWV9OwoKICAgIG15ICRQcmlvcml0eU9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpQcmlvcml0eScpOwogICAgbXkgJFByaW9yaXR5TmFtZSAgID0gJFByaW9yaXR5T2JqZWN0LT5Qcmlvcml0eUxvb2t1cCggUHJpb3JpdHlJRCA9PiAkUGFyYW17VmFsdWV9ICk7CiAgICByZXR1cm4gJFBhcmFte1ZhbHVlfSBpZiAhZGVmaW5lZCAkUHJpb3JpdHlOYW1lOwoKICAgIHJldHVybiAkUHJpb3JpdHlOYW1lOwp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OlF1ZXVlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06OlF1ZXVlJywKKTsKCnVzZSBwYXJlbnQgJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6WE1MOjpUeXBlOjpCYXNlU2VsZWN0YWJsZSc7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OlF1ZXVlIC0geG1sIGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgU1lOT1BTSVMKClhNTCBmdW5jdGlvbnMgZm9yIHF1ZXVlIG9iamVjdHMKCj1oZWFkMiBWYWx1ZUxvb2t1cCgpCgpnZXQgdGhlIHhtbCBkYXRhIG9mIGEgdmVyc2lvbgoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5WYWx1ZUxvb2t1cCgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIFZhbHVlTG9va3VwIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuICcnIGlmICEkUGFyYW17VmFsdWV9OwoKICAgIG15ICRRdWV1ZU9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpRdWV1ZScpOwogICAgbXkgJFF1ZXVlTmFtZSAgID0gJFF1ZXVlT2JqZWN0LT5RdWV1ZUxvb2t1cCggUXVldWVJRCA9PiAkUGFyYW17VmFsdWV9ICk7CiAgICByZXR1cm4gJFBhcmFte1ZhbHVlfSBpZiAhZGVmaW5lZCAkUXVldWVOYW1lOwoKICAgIHJldHVybiAkUXVldWVOYW1lOwp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OlNMQTsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpTTEEnLAopOwoKdXNlIHBhcmVudCAnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OkJhc2VTZWxlY3RhYmxlJzsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6U0xBIC0geG1sIGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgU1lOT1BTSVMKClhNTCBmdW5jdGlvbnMgZm9yIFNMQSBvYmplY3RzCgo9aGVhZDIgVmFsdWVMb29rdXAoKQoKZ2V0IHRoZSB4bWwgZGF0YSBvZiBhIHZlcnNpb24KCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+VmFsdWVMb29rdXAoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBWYWx1ZUxvb2t1cCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiAnJyBpZiAhJFBhcmFte1ZhbHVlfTsKCiAgICBteSAkU0xBT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlNMQScpOwogICAgbXkgJFNMQU5hbWUgICA9ICRTTEFPYmplY3QtPlNMQUxvb2t1cCggU0xBSUQgPT4gJFBhcmFte1ZhbHVlfSApOwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX0gaWYgIWRlZmluZWQgJFNMQU5hbWU7CgogICAgcmV0dXJuICRTTEFOYW1lOwp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OlNlcnZpY2U7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6U2VydmljZScsCik7Cgp1c2UgcGFyZW50ICdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6QmFzZVNlbGVjdGFibGUnOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6WE1MOjpUeXBlOjpTZXJ2aWNlIC0geG1sIGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgU1lOT1BTSVMKClhNTCBmdW5jdGlvbnMgZm9yIHNlcnZpY2Ugb2JqZWN0cwoKPWhlYWQyIFZhbHVlTG9va3VwKCkKCmdldCB0aGUgeG1sIGRhdGEgb2YgYSB2ZXJzaW9uCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPlZhbHVlTG9va3VwKAogICAgICAgIFZhbHVlID0+IDExLCAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgVmFsdWVMb29rdXAgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gJycgaWYgISRQYXJhbXtWYWx1ZX07CgogICAgbXkgJFNlcnZpY2VPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6U2VydmljZScpOwogICAgbXkgJFNlcnZpY2VOYW1lICAgPSAkU2VydmljZU9iamVjdC0+U2VydmljZUxvb2t1cCggU2VydmljZUlEID0+ICRQYXJhbXtWYWx1ZX0gKTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9IGlmICFkZWZpbmVkICRTZXJ2aWNlTmFtZTsKCiAgICByZXR1cm4gJFNlcnZpY2VOYW1lOwp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OlN0YXRlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06OlN0YXRlJywKKTsKCnVzZSBwYXJlbnQgJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6WE1MOjpUeXBlOjpCYXNlU2VsZWN0YWJsZSc7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OlN0YXRlIC0geG1sIGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgU1lOT1BTSVMKClhNTCBmdW5jdGlvbnMgZm9yIFN0YXRlIG9iamVjdHMKCj1oZWFkMiBWYWx1ZUxvb2t1cCgpCgpnZXQgdGhlIHhtbCBkYXRhIG9mIGEgdmVyc2lvbgoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5WYWx1ZUxvb2t1cCgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIFZhbHVlTG9va3VwIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuICcnIGlmICEkUGFyYW17VmFsdWV9OwoKICAgIG15ICRTdGF0ZU9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpTdGF0ZScpOwogICAgbXkgJFN0YXRlTmFtZSAgID0gJFN0YXRlT2JqZWN0LT5TdGF0ZUxvb2t1cCggU3RhdGVJRCA9PiAkUGFyYW17VmFsdWV9ICk7CiAgICByZXR1cm4gJFBhcmFte1ZhbHVlfSBpZiAhZGVmaW5lZCAkU3RhdGVOYW1lOwoKICAgIHJldHVybiAkU3RhdGVOYW1lOwp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6VGV4dDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpMb2cnLAopOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6WE1MOjpUeXBlOjpUZXh0IC0geG1sIGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgREVTQ1JJUFRJT04KCkFsbCB4bWwgZnVuY3Rpb25zIG9mIHRleHQgb2JqZWN0cwoKPWhlYWQyIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgdXNlIEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyOwogICAgbG9jYWwgJEtlcm5lbDo6T00gPSBLZXJuZWw6OlN5c3RlbTo6T2JqZWN0TWFuYWdlci0+bmV3KCk7CiAgICBteSAkWE1MVHlwZVRleHRCYWNrZW5kT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OlRleHQnKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aGVhZDIgVmFsdWVMb29rdXAoKQoKZ2V0IHRoZSB0ZXh0IGRhdGEgb2YgYSB2ZXJzaW9uCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPlZhbHVlTG9va3VwKAogICAgICAgIFZhbHVlID0+IDExLCAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIFZhbHVlTG9va3VwIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBTdGF0c0F0dHJpYnV0ZUNyZWF0ZSgpCgpjcmVhdGUgYSBhdHRyaWJ1dGUgYXJyYXkgZm9yIHRoZSBzdGF0cyBmcmFtZXdvcmsKCiAgICBteSAkQXR0cmlidXRlID0gJEJhY2tlbmRPYmplY3QtPlN0YXRzQXR0cmlidXRlQ3JlYXRlKAogICAgICAgIEtleSAgPT4gJ0tleTo6U3Via2V5JywKICAgICAgICBOYW1lID0+ICdOYW1lJywKICAgICAgICBJdGVtID0+ICRJdGVtUmVmLAogICAgKTsKCj1jdXQKCnN1YiBTdGF0c0F0dHJpYnV0ZUNyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBmb3IgbXkgJEFyZ3VtZW50IChxdyhLZXkgTmFtZSBJdGVtKSkgewogICAgICAgIGlmICggISRQYXJhbXskQXJndW1lbnR9ICkgewogICAgICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkQXJndW1lbnQhIiwKICAgICAgICAgICAgKTsKICAgICAgICAgICAgcmV0dXJuOwogICAgICAgIH0KICAgIH0KCiAgICAjIGNyZWF0ZSBhdHRyaWJ1dGUKICAgIG15ICRBdHRyaWJ1dGUgPSBbCiAgICAgICAgewogICAgICAgICAgICBOYW1lICAgICAgICAgICAgID0+ICRQYXJhbXtOYW1lfSwKICAgICAgICAgICAgVXNlQXNYdmFsdWUgICAgICA9PiAxLAogICAgICAgICAgICBVc2VBc1ZhbHVlU2VyaWVzID0+IDEsCiAgICAgICAgICAgIFVzZUFzUmVzdHJpY3Rpb24gPT4gMSwKICAgICAgICAgICAgRWxlbWVudCAgICAgICAgICA9PiAkUGFyYW17S2V5fSwKICAgICAgICAgICAgQmxvY2sgICAgICAgICAgICA9PiAnSW5wdXRGaWVsZCcsCiAgICAgICAgfSwKICAgIF07CgogICAgcmV0dXJuICRBdHRyaWJ1dGU7Cn0KCj1oZWFkMiBFeHBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSBzZWFyY2ggdmFsdWUgZm9yIGV4cG9ydAoKICAgIG15ICRBcnJheVJlZiA9ICRCYWNrZW5kT2JqZWN0LT5FeHBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBFeHBvcnRTZWFyY2hWYWx1ZVByZXBhcmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gaWYgIWRlZmluZWQgJFBhcmFte1ZhbHVlfTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9Owp9Cgo9aGVhZDIgRXhwb3J0VmFsdWVQcmVwYXJlKCkKCnByZXBhcmUgdmFsdWUgZm9yIGV4cG9ydAoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5FeHBvcnRWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBFeHBvcnRWYWx1ZVByZXBhcmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gaWYgIWRlZmluZWQgJFBhcmFte1ZhbHVlfTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9Owp9Cgo9aGVhZDIgSW1wb3J0U2VhcmNoVmFsdWVQcmVwYXJlKCkKCnByZXBhcmUgc2VhcmNoIHZhbHVlIGZvciBpbXBvcnQKCiAgICBteSAkQXJyYXlSZWYgPSAkQmFja2VuZE9iamVjdC0+SW1wb3J0U2VhcmNoVmFsdWVQcmVwYXJlKAogICAgICAgIFZhbHVlID0+IDExLCAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgSW1wb3J0U2VhcmNoVmFsdWVQcmVwYXJlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIGlmICFkZWZpbmVkICRQYXJhbXtWYWx1ZX07CiAgICByZXR1cm4gJFBhcmFte1ZhbHVlfTsKfQoKPWhlYWQyIEltcG9ydFZhbHVlUHJlcGFyZSgpCgpwcmVwYXJlIHZhbHVlIGZvciBpbXBvcnQKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+SW1wb3J0VmFsdWVQcmVwYXJlKAogICAgICAgIFZhbHVlID0+IDExLCAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgSW1wb3J0VmFsdWVQcmVwYXJlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIGlmICFkZWZpbmVkICRQYXJhbXtWYWx1ZX07CiAgICByZXR1cm4gJFBhcmFte1ZhbHVlfTsKfQoKMTsKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKEw8aHR0cHM6Ly9vdHJzLm9yZy8+KS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0Pi4KCj1jdXQK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6VGV4dEFyZWE7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6TG9nJywKKTsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6VGV4dEFyZWEgLSB4bWwgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBERVNDUklQVElPTgoKQWxsIHhtbCBmdW5jdGlvbnMgb2YgdGV4dGFyZWEgb2JqZWN0cwoKPWhlYWQyIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgdXNlIEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyOwogICAgbG9jYWwgJEtlcm5lbDo6T00gPSBLZXJuZWw6OlN5c3RlbTo6T2JqZWN0TWFuYWdlci0+bmV3KCk7CiAgICBteSAkWE1MVHlwZVRleHRBcmVhQmFja2VuZE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6WE1MOjpUeXBlOjpUZXh0QXJlYScpOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1oZWFkMiBWYWx1ZUxvb2t1cCgpCgpnZXQgdGhlIHRleHQgZGF0YSBvZiBhIHZlcnNpb24KCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+VmFsdWVMb29rdXAoCiAgICAgICAgVmFsdWUgPT4gMTEsICAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgVmFsdWVMb29rdXAgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gJFBhcmFte1ZhbHVlfTsKfQoKPWhlYWQyIFN0YXRzQXR0cmlidXRlQ3JlYXRlKCkKCmNyZWF0ZSBhIGF0dHJpYnV0ZSBhcnJheSBmb3IgdGhlIHN0YXRzIGZyYW1ld29yawoKICAgIG15ICRBdHRyaWJ1dGUgPSAkQmFja2VuZE9iamVjdC0+U3RhdHNBdHRyaWJ1dGVDcmVhdGUoCiAgICAgICAgS2V5ICA9PiAnS2V5OjpTdWJrZXknLAogICAgICAgIE5hbWUgPT4gJ05hbWUnLAogICAgICAgIEl0ZW0gPT4gJEl0ZW1SZWYsCiAgICApOwoKPWN1dAoKc3ViIFN0YXRzQXR0cmlidXRlQ3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGZvciBteSAkQXJndW1lbnQgKHF3KEtleSBOYW1lIEl0ZW0pKSB7CiAgICAgICAgaWYgKCAhJFBhcmFteyRBcmd1bWVudH0gKSB7CiAgICAgICAgICAgICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKS0+TG9nKAogICAgICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJOZWVkICRBcmd1bWVudCEiLAogICAgICAgICAgICApOwogICAgICAgICAgICByZXR1cm47CiAgICAgICAgfQogICAgfQoKICAgICMgY3JlYXRlIGF0dHJpYnV0ZQogICAgbXkgJEF0dHJpYnV0ZSA9IFsKICAgICAgICB7CiAgICAgICAgICAgIE5hbWUgICAgICAgICAgICAgPT4gJFBhcmFte05hbWV9LAogICAgICAgICAgICBVc2VBc1h2YWx1ZSAgICAgID0+IDAsCiAgICAgICAgICAgIFVzZUFzVmFsdWVTZXJpZXMgPT4gMCwKICAgICAgICAgICAgVXNlQXNSZXN0cmljdGlvbiA9PiAxLAogICAgICAgICAgICBFbGVtZW50ICAgICAgICAgID0+ICRQYXJhbXtLZXl9LAogICAgICAgICAgICBCbG9jayAgICAgICAgICAgID0+ICdJbnB1dEZpZWxkJywKICAgICAgICB9LAogICAgXTsKCiAgICByZXR1cm4gJEF0dHJpYnV0ZTsKfQoKPWhlYWQyIEV4cG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgpCgpwcmVwYXJlIHNlYXJjaCB2YWx1ZSBmb3IgZXhwb3J0CgogICAgbXkgJEFycmF5UmVmID0gJEJhY2tlbmRPYmplY3QtPkV4cG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEV4cG9ydFNlYXJjaFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBFeHBvcnRWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSB2YWx1ZSBmb3IgZXhwb3J0CgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkV4cG9ydFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEV4cG9ydFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBJbXBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSBzZWFyY2ggdmFsdWUgZm9yIGltcG9ydAoKICAgIG15ICRBcnJheVJlZiA9ICRCYWNrZW5kT2JqZWN0LT5JbXBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBJbXBvcnRTZWFyY2hWYWx1ZVByZXBhcmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gaWYgIWRlZmluZWQgJFBhcmFte1ZhbHVlfTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9Owp9Cgo9aGVhZDIgSW1wb3J0VmFsdWVQcmVwYXJlKCkKCnByZXBhcmUgdmFsdWUgZm9yIGltcG9ydAoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5JbXBvcnRWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBJbXBvcnRWYWx1ZVByZXBhcmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gaWYgIWRlZmluZWQgJFBhcmFte1ZhbHVlfTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9Owp9CgoxOwoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIHNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoTDxodHRwczovL290cnMub3JnLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQ+LgoKPWN1dAo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OlRleHRMaW5rOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHBhcmVudCBxdyggS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OlRleHQgKTsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKCk7CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OlR5cGU7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6VHlwZScsCik7Cgp1c2UgcGFyZW50ICdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OlhNTDo6VHlwZTo6QmFzZVNlbGVjdGFibGUnOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbTo6WE1MOjpUeXBlOjpUeXBlIC0geG1sIGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgU1lOT1BTSVMKClhNTCBmdW5jdGlvbnMgZm9yIHR5cGUgb2JqZWN0cwoKPWhlYWQyIFZhbHVlTG9va3VwKCkKCmdldCB0aGUgeG1sIGRhdGEgb2YgYSB2ZXJzaW9uCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPlZhbHVlTG9va3VwKAogICAgICAgIFZhbHVlID0+IDExLCAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgVmFsdWVMb29rdXAgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gJycgaWYgISRQYXJhbXtWYWx1ZX07CgogICAgbXkgJFR5cGVPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VHlwZScpOwogICAgbXkgJFR5cGVOYW1lICAgPSAkVHlwZU9iamVjdC0+VHlwZUxvb2t1cCggVHlwZUlEID0+ICRQYXJhbXtWYWx1ZX0gKTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9IGlmICFkZWZpbmVkICRUeXBlTmFtZTsKCiAgICByZXR1cm4gJFR5cGVOYW1lOwp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OlVzZXI7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6VXNlcicsCiAgICAnS2VybmVsOjpTeXN0ZW06OkxvZycsCik7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OlVzZXIgLSB4bWwgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKQWxsIHhtbCBmdW5jdGlvbnMgb2YgdXNlciBvYmplY3RzCgo9aGVhZDIgbmV3KCkKCmNyZWF0ZSBhbiBvYmplY3QKCiAgICBteSAkWE1MVHlwZVVzZXJCYWNrZW5kT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpYTUw6OlR5cGU6OlVzZXInKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aGVhZDIgVmFsdWVMb29rdXAoKQoKZ2V0IHRoZSB4bWwgZGF0YSBvZiBhIHZlcnNpb24KCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+VmFsdWVMb29rdXAoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBWYWx1ZUxvb2t1cCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiAnJyBpZiAhJFBhcmFte1ZhbHVlfTsKCiAgICBteSAkVXNlck9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpVc2VyJyk7CgogICAgbXkgJVVzZXIgPSAkVXNlck9iamVjdC0+R2V0VXNlckRhdGEoCiAgICAgICAgVXNlcklEID0+ICRQYXJhbXtWYWx1ZX0sCiAgICApOwoKICAgIGlmICglVXNlcikgewoKICAgICAgICBteSAkVXNlclZhbHVlID0gc3ByaW50ZiAnIiVzICVzIiA8JXM+JywKICAgICAgICAgICAgJFVzZXJ7VXNlckZpcnN0bmFtZX0sCiAgICAgICAgICAgICRVc2Vye1VzZXJMYXN0bmFtZX0sCiAgICAgICAgICAgICRVc2Vye1VzZXJFbWFpbH07CgogICAgICAgIHJldHVybiAkVXNlclZhbHVlOwogICAgfQoKICAgIHJldHVybiAkUGFyYW17VmFsdWV9OwoKfQoKPWhlYWQyIFN0YXRzQXR0cmlidXRlQ3JlYXRlKCkKCmNyZWF0ZSBhIGF0dHJpYnV0ZSBhcnJheSBmb3IgdGhlIHN0YXRzIGZyYW1ld29yawoKICAgIG15ICRBdHRyaWJ1dGUgPSAkQmFja2VuZE9iamVjdC0+U3RhdHNBdHRyaWJ1dGVDcmVhdGUoCiAgICAgICAgS2V5ICA9PiAnS2V5OjpTdWJrZXknLAogICAgICAgIE5hbWUgPT4gJ05hbWUnLAogICAgICAgIEl0ZW0gPT4gJEl0ZW1SZWYsCiAgICApOwoKPWN1dAoKc3ViIFN0YXRzQXR0cmlidXRlQ3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgTkVFREVEOgogICAgZm9yIG15ICROZWVkZWQgKHF3KEtleSBOYW1lIEl0ZW0pKSB7CgogICAgICAgIG5leHQgTkVFREVEIGlmIGRlZmluZWQgJFBhcmFteyROZWVkZWR9OwoKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gIk5lZWQgJE5lZWRlZCEiLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICMgY3JlYXRlIGF0dHJpYnV0ZQogICAgbXkgJEF0dHJpYnV0ZSA9IFsKICAgICAgICB7CiAgICAgICAgICAgIE5hbWUgICAgICAgICAgICAgPT4gJFBhcmFte05hbWV9LAogICAgICAgICAgICBVc2VBc1h2YWx1ZSAgICAgID0+IDEsCiAgICAgICAgICAgIFVzZUFzVmFsdWVTZXJpZXMgPT4gMSwKICAgICAgICAgICAgVXNlQXNSZXN0cmljdGlvbiA9PiAxLAogICAgICAgICAgICBFbGVtZW50ICAgICAgICAgID0+ICRQYXJhbXtLZXl9LAogICAgICAgICAgICBCbG9jayAgICAgICAgICAgID0+ICdJbnB1dEZpZWxkJywKICAgICAgICB9LAogICAgXTsKCiAgICByZXR1cm4gJEF0dHJpYnV0ZTsKfQoKPWhlYWQyIEV4cG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgpCgpwcmVwYXJlIHNlYXJjaCB2YWx1ZSBmb3IgZXhwb3J0CgogICAgbXkgJEFycmF5UmVmID0gJEJhY2tlbmRPYmplY3QtPkV4cG9ydFNlYXJjaFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEV4cG9ydFNlYXJjaFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBFeHBvcnRWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSB2YWx1ZSBmb3IgZXhwb3J0CgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkV4cG9ydFZhbHVlUHJlcGFyZSgKICAgICAgICBWYWx1ZSA9PiAxMSwgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEV4cG9ydFZhbHVlUHJlcGFyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiBpZiAhZGVmaW5lZCAkUGFyYW17VmFsdWV9OwogICAgcmV0dXJuICRQYXJhbXtWYWx1ZX07Cn0KCj1oZWFkMiBJbXBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoKQoKcHJlcGFyZSBzZWFyY2ggdmFsdWUgZm9yIGltcG9ydAoKICAgIG15ICRBcnJheVJlZiA9ICRCYWNrZW5kT2JqZWN0LT5JbXBvcnRTZWFyY2hWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBJbXBvcnRTZWFyY2hWYWx1ZVByZXBhcmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gaWYgIWRlZmluZWQgJFBhcmFte1ZhbHVlfTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9Owp9Cgo9aGVhZDIgSW1wb3J0VmFsdWVQcmVwYXJlKCkKCnByZXBhcmUgdmFsdWUgZm9yIGltcG9ydAoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5JbXBvcnRWYWx1ZVByZXBhcmUoCiAgICAgICAgVmFsdWUgPT4gMTEsICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBJbXBvcnRWYWx1ZVByZXBhcmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gaWYgIWRlZmluZWQgJFBhcmFte1ZhbHVlfTsKICAgIHJldHVybiAkUGFyYW17VmFsdWV9Owp9CgoxOwo=
# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::System::ITSMConfigItemCustomerCIs;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::LinkObject',
    'Kernel::System::Log',
    'Kernel::System::Ticket',
);

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

=head1 NAME

Kernel::System::ITSMConfigItemCustomerCIs - ITSMConfigItemCustomerCIs lib

=head1 SYNOPSIS

All ITSMConfigItemCustomerCIs functions

=head1 PUBLIC INTERFACE

=head2 new()

    Don't use the constructor directly, use the ObjectManager instead:

    my $ITSMConfigItemCustomerCIsObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItemCustomerCIs');

=cut

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

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

    return $Self;
}

=head2 GetCustomerUserCIs()

Searches for all customer user specific config items

    my @ConfigItemIDs = $ITSMConfigItemCustomerCIsObject->GetCustomerUserCIs(
        CustomerUserID => $CustomerUserLogin,                       # the CustomerUserID is the UserLogin
    );

Returns:

    my @ConfigItemIDs = [1,2,3,4];

=cut

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

    my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
    my $LogObject            = $Kernel::OM->Get('Kernel::System::Log');

    NEEDED:
    for my $Needed (qw(CustomerUserID)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed in GetCustomerUserCIs!",
        );
        return;
    }

    my %ConfigItemKey = %{
        $ConfigObject->Get('AgentCustomerUserInformationCenter::Backend')->{'0060-CUIC-ITSMConfigItemCustomerUser'}
            ->{ConfigItemKey}
            || {}
    };

    my $Classes = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );
    my %Classes = reverse %{$Classes};

    my @ConfigItemIDs;
    ENTRY:
    for my $Class ( sort keys %ConfigItemKey ) {

        # only valid classes
        next ENTRY if !$Classes{$Class};

        my $Attribute = $ConfigItemKey{$Class};

        my $ItemDataRef = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::Class',
            Name  => $Class,
        );
        next ENTRY if !IsHashRefWithData($ItemDataRef);

        my $What = '[%]{' . "'Version'" . '}[%]{' . "'" . $Attribute . "'" . '}[%]{' . "'Content'" . '}';

        my $ConfigItemIDs = $ConfigItemObject->ConfigItemSearchExtended(
            ClassIDs => [ $ItemDataRef->{ItemID} ],
            What     => [
                {
                    $What => $Param{CustomerUserID},
                },
            ],
            Limit => 100_000,
        );
        next ENTRY if !IsArrayRefWithData($ConfigItemIDs);

        push @ConfigItemIDs, @{$ConfigItemIDs};
    }

    return @ConfigItemIDs;
}

=head2 GetCustomerCIs()

Searches for all customer specific config items

    my @ConfigItemIDs = $ITSMConfigItemCustomerCIsObject->GetCustomerCIs(
        CustomerID => $CustomerID,
    );

Returns:

    my @ConfigItemIDs = [1,2,3,4];

=cut

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

    my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
    my $LogObject            = $Kernel::OM->Get('Kernel::System::Log');

    NEEDED:
    for my $Needed (qw(CustomerID)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed in GetCustomerCIs!",
        );
        return;
    }

    my %ConfigItemKey = %{
        $ConfigObject->Get('AgentCustomerInformationCenter::Backend')->{'0060-CIC-ITSMConfigItemCustomerCompany'}
            ->{ConfigItemKey}
            || {}
    };

    my $Classes = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );
    my %Classes = reverse %{$Classes};

    my @ConfigItemIDs;
    ENTRY:
    for my $Class ( sort keys %ConfigItemKey ) {

        # only valid classes
        next ENTRY if !$Classes{$Class};

        my $Attribute = $ConfigItemKey{$Class};

        my $ItemDataRef = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::Class',
            Name  => $Class,
        );
        next ENTRY if !IsHashRefWithData($ItemDataRef);

        my $What = '[%]{' . "'Version'" . '}[%]{' . "'" . $Attribute . "'" . '}[%]{' . "'Content'" . '}';

        my $ConfigItemIDs = $ConfigItemObject->ConfigItemSearchExtended(
            ClassIDs => [ $ItemDataRef->{ItemID} ],
            What     => [
                {
                    $What => $Param{CustomerID},
                },
            ],
            Limit => 100_000,
        );

        next ENTRY if !IsArrayRefWithData($ConfigItemIDs);

        push @ConfigItemIDs, @{$ConfigItemIDs};
    }

    return @ConfigItemIDs;
}

=head2 GetPossibleCustomerCIs()

Returns possible customer CIs.

    my @ConfigItems = $ITSMConfigItemCustomerCIsObject->GetPossibleCustomerCIs(
        CustomerUserID => 123,
        TicketID       => 123,      # optional
    );

Returns:

    my @ConfigItems = (
        {
            'Linked'       => '1',
            'ConfigItemID' => 123456,
            'Name'         => 'Computer xyz',
        }
    );

=cut

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

    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $ConfigObject     = $Kernel::OM->Get('Kernel::Config');
    my $LinkObject       = $Kernel::OM->Get('Kernel::System::LinkObject');
    my $TicketObject     = $Kernel::OM->Get('Kernel::System::Ticket');

    my $Config = $ConfigObject->Get('ITSMConfigItem::Frontend::AgentITSMConfigItemCustomerCIs');

    my %LinkKeyList;

    if ( $Param{TicketID} ) {
        my %Ticket = $TicketObject->TicketGet(
            TicketID => $Param{TicketID},
            UserID   => 1,
        );

        $Param{CustomerUserID} = $Ticket{CustomerUserID};
        $Param{CustomerID}     = $Ticket{CustomerID};

        %LinkKeyList = $LinkObject->LinkKeyList(
            Object1 => 'Ticket',
            Key1    => $Param{TicketID},
            Object2 => 'ITSMConfigItem',
            State   => 'Valid',
            UserID  => 1,
        );
    }

    my @ConfigItems;
    return @ConfigItems if !( $Param{CustomerUserID} || $Param{CustomerID} );

    my @CustomerUserConfigItemIDs;
    if ( $Config->{CustomerUser} ) {
        @CustomerUserConfigItemIDs = $Self->GetCustomerUserCIs(
            CustomerUserID => $Param{CustomerUserID},
            Limit          => $Config->{SearchLimit},
        );
    }
    my @CustomerConfigItemIDs;
    if ( $Config->{CustomerCompany} ) {
        @CustomerConfigItemIDs = $Self->GetCustomerCIs(
            CustomerID => $Param{CustomerID},
            Limit      => $Config->{SearchLimit},
        );
    }

    my @ConfigItemIDs   = ( @CustomerUserConfigItemIDs, @CustomerConfigItemIDs );
    my %UniqConfigItems = map { $_ => $_ } @ConfigItemIDs;
    @ConfigItemIDs = sort keys %UniqConfigItems;

    my %Mapping = %{ $Config->{Mapping} || {} };

    for my $ConfigItemID (@ConfigItemIDs) {
        my $Version = $ConfigItemObject->VersionGet(
            ConfigItemID => $ConfigItemID,
        );

        push @ConfigItems, {
            ConfigItemID => $ConfigItemID,
            Name         => $Version->{Name},
            Number       => $Version->{Number},
            Class        => $Version->{Class},
            Icon         => $Mapping{ $Version->{Class} } || '',
            Linked       => $LinkKeyList{$ConfigItemID} || 0,
        };
    }

    return @ConfigItems;
}
1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::System::ITSMConfigItemInvoker;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::System::CustomerCompany',
    'Kernel::System::CustomerUser',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::Log',
    'Kernel::System::XML',
);

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

=head1 NAME

Kernel::System::ITSMConfigItemInvoker

=head1 PUBLIC INTERFACE

=head2 new()

    Don't use the constructor directly, use the ObjectManager instead:

    my $ITSMConfigItemInvokerObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItemInvoker');

=cut

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

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

    return $Self;
}

=head2 GetConfigItemData()

    Returns hash with complete data of config item with given ID.

    my $ConfigItemData = $ITSMConfigItemInvokerObject->GetConfigItemData(
        ConfigItemID => 21,

        # OR
        VersionID => 52,

        Event => 'DeploymentStateUpdate', # optional; Will include data of previous version for certain events.
    );

=cut

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

    my $LogObject        = $Kernel::OM->Get('Kernel::System::Log');
    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

    if (
        ( !defined $Param{ConfigItemID} && !defined $Param{VersionID} )
        || ( defined $Param{ConfigItemID} && defined $Param{VersionID} )
        )
    {
        $LogObject->Log(
            Priority => 'error',
            Message  => 'Either parameter ConfigItemID or Version ID has to be given.',
        );
        return;
    }

    my $CurrentConfigItemVersion = $ConfigItemObject->VersionGet(
        ConfigItemID => $Param{ConfigItemID},
        VersionID    => $Param{VersionID},
        XMLDataGet   => 1,
    );
    return if !IsHashRefWithData($CurrentConfigItemVersion);

    my %ConfigItemData;
    $ConfigItemData{XMLData} ||= {};
    if ( IsHashRefWithData( $CurrentConfigItemVersion->{XMLData}->[1]->{Version}->[1] ) ) {
        $Self->_XML2Data(
            Result     => $ConfigItemData{XMLData},
            Data       => $CurrentConfigItemVersion->{XMLData}->[1]->{Version}->[1],
            Definition => $CurrentConfigItemVersion->{XMLDefinition},
        );
    }

    for my $Field (qw(ConfigItemID Name ClassID Class DefinitionID DeplStateID DeplState InciStateID InciState)) {
        $ConfigItemData{$Field} = $CurrentConfigItemVersion->{$Field};
    }

    return \%ConfigItemData if !defined $Param{Event};

    my %EventsToIncludePreviousVersion = (
        DeploymentStateUpdate => 1,
        IncidentStateUpdate   => 1,
        NameUpdate            => 1,
        ValueUpdate           => 1,
        VersionCreate         => 1,
    );

    return \%ConfigItemData if !$EventsToIncludePreviousVersion{ $Param{Event} };

    my $ConfigItemVersions = $ConfigItemObject->VersionList(
        ConfigItemID => $Param{ConfigItemID},
    );
    return if !IsArrayRefWithData($ConfigItemVersions);
    return if @{$ConfigItemVersions} < 2;

    my $PreviousVersionID = $ConfigItemVersions->[-2];

    $ConfigItemData{PreviousConfigItemVersion} = $Self->GetConfigItemData(
        VersionID => $PreviousVersionID,
    );

    if ( $Param{Event} eq 'DeploymentStateUpdate' ) {
        $ConfigItemData{DeploymentStateUpdate} = {
            Old => {
                ID      => $ConfigItemData{PreviousConfigItemVersion}->{DeplStateID},
                Content => $ConfigItemData{PreviousConfigItemVersion}->{DeplState},
            },
            New => {
                ID      => $ConfigItemData{DeplStateID},
                Content => $ConfigItemData{DeplState},
            },
        };
    }
    elsif ( $Param{Event} eq 'IncidentStateUpdate' ) {
        $ConfigItemData{IncidentStateUpdate} = {
            Old => {
                ID      => $ConfigItemData{PreviousConfigItemVersion}->{InciStateID},
                Content => $ConfigItemData{PreviousConfigItemVersion}->{InciState},
            },
            New => {
                ID      => $ConfigItemData{InciStateID},
                Content => $ConfigItemData{InciState},
            },
        };
    }
    elsif ( $Param{Event} eq 'NameUpdate' ) {
        $ConfigItemData{NameUpdate} = {
            Old => {
                Content => $ConfigItemData{PreviousConfigItemVersion}->{Name},
            },
            New => {
                Content => $ConfigItemData{Name},
            },
        };
    }
    elsif ( $Param{Event} eq 'ValueUpdate' ) {
        my $ChangedConfigItemValues = $Self->_FindChangedConfigItemValues(
            ConfigItemID => $Param{ConfigItemID},
        );

        $ConfigItemData{ValueUpdate} = $ChangedConfigItemValues // {};
    }

    return \%ConfigItemData;
}

=head2 _XML2Data()

    Turns XML into Perl structure.

    my $Success = $ITSMConfigItemInvokerObject->_XML2Data(
        Parent          => $Identifier,          # optional: contains the field name of the parent XML
        Result          => $Result,              # contains the reference to the result hash
        Data            => $Data{$Field}->[1],   # contains the XML hash to be parsed
    );

    Returns true value on success.

=cut

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

    my $Result = $Param{Result};
    my $Parent = $Param{Parent} || '';
    my %Data   = %{ $Param{Data} || {} };

    FIELD:
    for my $Field ( sort keys %Data ) {
        next FIELD if !IsArrayRefWithData( $Data{$Field} );

        my $FieldDefinitionIndex = 0;
        my $FieldType;

        FIELDDEFINITION:
        for my $FieldDefinition ( @{ $Param{Definition} // [] } ) {
            if ( defined $FieldDefinition->{Key} && $FieldDefinition->{Key} eq $Field ) {
                $FieldType = $FieldDefinition->{Input}->{Type};
                last FIELDDEFINITION;
            }

            $FieldDefinitionIndex++;
        }

        $Result->{$Field} = [];

        for my $Index ( 1 .. $#{ $Data{$Field} } ) {
            my $Value = $Data{$Field}->[$Index]->{Content};

            my $CurrentResult = {};

            my $Definition = $Param{Definition}->[$FieldDefinitionIndex]->{Sub} // [];

            $Self->_XML2Data(
                %Param,
                Parent     => $Field,
                Result     => $CurrentResult,
                Data       => $Data{$Field}->[$Index],
                Definition => $Definition,
            );

            if ( defined $Value ) {
                $CurrentResult->{Content} = $Value;

                my $ReadableValue = $Self->_GetReadableValue(
                    Value     => $Value,
                    FieldType => $FieldType,
                );

                if ( $ReadableValue ne $Value ) {
                    $CurrentResult->{ReadableValue} = $ReadableValue;
                }

                if ( keys %{$CurrentResult} ) {
                    push @{ $Result->{$Field} }, $CurrentResult;
                }
            }
        }
    }

    return 1;
}

=head2 _GetReadableValue()

    Maps value of certain field types (ID) to a value that can be read (name, text, etc.).

    my $ReadableValue = $ITSMConfigItemInvokerObject->_GetReadableValue(
        Value     => 4,
        FieldType => 'GeneralCatalog',
    );

    Returns the readable value depending on the field type or the original value
    if no readable value could be determined.

=cut

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

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

    my $Value = $Param{Value};

    return $Value if !defined $Param{Value};
    return $Value if !length $Param{Value};

    return $Value if !defined $Param{FieldType};
    return $Value if !length $Param{FieldType};

    my $ReadableValue = $Value;

    if ( $Param{FieldType} eq 'GeneralCatalog' ) {
        my $GeneralCatalogItemData = $GeneralCatalogObject->ItemGet(
            ItemID => $Value,
        );
        return $Value if !IsHashRefWithData($GeneralCatalogItemData);
        return $Value if !defined $GeneralCatalogItemData->{Name};

        $ReadableValue = $GeneralCatalogItemData->{Name};
    }
    elsif ( $Param{FieldType} eq 'CustomerCompany' ) {
        my %CustomerCompany = $CustomerCompanyObject->CustomerCompanyGet(
            CustomerID => $Value,
        );
        return $Value if !%CustomerCompany;

        delete $CustomerCompany{Config};
        delete $CustomerCompany{CompanyConfig};

        $ReadableValue = \%CustomerCompany;
    }
    elsif ( $Param{FieldType} eq 'Customer' ) {
        my %CustomerUser = $CustomerUserObject->CustomerUserDataGet(
            User => $Value,
        );
        return $Value if !%CustomerUser;

        delete $CustomerUser{Config};
        delete $CustomerUser{CompanyConfig};

        $ReadableValue = \%CustomerUser;
    }

    return $ReadableValue;
}

=head2 _FindChangedConfigItemValues()

The following function is based on Kernel::System::ITSMConfigurationManagement::_FindChangedXMLValues()

Origin: ITSMConfigurationManagement - f5afe974d300056e368a24237962d10bf25f1cd0 - Kernel/System/ITSMConfigItem/Version.pm
Copyright (C) 2001-2021 OTRS AG, https://otrs.com/

Compares XML data of current config item version with the previous one (if one exists).

    my $ChangedConfigItemValues = $ITSMConfigItemInvokerObject->_FindChangedConfigItemValues(
        ConfigItemID => 123,
    );

Returns:

    my $ChangedConfigItemValues = {
        'NIC::2::IPoverDHCP::1' => {
            New => {
                Content       => '38',
                ReadableValue => 'No',
            },
            Old => {
                Content => '',
            },
        },
    };

=cut

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

    my $LogObject        = $Kernel::OM->Get('Kernel::System::Log');
    my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
    my $XMLObject        = $Kernel::OM->Get('Kernel::System::XML');

    NEEDED:
    for my $Needed (qw(ConfigItemID)) {
        next NEEDED if $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Need $Needed!",
        );
        return;
    }

    my $Versions = $ConfigItemObject->VersionList(
        ConfigItemID => $Param{ConfigItemID},
    );

    return if !@{$Versions};
    return if @{$Versions} < 2;

    my $PreviousVersion = $ConfigItemObject->VersionGet(
        VersionID => $Versions->[-2],
    );
    return if !IsHashRefWithData($PreviousVersion);

    my $CurrentVersion = $ConfigItemObject->VersionGet(
        VersionID => $Versions->[-1],
    );
    return if !IsHashRefWithData($CurrentVersion);

    my $CurrentXMLData  = $CurrentVersion->{XMLData};
    my $PreviousXMLData = $PreviousVersion->{XMLData};

    my $Definition = $CurrentVersion->{XMLDefinition};

    # Get all tag keys in current and previous XML data.
    # Use a side effect of XMLHash2D() which adds the tag keys to the passed in data structure.
    $XMLObject->XMLHash2D( XMLHash => $CurrentXMLData );
    my @TagKeys = $ConfigItemObject->_GrabTagKeys(
        Data => [
            $PreviousXMLData,
            $CurrentXMLData,
        ],
    );

    my %UniqueTagKeys = map { $_ => 1 } @TagKeys;

    my %ChangedConfigItemValues;
    my %SuppressVersionAdd;

    TAGKEY:
    for my $TagKey ( sort keys %UniqueTagKeys ) {
        my $CurrentContent  = eval '$CurrentXMLData->' . $TagKey . '->{Content}'  || '';    ## no critic
        my $PreviousContent = eval '$PreviousXMLData->' . $TagKey . '->{Content}' || '';    ## no critic

        next TAGKEY if $CurrentContent eq $PreviousContent;

        # Remove leading 'Version' key.
        ( my $HashKey = $TagKey ) =~ s{\A\[\d+\]\{'Version'\}\[\d+\]}{};

        # Remove {''} around keys.
        $HashKey =~ s{(\{'|'\})}{}g;

        # Substitute [1] (index) with ::1::
        $HashKey =~ s{\[(\d+)\]}{::$1::}g;

        # Remove trailing ::
        $HashKey =~ s{::\z}{};

        $ChangedConfigItemValues{$HashKey} = {
            Old => {
                Content => $PreviousContent,
            },
            New => {
                Content => $CurrentContent,
            },
        };

        # Check for general catalog entry and add its name.
        ( my $AttributePath = $HashKey ) =~ s{::\d+}{}g;
        next TAGKEY if !defined $AttributePath;
        next TAGKEY if !length $AttributePath;

        my $AttributeInfo = $ConfigItemObject->DefinitionAttributeInfo(
            AttributePath => $AttributePath,
            Definition    => $Definition,
        );
        next TAGKEY if !IsHashRefWithData($AttributeInfo);

        my $PreviousReadableValue = $Self->_GetReadableValue(
            Value     => $PreviousContent,
            FieldType => $AttributeInfo->{Input}->{Type} // '',
        );

        my $CurrentReadableValue = $Self->_GetReadableValue(
            Value     => $CurrentContent,
            FieldType => $AttributeInfo->{Input}->{Type} // '',
        );

        if ( $PreviousReadableValue ne $PreviousContent ) {
            $ChangedConfigItemValues{$HashKey}->{Old}->{ReadableValue} = $PreviousReadableValue;
        }

        if ( $CurrentReadableValue ne $CurrentContent ) {
            $ChangedConfigItemValues{$HashKey}->{New}->{ReadableValue} = $CurrentReadableValue;
        }
    }

    return \%ChangedConfigItemValues;
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::ImportExport::ObjectBackend::ITSMConfigItem;

use strict;
use warnings;

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

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

=head1 NAME

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

=head1 DESCRIPTION

All functions to import and export ITSM config items.

=head1 PUBLIC INTERFACE

=head2 new()

create an object

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

=cut

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

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

    return $Self;
}

=head2 ObjectAttributesGet()

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

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

=cut

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

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

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    ) || {};

    my $Attributes = [
        {
            Key   => 'ClassID',
            Name  => Translatable('Class'),
            Input => {
                Type         => 'Selection',
                Data         => $ClassList,
                Required     => 1,
                Translation  => 0,
                PossibleNone => 1,
            },
        },
        {
            Key   => 'CountMax',
            Name  => Translatable('Maximum number of one element'),
            Input => {
                Type         => 'Text',
                ValueDefault => '10',
                Required     => 1,
                Regex        => qr{ \A \d+ \z }xms,
                Translation  => 0,
                Size         => 5,
                MaxLength    => 5,
                DataType     => 'IntegerBiggerThanZero',
            },
        },
        {
            Key   => 'EmptyFieldsLeaveTheOldValues',
            Name  => Translatable('Empty fields indicate that the current values are kept'),
            Input => {
                Type => 'Checkbox',
            },
        },
    ];

    return $Attributes;
}

=head2 MappingObjectAttributesGet()

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

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

=cut

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

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

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

    return [] if !$ObjectData;
    return [] if ref $ObjectData ne 'HASH';
    return [] if !$ObjectData->{ClassID};

    # get definition
    my $XMLDefinition = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->DefinitionGet(
        ClassID => $ObjectData->{ClassID},
    );

    return [] if !$XMLDefinition;
    return [] if ref $XMLDefinition ne 'HASH';
    return [] if !$XMLDefinition->{DefinitionRef};
    return [] if ref $XMLDefinition->{DefinitionRef} ne 'ARRAY';

    my $ElementList = [
        {
            Key   => 'Number',
            Value => Translatable('Number'),
        },
        {
            Key   => 'Name',
            Value => Translatable('Name'),
        },
        {
            Key   => 'DeplState',
            Value => Translatable('Deployment State'),
        },
        {
            Key   => 'InciState',
            Value => Translatable('Incident State'),
        },
    ];

    # add xml elements
    $Self->_MappingObjectAttributesGet(
        XMLDefinition => $XMLDefinition->{DefinitionRef},
        ElementList   => $ElementList,
        CountMaxLimit => $ObjectData->{CountMax} || 10,
    );

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

    return $Attributes;
}

=head2 SearchAttributesGet()

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

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

=cut

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

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

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

    return [] if !$ObjectData;
    return [] if ref $ObjectData ne 'HASH';
    return [] if !$ObjectData->{ClassID};

    # get definition
    my $XMLDefinition = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->DefinitionGet(
        ClassID => $ObjectData->{ClassID},
    );

    return [] if !$XMLDefinition;
    return [] if ref $XMLDefinition ne 'HASH';
    return [] if !$XMLDefinition->{DefinitionRef};
    return [] if ref $XMLDefinition->{DefinitionRef} ne 'ARRAY';

    # get deployment state list
    my $DeplStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    ) || {};

    # get incident state list
    my $InciStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::Core::IncidentState',
    ) || {};

    my $AttributeList = [
        {
            Key   => 'Number',
            Name  => Translatable('Number'),
            Input => {
                Type      => 'Text',
                Size      => 80,
                MaxLength => 255,
            },
        },
        {
            Key   => 'Name',
            Name  => Translatable('Name'),
            Input => {
                Type      => 'Text',
                Size      => 80,
                MaxLength => 255,
            },
        },
        {
            Key   => 'DeplStateIDs',
            Name  => Translatable('Deployment State'),
            Input => {
                Type        => 'Selection',
                Data        => $DeplStateList,
                Translation => 1,
                Size        => 5,
                Multiple    => 1,
            },
        },
        {
            Key   => 'InciStateIDs',
            Name  => Translatable('Incident State'),
            Input => {
                Type        => 'Selection',
                Data        => $InciStateList,
                Translation => 1,
                Size        => 5,
                Multiple    => 1,
            },
        },
    ];

    # add xml attributes
    $Self->_SearchAttributesGet(
        XMLDefinition => $XMLDefinition->{DefinitionRef},
        AttributeList => $AttributeList,
    );

    return $AttributeList;
}

=head2 ExportDataGet()

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

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

=cut

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

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

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

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

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    return if !$ClassList || ref $ClassList ne 'HASH';

    # check the class id
    if ( !$ObjectData->{ClassID} || !$ClassList->{ $ObjectData->{ClassID} } ) {

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

    # get the mapping list
    my $MappingList = $Kernel::OM->Get('Kernel::System::ImportExport')->MappingList(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

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

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

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

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

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

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

        push @MappingObjectList, $MappingObjectData;
    }

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

    return if !$SearchData || ref $SearchData ne 'HASH';

    # get deployment state list
    my $DeplStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    # check deployment state list
    if ( !$DeplStateList || ref $DeplStateList ne 'HASH' ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Can't get the general catalog list ITSM::ConfigItem::DeploymentState!",
        );
        return;
    }

    # get incident state list
    my $InciStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::Core::IncidentState',
    );

    # check incident state list
    if ( !$InciStateList || ref $InciStateList ne 'HASH' ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Can't get the general catalog list ITSM::Core::IncidentState!",
        );
        return;
    }

    # get current definition of this class
    my $DefinitionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->DefinitionGet(
        ClassID => $ObjectData->{ClassID},
        UserID  => $Param{UserID},
    );

    my %SearchParams;

    # add number to the search params
    if ( $SearchData->{Number} ) {
        $SearchParams{Number} = delete $SearchData->{Number};
    }

    # add name to the search params
    if ( $SearchData->{Name} ) {
        $SearchParams{Name} = delete $SearchData->{Name};
    }

    # add deployment state to the search params
    if ( $SearchData->{DeplStateIDs} ) {
        my @DeplStateIDs = split '#####', $SearchData->{DeplStateIDs};
        $SearchParams{DeplStateIDs} = \@DeplStateIDs;
        delete $SearchData->{DeplStateIDs};
    }

    # add incident state to the search params
    if ( $SearchData->{InciStateIDs} ) {
        my @InciStateIDs = split '#####', $SearchData->{InciStateIDs};
        $SearchParams{InciStateIDs} = \@InciStateIDs;
        delete $SearchData->{InciStateIDs};
    }

    # add all XML data to the search params
    my @SearchParamsWhat;
    $Self->_ExportXMLSearchDataPrepare(
        XMLDefinition => $DefinitionData->{DefinitionRef},
        What          => \@SearchParamsWhat,
        SearchData    => $SearchData,
    );

    # add XML search params to the search hash
    if (@SearchParamsWhat) {
        $SearchParams{What} = \@SearchParamsWhat;
    }

    # search the config items
    my $ConfigItemList = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemSearchExtended(
        %SearchParams,
        ClassIDs              => [ $ObjectData->{ClassID} ],
        PreviousVersionSearch => 0,
        UserID                => $Param{UserID},
    );

    my @ExportData;
    CONFIGITEMID:
    for my $ConfigItemID ( @{$ConfigItemList} ) {

        # get last version
        my $VersionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
            ConfigItemID => $ConfigItemID,
        );

        next CONFIGITEMID if !$VersionData;
        next CONFIGITEMID if ref $VersionData ne 'HASH';

        # translate xmldata to a 2d hash
        my %XMLData2D;
        $Self->_ExportXMLDataPrepare(
            XMLDefinition => $DefinitionData->{DefinitionRef},
            XMLData       => $VersionData->{XMLData}->[1]->{Version}->[1],
            XMLData2D     => \%XMLData2D,
        );

        # add data to the export data array
        my @Item;
        MAPPINGOBJECT:
        for my $MappingObject (@MappingObjectList) {

            # extract key
            my $Key = $MappingObject->{Key};

            # handle empty key
            if ( !$Key ) {
                push @Item, '';
                next MAPPINGOBJECT;
            }

            # handle config item number
            if ( $Key eq 'Number' ) {
                push @Item, $VersionData->{Number};
                next MAPPINGOBJECT;
            }

            # handle current config item name
            if ( $Key eq 'Name' ) {
                push @Item, $VersionData->{Name};
                next MAPPINGOBJECT;
            }

            # handle deployment state
            if ( $Key eq 'DeplState' ) {
                $VersionData->{DeplStateID} ||= 'DUMMY';
                push @Item, $DeplStateList->{ $VersionData->{DeplStateID} };
                next MAPPINGOBJECT;
            }

            # handle incident state
            if ( $Key eq 'InciState' ) {
                $VersionData->{InciStateID} ||= 'DUMMY';
                push @Item, $InciStateList->{ $VersionData->{InciStateID} };
                next MAPPINGOBJECT;
            }

            # handle all XML data elements
            push @Item, $XMLData2D{$Key};
        }

        push @ExportData, \@Item;
    }

    return \@ExportData;
}

=head2 ImportDataSave()

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

The current version of the config item will never be deleted. When there are no
changes in the data, the import will be skipped. When there is new or changed data,
then a new config item or a new version is created.

In the case of changed data, the new version of the config item will contain the
attributes of the C<ImportDataRow> plus the old attributes that are
not part of the import definition.
Thus ImportDataSave() guarantees to not overwrite undeclared attributes.

The behavior when imported attributes are empty depends on the setting in the object data.
When C<EmptyFieldsLeaveTheOldValues> is not set, then empty values will wipe out
the old data. This is the default behavior. When C<EmptyFieldsLeaveTheOldValues> is set,
then empty values will leave the old values.

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

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

An empty C<ConfigItemID> indicates failure. Otherwise it indicates the
location of the imported data.
C<RetCode> is either 'Created', 'Updated' or 'Skipped'. 'Created' means that a new
config item has been created. 'Updated' means that a new version has been added to
an existing config item. 'Skipped' means that no new version has been created,
as the new data is identical to the latest version of an existing config item.

No codes have yet been defined for the failure case.

=cut

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

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

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

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

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

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

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # check class list
    if ( !$ClassList || ref $ClassList ne 'HASH' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message =>
                "Can't import entity $Param{Counter}: "
                . "Can't get the general catalog list ITSM::ConfigItem::Class",
        );
        return;
    }

    # check the class id
    if ( !$ObjectData->{ClassID} || !$ClassList->{ $ObjectData->{ClassID} } ) {

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

    # get the mapping list
    my $MappingList = $Kernel::OM->Get('Kernel::System::ImportExport')->MappingList(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

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

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

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

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

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

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

        push @MappingObjectList, $MappingObjectData;
    }

    # check and remember the Identifiers
    # the Identifiers identify the config item that should be updated
    my %Identifier;
    my $RowIndex = 0;
    MAPPINGOBJECTDATA:
    for my $MappingObjectData (@MappingObjectList) {

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

        # check if identifier already exists
        if ( $Identifier{ $MappingObjectData->{Key} } ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message =>
                    "Can't import entity $Param{Counter}: "
                    . "'$MappingObjectData->{Key}' has been used multiple times as an identifier",
            );
            return;
        }

        # set identifier value
        $Identifier{ $MappingObjectData->{Key} } = $Param{ImportDataRow}->[$RowIndex];

        next MAPPINGOBJECTDATA if $MappingObjectData->{Key} && $Param{ImportDataRow}->[$RowIndex];

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message =>
                "Can't import entity $Param{Counter}: "
                . "Identifier field is empty",
        );

        return;
    }
    continue {
        $RowIndex++;
    }

    # get deployment state list
    my $DeplStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    # check deployment state list
    if ( !$DeplStateList || ref $DeplStateList ne 'HASH' ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message =>
                "Can't import entity $Param{Counter}: "
                . "Can't get the general catalog list ITSM::ConfigItem::DeploymentState!",
        );
        return;
    }

    # reverse the deployment state list
    my %DeplStateListReverse = reverse %{$DeplStateList};

    # get incident state list
    my $InciStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::Core::IncidentState',
    );

    # check incident state list
    if ( !$InciStateList || ref $InciStateList ne 'HASH' ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message =>
                "Can't import entity $Param{Counter}: "
                . "Can't get the general catalog list ITSM::Core::IncidentState",
        );
        return;
    }

    # reverse the incident state list
    my %InciStateListReverse = reverse %{$InciStateList};

    # get current definition of this class
    my $DefinitionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->DefinitionGet(
        ClassID => $ObjectData->{ClassID},
        UserID  => $Param{UserID},
    );

    # check definition data
    if ( !$DefinitionData || ref $DefinitionData ne 'HASH' ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message =>
                "Can't import entity $Param{Counter}: "
                . "Can't get the definition of class id $ObjectData->{ClassID}",
        );
        return;
    }

    # try to get config item ids, when there are identifiers
    my $ConfigItemID;
    if (%Identifier) {

        my %SearchParams;

        # add number to the search params
        if ( $Identifier{Number} ) {
            $SearchParams{Number} = delete $Identifier{Number};
        }

        # add name to the search params
        if ( $Identifier{Name} ) {
            $SearchParams{Name} = delete $Identifier{Name};
        }

        # add deployment state to the search params
        if ( $Identifier{DeplState} ) {

            # extract deployment state id
            my $DeplStateID = $DeplStateListReverse{ $Identifier{DeplState} } || '';

            if ( !$DeplStateID ) {

                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message =>
                        "Can't import entity $Param{Counter}: "
                        . "The deployment state '$Identifier{DeplState}' is invalid",
                );
                return;
            }

            $SearchParams{DeplStateIDs} = [$DeplStateID];
            delete $Identifier{DeplState};
        }

        # add incident state to the search params
        if ( $Identifier{InciState} ) {

            # extract incident state id
            my $InciStateID = $InciStateListReverse{ $Identifier{InciState} } || '';

            if ( !$InciStateID ) {

                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message =>
                        "Can't import entity $Param{Counter}: "
                        . "The incident state '$Identifier{InciState}' is invalid",
                );
                return;
            }

            $SearchParams{InciStateIDs} = [$InciStateID];
            delete $Identifier{InciState};
        }

        # add all XML data to the search params
        my @SearchParamsWhat;
        $Self->_ImportXMLSearchDataPrepare(
            XMLDefinition => $DefinitionData->{DefinitionRef},
            What          => \@SearchParamsWhat,
            Identifier    => \%Identifier,
        );

        # add XML search params to the search hash
        if (@SearchParamsWhat) {
            $SearchParams{What} = \@SearchParamsWhat;
        }

        # search existing config item with the same identifiers
        my $ConfigItemList = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemSearchExtended(
            %SearchParams,
            ClassIDs              => [ $ObjectData->{ClassID} ],
            PreviousVersionSearch => 0,
            UsingWildcards        => 0,
            UserID                => $Param{UserID},
        );

        if ( scalar @{$ConfigItemList} > 1 ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message =>
                    "Can't import entity $Param{Counter}: "
                    . "Identifier fields NOT unique!",
            );
            return;
        }

        $ConfigItemID = $ConfigItemList->[0];
    }

    # get version data of the config item
    my $VersionData = {};
    if ($ConfigItemID) {

        # get latest version
        $VersionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
            ConfigItemID => $ConfigItemID,
        );

        # remove empty xml data
        if (
            !$VersionData->{XMLData}
            || ref $VersionData->{XMLData} ne 'ARRAY'
            || !@{ $VersionData->{XMLData} }
            )
        {
            delete $VersionData->{XMLData};
        }
    }

    # Check if current definition of this class has required attribute which does not exist in mapping list.
    for my $DefinitionRef ( @{ $DefinitionData->{DefinitionRef} } ) {
        my $Key = $DefinitionRef->{Key};

        if (
            $DefinitionRef->{Input}->{Required}
            && !scalar( grep { $_->{Key} =~ /\Q$DefinitionRef->{Key}\E(::(\d+))?$/ } @MappingObjectList )
            && (
                !defined $DefinitionRef->{CountMin}
                || ( defined $DefinitionRef->{CountMin} && $DefinitionRef->{CountMin} > 0 )
            )
            )
        {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message =>
                    "Can't import entity $Param{Counter}: "
                    . "Attribute '$Key' is required, but does not exist in mapping list!",
            );
            return;
        }
    }

    # set up fields in VersionData and in the XML attributes
    my %XMLData2D;
    $RowIndex = 0;
    for my $MappingObjectData (@MappingObjectList) {

        # just for convenience
        my $Key   = $MappingObjectData->{Key};
        my $Value = $Param{ImportDataRow}->[ $RowIndex++ ];

        if ( $Key eq 'Number' ) {

            # do nothing
            # Import does not override the config item number
        }
        elsif ( $Key eq 'Name' ) {

            if ( $EmptyFieldsLeaveTheOldValues && ( !defined $Value || $Value eq '' ) ) {

                # do nothing, keep the old value
            }
            else {
                if ( !$Value ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message =>
                            "Can't import entity $Param{Counter}: "
                            . "The name '$Value' is invalid!",
                    );
                    return;
                }

                $VersionData->{$Key} = $Value;
            }
        }
        elsif ( $Key eq 'DeplState' ) {

            if ( $EmptyFieldsLeaveTheOldValues && ( !defined $Value || $Value eq '' ) ) {

                # do nothing, keep the old value
            }
            else {

                # extract deployment state id
                my $DeplStateID = $DeplStateListReverse{$Value} || '';
                if ( !$DeplStateID ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message =>
                            "Can't import entity $Param{Counter}: "
                            . "The deployment state '$Value' is invalid!",
                    );
                    return;
                }

                $VersionData->{DeplStateID} = $DeplStateID;
            }
        }
        elsif ( $Key eq 'InciState' ) {

            if ( $EmptyFieldsLeaveTheOldValues && ( !defined $Value || $Value eq '' ) ) {

                # do nothing, keep the old value
            }
            else {

                # extract the deployment state id
                my $InciStateID = $InciStateListReverse{$Value} || '';
                if ( !$InciStateID ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message =>
                            "Can't import entity $Param{Counter}: "
                            . "The incident state '$Value' is invalid!",
                    );
                    return;
                }

                $VersionData->{InciStateID} = $InciStateID;
            }
        }
        else {

            # handle xml data
            $XMLData2D{$Key} = $Value;
        }
    }

    # set up empty container, in case there is no previous data
    $VersionData->{XMLData}->[1]->{Version}->[1] ||= {};

    # Edit XMLDataPrev, so that the values in XMLData2D take precedence.
    my $MergeOk = $Self->_ImportXMLDataMerge(
        XMLDefinition                => $DefinitionData->{DefinitionRef},
        XMLDataPrev                  => $VersionData->{XMLData}->[1]->{Version}->[1],
        XMLData2D                    => \%XMLData2D,
        EmptyFieldsLeaveTheOldValues => $EmptyFieldsLeaveTheOldValues,
    );

    # bail out, when the was a problem in _ImportXMLDataMerge()
    if ( !$MergeOk ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message =>
                "Can't import entity $Param{Counter}: "
                . "Could not prepare the input!",
        );
        return;
    }

    my $RetCode = $ConfigItemID ? Translatable('Changed') : Translatable('Created');

    # check if the feature to check for a unique name is enabled
    if (
        IsStringWithData( $VersionData->{Name} )
        && $Kernel::OM->Get('Kernel::Config')->Get('UniqueCIName::EnableUniquenessCheck')
        )
    {

        if ( $Kernel::OM->Get('Kernel::Config')->{Debug} > 0 ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'debug',
                Message  => "Checking for duplicate names (ClassID: $ObjectData->{ClassID}, "
                    . "Name: $VersionData->{Name}, ConfigItemID: " . $ConfigItemID || 'NEW' . ')',
            );
        }

        my $NameDuplicates = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->UniqueNameCheck(
            ConfigItemID => $ConfigItemID || 'NEW',
            ClassID      => $ObjectData->{ClassID},
            Name         => $VersionData->{Name},
        );

        # stop processing if the name is not unique
        if ( IsArrayRefWithData($NameDuplicates) ) {

            # build a string of all duplicate IDs
            my $NameDuplicatesString = join ', ', @{$NameDuplicates};

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message =>
                    "The name $VersionData->{Name} is already in use by the ConfigItemID(s): "
                    . $NameDuplicatesString,
            );

            # set the return code to also include the duplicate name
            $RetCode = "DuplicateName '$VersionData->{Name}'";

            # return undef for the config item id so it will be counted as 'Failed'
            return undef, $RetCode;    ## no critic
        }
    }

    my $LatestVersionID = 0;
    if ($ConfigItemID) {

        # the specified config item already exists
        # get id of the latest version, for checking later whether a version was created
        my $VersionList = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionList(
            ConfigItemID => $ConfigItemID,
        ) || [];
        if ( scalar @{$VersionList} ) {
            $LatestVersionID = $VersionList->[-1];
        }
    }
    else {

        # no config item was found, so add new config item
        $ConfigItemID = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemAdd(
            ClassID => $ObjectData->{ClassID},
            UserID  => $Param{UserID},
        );

        # check the new config item id
        if ( !$ConfigItemID ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message =>
                    "Can't import entity $Param{Counter}: "
                    . "Error when adding the new config item.",
            );
            return;
        }
    }

    # add new version
    my $VersionID = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionAdd(
        ConfigItemID => $ConfigItemID,
        Name         => $VersionData->{Name},
        DefinitionID => $DefinitionData->{DefinitionID},
        DeplStateID  => $VersionData->{DeplStateID},
        InciStateID  => $VersionData->{InciStateID},
        XMLData      => $VersionData->{XMLData},
        UserID       => $Param{UserID},
    );

    # the import was successful, when we get a version id
    if ($VersionID) {

        # When VersionAdd() returns the previous latest version ID, we know that
        # no new version has been added.
        # The import of this config item has been skipped.
        if ( $LatestVersionID && $VersionID == $LatestVersionID ) {
            $RetCode = Translatable('Skipped');
        }

        return $ConfigItemID, $RetCode;
    }

    if ( $RetCode eq 'Created' ) {

        # delete the new config item
        $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemDelete(
            ConfigItemID => $ConfigItemID,
            UserID       => $Param{UserID},
        );
    }

    $Kernel::OM->Get('Kernel::System::Log')->Log(
        Priority => 'error',
        Message =>
            "Can't import entity $Param{Counter}: "
            . "Error when adding the new config item version.",
    );

    return;
}

=head1 INTERNAL INTERFACE

=head2 _MappingObjectAttributesGet()

recursion function for MappingObjectAttributesGet().
Definitions for object attributes are passed in C<XMLDefinition>.
The new object attributes are appended to C<ElementList>.
C<CountMaxLimit> limits the max length of importable arrays.

    $ObjectBackend->_MappingObjectAttributesGet(
        XMLDefinition => $ArrayRef,
        ElementList   => $ArrayRef,
        CountMaxLimit => 10,
    );

=cut

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

    return if !$Param{CountMaxLimit};
    return if !$Param{XMLDefinition};
    return if !$Param{ElementList};
    return if ref $Param{XMLDefinition} ne 'ARRAY';
    return if ref $Param{ElementList} ne 'ARRAY';

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        # limit the length of importable arrays, even if more elements can be set via the GUI
        my $CountMax = min( $Item->{CountMax}, $Param{CountMaxLimit} );

        COUNT:
        for my $Count ( 1 .. $CountMax ) {

            # create key string
            my $Key = $Item->{Key} . '::' . $Count;

            # add prefix to key
            if ( $Param{KeyPrefix} ) {
                $Key = $Param{KeyPrefix} . '::' . $Key;
            }

            # create value string
            my $Value = $Item->{Key};

            # add count if required
            if ( $CountMax > 1 || $Item->{Sub} ) {
                $Value .= '::' . $Count;
            }

            # add prefix to key
            if ( $Param{ValuePrefix} ) {
                $Value = $Param{ValuePrefix} . '::' . $Value;
            }

            # add row
            my %Row = (
                Key   => $Key,
                Value => $Value,
            );
            push @{ $Param{ElementList} }, \%Row;

            next COUNT if !$Item->{Sub};

            # start recursion
            $Self->_MappingObjectAttributesGet(
                XMLDefinition => $Item->{Sub},
                ElementList   => $Param{ElementList},
                KeyPrefix     => $Key,
                ValuePrefix   => $Value,
                CountMaxLimit => $Param{CountMaxLimit} || '10',
            );
        }
    }

    return 1;
}

=head2 _SearchAttributesGet()

recursion function for MappingObjectAttributesGet()

    $ObjectBackend->_SearchAttributesGet(
        XMLDefinition => $ArrayRef,
        AttributeList => $ArrayRef,
    );

=cut

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

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if !$Param{AttributeList};
    return if ref $Param{XMLDefinition} ne 'ARRAY';
    return if ref $Param{AttributeList} ne 'ARRAY';

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        # set prefix
        my $Key  = $Item->{Key};
        my $Name = $Item->{Name};

        if ( $Param{KeyPrefix} ) {
            $Key = $Param{KeyPrefix} . '::' . $Key;
        }

        if ( $Param{NamePrefix} ) {
            $Name = $Param{NamePrefix} . '::' . $Name;
        }

        # add attribute, if marked as searchable
        if ( $Item->{Searchable} ) {

            if ( $Item->{Input}->{Type} eq 'Text' || $Item->{Input}->{Type} eq 'TextArea' ) {

                my %Row = (
                    Key   => $Key,
                    Name  => $Name,
                    Input => {
                        Type        => 'Text',
                        Translation => $Item->{Input}->{Input}->{Translation},
                        Size        => $Item->{Input}->{Input}->{Size} || 60,
                        MaxLength   => $Item->{Input}->{Input}->{MaxLength},
                    },
                );

                push @{ $Param{AttributeList} }, \%Row;
            }
            elsif ( $Item->{Input}->{Type} eq 'GeneralCatalog' ) {

                # get general catalog list
                my $GeneralCatalogList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
                    Class => $Item->{Input}->{Class},
                ) || {};

                my %Row = (
                    Key   => $Key,
                    Name  => $Name,
                    Input => {
                        Type        => 'Selection',
                        Data        => $GeneralCatalogList,
                        Translation => $Item->{Input}->{Input}->{Translation},
                        Size        => 5,
                        Multiple    => 1,
                    },
                );

                push @{ $Param{AttributeList} }, \%Row;
            }
        }

        next ITEM if !$Item->{Sub};

        # start recursion, if "Sub" was found
        $Self->_SearchAttributesGet(
            XMLDefinition => $Item->{Sub},
            AttributeList => $Param{AttributeList},
            KeyPrefix     => $Key,
            NamePrefix    => $Name,
        );
    }

    return 1;
}

=head2 _ExportXMLSearchDataPrepare()

recursion function to prepare the export XML search params

    $ObjectBackend->_ExportXMLSearchDataPrepare(
        XMLDefinition => $ArrayRef,
        What          => $ArrayRef,
        SearchData    => $HashRef,
    );

=cut

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

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if !$Param{What};
    return if !$Param{SearchData};
    return if ref $Param{XMLDefinition} ne 'ARRAY';
    return if ref $Param{What} ne 'ARRAY';
    return if ref $Param{SearchData} ne 'HASH';

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        # create key
        my $Key = $Param{Prefix} ? $Param{Prefix} . '::' . $Item->{Key} : $Item->{Key};

        # prepare value
        my $Values = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->XMLExportSearchValuePrepare(
            Item  => $Item,
            Value => $Param{SearchData}->{$Key},
        );

        if ($Values) {

            # create search key
            my $SearchKey = $Key;
            $SearchKey =~ s{ :: }{\'\}[%]\{\'}xmsg;

            # create search hash
            my $SearchHash = {
                '[1]{\'Version\'}[1]{\'' . $SearchKey . '\'}[%]{\'Content\'}' => $Values,
            };

            push @{ $Param{What} }, $SearchHash;
        }

        next ITEM if !$Item->{Sub};

        # start recursion, if "Sub" was found
        $Self->_ExportXMLSearchDataPrepare(
            XMLDefinition => $Item->{Sub},
            What          => $Param{What},
            SearchData    => $Param{SearchData},
            Prefix        => $Key,
        );
    }

    return 1;
}

=head2 _ExportXMLDataPrepare()

recursion function to prepare the export XML data

    $ObjectBackend->_ExportXMLDataPrepare(
        XMLDefinition => $ArrayRef,
        XMLData       => $HashRef,
        XMLData2D     => $HashRef,
    );

=cut

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

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if !$Param{XMLData};
    return if !$Param{XMLData2D};
    return if ref $Param{XMLDefinition} ne 'ARRAY';
    return if ref $Param{XMLData} ne 'HASH';
    return if ref $Param{XMLData2D} ne 'HASH';

    if ( $Param{Prefix} ) {
        $Param{Prefix} .= '::';
    }
    $Param{Prefix} ||= '';

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {
        COUNTER:
        for my $Counter ( 1 .. $Item->{CountMax} ) {

            # stop loop, if no content was given
            last COUNTER if !defined $Param{XMLData}->{ $Item->{Key} }->[$Counter]->{Content};

            # create key
            my $Key = $Param{Prefix} . $Item->{Key} . '::' . $Counter;

            # prepare value
            $Param{XMLData2D}->{$Key} = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->XMLExportValuePrepare(
                Item  => $Item,
                Value => $Param{XMLData}->{ $Item->{Key} }->[$Counter]->{Content},
            );

            next COUNTER if !$Item->{Sub};

            # start recursion, if "Sub" was found
            $Self->_ExportXMLDataPrepare(
                XMLDefinition => $Item->{Sub},
                XMLData       => $Param{XMLData}->{ $Item->{Key} }->[$Counter],
                XMLData2D     => $Param{XMLData2D},
                Prefix        => $Key,
            );
        }
    }

    return 1;
}

=head2 _ImportXMLSearchDataPrepare()

recursion function to prepare the import XML search params

    $ObjectBackend->_ImportXMLSearchDataPrepare(
        XMLDefinition => $ArrayRef,
        What          => $ArrayRef,
        Identifier    => $HashRef,
    );

=cut

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

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if !$Param{What};
    return if !$Param{Identifier};
    return if ref $Param{XMLDefinition} ne 'ARRAY';
    return if ref $Param{What} ne 'ARRAY';
    return if ref $Param{Identifier} ne 'HASH';

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        # create key
        my $Key = $Param{Prefix} ? $Param{Prefix} . '::\d+::' . $Item->{Key} : $Item->{Key};
        $Key .= '::\d+';

        my $IdentifierKey;
        IDENTIFIERKEY:
        for my $IdentKey ( sort keys %{ $Param{Identifier} } ) {

            next IDENTIFIERKEY if $IdentKey !~ m{ \A $Key \z }xms;

            $IdentifierKey = $IdentKey;
        }

        if ($IdentifierKey) {

            # prepare value
            my $Value = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->XMLImportSearchValuePrepare(
                Item  => $Item,
                Value => $Param{Identifier}->{$IdentifierKey},
            );

            if ($Value) {

                # prepare key
                my $Counter = 0;
                while ( $IdentifierKey =~ m{ :: }xms ) {

                    if ( $Counter % 2 ) {
                        $IdentifierKey =~ s{ :: }{]\{'}xms;
                    }
                    else {
                        $IdentifierKey =~ s{ :: }{'\}[}xms;
                    }

                    $Counter++;
                }

                # create search hash
                my $SearchHash = {
                    '[1]{\'Version\'}[1]{\'' . $IdentifierKey . ']{\'Content\'}' => $Value,
                };

                push @{ $Param{What} }, $SearchHash;
            }
        }

        next ITEM if !$Item->{Sub};

        # start recursion, if "Sub" was found
        $Self->_ImportXMLSearchDataPrepare(
            XMLDefinition => $Item->{Sub},
            What          => $Param{What},
            Identifier    => $Param{Identifier},
            Prefix        => $Key,
        );
    }

    return 1;
}

=head2 _ImportXMLDataMerge()

recursive function to inplace edit the import XML data.

    my $MergeOk = $ObjectBackend->_ImportXMLDataMerge(
        XMLDefinition => $ArrayRef,
        XMLDataPrev   => $HashRef,
        XMLData2D     => $HashRef,
    );

The return value indicates whether the merge was successful.
A merge fails when for example a general catalog item name can't be mapped to an id.

=cut

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

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if !$Param{XMLData2D};
    return if !$Param{XMLDataPrev};
    return if ref $Param{XMLDefinition} ne 'ARRAY';    # the attributes of the config item class
    return if ref $Param{XMLData2D} ne 'HASH';         # hash with values that should be imported
    return if ref $Param{XMLDataPrev} ne 'HASH';       # hash with current values of the config item

    my $XMLData = $Param{XMLDataPrev};

    # default value for prefix
    $Param{Prefix} ||= '';

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        COUNTER:
        for my $Counter ( 1 .. $Item->{CountMax} ) {

            # create inputkey
            my $Key = $Param{Prefix} . $Item->{Key} . '::' . $Counter;

            # start recursion, if "Sub" was found
            if ( $Item->{Sub} ) {
                $XMLData->{ $Item->{Key} }->[$Counter]
                    ||= {};    # empty container, in case there is no previous data
                my $MergeOk = $Self->_ImportXMLDataMerge(
                    XMLDefinition                => $Item->{Sub},
                    XMLData2D                    => $Param{XMLData2D},
                    XMLDataPrev                  => $XMLData->{ $Item->{Key} }->[$Counter],
                    Prefix                       => $Key . '::',
                    EmptyFieldsLeaveTheOldValues => $Param{EmptyFieldsLeaveTheOldValues},
                );

                return if !$MergeOk;
            }

            # When the data point is not part of the input definition,
            # then do not overwrite the previous setting.
            # False values are OK.
            next COUNTER if !exists $Param{XMLData2D}->{$Key};

            if ( $Param{EmptyFieldsLeaveTheOldValues} ) {

                # do not override old value with an empty field is imported
                next COUNTER if !defined $Param{XMLData2D}->{$Key};
                next COUNTER if $Param{XMLData2D}->{$Key} eq '';
            }

            # prepare value
            my $Value = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->XMLImportValuePrepare(
                Item  => $Item,
                Value => $Param{XMLData2D}->{$Key},
            );

            # let merge fail, when a value cannot be prepared
            return if !defined $Value;

            # save the prepared value
            $XMLData->{ $Item->{Key} }->[$Counter]->{Content} = $Value;
        }
    }

    return 1;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::ImportExport::ObjectBackend::ITSMConfigItem;

use strict;
use warnings;

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

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

=head1 NAME

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

=head1 DESCRIPTION

All functions to import and export ITSM config items.

=head1 PUBLIC INTERFACE

=head2 new()

create an object

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

=cut

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

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

    return $Self;
}

=head2 ObjectAttributesGet()

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

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

=cut

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

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

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    ) || {};

    my $Attributes = [
        {
            Key   => 'ClassID',
            Name  => Translatable('Class'),
            Input => {
                Type         => 'Selection',
                Data         => $ClassList,
                Required     => 1,
                Translation  => 0,
                PossibleNone => 1,
            },
        },
        {
            Key   => 'CountMax',
            Name  => Translatable('Maximum number of one element'),
            Input => {
                Type         => 'Text',
                ValueDefault => '10',
                Required     => 1,
                Regex        => qr{ \A \d+ \z }xms,
                Translation  => 0,
                Size         => 5,
                MaxLength    => 5,
                DataType     => 'IntegerBiggerThanZero',
            },
        },
        {
            Key   => 'EmptyFieldsLeaveTheOldValues',
            Name  => Translatable('Empty fields indicate that the current values are kept'),
            Input => {
                Type => 'Checkbox',
            },
        },
    ];

    return $Attributes;
}

=head2 MappingObjectAttributesGet()

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

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

=cut

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

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

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

    return [] if !$ObjectData;
    return [] if ref $ObjectData ne 'HASH';
    return [] if !$ObjectData->{ClassID};

    # get definition
    my $XMLDefinition = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->DefinitionGet(
        ClassID => $ObjectData->{ClassID},
    );

    return [] if !$XMLDefinition;
    return [] if ref $XMLDefinition ne 'HASH';
    return [] if !$XMLDefinition->{DefinitionRef};
    return [] if ref $XMLDefinition->{DefinitionRef} ne 'ARRAY';

    my $ElementList = [
        {
            Key   => 'Number',
            Value => Translatable('Number'),
        },
        {
            Key   => 'Name',
            Value => Translatable('Name'),
        },
        {
            Key   => 'DeplState',
            Value => Translatable('Deployment State'),
        },
        {
            Key   => 'InciState',
            Value => Translatable('Incident State'),
        },
    ];

    # add xml elements
    $Self->_MappingObjectAttributesGet(
        XMLDefinition => $XMLDefinition->{DefinitionRef},
        ElementList   => $ElementList,
        CountMaxLimit => $ObjectData->{CountMax} || 10,
    );

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

    return $Attributes;
}

=head2 SearchAttributesGet()

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

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

=cut

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

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

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

    return [] if !$ObjectData;
    return [] if ref $ObjectData ne 'HASH';
    return [] if !$ObjectData->{ClassID};

    # get definition
    my $XMLDefinition = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->DefinitionGet(
        ClassID => $ObjectData->{ClassID},
    );

    return [] if !$XMLDefinition;
    return [] if ref $XMLDefinition ne 'HASH';
    return [] if !$XMLDefinition->{DefinitionRef};
    return [] if ref $XMLDefinition->{DefinitionRef} ne 'ARRAY';

    # get deployment state list
    my $DeplStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    ) || {};

    # get incident state list
    my $InciStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::Core::IncidentState',
    ) || {};

    my $AttributeList = [
        {
            Key   => 'Number',
            Name  => Translatable('Number'),
            Input => {
                Type      => 'Text',
                Size      => 80,
                MaxLength => 255,
            },
        },
        {
            Key   => 'Name',
            Name  => Translatable('Name'),
            Input => {
                Type      => 'Text',
                Size      => 80,
                MaxLength => 255,
            },
        },
        {
            Key   => 'DeplStateIDs',
            Name  => Translatable('Deployment State'),
            Input => {
                Type        => 'Selection',
                Data        => $DeplStateList,
                Translation => 1,
                Size        => 5,
                Multiple    => 1,
            },
        },
        {
            Key   => 'InciStateIDs',
            Name  => Translatable('Incident State'),
            Input => {
                Type        => 'Selection',
                Data        => $InciStateList,
                Translation => 1,
                Size        => 5,
                Multiple    => 1,
            },
        },
    ];

    # add xml attributes
    $Self->_SearchAttributesGet(
        XMLDefinition => $XMLDefinition->{DefinitionRef},
        AttributeList => $AttributeList,
    );

    return $AttributeList;
}

=head2 ExportDataGet()

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

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

=cut

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

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

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

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

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    return if !$ClassList || ref $ClassList ne 'HASH';

    # check the class id
    if ( !$ObjectData->{ClassID} || !$ClassList->{ $ObjectData->{ClassID} } ) {

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

    # get the mapping list
    my $MappingList = $Kernel::OM->Get('Kernel::System::ImportExport')->MappingList(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

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

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

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

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

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

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

        push @MappingObjectList, $MappingObjectData;
    }

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

    return if !$SearchData || ref $SearchData ne 'HASH';

    # get deployment state list
    my $DeplStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    # check deployment state list
    if ( !$DeplStateList || ref $DeplStateList ne 'HASH' ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Can't get the general catalog list ITSM::ConfigItem::DeploymentState!",
        );
        return;
    }

    # get incident state list
    my $InciStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::Core::IncidentState',
    );

    # check incident state list
    if ( !$InciStateList || ref $InciStateList ne 'HASH' ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Can't get the general catalog list ITSM::Core::IncidentState!",
        );
        return;
    }

    # get current definition of this class
    my $DefinitionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->DefinitionGet(
        ClassID => $ObjectData->{ClassID},
        UserID  => $Param{UserID},
    );

    my %SearchParams;

    # add number to the search params
    if ( $SearchData->{Number} ) {
        $SearchParams{Number} = delete $SearchData->{Number};
    }

    # add name to the search params
    if ( $SearchData->{Name} ) {
        $SearchParams{Name} = delete $SearchData->{Name};
    }

    # add deployment state to the search params
    if ( $SearchData->{DeplStateIDs} ) {
        my @DeplStateIDs = split '#####', $SearchData->{DeplStateIDs};
        $SearchParams{DeplStateIDs} = \@DeplStateIDs;
        delete $SearchData->{DeplStateIDs};
    }

    # add incident state to the search params
    if ( $SearchData->{InciStateIDs} ) {
        my @InciStateIDs = split '#####', $SearchData->{InciStateIDs};
        $SearchParams{InciStateIDs} = \@InciStateIDs;
        delete $SearchData->{InciStateIDs};
    }

    # add all XML data to the search params
    my @SearchParamsWhat;
    $Self->_ExportXMLSearchDataPrepare(
        XMLDefinition => $DefinitionData->{DefinitionRef},
        What          => \@SearchParamsWhat,
        SearchData    => $SearchData,
    );

    # add XML search params to the search hash
    if (@SearchParamsWhat) {
        $SearchParams{What} = \@SearchParamsWhat;
    }

    # search the config items
    my $ConfigItemList = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemSearchExtended(
        %SearchParams,
        ClassIDs              => [ $ObjectData->{ClassID} ],
        PreviousVersionSearch => 0,
        UserID                => $Param{UserID},
    );

    my @ExportData;
    CONFIGITEMID:
    for my $ConfigItemID ( @{$ConfigItemList} ) {

        # get last version
        my $VersionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
            ConfigItemID => $ConfigItemID,
        );

        next CONFIGITEMID if !$VersionData;
        next CONFIGITEMID if ref $VersionData ne 'HASH';

        # translate xmldata to a 2d hash
        my %XMLData2D;
        $Self->_ExportXMLDataPrepare(
            XMLDefinition => $DefinitionData->{DefinitionRef},
            XMLData       => $VersionData->{XMLData}->[1]->{Version}->[1],
            XMLData2D     => \%XMLData2D,
        );

        # add data to the export data array
        my @Item;
        MAPPINGOBJECT:
        for my $MappingObject (@MappingObjectList) {

            # extract key
            my $Key = $MappingObject->{Key};

            # handle empty key
            if ( !$Key ) {
                push @Item, '';
                next MAPPINGOBJECT;
            }

            # handle config item number
            if ( $Key eq 'Number' ) {
                push @Item, $VersionData->{Number};
                next MAPPINGOBJECT;
            }

            # handle current config item name
            if ( $Key eq 'Name' ) {
                push @Item, $VersionData->{Name};
                next MAPPINGOBJECT;
            }

            # handle deployment state
            if ( $Key eq 'DeplState' ) {
                $VersionData->{DeplStateID} ||= 'DUMMY';
                push @Item, $DeplStateList->{ $VersionData->{DeplStateID} };
                next MAPPINGOBJECT;
            }

            # handle incident state
            if ( $Key eq 'InciState' ) {
                $VersionData->{InciStateID} ||= 'DUMMY';
                push @Item, $InciStateList->{ $VersionData->{InciStateID} };
                next MAPPINGOBJECT;
            }

            # handle all XML data elements
            push @Item, $XMLData2D{$Key};
        }

        push @ExportData, \@Item;
    }

    return \@ExportData;
}

=head2 ImportDataSave()

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

The current version of the config item will never be deleted. When there are no
changes in the data, the import will be skipped. When there is new or changed data,
then a new config item or a new version is created.

In the case of changed data, the new version of the config item will contain the
attributes of the C<ImportDataRow> plus the old attributes that are
not part of the import definition.
Thus ImportDataSave() guarantees to not overwrite undeclared attributes.

The behavior when imported attributes are empty depends on the setting in the object data.
When C<EmptyFieldsLeaveTheOldValues> is not set, then empty values will wipe out
the old data. This is the default behavior. When C<EmptyFieldsLeaveTheOldValues> is set,
then empty values will leave the old values.

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

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

An empty C<ConfigItemID> indicates failure. Otherwise it indicates the
location of the imported data.
C<RetCode> is either 'Created', 'Updated' or 'Skipped'. 'Created' means that a new
config item has been created. 'Updated' means that a new version has been added to
an existing config item. 'Skipped' means that no new version has been created,
as the new data is identical to the latest version of an existing config item.

No codes have yet been defined for the failure case.

=cut

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

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

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

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

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

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

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # check class list
    if ( !$ClassList || ref $ClassList ne 'HASH' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message =>
                "Can't import entity $Param{Counter}: "
                . "Can't get the general catalog list ITSM::ConfigItem::Class",
        );
        return;
    }

    # check the class id
    if ( !$ObjectData->{ClassID} || !$ClassList->{ $ObjectData->{ClassID} } ) {

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

    # get the mapping list
    my $MappingList = $Kernel::OM->Get('Kernel::System::ImportExport')->MappingList(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

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

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

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

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

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

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

        push @MappingObjectList, $MappingObjectData;
    }

    # check and remember the Identifiers
    # the Identifiers identify the config item that should be updated
    my %Identifier;
    my $RowIndex = 0;
    MAPPINGOBJECTDATA:
    for my $MappingObjectData (@MappingObjectList) {

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

        # check if identifier already exists
        if ( $Identifier{ $MappingObjectData->{Key} } ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message =>
                    "Can't import entity $Param{Counter}: "
                    . "'$MappingObjectData->{Key}' has been used multiple times as an identifier",
            );
            return;
        }

        # set identifier value
        $Identifier{ $MappingObjectData->{Key} } = $Param{ImportDataRow}->[$RowIndex];

        next MAPPINGOBJECTDATA if $MappingObjectData->{Key} && $Param{ImportDataRow}->[$RowIndex];

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message =>
                "Can't import entity $Param{Counter}: "
                . "Identifier field is empty",
        );

        return;
    }
    continue {
        $RowIndex++;
    }

    # get deployment state list
    my $DeplStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    # check deployment state list
    if ( !$DeplStateList || ref $DeplStateList ne 'HASH' ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message =>
                "Can't import entity $Param{Counter}: "
                . "Can't get the general catalog list ITSM::ConfigItem::DeploymentState!",
        );
        return;
    }

    # reverse the deployment state list
    my %DeplStateListReverse = reverse %{$DeplStateList};

    # get incident state list
    my $InciStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::Core::IncidentState',
    );

    # check incident state list
    if ( !$InciStateList || ref $InciStateList ne 'HASH' ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message =>
                "Can't import entity $Param{Counter}: "
                . "Can't get the general catalog list ITSM::Core::IncidentState",
        );
        return;
    }

    # reverse the incident state list
    my %InciStateListReverse = reverse %{$InciStateList};

    # get current definition of this class
    my $DefinitionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->DefinitionGet(
        ClassID => $ObjectData->{ClassID},
        UserID  => $Param{UserID},
    );

    # check definition data
    if ( !$DefinitionData || ref $DefinitionData ne 'HASH' ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message =>
                "Can't import entity $Param{Counter}: "
                . "Can't get the definition of class id $ObjectData->{ClassID}",
        );
        return;
    }

    # try to get config item ids, when there are identifiers
    my $ConfigItemID;
    if (%Identifier) {

        my %SearchParams;

        # add number to the search params
        if ( $Identifier{Number} ) {
            $SearchParams{Number} = delete $Identifier{Number};
        }

        # add name to the search params
        if ( $Identifier{Name} ) {
            $SearchParams{Name} = delete $Identifier{Name};
        }

        # add deployment state to the search params
        if ( $Identifier{DeplState} ) {

            # extract deployment state id
            my $DeplStateID = $DeplStateListReverse{ $Identifier{DeplState} } || '';

            if ( !$DeplStateID ) {

                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message =>
                        "Can't import entity $Param{Counter}: "
                        . "The deployment state '$Identifier{DeplState}' is invalid",
                );
                return;
            }

            $SearchParams{DeplStateIDs} = [$DeplStateID];
            delete $Identifier{DeplState};
        }

        # add incident state to the search params
        if ( $Identifier{InciState} ) {

            # extract incident state id
            my $InciStateID = $InciStateListReverse{ $Identifier{InciState} } || '';

            if ( !$InciStateID ) {

                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message =>
                        "Can't import entity $Param{Counter}: "
                        . "The incident state '$Identifier{InciState}' is invalid",
                );
                return;
            }

            $SearchParams{InciStateIDs} = [$InciStateID];
            delete $Identifier{InciState};
        }

        # add all XML data to the search params
        my @SearchParamsWhat;
        $Self->_ImportXMLSearchDataPrepare(
            XMLDefinition => $DefinitionData->{DefinitionRef},
            What          => \@SearchParamsWhat,
            Identifier    => \%Identifier,
        );

        # add XML search params to the search hash
        if (@SearchParamsWhat) {
            $SearchParams{What} = \@SearchParamsWhat;
        }

        # search existing config item with the same identifiers
        my $ConfigItemList = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemSearchExtended(
            %SearchParams,
            ClassIDs              => [ $ObjectData->{ClassID} ],
            PreviousVersionSearch => 0,
            UsingWildcards        => 0,
            UserID                => $Param{UserID},
        );

        if ( scalar @{$ConfigItemList} > 1 ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message =>
                    "Can't import entity $Param{Counter}: "
                    . "Identifier fields NOT unique!",
            );
            return;
        }

        $ConfigItemID = $ConfigItemList->[0];
    }

    # get version data of the config item
    my $VersionData = {};
    if ($ConfigItemID) {

        # get latest version
        $VersionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
            ConfigItemID => $ConfigItemID,
        );

        # remove empty xml data
        if (
            !$VersionData->{XMLData}
            || ref $VersionData->{XMLData} ne 'ARRAY'
            || !@{ $VersionData->{XMLData} }
            )
        {
            delete $VersionData->{XMLData};
        }
    }

    # Check if current definition of this class has required attribute which does not exist in mapping list.
    for my $DefinitionRef ( @{ $DefinitionData->{DefinitionRef} } ) {
        my $Key = $DefinitionRef->{Key};

        if (
            $DefinitionRef->{Input}->{Required}
            && !scalar( grep { $_->{Key} =~ /\Q$DefinitionRef->{Key}\E(::(\d+))?$/ } @MappingObjectList )
            && (
                !defined $DefinitionRef->{CountMin}
                || ( defined $DefinitionRef->{CountMin} && $DefinitionRef->{CountMin} > 0 )
            )
            )
        {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message =>
                    "Can't import entity $Param{Counter}: "
                    . "Attribute '$Key' is required, but does not exist in mapping list!",
            );
            return;
        }
    }

    # set up fields in VersionData and in the XML attributes
    my %XMLData2D;
    $RowIndex = 0;
    for my $MappingObjectData (@MappingObjectList) {

        # just for convenience
        my $Key   = $MappingObjectData->{Key};
        my $Value = $Param{ImportDataRow}->[ $RowIndex++ ];

        if ( $Key eq 'Number' ) {

            # do nothing
            # Import does not override the config item number
        }
        elsif ( $Key eq 'Name' ) {

            if ( $EmptyFieldsLeaveTheOldValues && ( !defined $Value || $Value eq '' ) ) {

                # do nothing, keep the old value
            }
            else {
                if ( !$Value ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message =>
                            "Can't import entity $Param{Counter}: "
                            . "The name '$Value' is invalid!",
                    );
                    return;
                }

                $VersionData->{$Key} = $Value;
            }
        }
        elsif ( $Key eq 'DeplState' ) {

            if ( $EmptyFieldsLeaveTheOldValues && ( !defined $Value || $Value eq '' ) ) {

                # do nothing, keep the old value
            }
            else {

                # extract deployment state id
                my $DeplStateID = $DeplStateListReverse{$Value} || '';
                if ( !$DeplStateID ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message =>
                            "Can't import entity $Param{Counter}: "
                            . "The deployment state '$Value' is invalid!",
                    );
                    return;
                }

                $VersionData->{DeplStateID} = $DeplStateID;
            }
        }
        elsif ( $Key eq 'InciState' ) {

            if ( $EmptyFieldsLeaveTheOldValues && ( !defined $Value || $Value eq '' ) ) {

                # do nothing, keep the old value
            }
            else {

                # extract the deployment state id
                my $InciStateID = $InciStateListReverse{$Value} || '';
                if ( !$InciStateID ) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message =>
                            "Can't import entity $Param{Counter}: "
                            . "The incident state '$Value' is invalid!",
                    );
                    return;
                }

                $VersionData->{InciStateID} = $InciStateID;
            }
        }
        else {

            # handle xml data
            $XMLData2D{$Key} = $Value;
        }
    }

    # set up empty container, in case there is no previous data
    $VersionData->{XMLData}->[1]->{Version}->[1] ||= {};

    # Edit XMLDataPrev, so that the values in XMLData2D take precedence.
    my $MergeOk = $Self->_ImportXMLDataMerge(
        XMLDefinition                => $DefinitionData->{DefinitionRef},
        XMLDataPrev                  => $VersionData->{XMLData}->[1]->{Version}->[1],
        XMLData2D                    => \%XMLData2D,
        EmptyFieldsLeaveTheOldValues => $EmptyFieldsLeaveTheOldValues,
    );

    # bail out, when the was a problem in _ImportXMLDataMerge()
    if ( !$MergeOk ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message =>
                "Can't import entity $Param{Counter}: "
                . "Could not prepare the input!",
        );
        return;
    }

    my $RetCode = $ConfigItemID ? Translatable('Changed') : Translatable('Created');

    # check if the feature to check for a unique name is enabled
    if (
        IsStringWithData( $VersionData->{Name} )
        && $Kernel::OM->Get('Kernel::Config')->Get('UniqueCIName::EnableUniquenessCheck')
        )
    {

        if ( $Kernel::OM->Get('Kernel::Config')->{Debug} > 0 ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'debug',
                Message  => "Checking for duplicate names (ClassID: $ObjectData->{ClassID}, "
                    . "Name: $VersionData->{Name}, ConfigItemID: " . $ConfigItemID || 'NEW' . ')',
            );
        }

        my $NameDuplicates = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->UniqueNameCheck(
            ConfigItemID => $ConfigItemID || 'NEW',
            ClassID      => $ObjectData->{ClassID},
            Name         => $VersionData->{Name},
        );

        # stop processing if the name is not unique
        if ( IsArrayRefWithData($NameDuplicates) ) {

            # build a string of all duplicate IDs
            my $NameDuplicatesString = join ', ', @{$NameDuplicates};

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message =>
                    "The name $VersionData->{Name} is already in use by the ConfigItemID(s): "
                    . $NameDuplicatesString,
            );

            # set the return code to also include the duplicate name
            $RetCode = "DuplicateName '$VersionData->{Name}'";

            # return undef for the config item id so it will be counted as 'Failed'
            return undef, $RetCode;    ## no critic
        }
    }

    my $LatestVersionID = 0;
    if ($ConfigItemID) {

        # the specified config item already exists
        # get id of the latest version, for checking later whether a version was created
        my $VersionList = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionList(
            ConfigItemID => $ConfigItemID,
        ) || [];
        if ( scalar @{$VersionList} ) {
            $LatestVersionID = $VersionList->[-1];
        }
    }
    else {

        # no config item was found, so add new config item
        $ConfigItemID = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemAdd(
            ClassID => $ObjectData->{ClassID},
            UserID  => $Param{UserID},
        );

        # check the new config item id
        if ( !$ConfigItemID ) {

            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message =>
                    "Can't import entity $Param{Counter}: "
                    . "Error when adding the new config item.",
            );
            return;
        }
    }

    # add new version
    my $VersionID = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionAdd(
        ConfigItemID => $ConfigItemID,
        Name         => $VersionData->{Name},
        DefinitionID => $DefinitionData->{DefinitionID},
        DeplStateID  => $VersionData->{DeplStateID},
        InciStateID  => $VersionData->{InciStateID},
        XMLData      => $VersionData->{XMLData},
        UserID       => $Param{UserID},
    );

    # the import was successful, when we get a version id
    if ($VersionID) {

        # When VersionAdd() returns the previous latest version ID, we know that
        # no new version has been added.
        # The import of this config item has been skipped.
        if ( $LatestVersionID && $VersionID == $LatestVersionID ) {
            $RetCode = Translatable('Skipped');
        }

        return $ConfigItemID, $RetCode;
    }

    if ( $RetCode eq 'Created' ) {

        # delete the new config item
        $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemDelete(
            ConfigItemID => $ConfigItemID,
            UserID       => $Param{UserID},
        );
    }

    $Kernel::OM->Get('Kernel::System::Log')->Log(
        Priority => 'error',
        Message =>
            "Can't import entity $Param{Counter}: "
            . "Error when adding the new config item version.",
    );

    return;
}

=head1 INTERNAL INTERFACE

=head2 _MappingObjectAttributesGet()

recursion function for MappingObjectAttributesGet().
Definitions for object attributes are passed in C<XMLDefinition>.
The new object attributes are appended to C<ElementList>.
C<CountMaxLimit> limits the max length of importable arrays.

    $ObjectBackend->_MappingObjectAttributesGet(
        XMLDefinition => $ArrayRef,
        ElementList   => $ArrayRef,
        CountMaxLimit => 10,
    );

=cut

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

    return if !$Param{CountMaxLimit};
    return if !$Param{XMLDefinition};
    return if !$Param{ElementList};
    return if ref $Param{XMLDefinition} ne 'ARRAY';
    return if ref $Param{ElementList} ne 'ARRAY';

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        # limit the length of importable arrays, even if more elements can be set via the GUI
        my $CountMax = min( $Item->{CountMax}, $Param{CountMaxLimit} );

        COUNT:
        for my $Count ( 1 .. $CountMax ) {

            # create key string
            my $Key = $Item->{Key} . '::' . $Count;

            # add prefix to key
            if ( $Param{KeyPrefix} ) {
                $Key = $Param{KeyPrefix} . '::' . $Key;
            }

            # create value string
            my $Value = $Item->{Key};

            # add count if required
            if ( $CountMax > 1 || $Item->{Sub} ) {
                $Value .= '::' . $Count;
            }

            # add prefix to key
            if ( $Param{ValuePrefix} ) {
                $Value = $Param{ValuePrefix} . '::' . $Value;
            }

            # add row
            my %Row = (
                Key   => $Key,
                Value => $Value,
            );
            push @{ $Param{ElementList} }, \%Row;

            next COUNT if !$Item->{Sub};

            # start recursion
            $Self->_MappingObjectAttributesGet(
                XMLDefinition => $Item->{Sub},
                ElementList   => $Param{ElementList},
                KeyPrefix     => $Key,
                ValuePrefix   => $Value,
                CountMaxLimit => $Param{CountMaxLimit} || '10',
            );
        }
    }

    return 1;
}

=head2 _SearchAttributesGet()

recursion function for MappingObjectAttributesGet()

    $ObjectBackend->_SearchAttributesGet(
        XMLDefinition => $ArrayRef,
        AttributeList => $ArrayRef,
    );

=cut

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

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if !$Param{AttributeList};
    return if ref $Param{XMLDefinition} ne 'ARRAY';
    return if ref $Param{AttributeList} ne 'ARRAY';

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        # set prefix
        my $Key  = $Item->{Key};
        my $Name = $Item->{Name};

        if ( $Param{KeyPrefix} ) {
            $Key = $Param{KeyPrefix} . '::' . $Key;
        }

        if ( $Param{NamePrefix} ) {
            $Name = $Param{NamePrefix} . '::' . $Name;
        }

        # add attribute, if marked as searchable
        if ( $Item->{Searchable} ) {

            if ( $Item->{Input}->{Type} eq 'Text' || $Item->{Input}->{Type} eq 'TextArea' ) {

                my %Row = (
                    Key   => $Key,
                    Name  => $Name,
                    Input => {
                        Type        => 'Text',
                        Translation => $Item->{Input}->{Input}->{Translation},
                        Size        => $Item->{Input}->{Input}->{Size} || 60,
                        MaxLength   => $Item->{Input}->{Input}->{MaxLength},
                    },
                );

                push @{ $Param{AttributeList} }, \%Row;
            }
            elsif ( $Item->{Input}->{Type} eq 'GeneralCatalog' ) {

                # get general catalog list
                my $GeneralCatalogList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
                    Class => $Item->{Input}->{Class},
                ) || {};

                my %Row = (
                    Key   => $Key,
                    Name  => $Name,
                    Input => {
                        Type        => 'Selection',
                        Data        => $GeneralCatalogList,
                        Translation => $Item->{Input}->{Input}->{Translation},
                        Size        => 5,
                        Multiple    => 1,
                    },
                );

                push @{ $Param{AttributeList} }, \%Row;
            }
        }

        next ITEM if !$Item->{Sub};

        # start recursion, if "Sub" was found
        $Self->_SearchAttributesGet(
            XMLDefinition => $Item->{Sub},
            AttributeList => $Param{AttributeList},
            KeyPrefix     => $Key,
            NamePrefix    => $Name,
        );
    }

    return 1;
}

=head2 _ExportXMLSearchDataPrepare()

recursion function to prepare the export XML search params

    $ObjectBackend->_ExportXMLSearchDataPrepare(
        XMLDefinition => $ArrayRef,
        What          => $ArrayRef,
        SearchData    => $HashRef,
    );

=cut

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

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if !$Param{What};
    return if !$Param{SearchData};
    return if ref $Param{XMLDefinition} ne 'ARRAY';
    return if ref $Param{What} ne 'ARRAY';
    return if ref $Param{SearchData} ne 'HASH';

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        # create key
        my $Key = $Param{Prefix} ? $Param{Prefix} . '::' . $Item->{Key} : $Item->{Key};

        # prepare value
        my $Values = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->XMLExportSearchValuePrepare(
            Item  => $Item,
            Value => $Param{SearchData}->{$Key},
        );

        if ($Values) {

            # create search key
            my $SearchKey = $Key;
            $SearchKey =~ s{ :: }{\'\}[%]\{\'}xmsg;

            # create search hash
            my $SearchHash = {
                '[1]{\'Version\'}[1]{\'' . $SearchKey . '\'}[%]{\'Content\'}' => $Values,
            };

            push @{ $Param{What} }, $SearchHash;
        }

        next ITEM if !$Item->{Sub};

        # start recursion, if "Sub" was found
        $Self->_ExportXMLSearchDataPrepare(
            XMLDefinition => $Item->{Sub},
            What          => $Param{What},
            SearchData    => $Param{SearchData},
            Prefix        => $Key,
        );
    }

    return 1;
}

=head2 _ExportXMLDataPrepare()

recursion function to prepare the export XML data

    $ObjectBackend->_ExportXMLDataPrepare(
        XMLDefinition => $ArrayRef,
        XMLData       => $HashRef,
        XMLData2D     => $HashRef,
    );

=cut

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

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if !$Param{XMLData};
    return if !$Param{XMLData2D};
    return if ref $Param{XMLDefinition} ne 'ARRAY';
    return if ref $Param{XMLData} ne 'HASH';
    return if ref $Param{XMLData2D} ne 'HASH';

    if ( $Param{Prefix} ) {
        $Param{Prefix} .= '::';
    }
    $Param{Prefix} ||= '';

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {
        COUNTER:
        for my $Counter ( 1 .. $Item->{CountMax} ) {

            # stop loop, if no content was given
            last COUNTER if !defined $Param{XMLData}->{ $Item->{Key} }->[$Counter]->{Content};

            # create key
            my $Key = $Param{Prefix} . $Item->{Key} . '::' . $Counter;

            # prepare value
            $Param{XMLData2D}->{$Key} = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->XMLExportValuePrepare(
                Item  => $Item,
                Value => $Param{XMLData}->{ $Item->{Key} }->[$Counter]->{Content},
            );

            next COUNTER if !$Item->{Sub};

            # start recursion, if "Sub" was found
            $Self->_ExportXMLDataPrepare(
                XMLDefinition => $Item->{Sub},
                XMLData       => $Param{XMLData}->{ $Item->{Key} }->[$Counter],
                XMLData2D     => $Param{XMLData2D},
                Prefix        => $Key,
            );
        }
    }

    return 1;
}

=head2 _ImportXMLSearchDataPrepare()

recursion function to prepare the import XML search params

    $ObjectBackend->_ImportXMLSearchDataPrepare(
        XMLDefinition => $ArrayRef,
        What          => $ArrayRef,
        Identifier    => $HashRef,
    );

=cut

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

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if !$Param{What};
    return if !$Param{Identifier};
    return if ref $Param{XMLDefinition} ne 'ARRAY';
    return if ref $Param{What} ne 'ARRAY';
    return if ref $Param{Identifier} ne 'HASH';

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        # create key
        my $Key = $Param{Prefix} ? $Param{Prefix} . '::\d+::' . $Item->{Key} : $Item->{Key};
        $Key .= '::\d+';

        my $IdentifierKey;
        IDENTIFIERKEY:
        for my $IdentKey ( sort keys %{ $Param{Identifier} } ) {

            next IDENTIFIERKEY if $IdentKey !~ m{ \A $Key \z }xms;

            $IdentifierKey = $IdentKey;
        }

        if ($IdentifierKey) {

            # prepare value
            my $Value = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->XMLImportSearchValuePrepare(
                Item  => $Item,
                Value => $Param{Identifier}->{$IdentifierKey},
            );

            if ($Value) {

                # prepare key
                my $Counter = 0;
                while ( $IdentifierKey =~ m{ :: }xms ) {

                    if ( $Counter % 2 ) {
                        $IdentifierKey =~ s{ :: }{]\{'}xms;
                    }
                    else {
                        $IdentifierKey =~ s{ :: }{'\}[}xms;
                    }

                    $Counter++;
                }

                # create search hash
                my $SearchHash = {
                    '[1]{\'Version\'}[1]{\'' . $IdentifierKey . ']{\'Content\'}' => $Value,
                };

                push @{ $Param{What} }, $SearchHash;
            }
        }

        next ITEM if !$Item->{Sub};

        # start recursion, if "Sub" was found
        $Self->_ImportXMLSearchDataPrepare(
            XMLDefinition => $Item->{Sub},
            What          => $Param{What},
            Identifier    => $Param{Identifier},
            Prefix        => $Key,
        );
    }

    return 1;
}

=head2 _ImportXMLDataMerge()

recursive function to inplace edit the import XML data.

    my $MergeOk = $ObjectBackend->_ImportXMLDataMerge(
        XMLDefinition => $ArrayRef,
        XMLDataPrev   => $HashRef,
        XMLData2D     => $HashRef,
    );

The return value indicates whether the merge was successful.
A merge fails when for example a general catalog item name can't be mapped to an id.

=cut

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

    # check needed stuff
    return if !$Param{XMLDefinition};
    return if !$Param{XMLData2D};
    return if !$Param{XMLDataPrev};
    return if ref $Param{XMLDefinition} ne 'ARRAY';    # the attributes of the config item class
    return if ref $Param{XMLData2D} ne 'HASH';         # hash with values that should be imported
    return if ref $Param{XMLDataPrev} ne 'HASH';       # hash with current values of the config item

    my $XMLData = $Param{XMLDataPrev};

    # default value for prefix
    $Param{Prefix} ||= '';

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        COUNTER:
        for my $Counter ( 1 .. $Item->{CountMax} ) {

            # create inputkey
            my $Key = $Param{Prefix} . $Item->{Key} . '::' . $Counter;

            # start recursion, if "Sub" was found
            if ( $Item->{Sub} ) {
                $XMLData->{ $Item->{Key} }->[$Counter]
                    ||= {};    # empty container, in case there is no previous data
                my $MergeOk = $Self->_ImportXMLDataMerge(
                    XMLDefinition                => $Item->{Sub},
                    XMLData2D                    => $Param{XMLData2D},
                    XMLDataPrev                  => $XMLData->{ $Item->{Key} }->[$Counter],
                    Prefix                       => $Key . '::',
                    EmptyFieldsLeaveTheOldValues => $Param{EmptyFieldsLeaveTheOldValues},
                );

                return if !$MergeOk;
            }

            # When the data point is not part of the input definition,
            # then do not overwrite the previous setting.
            # False values are OK.
            next COUNTER if !exists $Param{XMLData2D}->{$Key};

            if ( $Param{EmptyFieldsLeaveTheOldValues} ) {

                # do not override old value with an empty field is imported
                next COUNTER if !defined $Param{XMLData2D}->{$Key};
                next COUNTER if $Param{XMLData2D}->{$Key} eq '';
            }

            # prepare value
            my $Value = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->XMLImportValuePrepare(
                Item  => $Item,
                Value => $Param{XMLData2D}->{$Key},
            );

            # let merge fail, when a value cannot be prepared
            return if !defined $Value;

            # save the prepared value
            $XMLData->{ $Item->{Key} }->[$Counter]->{Content} = $Value;
        }
    }

    return 1;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::LinkObject::ITSMConfigItem;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::Log',
);

=head1 NAME

Kernel::System::LinkObject::ITSMConfigItem - LinkObject module for ITSMConfigItem

=head2 new()

create an object. Do not use it directly, instead use:

    use Kernel::System::ObjectManager;
    local $Kernel::OM = Kernel::System::ObjectManager->new();
    my $LinkObjectITSMConfigItemObject = $Kernel::OM->Get('Kernel::System::LinkObject::ITSMConfigItem');

=cut

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

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

    return $Self;
}

=head2 LinkListWithData()

fill up the link list with data

    $Success = $LinkObjectBackend->LinkListWithData(
        LinkList => $HashRef,
        UserID   => 1,
    );

=cut

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

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

    # check link list
    if ( ref $Param{LinkList} ne 'HASH' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'LinkList must be a hash reference!',
        );
        return;
    }

    for my $LinkType ( sort keys %{ $Param{LinkList} } ) {

        for my $Direction ( sort keys %{ $Param{LinkList}->{$LinkType} } ) {

            CONFIGITEMID:
            for my $ConfigItemID ( sort keys %{ $Param{LinkList}->{$LinkType}->{$Direction} } ) {

                # get last version data
                my $VersionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
                    ConfigItemID => $ConfigItemID,
                    XMLDataGet   => 0,
                    UserID       => $Param{UserID},
                );

                # remove id from hash if config item can not get
                if ( !$VersionData || ref $VersionData ne 'HASH' || !%{$VersionData} ) {
                    delete $Param{LinkList}->{$LinkType}->{$Direction}->{$ConfigItemID};
                    next CONFIGITEMID;
                }

                # add version data
                $Param{LinkList}->{$LinkType}->{$Direction}->{$ConfigItemID} = $VersionData;
            }
        }
    }

    return 1;
}

=head2 ObjectPermission()

checks read permission for a given object and UserID.

    $Permission = $LinkObject->ObjectPermission(
        Object  => 'ITSMConfigItem',
        Key     => 123,
        UserID  => 1,
    );

=cut

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

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

    # get config of configitem zoom frontend module
    $Self->{Config} = $Kernel::OM->Get('Kernel::Config')->Get('ITSMConfigItem::Frontend::AgentITSMConfigItemZoom');

    # check for access rights
    my $Access = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->Permission(
        Scope  => 'Item',
        ItemID => $Param{Key},
        UserID => $Param{UserID},
        Type   => $Self->{Config}->{Permission},
    );

    return $Access;
}

=head2 ObjectDescriptionGet()

return a hash of object descriptions

Return:

    %Description = (
        Normal => "ConfigItem# 1234455",
        Long   => "ConfigItem# 1234455: The Config Item Title",
    );

    %Description = $LinkObject->ObjectDescriptionGet(
        Key     => 123,
        UserID  => 1,
    );

=cut

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

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

    # create description
    my %Description = (
        Normal => 'ConfigItem',
        Long   => 'ConfigItem',
    );

    return %Description if $Param{Mode} && $Param{Mode} eq 'Temporary';

    # get last version data
    my $VersionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
        ConfigItemID => $Param{Key},
        XMLDataGet   => 0,
        UserID       => $Param{UserID},
    );

    return if !$VersionData;
    return if ref $VersionData ne 'HASH';
    return if !%{$VersionData};

    # create description
    %Description = (
        Normal => "ConfigItem# $VersionData->{Number}",
        Long   => "ConfigItem# $VersionData->{Number}: $VersionData->{Name}",
    );

    return %Description;
}

=head2 ObjectSearch()

return a hash list of the search results

Return:

    $SearchList = {
        C<NOTLINKED> => {
            Source => {
                12  => $DataOfItem12,
                212 => $DataOfItem212,
                332 => $DataOfItem332,
            },
        },
    };

    $SearchList = $LinkObjectBackend->ObjectSearch(
        SubObject    => '25',        # (optional)
        SearchParams => $HashRef,    # (optional)
        UserID       => 1,
    );

=cut

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

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

    # set default params
    $Param{SearchParams} ||= {};

    # set focus
    my %Search;
    for my $Element (qw(Number Name)) {
        if ( $Param{SearchParams}->{$Element} ) {
            $Search{$Element} = '*' . $Param{SearchParams}->{$Element} . '*';
        }
    }

    if ( !$Param{SubObject} ) {

        # get the config with the default subobjects
        my $DefaultSubobject = $Kernel::OM->Get('Kernel::Config')->Get('LinkObject::DefaultSubObject') || {};

        # extract default class name
        my $DefaultClass = $DefaultSubobject->{ITSMConfigItem} || '';

        # get class list
        my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
            Class => 'ITSM::ConfigItem::Class',
        );

        return if !$ClassList;
        return if ref $ClassList ne 'HASH';

        # lookup the class id
        my %ClassListReverse = reverse %{$ClassList};
        $Param{SubObject} = $ClassListReverse{$DefaultClass} || '';
    }

    return if !$Param{SubObject};

    # search the config items
    my $ConfigItemIDs = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemSearchExtended(
        %{ $Param{SearchParams} },
        %Search,
        ClassIDs              => [ $Param{SubObject} ],
        PreviousVersionSearch => 0,
        UsingWildcards        => 1,
        OrderBy               => ['Number'],
        OrderByDirection      => ['Up'],
        Limit                 => 50,
        UserID                => $Param{UserID},
    );

    my %SearchList;
    CONFIGITEMID:
    for my $ConfigItemID ( @{$ConfigItemIDs} ) {

        # get last version data
        my $VersionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
            ConfigItemID => $ConfigItemID,
            XMLDataGet   => 0,
            UserID       => $Param{UserID},
        );

        next CONFIGITEMID if !$VersionData;
        next CONFIGITEMID if ref $VersionData ne 'HASH';
        next CONFIGITEMID if !%{$VersionData};

        # add version data
        $SearchList{NOTLINKED}->{Source}->{$ConfigItemID} = $VersionData;
    }

    return \%SearchList;
}

=head2 LinkAddPre()

link add pre event module

    $True = $LinkObject->LinkAddPre(
        Key          => 123,
        SourceObject => 'ITSMConfigItem',
        SourceKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

    or

    $True = $LinkObject->LinkAddPre(
        Key          => 123,
        TargetObject => 'ITSMConfigItem',
        TargetKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

=cut

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

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

    # do not trigger event for temporary links
    return 1 if $Param{State} eq 'Temporary';

    return 1;
}

=head2 LinkAddPost()

link add pre event module

    $True = $LinkObject->LinkAddPost(
        Key          => 123,
        SourceObject => 'ITSMConfigItem',
        SourceKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

    or

    $True = $LinkObject->LinkAddPost(
        Key          => 123,
        TargetObject => 'ITSMConfigItem',
        TargetKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

=cut

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

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

    # do not trigger event for temporary links
    return 1 if $Param{State} eq 'Temporary';

    # get information about linked object
    my $ID     = $Param{TargetKey}    || $Param{SourceKey};
    my $Object = $Param{TargetObject} || $Param{SourceObject};

    # recalculate the current incident state of this CI
    $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->CurInciStateRecalc(
        ConfigItemID => $Param{Key},
    );

    # trigger LinkAdd event
    $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->EventHandler(
        Event => 'LinkAdd',
        Data  => {
            ConfigItemID => $Param{Key},
            Comment      => $ID . '%%' . $Object,
            Type         => $Param{Type},
        },
        UserID => $Param{UserID},
    );

    return 1;
}

=head2 LinkDeletePre()

link delete pre event module

    $True = $LinkObject->LinkDeletePre(
        Key          => 123,
        SourceObject => 'ITSMConfigItem',
        SourceKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

    or

    $True = $LinkObject->LinkDeletePre(
        Key          => 123,
        TargetObject => 'ITSMConfigItem',
        TargetKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

=cut

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

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

    # do not trigger event for temporary links
    return 1 if $Param{State} eq 'Temporary';

    return 1;
}

=head2 LinkDeletePost()

link delete post event module

    $True = $LinkObject->LinkDeletePost(
        Key          => 123,
        SourceObject => 'ITSMConfigItem',
        SourceKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

    or

    $True = $LinkObject->LinkDeletePost(
        Key          => 123,
        TargetObject => 'ITSMConfigItem',
        TargetKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

=cut

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

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

    # do not trigger event for temporary links
    return 1 if $Param{State} eq 'Temporary';

    # get information about linked object
    my $ID     = $Param{TargetKey}    || $Param{SourceKey};
    my $Object = $Param{TargetObject} || $Param{SourceObject};

    # recalculate the current incident state of this CI
    $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->CurInciStateRecalc(
        ConfigItemID => $Param{Key},
    );

    # trigger LinkDelete event
    $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->EventHandler(
        Event => 'LinkDelete',
        Data  => {
            ConfigItemID => $Param{Key},
            Comment      => $ID . '%%' . $Object,
        },
        UserID => $Param{UserID},
    );

    return 1;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::LinkObject::ITSMConfigItem;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::Log',
);

=head1 NAME

Kernel::System::LinkObject::ITSMConfigItem - LinkObject module for ITSMConfigItem

=head2 new()

create an object. Do not use it directly, instead use:

    use Kernel::System::ObjectManager;
    local $Kernel::OM = Kernel::System::ObjectManager->new();
    my $LinkObjectITSMConfigItemObject = $Kernel::OM->Get('Kernel::System::LinkObject::ITSMConfigItem');

=cut

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

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

    return $Self;
}

=head2 LinkListWithData()

fill up the link list with data

    $Success = $LinkObjectBackend->LinkListWithData(
        LinkList => $HashRef,
        UserID   => 1,
    );

=cut

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

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

    # check link list
    if ( ref $Param{LinkList} ne 'HASH' ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'LinkList must be a hash reference!',
        );
        return;
    }

    for my $LinkType ( sort keys %{ $Param{LinkList} } ) {

        for my $Direction ( sort keys %{ $Param{LinkList}->{$LinkType} } ) {

            CONFIGITEMID:
            for my $ConfigItemID ( sort keys %{ $Param{LinkList}->{$LinkType}->{$Direction} } ) {

                # get last version data
                my $VersionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
                    ConfigItemID => $ConfigItemID,
                    XMLDataGet   => 0,
                    UserID       => $Param{UserID},
                );

                # remove id from hash if config item can not get
                if ( !$VersionData || ref $VersionData ne 'HASH' || !%{$VersionData} ) {
                    delete $Param{LinkList}->{$LinkType}->{$Direction}->{$ConfigItemID};
                    next CONFIGITEMID;
                }

                # add version data
                $Param{LinkList}->{$LinkType}->{$Direction}->{$ConfigItemID} = $VersionData;
            }
        }
    }

    return 1;
}

=head2 ObjectPermission()

checks read permission for a given object and UserID.

    $Permission = $LinkObject->ObjectPermission(
        Object  => 'ITSMConfigItem',
        Key     => 123,
        UserID  => 1,
    );

=cut

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

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

    # get config of configitem zoom frontend module
    $Self->{Config} = $Kernel::OM->Get('Kernel::Config')->Get('ITSMConfigItem::Frontend::AgentITSMConfigItemZoom');

    # check for access rights
    my $Access = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->Permission(
        Scope  => 'Item',
        ItemID => $Param{Key},
        UserID => $Param{UserID},
        Type   => $Self->{Config}->{Permission},
    );

    return $Access;
}

=head2 ObjectDescriptionGet()

return a hash of object descriptions

Return:

    %Description = (
        Normal => "ConfigItem# 1234455",
        Long   => "ConfigItem# 1234455: The Config Item Title",
    );

    %Description = $LinkObject->ObjectDescriptionGet(
        Key     => 123,
        UserID  => 1,
    );

=cut

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

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

    # create description
    my %Description = (
        Normal => 'ConfigItem',
        Long   => 'ConfigItem',
    );

    return %Description if $Param{Mode} && $Param{Mode} eq 'Temporary';

    # get last version data
    my $VersionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
        ConfigItemID => $Param{Key},
        XMLDataGet   => 0,
        UserID       => $Param{UserID},
    );

    return if !$VersionData;
    return if ref $VersionData ne 'HASH';
    return if !%{$VersionData};

    # create description
    %Description = (
        Normal => "ConfigItem# $VersionData->{Number}",
        Long   => "ConfigItem# $VersionData->{Number}: $VersionData->{Name}",
    );

    return %Description;
}

=head2 ObjectSearch()

return a hash list of the search results

Return:

    $SearchList = {
        C<NOTLINKED> => {
            Source => {
                12  => $DataOfItem12,
                212 => $DataOfItem212,
                332 => $DataOfItem332,
            },
        },
    };

    $SearchList = $LinkObjectBackend->ObjectSearch(
        SubObject    => '25',        # (optional)
        SearchParams => $HashRef,    # (optional)
        UserID       => 1,
    );

=cut

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

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

    # set default params
    $Param{SearchParams} ||= {};

    # set focus
    my %Search;
    for my $Element (qw(Number Name)) {
        if ( $Param{SearchParams}->{$Element} ) {
            $Search{$Element} = '*' . $Param{SearchParams}->{$Element} . '*';
        }
    }

    if ( !$Param{SubObject} ) {

        # get the config with the default subobjects
        my $DefaultSubobject = $Kernel::OM->Get('Kernel::Config')->Get('LinkObject::DefaultSubObject') || {};

        # extract default class name
        my $DefaultClass = $DefaultSubobject->{ITSMConfigItem} || '';

        # get class list
        my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
            Class => 'ITSM::ConfigItem::Class',
        );

        return if !$ClassList;
        return if ref $ClassList ne 'HASH';

        # lookup the class id
        my %ClassListReverse = reverse %{$ClassList};
        $Param{SubObject} = $ClassListReverse{$DefaultClass} || '';
    }

    return if !$Param{SubObject};

    # search the config items
    my $ConfigItemIDs = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemSearchExtended(
        %{ $Param{SearchParams} },
        %Search,
        ClassIDs              => [ $Param{SubObject} ],
        PreviousVersionSearch => 0,
        UsingWildcards        => 1,
        OrderBy               => ['Number'],
        OrderByDirection      => ['Up'],
        Limit                 => 50,
        UserID                => $Param{UserID},
    );

    my %SearchList;
    CONFIGITEMID:
    for my $ConfigItemID ( @{$ConfigItemIDs} ) {

        # get last version data
        my $VersionData = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
            ConfigItemID => $ConfigItemID,
            XMLDataGet   => 0,
            UserID       => $Param{UserID},
        );

        next CONFIGITEMID if !$VersionData;
        next CONFIGITEMID if ref $VersionData ne 'HASH';
        next CONFIGITEMID if !%{$VersionData};

        # add version data
        $SearchList{NOTLINKED}->{Source}->{$ConfigItemID} = $VersionData;
    }

    return \%SearchList;
}

=head2 LinkAddPre()

link add pre event module

    $True = $LinkObject->LinkAddPre(
        Key          => 123,
        SourceObject => 'ITSMConfigItem',
        SourceKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

    or

    $True = $LinkObject->LinkAddPre(
        Key          => 123,
        TargetObject => 'ITSMConfigItem',
        TargetKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

=cut

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

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

    # do not trigger event for temporary links
    return 1 if $Param{State} eq 'Temporary';

    return 1;
}

=head2 LinkAddPost()

link add pre event module

    $True = $LinkObject->LinkAddPost(
        Key          => 123,
        SourceObject => 'ITSMConfigItem',
        SourceKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

    or

    $True = $LinkObject->LinkAddPost(
        Key          => 123,
        TargetObject => 'ITSMConfigItem',
        TargetKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

=cut

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

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

    # do not trigger event for temporary links
    return 1 if $Param{State} eq 'Temporary';

    # get information about linked object
    my $ID     = $Param{TargetKey}    || $Param{SourceKey};
    my $Object = $Param{TargetObject} || $Param{SourceObject};

    # recalculate the current incident state of this CI
    $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->CurInciStateRecalc(
        ConfigItemID => $Param{Key},
    );

    # trigger LinkAdd event
    $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->EventHandler(
        Event => 'LinkAdd',
        Data  => {
            ConfigItemID => $Param{Key},
            Comment      => $ID . '%%' . $Object,
            Type         => $Param{Type},
        },
        UserID => $Param{UserID},
    );

    return 1;
}

=head2 LinkDeletePre()

link delete pre event module

    $True = $LinkObject->LinkDeletePre(
        Key          => 123,
        SourceObject => 'ITSMConfigItem',
        SourceKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

    or

    $True = $LinkObject->LinkDeletePre(
        Key          => 123,
        TargetObject => 'ITSMConfigItem',
        TargetKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

=cut

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

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

    # do not trigger event for temporary links
    return 1 if $Param{State} eq 'Temporary';

    return 1;
}

=head2 LinkDeletePost()

link delete post event module

    $True = $LinkObject->LinkDeletePost(
        Key          => 123,
        SourceObject => 'ITSMConfigItem',
        SourceKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

    or

    $True = $LinkObject->LinkDeletePost(
        Key          => 123,
        TargetObject => 'ITSMConfigItem',
        TargetKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

=cut

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

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

    # do not trigger event for temporary links
    return 1 if $Param{State} eq 'Temporary';

    # get information about linked object
    my $ID     = $Param{TargetKey}    || $Param{SourceKey};
    my $Object = $Param{TargetObject} || $Param{SourceObject};

    # recalculate the current incident state of this CI
    $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->CurInciStateRecalc(
        ConfigItemID => $Param{Key},
    );

    # trigger LinkDelete event
    $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->EventHandler(
        Event => 'LinkDelete',
        Data  => {
            ConfigItemID => $Param{Key},
            Comment      => $ID . '%%' . $Object,
        },
        UserID => $Param{UserID},
    );

    return 1;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::Stats::Dynamic::ITSMConfigItem;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::System::GeneralCatalog',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::DateTime',
);

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

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

    return $Self;
}

sub GetObjectName {
    my $Self = shift;

    return 'ITSMConfigItem';
}

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

    my %Behaviours = (
        ProvidesDashboardWidget => 1,
    );

    return %Behaviours;
}

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

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # get deployment state list
    my $DeplStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    # get incident state list
    my $InciStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::Core::IncidentState',
    );

    # get current time to fix bug#3830
    my $TimeStamp = $Kernel::OM->Create('Kernel::System::DateTime')->ToString();

    # create object attribute array
    my @ObjectAttributes = (
        {
            Name                => 'Class',
            UseAsXvalue         => 1,
            UseAsValueSeries    => 1,
            UseAsRestriction    => 1,
            Element             => 'ClassIDs',
            Block               => 'MultiSelectField',
            LanguageTranslation => 0,
            Values              => $ClassList,
        },
        {
            Name             => 'Deployment State',
            UseAsXvalue      => 1,
            UseAsValueSeries => 1,
            UseAsRestriction => 1,
            Element          => 'DeplStateIDs',
            Block            => 'MultiSelectField',
            Values           => $DeplStateList,
        },
        {
            Name             => 'Incident State',
            UseAsXvalue      => 1,
            UseAsValueSeries => 1,
            UseAsRestriction => 1,
            Element          => 'InciStateIDs',
            Block            => 'MultiSelectField',
            Values           => $InciStateList,
        },
        {
            Name             => 'Number',
            UseAsXvalue      => 0,
            UseAsValueSeries => 0,
            UseAsRestriction => 1,
            Element          => 'Number',
            Block            => 'InputField',
        },
        {
            Name             => 'Name',
            UseAsXvalue      => 0,
            UseAsValueSeries => 0,
            UseAsRestriction => 1,
            Element          => 'Name',
            Block            => 'InputField',
        },
        {
            Name             => 'Create Time',
            UseAsXvalue      => 1,
            UseAsValueSeries => 1,
            UseAsRestriction => 1,
            Element          => 'CreateTime',
            TimePeriodFormat => 'DateInputFormat',
            Block            => 'Time',
            Values           => {
                TimeStart => 'ConfigItemCreateTimeNewerDate',
                TimeStop  => 'ConfigItemCreateTimeOlderDate',
            },
        },
        {
            Name             => 'Change Time',
            UseAsXvalue      => 1,
            UseAsValueSeries => 1,
            UseAsRestriction => 1,
            Element          => 'ChangeTime',
            TimePeriodFormat => 'DateInputFormat',
            Block            => 'Time',
            TimeStop         => $TimeStamp,
            Values           => {
                TimeStart => 'ConfigItemChangeTimeNewerDate',
                TimeStop  => 'ConfigItemChangeTimeOlderDate',
            },
        },
    );

    # add the xml data
    CLASSID:
    for my $ClassID ( sort keys %{$ClassList} ) {

        # get xml definition hash
        my $XMLDefinition = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->DefinitionGet(
            ClassID => $ClassID,
        );

        next CLASSID if !$XMLDefinition->{DefinitionID};

        $Self->_XMLAttributeAdd(
            ObjectAttributes => \@ObjectAttributes,
            XMLDefinition    => $XMLDefinition->{DefinitionRef},
            Prefix           => 'XML::' . $ClassID,
            PrefixName       => $ClassList->{$ClassID},
        );
    }

    return @ObjectAttributes;
}

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

    return if !$Param{ObjectAttributes};
    return if !$Param{XMLDefinition};
    return if ref $Param{XMLDefinition} ne 'ARRAY';

    if ( $Param{Prefix} ) {
        $Param{Prefix} .= '::';
    }
    if ( $Param{PrefixName} ) {
        $Param{PrefixName} .= '::';
    }

    $Param{Level}      ||= 0;
    $Param{Prefix}     ||= '';
    $Param{PrefixName} ||= '';

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        next ITEM if !$Item->{Searchable} && !$Item->{Sub};

        # create key and name
        my $Key  = $Param{Prefix} . $Item->{Key};
        my $Name = $Param{PrefixName} . $Item->{Name};

        # add attribute
        my $Attribute = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->XMLStatsAttributeCreate(
            Key  => $Key,
            Item => $Item,
            Name => $Name,
        );

        next ITEM if !$Attribute;
        next ITEM if ref $Attribute ne 'ARRAY';
        next ITEM if !scalar @{$Attribute};

        # add attributes to object array
        push @{ $Param{ObjectAttributes} }, @{$Attribute};

        next ITEM if !$Item->{Sub};

        # start recursion, if "Sub" was found
        $Self->_XMLAttributeAdd(
            ObjectAttributes => $Param{ObjectAttributes},
            XMLDefinition    => $Item->{Sub},
            Level            => $Param{Level} + 1,
            Prefix           => $Key,
            PrefixName       => $Name,
        );
    }

    return 1;
}

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

    return ( int rand 500 ) + 1;
}

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

    # set limit
    $Param{Limit} = 1_000_000;

    # extract all xml param keys from the param hash
    my @XMLParams = grep { $_ =~ m{\A XML::}xms } sort keys %Param;

    if (@XMLParams) {
        return 'You must define a class in one axis.' if !$Param{ClassIDs};
        return 'You must define a class in one axis.' if ref $Param{ClassIDs} ne 'ARRAY';
    }

    # to collect date fields separately
    my %XMLParamsDateFields;

    my %XMLClassIDs;
    PARAMKEY:
    for my $ParamKey (@XMLParams) {

        # extract search values
        my $SearchValues = $Param{$ParamKey};

        # prepare param value
        if ( !ref $SearchValues ) {
            $SearchValues = [$SearchValues];
        }

        next PARAMKEY if !@{$SearchValues};

        # split param key
        my ( $ClassID, $SearchKey ) = $ParamKey =~ m{ \A XML:: ( \d+ ) :: (.+) \z }xms;

        # prepare search key
        $SearchKey =~ s[ :: ]['}[%]{']xmsg;

        # Add class id to xml class id hash
        $XMLClassIDs{$ClassID} = 1;

        # collect date fields separately (e.g WarrantyExpirationDateNewerDate)
        if ( $SearchKey =~ m{ (.+) ( NewerDate | OlderDate ) \z }xms ) {
            $XMLParamsDateFields{$1}->{$2} = $SearchValues->[0];

            next PARAMKEY;
        }

        # create search hash
        my $SearchHash = {
            '[1]{\'Version\'}[1]{\'' . $SearchKey . '\'}[%]{\'Content\'}' => $SearchValues,
        };

        push @{ $Param{What} }, $SearchHash;
    }

    # build search hash for date fields
    for my $DateFieldName ( sort keys %XMLParamsDateFields ) {

        my $SearchHash = {
            '[1]{\'Version\'}[1]{\'' . $DateFieldName . '\'}[%]{\'Content\'}' => {
                '-between' => [
                    $XMLParamsDateFields{$DateFieldName}->{NewerDate},
                    $XMLParamsDateFields{$DateFieldName}->{OlderDate},
                ],
            },
        };

        push @{ $Param{What} }, $SearchHash;
    }

    if (%XMLClassIDs) {
        my @Exists = grep { $XMLClassIDs{$_} } @{ $Param{ClassIDs} };

        return 0 if !@Exists;
    }

    # start config item extended search
    my $ConfigItemIDs = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemSearchExtended(%Param);

    return scalar @{$ConfigItemIDs};
}

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

    return \%Param;
}

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

    return \%Param;
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::Stats::Dynamic::ITSMConfigItem;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::System::GeneralCatalog',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::DateTime',
);

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

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

    return $Self;
}

sub GetObjectName {
    my $Self = shift;

    return 'ITSMConfigItem';
}

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

    my %Behaviours = (
        ProvidesDashboardWidget => 1,
    );

    return %Behaviours;
}

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

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # get deployment state list
    my $DeplStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    # get incident state list
    my $InciStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::Core::IncidentState',
    );

    # get current time to fix bug#3830
    my $TimeStamp = $Kernel::OM->Create('Kernel::System::DateTime')->ToString();

    # create object attribute array
    my @ObjectAttributes = (
        {
            Name                => 'Class',
            UseAsXvalue         => 1,
            UseAsValueSeries    => 1,
            UseAsRestriction    => 1,
            Element             => 'ClassIDs',
            Block               => 'MultiSelectField',
            LanguageTranslation => 0,
            Values              => $ClassList,
        },
        {
            Name             => 'Deployment State',
            UseAsXvalue      => 1,
            UseAsValueSeries => 1,
            UseAsRestriction => 1,
            Element          => 'DeplStateIDs',
            Block            => 'MultiSelectField',
            Values           => $DeplStateList,
        },
        {
            Name             => 'Incident State',
            UseAsXvalue      => 1,
            UseAsValueSeries => 1,
            UseAsRestriction => 1,
            Element          => 'InciStateIDs',
            Block            => 'MultiSelectField',
            Values           => $InciStateList,
        },
        {
            Name             => 'Number',
            UseAsXvalue      => 0,
            UseAsValueSeries => 0,
            UseAsRestriction => 1,
            Element          => 'Number',
            Block            => 'InputField',
        },
        {
            Name             => 'Name',
            UseAsXvalue      => 0,
            UseAsValueSeries => 0,
            UseAsRestriction => 1,
            Element          => 'Name',
            Block            => 'InputField',
        },
        {
            Name             => 'Create Time',
            UseAsXvalue      => 1,
            UseAsValueSeries => 1,
            UseAsRestriction => 1,
            Element          => 'CreateTime',
            TimePeriodFormat => 'DateInputFormat',
            Block            => 'Time',
            Values           => {
                TimeStart => 'ConfigItemCreateTimeNewerDate',
                TimeStop  => 'ConfigItemCreateTimeOlderDate',
            },
        },
        {
            Name             => 'Change Time',
            UseAsXvalue      => 1,
            UseAsValueSeries => 1,
            UseAsRestriction => 1,
            Element          => 'ChangeTime',
            TimePeriodFormat => 'DateInputFormat',
            Block            => 'Time',
            TimeStop         => $TimeStamp,
            Values           => {
                TimeStart => 'ConfigItemChangeTimeNewerDate',
                TimeStop  => 'ConfigItemChangeTimeOlderDate',
            },
        },
    );

    # add the xml data
    CLASSID:
    for my $ClassID ( sort keys %{$ClassList} ) {

        # get xml definition hash
        my $XMLDefinition = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->DefinitionGet(
            ClassID => $ClassID,
        );

        next CLASSID if !$XMLDefinition->{DefinitionID};

        $Self->_XMLAttributeAdd(
            ObjectAttributes => \@ObjectAttributes,
            XMLDefinition    => $XMLDefinition->{DefinitionRef},
            Prefix           => 'XML::' . $ClassID,
            PrefixName       => $ClassList->{$ClassID},
        );
    }

    return @ObjectAttributes;
}

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

    return if !$Param{ObjectAttributes};
    return if !$Param{XMLDefinition};
    return if ref $Param{XMLDefinition} ne 'ARRAY';

    if ( $Param{Prefix} ) {
        $Param{Prefix} .= '::';
    }
    if ( $Param{PrefixName} ) {
        $Param{PrefixName} .= '::';
    }

    $Param{Level}      ||= 0;
    $Param{Prefix}     ||= '';
    $Param{PrefixName} ||= '';

    ITEM:
    for my $Item ( @{ $Param{XMLDefinition} } ) {

        next ITEM if !$Item->{Searchable} && !$Item->{Sub};

        # create key and name
        my $Key  = $Param{Prefix} . $Item->{Key};
        my $Name = $Param{PrefixName} . $Item->{Name};

        # add attribute
        my $Attribute = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->XMLStatsAttributeCreate(
            Key  => $Key,
            Item => $Item,
            Name => $Name,
        );

        next ITEM if !$Attribute;
        next ITEM if ref $Attribute ne 'ARRAY';
        next ITEM if !scalar @{$Attribute};

        # add attributes to object array
        push @{ $Param{ObjectAttributes} }, @{$Attribute};

        next ITEM if !$Item->{Sub};

        # start recursion, if "Sub" was found
        $Self->_XMLAttributeAdd(
            ObjectAttributes => $Param{ObjectAttributes},
            XMLDefinition    => $Item->{Sub},
            Level            => $Param{Level} + 1,
            Prefix           => $Key,
            PrefixName       => $Name,
        );
    }

    return 1;
}

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

    return ( int rand 500 ) + 1;
}

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

    # set limit
    $Param{Limit} = 1_000_000;

    # extract all xml param keys from the param hash
    my @XMLParams = grep { $_ =~ m{\A XML::}xms } sort keys %Param;

    if (@XMLParams) {
        return 'You must define a class in one axis.' if !$Param{ClassIDs};
        return 'You must define a class in one axis.' if ref $Param{ClassIDs} ne 'ARRAY';
    }

    # to collect date fields separately
    my %XMLParamsDateFields;

    my %XMLClassIDs;
    PARAMKEY:
    for my $ParamKey (@XMLParams) {

        # extract search values
        my $SearchValues = $Param{$ParamKey};

        # prepare param value
        if ( !ref $SearchValues ) {
            $SearchValues = [$SearchValues];
        }

        next PARAMKEY if !@{$SearchValues};

        # split param key
        my ( $ClassID, $SearchKey ) = $ParamKey =~ m{ \A XML:: ( \d+ ) :: (.+) \z }xms;

        # prepare search key
        $SearchKey =~ s[ :: ]['}[%]{']xmsg;

        # Add class id to xml class id hash
        $XMLClassIDs{$ClassID} = 1;

        # collect date fields separately (e.g WarrantyExpirationDateNewerDate)
        if ( $SearchKey =~ m{ (.+) ( NewerDate | OlderDate ) \z }xms ) {
            $XMLParamsDateFields{$1}->{$2} = $SearchValues->[0];

            next PARAMKEY;
        }

        # create search hash
        my $SearchHash = {
            '[1]{\'Version\'}[1]{\'' . $SearchKey . '\'}[%]{\'Content\'}' => $SearchValues,
        };

        push @{ $Param{What} }, $SearchHash;
    }

    # build search hash for date fields
    for my $DateFieldName ( sort keys %XMLParamsDateFields ) {

        my $SearchHash = {
            '[1]{\'Version\'}[1]{\'' . $DateFieldName . '\'}[%]{\'Content\'}' => {
                '-between' => [
                    $XMLParamsDateFields{$DateFieldName}->{NewerDate},
                    $XMLParamsDateFields{$DateFieldName}->{OlderDate},
                ],
            },
        };

        push @{ $Param{What} }, $SearchHash;
    }

    if (%XMLClassIDs) {
        my @Exists = grep { $XMLClassIDs{$_} } @{ $Param{ClassIDs} };

        return 0 if !@Exists;
    }

    # start config item extended search
    my $ConfigItemIDs = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemSearchExtended(%Param);

    return scalar @{$ConfigItemIDs};
}

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

    return \%Param;
}

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

    return \%Param;
}

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OlRpY2tldDo6RXZlbnQ6OkR5bmFtaWNGaWVsZENvbmZpZ0l0ZW1BZGRpdGlvbmFsREZTdG9yYWdlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIEtlcm5lbDo6U3lzdGVtOjpWYXJpYWJsZUNoZWNrIHF3KDphbGwpOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06OkR5bmFtaWNGaWVsZCcsCiAgICAnS2VybmVsOjpTeXN0ZW06OkxvZycsCiAgICAnS2VybmVsOjpTeXN0ZW06OkR5bmFtaWNGaWVsZENvbmZpZ0l0ZW0nLAopOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJExvZ09iamVjdCAgICAgICAgICAgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyk7CiAgICBteSAkRHluYW1pY0ZpZWxkT2JqZWN0ICAgICAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpEeW5hbWljRmllbGQnKTsKICAgIG15ICREeW5hbWljRmllbGRDb25maWdJdGVtT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkR5bmFtaWNGaWVsZDo6Q29uZmlnSXRlbScpOwoKICAgIE5FRURFRDoKICAgIGZvciBteSAkTmVlZGVkIChxdyggRGF0YSBFdmVudCBDb25maWcgVXNlcklEICkpIHsKICAgICAgICBuZXh0IE5FRURFRCBpZiAkUGFyYW17JE5lZWRlZH07CgogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gIk5lZWQgJE5lZWRlZCEiCiAgICAgICAgKTsKICAgICAgICByZXR1cm4gMTsKICAgIH0KCiAgICBORUVERUQ6CiAgICBmb3IgbXkgJE5lZWRlZCAocXcoIFRpY2tldElEICkpIHsKICAgICAgICBuZXh0IE5FRURFRCBpZiAkUGFyYW17RGF0YX0tPnskTmVlZGVkfTsKCiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkTmVlZGVkIGluIERhdGEhIgogICAgICAgICk7CiAgICAgICAgcmV0dXJuIDE7CiAgICB9CgogICAgbXkgJER5bmFtaWNGaWVsZENvbmZpZyA9ICREeW5hbWljRmllbGRPYmplY3QtPkR5bmFtaWNGaWVsZEdldCgKICAgICAgICBOYW1lID0+ICRQYXJhbXtEYXRhfS0+e0ZpZWxkTmFtZX0sCiAgICApOwogICAgcmV0dXJuIDEgaWYgIUlzSGFzaFJlZldpdGhEYXRhKCREeW5hbWljRmllbGRDb25maWcpOwogICAgcmV0dXJuIDEgaWYgJER5bmFtaWNGaWVsZENvbmZpZy0+e0ZpZWxkVHlwZX0gbmUgJ0NvbmZpZ0l0ZW1Ecm9wZG93bicKICAgICAgICAmJiAkRHluYW1pY0ZpZWxkQ29uZmlnLT57RmllbGRUeXBlfSBuZSAnQ29uZmlnSXRlbU11bHRpc2VsZWN0JzsKCiAgICByZXR1cm4gMSBpZiAkRHluYW1pY0ZpZWxkQ29uZmlnLT57T2JqZWN0VHlwZX0gbmUgJ1RpY2tldCc7CiAgICByZXR1cm4gMSBpZiAhSXNBcnJheVJlZldpdGhEYXRhKCAkRHluYW1pY0ZpZWxkQ29uZmlnLT57Q29uZmlnfS0+e0FkZGl0aW9uYWxERlN0b3JhZ2V9ICk7CgogICAgbXkgJENvbmZpZ0l0ZW1JRHMgPSAkUGFyYW17RGF0YX0tPntWYWx1ZX07ICAgICMgY2FuIGFsc28gYmUgb25seSBvbmUgKHNjYWxhcikuCiAgICByZXR1cm4gMSBpZiAhZGVmaW5lZCAkQ29uZmlnSXRlbUlEczsKCiAgICBpZiAoICFJc0FycmF5UmVmV2l0aERhdGEoJENvbmZpZ0l0ZW1JRHMpICkgewogICAgICAgICRDb25maWdJdGVtSURzID0gWyAkQ29uZmlnSXRlbUlEcywgXTsKICAgIH0KCiAgICBteSAkQWRkaXRpb25hbERGU3RvcmFnZURhdGEgPSAkRHluYW1pY0ZpZWxkQ29uZmlnSXRlbU9iamVjdC0+R2V0QWRkaXRpb25hbERGU3RvcmFnZURhdGEoCiAgICAgICAgU291cmNlRHluYW1pY0ZpZWxkTmFtZSA9PiAkUGFyYW17RGF0YX0tPntGaWVsZE5hbWV9LAogICAgICAgIFNlbGVjdGVkQ29uZmlnSXRlbUlEcyAgPT4gJENvbmZpZ0l0ZW1JRHMsCiAgICAgICAgU3RvcmFnZVR5cGUgICAgICAgICAgICA9PiAnQmFja2VuZCcsCiAgICAgICAgVXNlcklEICAgICAgICAgICAgICAgICA9PiAkUGFyYW17VXNlcklEfSwKICAgICk7CiAgICByZXR1cm4gMSBpZiAhSXNIYXNoUmVmV2l0aERhdGEoJEFkZGl0aW9uYWxERlN0b3JhZ2VEYXRhKTsKCiAgICBteSAkQWRkaXRpb25hbER5bmFtaWNGaWVsZFZhbHVlc1N0b3JlZCA9ICREeW5hbWljRmllbGRDb25maWdJdGVtT2JqZWN0LT5TdG9yZUR5bmFtaWNGaWVsZFZhbHVlcygKICAgICAgICBUaWNrZXRJRCAgICAgICAgICAgICAgICA9PiAkUGFyYW17RGF0YX0tPntUaWNrZXRJRH0sCiAgICAgICAgQWRkaXRpb25hbERGU3RvcmFnZURhdGEgPT4gJEFkZGl0aW9uYWxERlN0b3JhZ2VEYXRhLAogICAgICAgIFVzZXJJRCAgICAgICAgICAgICAgICAgID0+ICRQYXJhbXtVc2VySUR9LAogICAgKTsKCiAgICByZXR1cm4gaWYgISRBZGRpdGlvbmFsRHluYW1pY0ZpZWxkVmFsdWVzU3RvcmVkOwoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OlRpY2tldDo6RXZlbnQ6OkR5bmFtaWNGaWVsZENvbmZpZ0l0ZW1UaWNrZXRMaW5rOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIEtlcm5lbDo6U3lzdGVtOjpWYXJpYWJsZUNoZWNrIHF3KDphbGwpOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06OkR5bmFtaWNGaWVsZCcsCiAgICAnS2VybmVsOjpTeXN0ZW06OkxpbmtPYmplY3QnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpMb2cnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpUaWNrZXQnLAopOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJExvZ09iamVjdCAgICAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKTsKICAgIG15ICRUaWNrZXRPYmplY3QgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VGlja2V0Jyk7CiAgICBteSAkTGlua09iamVjdCAgICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxpbmtPYmplY3QnKTsKICAgIG15ICREeW5hbWljRmllbGRPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6RHluYW1pY0ZpZWxkJyk7CgogICAgTkVFREVEOgogICAgZm9yIG15ICROZWVkZWQgKHF3KCBEYXRhIEV2ZW50IENvbmZpZyBVc2VySUQgKSkgewogICAgICAgIG5leHQgTkVFREVEIGlmICRQYXJhbXskTmVlZGVkfTsKCiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkTmVlZGVkISIKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICBORUVERUQ6CiAgICBmb3IgbXkgJE5lZWRlZCAocXcoIFRpY2tldElEICkpIHsKICAgICAgICBuZXh0IE5FRURFRCBpZiAkUGFyYW17RGF0YX0tPnskTmVlZGVkfTsKCiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkTmVlZGVkIGluIERhdGEhIgogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIG15ICVWYWxpZER5bmFtaWNGaWVsZFR5cGVzID0gKAogICAgICAgIENvbmZpZ0l0ZW1Ecm9wZG93biAgICA9PiAxLAogICAgICAgIENvbmZpZ0l0ZW1NdWx0aXNlbGVjdCA9PiAxLAogICAgKTsKCiAgICBteSAkRHluYW1pY0ZpZWxkID0gJER5bmFtaWNGaWVsZE9iamVjdC0+RHluYW1pY0ZpZWxkR2V0KAogICAgICAgIE5hbWUgPT4gJFBhcmFte0RhdGF9LT57RmllbGROYW1lfQogICAgKTsKICAgIHJldHVybiBpZiAhJFZhbGlkRHluYW1pY0ZpZWxkVHlwZXN7ICREeW5hbWljRmllbGQtPntGaWVsZFR5cGV9IH07CgogICAgIyBTa2lwLCBpZiBubyBsaW5rIHR5cGUgaXMgY29uZmlndXJlZC4KICAgIG15ICRMaW5rVHlwZSA9ICREeW5hbWljRmllbGQtPntDb25maWd9LT57Q29uZmlnSXRlbUxpbmtUeXBlfTsKICAgIHJldHVybiBpZiAhJExpbmtUeXBlOwoKICAgIG15ICRUaWNrZXRJRCA9ICRQYXJhbXtEYXRhfS0+e1RpY2tldElEfTsKCiAgICBteSAkTGlua0xpc3QgPSAkTGlua09iamVjdC0+TGlua0xpc3QoCiAgICAgICAgT2JqZWN0ICA9PiAnVGlja2V0JywKICAgICAgICBLZXkgICAgID0+ICRUaWNrZXRJRCwKICAgICAgICBPYmplY3QyID0+ICdJVFNNQ29uZmlnSXRlbScsCiAgICAgICAgU3RhdGUgICA9PiAnVmFsaWQnLAogICAgICAgIFVzZXJJRCAgPT4gMSwKICAgICk7CgogICAgbXkgQE5ld1ZhbHVlczsKICAgIGlmICggcmVmICRQYXJhbXtEYXRhfS0+e1ZhbHVlfSBlcSAnQVJSQVknICkgewogICAgICAgIEBOZXdWYWx1ZXMgPSBAeyAkUGFyYW17RGF0YX0tPntWYWx1ZX0gfTsKICAgIH0KICAgIGVsc2UgewogICAgICAgIEBOZXdWYWx1ZXMgPSAkUGFyYW17RGF0YX0tPntWYWx1ZX07CiAgICB9CgogICAgbXkgQE9sZFZhbHVlczsKICAgIGlmICggJFBhcmFte0RhdGF9LT57T2xkVmFsdWV9ICYmIHJlZiAkUGFyYW17RGF0YX0tPntPbGRWYWx1ZX0gZXEgJ0FSUkFZJyApIHsKICAgICAgICBAT2xkVmFsdWVzID0gQHsgJFBhcmFte0RhdGF9LT57T2xkVmFsdWV9IH07CiAgICB9CiAgICBlbHNpZiAoICRQYXJhbXtEYXRhfS0+e09sZFZhbHVlfSApIHsKICAgICAgICBAT2xkVmFsdWVzID0gJFBhcmFte0RhdGF9LT57T2xkVmFsdWV9OwogICAgfQoKICAgIG15ICRTb3VyY2VPYmplY3QgPSAkRHluYW1pY0ZpZWxkLT57Q29uZmlnfS0+e0NvbmZpZ0l0ZW1MaW5rU291cmNlfTsKICAgIG15ICRTb3VyY2VLZXk7CiAgICBteSAkVGFyZ2V0T2JqZWN0OwogICAgbXkgJFRhcmdldEtleTsKCiAgICBpZiAoICRTb3VyY2VPYmplY3QgZXEgJ0lUU01Db25maWdJdGVtJyApIHsKICAgICAgICAkVGFyZ2V0T2JqZWN0ID0gJER5bmFtaWNGaWVsZC0+e09iamVjdFR5cGV9OwogICAgICAgICRUYXJnZXRLZXkgICAgPSAkVGlja2V0SUQ7CiAgICB9CiAgICBlbHNlIHsKICAgICAgICAkU291cmNlS2V5ICAgID0gJFRpY2tldElEOwogICAgICAgICRUYXJnZXRPYmplY3QgPSAnSVRTTUNvbmZpZ0l0ZW0nOwogICAgfQoKICAgICMgUmVtb3ZlIGxpbmtzIGZvciByZW1vdmVkIGNvbmZpZyBpdGVtcyAob25seSBpZiBhY3RpdmF0ZWQpLgogICAgbXkgJENvbmZpZ0l0ZW1MaW5rUmVtb3ZhbCA9ICREeW5hbWljRmllbGQtPntDb25maWd9LT57Q29uZmlnSXRlbUxpbmtSZW1vdmFsfTsKICAgIGlmICggQE9sZFZhbHVlcyAmJiAkQ29uZmlnSXRlbUxpbmtSZW1vdmFsICkgewogICAgICAgIG15IEBEZXNlbGVjdGVkQ29uZmlnSXRlbUlEczsKCiAgICAgICAgQ09ORklHSVRFTUlEOgogICAgICAgIGZvciBteSAkQ29uZmlnSXRlbUlEIChAT2xkVmFsdWVzKSB7CiAgICAgICAgICAgIG15ICRDb25maWdJdGVtSURTdGlsbFNlbGVjdGVkID0gZ3JlcCB7ICRDb25maWdJdGVtSUQgZXEgJF8gfSBATmV3VmFsdWVzOwogICAgICAgICAgICBuZXh0IENPTkZJR0lURU1JRCBpZiAkQ29uZmlnSXRlbUlEU3RpbGxTZWxlY3RlZDsKCiAgICAgICAgICAgIHB1c2ggQERlc2VsZWN0ZWRDb25maWdJdGVtSURzLCAkQ29uZmlnSXRlbUlEOwogICAgICAgIH0KCiAgICAgICAgQ09ORklHSVRFTUlEOgogICAgICAgIGZvciBteSAkQ29uZmlnSXRlbUlEIChARGVzZWxlY3RlZENvbmZpZ0l0ZW1JRHMpIHsKICAgICAgICAgICAgbmV4dCBDT05GSUdJVEVNSUQgaWYgISRMaW5rTGlzdC0+e0lUU01Db25maWdJdGVtfS0+eyRMaW5rVHlwZX0tPntTb3VyY2V9LT57JENvbmZpZ0l0ZW1JRH0KICAgICAgICAgICAgICAgICYmICEkTGlua0xpc3QtPntJVFNNQ29uZmlnSXRlbX0tPnskTGlua1R5cGV9LT57VGFyZ2V0fS0+eyRDb25maWdJdGVtSUR9OwoKICAgICAgICAgICAgaWYgKCAkU291cmNlT2JqZWN0IGVxICdJVFNNQ29uZmlnSXRlbScgKSB7CiAgICAgICAgICAgICAgICAkU291cmNlS2V5ID0gJENvbmZpZ0l0ZW1JRDsKICAgICAgICAgICAgfQogICAgICAgICAgICBlbHNlIHsKICAgICAgICAgICAgICAgICRUYXJnZXRLZXkgPSAkQ29uZmlnSXRlbUlEOwogICAgICAgICAgICB9CgogICAgICAgICAgICAkTGlua09iamVjdC0+TGlua0RlbGV0ZSgKICAgICAgICAgICAgICAgIE9iamVjdDEgPT4gJFNvdXJjZU9iamVjdCwKICAgICAgICAgICAgICAgIEtleTEgICAgPT4gJFNvdXJjZUtleSwKICAgICAgICAgICAgICAgIE9iamVjdDIgPT4gJFRhcmdldE9iamVjdCwKICAgICAgICAgICAgICAgIEtleTIgICAgPT4gJFRhcmdldEtleSwKICAgICAgICAgICAgICAgIFR5cGUgICAgPT4gJExpbmtUeXBlLAogICAgICAgICAgICAgICAgVXNlcklEICA9PiAxLAogICAgICAgICAgICApOwogICAgICAgIH0KICAgIH0KCiAgICAjIEFkZCBsaW5rcyBmb3IgYWRkZWQgY29uZmlnIGl0ZW1zLgogICAgQ09ORklHSVRFTUlEOgogICAgZm9yIG15ICRDb25maWdJdGVtSUQgKEBOZXdWYWx1ZXMpIHsKICAgICAgICBuZXh0IENPTkZJR0lURU1JRCBpZiAkTGlua0xpc3QtPntJVFNNQ29uZmlnSXRlbX0tPnskTGlua1R5cGV9LT57U291cmNlfS0+eyRDb25maWdJdGVtSUR9OwoKICAgICAgICBpZiAoICRTb3VyY2VPYmplY3QgZXEgJ0lUU01Db25maWdJdGVtJyApIHsKICAgICAgICAgICAgJFNvdXJjZUtleSA9ICRDb25maWdJdGVtSUQ7CiAgICAgICAgfQogICAgICAgIGVsc2UgewogICAgICAgICAgICAkVGFyZ2V0S2V5ID0gJENvbmZpZ0l0ZW1JRDsKICAgICAgICB9CgogICAgICAgICRMaW5rT2JqZWN0LT5MaW5rQWRkKAogICAgICAgICAgICBTb3VyY2VPYmplY3QgPT4gJFNvdXJjZU9iamVjdCwKICAgICAgICAgICAgU291cmNlS2V5ICAgID0+ICRTb3VyY2VLZXksCiAgICAgICAgICAgIFRhcmdldE9iamVjdCA9PiAkVGFyZ2V0T2JqZWN0LAogICAgICAgICAgICBUYXJnZXRLZXkgICAgPT4gJFRhcmdldEtleSwKICAgICAgICAgICAgVHlwZSAgICAgICAgID0+ICRMaW5rVHlwZSwKICAgICAgICAgICAgU3RhdGUgICAgICAgID0+ICdWYWxpZCcsCiAgICAgICAgICAgIFVzZXJJRCAgICAgICA9PiAxLAogICAgICAgICk7CiAgICB9CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KIyMgbm9maWx0ZXIoVGlkeUFsbDo6UGx1Z2luOjpabnVueTo6UGVybDo6UGFyYW1PYmplY3QpCgpwYWNrYWdlIEtlcm5lbDo6U3lzdGVtOjpUaWNrZXQ6OkV2ZW50OjpJVFNNQ29uZmlnSXRlbUN1c3RvbWVyQ0lzOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIEtlcm5lbDo6U3lzdGVtOjpWYXJpYWJsZUNoZWNrIHF3KDphbGwpOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06OkxpbmtPYmplY3QnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpMb2cnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpXZWI6OlJlcXVlc3QnLAopOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJExvZ09iamVjdCAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpOwogICAgbXkgJExpbmtPYmplY3QgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxpbmtPYmplY3QnKTsKICAgIG15ICRQYXJhbU9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpXZWI6OlJlcXVlc3QnKTsKCiAgICBORUVERUQ6CiAgICBmb3IgbXkgJE5lZWRlZCAocXcoIERhdGEgRXZlbnQgQ29uZmlnIFVzZXJJRCApKSB7CiAgICAgICAgbmV4dCBORUVERUQgaWYgJFBhcmFteyROZWVkZWR9OwoKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJOZWVkICROZWVkZWQhIgogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIE5FRURFRDoKICAgIGZvciBteSAkTmVlZGVkIChxdyggVGlja2V0SUQgKSkgewogICAgICAgIG5leHQgTkVFREVEIGlmICRQYXJhbXtEYXRhfS0+eyROZWVkZWR9OwoKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJOZWVkICROZWVkZWQgaW4gRGF0YSEiCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgbXkgJENJTGlzdCA9ICRQYXJhbU9iamVjdC0+R2V0UGFyYW0oIFBhcmFtID0+ICdJVFNNQ29uZmlnSXRlbUN1c3RvbWVyQ0lMaXN0JyApIHx8ICcnOwogICAgcmV0dXJuIDEgaWYgISRDSUxpc3Q7CgogICAgbXkgQENvbmZpZ0l0ZW1JRHMgPSBzcGxpdCAnLCcsICRDSUxpc3Q7CgogICAgZm9yIG15ICRDb25maWdJdGVtSUQgKEBDb25maWdJdGVtSURzKSB7CiAgICAgICAgJExpbmtPYmplY3QtPkxpbmtBZGQoCiAgICAgICAgICAgIFNvdXJjZU9iamVjdCA9PiAnVGlja2V0JywKICAgICAgICAgICAgU291cmNlS2V5ICAgID0+ICRQYXJhbXtEYXRhfS0+e1RpY2tldElEfSwKICAgICAgICAgICAgVGFyZ2V0T2JqZWN0ID0+ICdJVFNNQ29uZmlnSXRlbScsCiAgICAgICAgICAgIFRhcmdldEtleSAgICA9PiAkQ29uZmlnSXRlbUlELAogICAgICAgICAgICBUeXBlICAgICAgICAgPT4gJ1JlbGV2YW50VG8nLAogICAgICAgICAgICBTdGF0ZSAgICAgICAgPT4gJ1ZhbGlkJywKICAgICAgICAgICAgVXNlcklEICAgICAgID0+ICRQYXJhbXtVc2VySUR9LAogICAgICAgICk7CiAgICB9CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::Ticket::Event::TicketStatusLink;

use strict;
use warnings;

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

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::Log',
    'Kernel::System::Ticket',
    'Kernel::System::User',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::LinkObject',
);

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

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

    return $Self;
}

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

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

    # check if functionality is enabled
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');
    return 1 if !$ConfigObject->Get('ITSMConfigItem::SetIncidentStateOnLink');

    # get ticket id depending on event
    my $TicketID;
    if ( $Param{Event} eq 'LinkAdd' || $Param{Event} eq 'LinkDelete' ) {
        for my $Needed (qw(ConfigItemID Comment)) {
            if ( !$Param{Data}->{$Needed} ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  => "Need $Needed in Data!",
                );
                return;
            }
        }

        if ( $Param{Event} eq 'LinkAdd' ) {
            if ( !$Param{Data}->{Type} ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  => "Need Type in Data!",
                );
                return;
            }
        }

        # check if link event concerns a ticket
        return 1 if $Param{Data}->{Comment} !~ m{ \A ( \d+ ) %%Ticket \z }xms;
        $TicketID = $1;
    }
    else {
        if ( !$Param{Data}->{TicketID} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need TicketID in Data!",
            );
            return;
        }
        $TicketID = $Param{Data}->{TicketID};
    }

    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');
    my %Ticket       = $TicketObject->TicketGet(
        TicketID => $TicketID,
        UserID   => 1,
    );
    return 1 if !%Ticket;

    # check if ticket type is relevant (optional functionality)
    my $OldTicketTypeRelevant;
    my $NewTicketTypeRelevant;
    my $TicketTypes = $ConfigObject->Get('ITSMConfigItem::LinkStatus::TicketTypes');
    if ( $Param{Event} eq 'TicketTypeUpdate' ) {

        # ticket types are not used, changes don't affect CI incident states
        return 1 if !IsArrayRefWithData($TicketTypes);

        # determine type of ticket before last update
        my $OldTicketType;
        my @HistoryLines = $TicketObject->HistoryGet(
            TicketID => $TicketID,
            UserID   => 1,
        );
        LINE:
        for my $Line ( reverse @HistoryLines ) {
            next LINE if $Line->{HistoryType} ne 'TypeUpdate';
            my @CommentParts = split '%%', $Line->{Name};
            $OldTicketType = $CommentParts[3];
            last LINE;
        }
        if ( !$OldTicketType ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Found no previous ticket type for ticket id $TicketID!",
            );
            return;
        }

        # only changes relevant->irrelevant and irrelevant->relevant are of interest
        $OldTicketTypeRelevant = 1 if grep { $_ eq $OldTicketType } @{$TicketTypes};
        $NewTicketTypeRelevant = 1 if grep { $_ eq $Ticket{Type} } @{$TicketTypes};
        return 1 if ( $OldTicketTypeRelevant || '' ) eq ( $NewTicketTypeRelevant || '' );
    }

    # shortcut for other events (check if ticket type is in scope)
    elsif ( IsArrayRefWithData($TicketTypes) ) {
        return 1 if !$Ticket{Type};
        return 1 if !grep { $_ eq $Ticket{Type} } @{$TicketTypes};
    }

    # check if ticket state is relevant
    my $OldTicketStateRelevant;
    my $NewTicketStateRelevant;
    my @OpenStateTypes = ( 'new', 'open', 'pending reminder', 'pending auto' );
    if ( $Param{Event} eq 'TicketStateUpdate' ) {
        if ( !$Param{Data}->{OldTicketData} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need TicketID in OldTicketData!",
            );
            return;
        }

        # only changes open->close and close->open are of interest
        $NewTicketStateRelevant = 1 if grep { $_ eq $Ticket{StateType} } @OpenStateTypes;
        $OldTicketStateRelevant = 1 if grep { $_ eq $Param{Data}->{OldTicketData}->{StateType} } @OpenStateTypes;
        return 1 if ( $NewTicketStateRelevant || '' ) eq ( $OldTicketStateRelevant || '' );
    }

    # shortcut for other events (check if ticket state is in scope)
    else {
        return 1 if !grep { $_ eq $Ticket{StateType} } @OpenStateTypes;
    }

    # shortcut for ticket events (check if ticket has any potentially relevant links)
    if ( $Param{Event} eq 'TicketStateUpdate' || $Param{Event} eq 'TicketTypeUpdate' ) {
        my %LinkKeyList = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkKeyList(
            Object1 => 'Ticket',
            Key1    => $TicketID,
            Object2 => 'ITSMConfigItem',
            State   => 'Valid',
            UserID  => 1,
        );
        return 1 if !IsHashRefWithData( \%LinkKeyList );
    }

    # prepare incident states and relevant link types per incident state
    my $IncidentStates = $ConfigObject->Get('ITSMConfigItem::LinkStatus::IncidentStates');
    my $LinkTypes      = $ConfigObject->Get('ITSMConfigItem::LinkStatus::LinkTypes');
    if ( !IsHashRefWithData($LinkTypes) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need configuration for 'ITSMConfigItem::LinkStatus::LinkTypes'!",
        );
        return;
    }
    my %LinkTypesByIncidentState;

    if ( $Param{Event} eq 'LinkAdd' ) {
        for my $LinkType ( sort keys %{$LinkTypes} ) {
            if ( $LinkType eq $Param{Data}->{Type} ) {
                push @{ $LinkTypesByIncidentState{ $LinkTypes->{$LinkType} } }, $LinkType;
            }
        }
        return 1 if !IsHashRefWithData( \%LinkTypesByIncidentState );
    }
    else {
        for my $LinkType ( sort keys %{$LinkTypes} ) {
            push @{ $LinkTypesByIncidentState{ $LinkTypes->{$LinkType} } }, $LinkType;
        }
    }

    # handle added or removed links
    if ( $Param{Event} eq 'LinkAdd' || $Param{Event} eq 'LinkDelete' ) {
        my $ConfigItemID = $Param{Data}->{ConfigItemID};
        my $Version      = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
            ConfigItemID => $ConfigItemID,
        );

        # optional: check if CI deployment state is relevant
        my $DeploymentStates = $ConfigObject->Get('ITSMConfigItem::LinkStatus::DeploymentStates');
        if ( IsArrayRefWithData($DeploymentStates) ) {
            return 1 if !grep { $_ eq $Version->{DeplState} } @{$DeploymentStates};
        }

        # raise incident state if necessary
        if ( $Param{Event} eq 'LinkAdd' ) {
            return $Self->_CheckRaiseIncidentState(
                Version                  => $Version,
                IncidentStates           => $IncidentStates,
                LinkTypesByIncidentState => \%LinkTypesByIncidentState,
                TicketID                 => $TicketID,
            );
        }

        # lower incident state if necessary
        return $Self->_CheckLowerIncidentState(
            Version                  => $Version,
            IncidentStates           => $IncidentStates,
            LinkTypesByIncidentState => \%LinkTypesByIncidentState,
            TicketID                 => $TicketID,
        );
    }

    # handle ticket closure/re-open and relevant/irrelevant type change
    elsif ( $Param{Event} eq 'TicketStateUpdate' || $Param{Event} eq 'TicketTypeUpdate' ) {

        # relevant ticket state and type updates are handled the same way
        # both mean a ticket is relevant for CI incident state changes or not
        # ticket re-open or change to relevant type = might have to raise incident state of CIs
        # ticket closure or change to irrelevant type = might have to lower incident state of CIs
        my $RelevantNow;
        if (
            ( $Param{Event} eq 'TicketStateUpdate' && $NewTicketStateRelevant )
            || ( $Param{Event} eq 'TicketTypeUpdate' && $NewTicketTypeRelevant )
            )
        {
            $RelevantNow = 1;
        }

        # loop through all link types and check links to see if there are possibly affected CIs
        my $DeploymentStates = $ConfigObject->Get('ITSMConfigItem::LinkStatus::DeploymentStates');
        INCIDENTSTATE:
        for my $IncidentState ( @{$IncidentStates} ) {
            LINKTYPE:
            for my $LinkType ( @{ $LinkTypesByIncidentState{$IncidentState} } ) {
                my @LinkedCIs = $Self->_CheckTicketLinks(
                    TicketID => $TicketID,
                    Type     => $LinkType,
                );
                next LINKTYPE if !@LinkedCIs;

                CONFIGITEMID:
                for my $ConfigItemID (@LinkedCIs) {
                    my $Version = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
                        ConfigItemID => $ConfigItemID,
                    );

                    # optional: check if CI deployment state is relevant
                    if ( IsArrayRefWithData($DeploymentStates) ) {
                        next CONFIGITEMID if !grep { $_ eq $Version->{DeplState} } @{$DeploymentStates};
                    }

                    # check current incident state vs state caused by link

                    # current incident state is lower than caused by link -> set
                    if ($RelevantNow) {

                        # shortcut: nothing to do if CI is already in highest incident state
                        next CONFIGITEMID if $Version->{InciState} eq $IncidentStates->[0];

                        # shortcut: nothing to do if CI is already in same incident state
                        next CONFIGITEMID if $Version->{InciState} eq $IncidentState;

                        # check if CI is in higher incident state already
                        TMPINCIDENTSTATE:
                        for my $TmpIncidentState ( @{$IncidentStates} ) {

                            # all further incident states are lower than current one -> set new state
                            last TMPINCIDENTSTATE if $TmpIncidentState eq $IncidentState;

                            next CONFIGITEMID if $Version->{InciState} eq $TmpIncidentState;
                        }

                        # set new incident state
                        $Self->_SetCIStatus(
                            Version       => $Version,
                            IncidentState => $IncidentState,
                            TicketID      => $TicketID,
                        );
                    }

                    # current incident state is same as caused by link -> re-calculate
                    else {

                        # shortcut: nothing to do if CI is already in lowest incident state
                        next CONFIGITEMID if $Version->{InciState} eq $IncidentStates->[-1];

                        # shortcut: nothing to do if CI is not in same incident state
                        next CONFIGITEMID if $Version->{InciState} ne $IncidentState;

                        # recalculate incident state exactly
                        $Self->_CheckLowerIncidentState(
                            Version                  => $Version,
                            IncidentStates           => $IncidentStates,
                            LinkTypesByIncidentState => \%LinkTypesByIncidentState,
                            TicketID                 => $TicketID,
                        );
                    }
                }
            }
        }
    }

    # unknown/irrelevant event
    else {
        return 1;
    }

    return 1;
}

# Rationale:
# Start at highest (worst) incident state and loop down to current incident state,
#   checking for links on all types configured for the corresponding incident state.
# If any link for a higher incident state than the current one is found,
#   set this state and exit.
# Due to top-to-bottom method, no higher incident state can occur later.
sub _CheckRaiseIncidentState {
    my ( $Self, %Param ) = @_;

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

    # loop through target states and check if it needs to be changed
    INCIDENTSTATE:
    for my $IncidentState ( @{ $Param{IncidentStates} } ) {

        # CI already in target state - nothing to do
        return 1 if $Param{Version}->{InciState} eq $IncidentState;

        # check if we have linked tickets that lead to incident state
        LINKTYPE:
        for my $LinkType ( @{ $Param{LinkTypesByIncidentState}->{$IncidentState} } ) {
            next LINKTYPE if !$Self->_CheckConfigItemLinks(
                ConfigItemID => $Param{Version}->{ConfigItemID},
                Type         => $LinkType,
            );

            # we have at least one linked ticket - set incident state and exit
            return $Self->_SetCIStatus(
                Version       => $Param{Version},
                IncidentState => $IncidentState,
                TicketID      => $Param{TicketID},
            );
        }
    }

    return 1;
}

# Rationale:
# Shortcut if CI is alredy in lowest (best) incident state = can't have changed after link delete.
# Start at highest (worst) incident state and loop down to current incident state,
#   checking for links on all types configured for the corresponding incident states.
# If a link for the current incident state is found, exit immediately (no change).
# Otherwise set incident state for first occurance of a link, then exit.
# If no link is found, set the lowest state.
# Due to top-to-bottom method, first remembered incident state is the highest possible.
sub _CheckLowerIncidentState {
    my ( $Self, %Param ) = @_;

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

    # shortcut for lowest level
    return 1 if $Param{Version}->{InciState} eq $Param{IncidentStates}->[-1];

    # loop through target states and check if it needs to be changed
    my $LowerIncidentState;
    INCIDENTSTATE:
    for my $IncidentState ( @{ $Param{IncidentStates} } ) {

        # skip level if current CI incident state is not on this or any higher level
        next INCIDENTSTATE if $Param{Version}->{InciState} ne $IncidentState && !$LowerIncidentState;

        # check if we have linked tickets that keep incident state
        LINKTYPE:
        for my $LinkType ( @{ $Param{LinkTypesByIncidentState}->{$IncidentState} } ) {
            next LINKTYPE if !$Self->_CheckConfigItemLinks(
                ConfigItemID => $Param{Version}->{ConfigItemID},
                Type         => $LinkType,
            );

            # we have at least one linked ticket

            # keep incident state
            return 1 if $Param{Version}->{InciState} eq $IncidentState;

            # otherwise set new (lower) incident state and exit
            return $Self->_SetCIStatus(
                Version       => $Param{Version},
                IncidentState => $IncidentState,
                TicketID      => $Param{TicketID},
            );
        }

        # remember that state needs to be changed (= current level isn'T justified by links anymore)
        $LowerIncidentState = 1;
    }

    # no relevant links have been found, but CI wasn't in lowest incident state before - set it now and exit
    return $Self->_SetCIStatus(
        Version       => $Param{Version},
        IncidentState => $Param{IncidentStates}->[-1],
        TicketID      => $Param{TicketID},
    );
}

# set CI to a specified incident state and log change in corresponding ticket
sub _SetCIStatus {
    my ( $Self, %Param ) = @_;

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

            return;
        }
    }

    # get incident state list
    my $IncidentStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::Core::IncidentState',
    );
    return if !IsHashRefWithData($IncidentStateList);

    # check if incident state is valid
    my %ReverseIncidentStateList = reverse %{$IncidentStateList};
    if ( !$ReverseIncidentStateList{ $Param{IncidentState} } ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Have invalid incident state '$Param{IncidentState}'!",
        );
        return;
    }

    # add a new version with the new incident state
    my $VersionID = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionAdd(
        %{ $Param{Version} },
        InciStateID => $ReverseIncidentStateList{ $Param{IncidentState} },
        UserID      => 1,
    );
    return if !$VersionID;

    # log change in ticket
    $Kernel::OM->Get('Kernel::System::Ticket')->HistoryAdd(
        TicketID     => $Param{TicketID},
        HistoryType  => 'Misc',
        Name         => "Updated incident state of config item '$Param{Version}->{Number}' to '$Param{IncidentState}'.",
        CreateUserID => 1,
    );

    return 1;
}

# check if relevant CIs are linked to a ticket
sub _CheckTicketLinks {
    my ( $Self, %Param ) = @_;

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

    # get link type lookup
    my $LinkTypeLookup = $Self->_LinkTypeLookupGet();

    my %LinkKeyList = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkKeyList(
        Object1   => 'Ticket',
        Key1      => $Param{TicketID},
        Object2   => 'ITSMConfigItem',
        State     => 'Valid',
        Type      => $LinkTypeLookup->{ $Param{Type} }->{Name},
        Direction => $LinkTypeLookup->{ $Param{Type} }->{Direction}->{Ticket},
        UserID    => 1,
    );
    return if !IsHashRefWithData( \%LinkKeyList );
    return ( sort keys %LinkKeyList );
}

# check if relevant open tickets are linked to a CI
sub _CheckConfigItemLinks {
    my ( $Self, %Param ) = @_;

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

    # get link type lookup
    my $LinkTypeLookup = $Self->_LinkTypeLookupGet();

    my %LinkKeyList = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkKeyList(
        Object1   => 'ITSMConfigItem',
        Key1      => $Param{ConfigItemID},
        Object2   => 'Ticket',
        State     => 'Valid',
        Type      => $LinkTypeLookup->{ $Param{Type} }->{Name},
        Direction => $LinkTypeLookup->{ $Param{Type} }->{Direction}->{ITSMConfigItem},
        UserID    => 1,
    );
    return if !IsHashRefWithData( \%LinkKeyList );

    # list of relevant state types
    my @OpenStateTypes = ( 'new', 'open', 'pending reminder', 'pending auto' );

    # use ticket type check?
    my $TicketTypes = $Kernel::OM->Get('Kernel::Config')->Get('ITSMConfigItem::LinkStatus::TicketTypes');
    my $CheckTicketTypes;
    $CheckTicketTypes = 1 if IsArrayRefWithData($TicketTypes);

    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');
    TICKETID:
    for my $TicketID ( sort keys %LinkKeyList ) {
        my %Ticket = $TicketObject->TicketGet(
            TicketID => $TicketID,
            UserID   => 1,
        );
        next TICKETID if !grep { $_ eq $Ticket{StateType} } @OpenStateTypes;
        if ($CheckTicketTypes) {
            next TICKETID if !$Ticket{Type};
            next TICKETID if !grep { $_ eq $Ticket{Type} } @{$TicketTypes};
        }
        return 1;
    }

    return;
}

# compile list of internal link type names and directions to use for LinkKeyList
sub _LinkTypeLookupGet {
    my ( $Self, %Param ) = @_;

    # if we have generated lookup before
    return $Self->{LinkTypeLookup} if $Self->{LinkTypeLookup};

    my %LinkTypeLookup;
    my %TypeList = $Kernel::OM->Get('Kernel::System::LinkObject')->TypeList();
    for my $TypeNameInternal ( sort keys %TypeList ) {
        my $SourceName = $TypeList{$TypeNameInternal}->{SourceName};
        my $TargetName = $TypeList{$TypeNameInternal}->{TargetName};
        $LinkTypeLookup{$SourceName} = {
            Name      => $TypeNameInternal,
            Direction => {
                ITSMConfigItem => $SourceName eq $TargetName ? 'Both' : 'Target',
                Ticket         => $SourceName eq $TargetName ? 'Both' : 'Source',
            },
        };
        $LinkTypeLookup{$TargetName} = {
            Name      => $TypeNameInternal,
            Direction => {
                ITSMConfigItem => $SourceName eq $TargetName ? 'Both' : 'Source',
                Ticket         => $SourceName eq $TargetName ? 'Both' : 'Target',
            },
        };
    }

    # remember result
    $Self->{LinkTypeLookup} = \%LinkTypeLookup;

    return \%LinkTypeLookup;
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::Ticket::Event::TicketStatusLink;

use strict;
use warnings;

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

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::Log',
    'Kernel::System::Ticket',
    'Kernel::System::User',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::LinkObject',
);

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

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

    return $Self;
}

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

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

    # check if functionality is enabled
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');
    return 1 if !$ConfigObject->Get('ITSMConfigItem::SetIncidentStateOnLink');

    # get ticket id depending on event
    my $TicketID;
    if ( $Param{Event} eq 'LinkAdd' || $Param{Event} eq 'LinkDelete' ) {
        for my $Needed (qw(ConfigItemID Comment)) {
            if ( !$Param{Data}->{$Needed} ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  => "Need $Needed in Data!",
                );
                return;
            }
        }

        if ( $Param{Event} eq 'LinkAdd' ) {
            if ( !$Param{Data}->{Type} ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  => "Need Type in Data!",
                );
                return;
            }
        }

        # check if link event concerns a ticket
        return 1 if $Param{Data}->{Comment} !~ m{ \A ( \d+ ) %%Ticket \z }xms;
        $TicketID = $1;
    }
    else {
        if ( !$Param{Data}->{TicketID} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need TicketID in Data!",
            );
            return;
        }
        $TicketID = $Param{Data}->{TicketID};
    }

    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');
    my %Ticket       = $TicketObject->TicketGet(
        TicketID => $TicketID,
        UserID   => 1,
    );
    return 1 if !%Ticket;

    # check if ticket type is relevant (optional functionality)
    my $OldTicketTypeRelevant;
    my $NewTicketTypeRelevant;
    my $TicketTypes = $ConfigObject->Get('ITSMConfigItem::LinkStatus::TicketTypes');
    if ( $Param{Event} eq 'TicketTypeUpdate' ) {

        # ticket types are not used, changes don't affect CI incident states
        return 1 if !IsArrayRefWithData($TicketTypes);

        # determine type of ticket before last update
        my $OldTicketType;
        my @HistoryLines = $TicketObject->HistoryGet(
            TicketID => $TicketID,
            UserID   => 1,
        );
        LINE:
        for my $Line ( reverse @HistoryLines ) {
            next LINE if $Line->{HistoryType} ne 'TypeUpdate';
            my @CommentParts = split '%%', $Line->{Name};
            $OldTicketType = $CommentParts[3];
            last LINE;
        }
        if ( !$OldTicketType ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Found no previous ticket type for ticket id $TicketID!",
            );
            return;
        }

        # only changes relevant->irrelevant and irrelevant->relevant are of interest
        $OldTicketTypeRelevant = 1 if grep { $_ eq $OldTicketType } @{$TicketTypes};
        $NewTicketTypeRelevant = 1 if grep { $_ eq $Ticket{Type} } @{$TicketTypes};
        return 1 if ( $OldTicketTypeRelevant || '' ) eq ( $NewTicketTypeRelevant || '' );
    }

    # shortcut for other events (check if ticket type is in scope)
    elsif ( IsArrayRefWithData($TicketTypes) ) {
        return 1 if !$Ticket{Type};
        return 1 if !grep { $_ eq $Ticket{Type} } @{$TicketTypes};
    }

    # check if ticket state is relevant
    my $OldTicketStateRelevant;
    my $NewTicketStateRelevant;
    my @OpenStateTypes = ( 'new', 'open', 'pending reminder', 'pending auto' );
    if ( $Param{Event} eq 'TicketStateUpdate' ) {
        if ( !$Param{Data}->{OldTicketData} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need TicketID in OldTicketData!",
            );
            return;
        }

        # only changes open->close and close->open are of interest
        $NewTicketStateRelevant = 1 if grep { $_ eq $Ticket{StateType} } @OpenStateTypes;
        $OldTicketStateRelevant = 1 if grep { $_ eq $Param{Data}->{OldTicketData}->{StateType} } @OpenStateTypes;
        return 1 if ( $NewTicketStateRelevant || '' ) eq ( $OldTicketStateRelevant || '' );
    }

    # shortcut for other events (check if ticket state is in scope)
    else {
        return 1 if !grep { $_ eq $Ticket{StateType} } @OpenStateTypes;
    }

    # shortcut for ticket events (check if ticket has any potentially relevant links)
    if ( $Param{Event} eq 'TicketStateUpdate' || $Param{Event} eq 'TicketTypeUpdate' ) {
        my %LinkKeyList = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkKeyList(
            Object1 => 'Ticket',
            Key1    => $TicketID,
            Object2 => 'ITSMConfigItem',
            State   => 'Valid',
            UserID  => 1,
        );
        return 1 if !IsHashRefWithData( \%LinkKeyList );
    }

    # prepare incident states and relevant link types per incident state
    my $IncidentStates = $ConfigObject->Get('ITSMConfigItem::LinkStatus::IncidentStates');
    my $LinkTypes      = $ConfigObject->Get('ITSMConfigItem::LinkStatus::LinkTypes');
    if ( !IsHashRefWithData($LinkTypes) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need configuration for 'ITSMConfigItem::LinkStatus::LinkTypes'!",
        );
        return;
    }
    my %LinkTypesByIncidentState;

    if ( $Param{Event} eq 'LinkAdd' ) {
        for my $LinkType ( sort keys %{$LinkTypes} ) {
            if ( $LinkType eq $Param{Data}->{Type} ) {
                push @{ $LinkTypesByIncidentState{ $LinkTypes->{$LinkType} } }, $LinkType;
            }
        }
        return 1 if !IsHashRefWithData( \%LinkTypesByIncidentState );
    }
    else {
        for my $LinkType ( sort keys %{$LinkTypes} ) {
            push @{ $LinkTypesByIncidentState{ $LinkTypes->{$LinkType} } }, $LinkType;
        }
    }

    # handle added or removed links
    if ( $Param{Event} eq 'LinkAdd' || $Param{Event} eq 'LinkDelete' ) {
        my $ConfigItemID = $Param{Data}->{ConfigItemID};
        my $Version      = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
            ConfigItemID => $ConfigItemID,
        );

        # optional: check if CI deployment state is relevant
        my $DeploymentStates = $ConfigObject->Get('ITSMConfigItem::LinkStatus::DeploymentStates');
        if ( IsArrayRefWithData($DeploymentStates) ) {
            return 1 if !grep { $_ eq $Version->{DeplState} } @{$DeploymentStates};
        }

        # raise incident state if necessary
        if ( $Param{Event} eq 'LinkAdd' ) {
            return $Self->_CheckRaiseIncidentState(
                Version                  => $Version,
                IncidentStates           => $IncidentStates,
                LinkTypesByIncidentState => \%LinkTypesByIncidentState,
                TicketID                 => $TicketID,
            );
        }

        # lower incident state if necessary
        return $Self->_CheckLowerIncidentState(
            Version                  => $Version,
            IncidentStates           => $IncidentStates,
            LinkTypesByIncidentState => \%LinkTypesByIncidentState,
            TicketID                 => $TicketID,
        );
    }

    # handle ticket closure/re-open and relevant/irrelevant type change
    elsif ( $Param{Event} eq 'TicketStateUpdate' || $Param{Event} eq 'TicketTypeUpdate' ) {

        # relevant ticket state and type updates are handled the same way
        # both mean a ticket is relevant for CI incident state changes or not
        # ticket re-open or change to relevant type = might have to raise incident state of CIs
        # ticket closure or change to irrelevant type = might have to lower incident state of CIs
        my $RelevantNow;
        if (
            ( $Param{Event} eq 'TicketStateUpdate' && $NewTicketStateRelevant )
            || ( $Param{Event} eq 'TicketTypeUpdate' && $NewTicketTypeRelevant )
            )
        {
            $RelevantNow = 1;
        }

        # loop through all link types and check links to see if there are possibly affected CIs
        my $DeploymentStates = $ConfigObject->Get('ITSMConfigItem::LinkStatus::DeploymentStates');
        INCIDENTSTATE:
        for my $IncidentState ( @{$IncidentStates} ) {
            LINKTYPE:
            for my $LinkType ( @{ $LinkTypesByIncidentState{$IncidentState} } ) {
                my @LinkedCIs = $Self->_CheckTicketLinks(
                    TicketID => $TicketID,
                    Type     => $LinkType,
                );
                next LINKTYPE if !@LinkedCIs;

                CONFIGITEMID:
                for my $ConfigItemID (@LinkedCIs) {
                    my $Version = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
                        ConfigItemID => $ConfigItemID,
                    );

                    # optional: check if CI deployment state is relevant
                    if ( IsArrayRefWithData($DeploymentStates) ) {
                        next CONFIGITEMID if !grep { $_ eq $Version->{DeplState} } @{$DeploymentStates};
                    }

                    # check current incident state vs state caused by link

                    # current incident state is lower than caused by link -> set
                    if ($RelevantNow) {

                        # shortcut: nothing to do if CI is already in highest incident state
                        next CONFIGITEMID if $Version->{InciState} eq $IncidentStates->[0];

                        # shortcut: nothing to do if CI is already in same incident state
                        next CONFIGITEMID if $Version->{InciState} eq $IncidentState;

                        # check if CI is in higher incident state already
                        TMPINCIDENTSTATE:
                        for my $TmpIncidentState ( @{$IncidentStates} ) {

                            # all further incident states are lower than current one -> set new state
                            last TMPINCIDENTSTATE if $TmpIncidentState eq $IncidentState;

                            next CONFIGITEMID if $Version->{InciState} eq $TmpIncidentState;
                        }

                        # set new incident state
                        $Self->_SetCIStatus(
                            Version       => $Version,
                            IncidentState => $IncidentState,
                            TicketID      => $TicketID,
                        );
                    }

                    # current incident state is same as caused by link -> re-calculate
                    else {

                        # shortcut: nothing to do if CI is already in lowest incident state
                        next CONFIGITEMID if $Version->{InciState} eq $IncidentStates->[-1];

                        # shortcut: nothing to do if CI is not in same incident state
                        next CONFIGITEMID if $Version->{InciState} ne $IncidentState;

                        # recalculate incident state exactly
                        $Self->_CheckLowerIncidentState(
                            Version                  => $Version,
                            IncidentStates           => $IncidentStates,
                            LinkTypesByIncidentState => \%LinkTypesByIncidentState,
                            TicketID                 => $TicketID,
                        );
                    }
                }
            }
        }
    }

    # unknown/irrelevant event
    else {
        return 1;
    }

    return 1;
}

# Rationale:
# Start at highest (worst) incident state and loop down to current incident state,
#   checking for links on all types configured for the corresponding incident state.
# If any link for a higher incident state than the current one is found,
#   set this state and exit.
# Due to top-to-bottom method, no higher incident state can occur later.
sub _CheckRaiseIncidentState {
    my ( $Self, %Param ) = @_;

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

    # loop through target states and check if it needs to be changed
    INCIDENTSTATE:
    for my $IncidentState ( @{ $Param{IncidentStates} } ) {

        # CI already in target state - nothing to do
        return 1 if $Param{Version}->{InciState} eq $IncidentState;

        # check if we have linked tickets that lead to incident state
        LINKTYPE:
        for my $LinkType ( @{ $Param{LinkTypesByIncidentState}->{$IncidentState} } ) {
            next LINKTYPE if !$Self->_CheckConfigItemLinks(
                ConfigItemID => $Param{Version}->{ConfigItemID},
                Type         => $LinkType,
            );

            # we have at least one linked ticket - set incident state and exit
            return $Self->_SetCIStatus(
                Version       => $Param{Version},
                IncidentState => $IncidentState,
                TicketID      => $Param{TicketID},
            );
        }
    }

    return 1;
}

# Rationale:
# Shortcut if CI is alredy in lowest (best) incident state = can't have changed after link delete.
# Start at highest (worst) incident state and loop down to current incident state,
#   checking for links on all types configured for the corresponding incident states.
# If a link for the current incident state is found, exit immediately (no change).
# Otherwise set incident state for first occurance of a link, then exit.
# If no link is found, set the lowest state.
# Due to top-to-bottom method, first remembered incident state is the highest possible.
sub _CheckLowerIncidentState {
    my ( $Self, %Param ) = @_;

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

    # shortcut for lowest level
    return 1 if $Param{Version}->{InciState} eq $Param{IncidentStates}->[-1];

    # loop through target states and check if it needs to be changed
    my $LowerIncidentState;
    INCIDENTSTATE:
    for my $IncidentState ( @{ $Param{IncidentStates} } ) {

        # skip level if current CI incident state is not on this or any higher level
        next INCIDENTSTATE if $Param{Version}->{InciState} ne $IncidentState && !$LowerIncidentState;

        # check if we have linked tickets that keep incident state
        LINKTYPE:
        for my $LinkType ( @{ $Param{LinkTypesByIncidentState}->{$IncidentState} } ) {
            next LINKTYPE if !$Self->_CheckConfigItemLinks(
                ConfigItemID => $Param{Version}->{ConfigItemID},
                Type         => $LinkType,
            );

            # we have at least one linked ticket

            # keep incident state
            return 1 if $Param{Version}->{InciState} eq $IncidentState;

            # otherwise set new (lower) incident state and exit
            return $Self->_SetCIStatus(
                Version       => $Param{Version},
                IncidentState => $IncidentState,
                TicketID      => $Param{TicketID},
            );
        }

        # remember that state needs to be changed (= current level isn'T justified by links anymore)
        $LowerIncidentState = 1;
    }

    # no relevant links have been found, but CI wasn't in lowest incident state before - set it now and exit
    return $Self->_SetCIStatus(
        Version       => $Param{Version},
        IncidentState => $Param{IncidentStates}->[-1],
        TicketID      => $Param{TicketID},
    );
}

# set CI to a specified incident state and log change in corresponding ticket
sub _SetCIStatus {
    my ( $Self, %Param ) = @_;

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

            return;
        }
    }

    # get incident state list
    my $IncidentStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::Core::IncidentState',
    );
    return if !IsHashRefWithData($IncidentStateList);

    # check if incident state is valid
    my %ReverseIncidentStateList = reverse %{$IncidentStateList};
    if ( !$ReverseIncidentStateList{ $Param{IncidentState} } ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Have invalid incident state '$Param{IncidentState}'!",
        );
        return;
    }

    # add a new version with the new incident state
    my $VersionID = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionAdd(
        %{ $Param{Version} },
        InciStateID => $ReverseIncidentStateList{ $Param{IncidentState} },
        UserID      => 1,
    );
    return if !$VersionID;

    # log change in ticket
    $Kernel::OM->Get('Kernel::System::Ticket')->HistoryAdd(
        TicketID     => $Param{TicketID},
        HistoryType  => 'Misc',
        Name         => "Updated incident state of config item '$Param{Version}->{Number}' to '$Param{IncidentState}'.",
        CreateUserID => 1,
    );

    return 1;
}

# check if relevant CIs are linked to a ticket
sub _CheckTicketLinks {
    my ( $Self, %Param ) = @_;

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

    # get link type lookup
    my $LinkTypeLookup = $Self->_LinkTypeLookupGet();

    my %LinkKeyList = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkKeyList(
        Object1   => 'Ticket',
        Key1      => $Param{TicketID},
        Object2   => 'ITSMConfigItem',
        State     => 'Valid',
        Type      => $LinkTypeLookup->{ $Param{Type} }->{Name},
        Direction => $LinkTypeLookup->{ $Param{Type} }->{Direction}->{Ticket},
        UserID    => 1,
    );
    return if !IsHashRefWithData( \%LinkKeyList );
    return ( sort keys %LinkKeyList );
}

# check if relevant open tickets are linked to a CI
sub _CheckConfigItemLinks {
    my ( $Self, %Param ) = @_;

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

    # get link type lookup
    my $LinkTypeLookup = $Self->_LinkTypeLookupGet();

    my %LinkKeyList = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkKeyList(
        Object1   => 'ITSMConfigItem',
        Key1      => $Param{ConfigItemID},
        Object2   => 'Ticket',
        State     => 'Valid',
        Type      => $LinkTypeLookup->{ $Param{Type} }->{Name},
        Direction => $LinkTypeLookup->{ $Param{Type} }->{Direction}->{ITSMConfigItem},
        UserID    => 1,
    );
    return if !IsHashRefWithData( \%LinkKeyList );

    # list of relevant state types
    my @OpenStateTypes = ( 'new', 'open', 'pending reminder', 'pending auto' );

    # use ticket type check?
    my $TicketTypes = $Kernel::OM->Get('Kernel::Config')->Get('ITSMConfigItem::LinkStatus::TicketTypes');
    my $CheckTicketTypes;
    $CheckTicketTypes = 1 if IsArrayRefWithData($TicketTypes);

    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');
    TICKETID:
    for my $TicketID ( sort keys %LinkKeyList ) {
        my %Ticket = $TicketObject->TicketGet(
            TicketID => $TicketID,
            UserID   => 1,
        );
        next TICKETID if !grep { $_ eq $Ticket{StateType} } @OpenStateTypes;
        if ($CheckTicketTypes) {
            next TICKETID if !$Ticket{Type};
            next TICKETID if !grep { $_ eq $Ticket{Type} } @{$TicketTypes};
        }
        return 1;
    }

    return;
}

# compile list of internal link type names and directions to use for LinkKeyList
sub _LinkTypeLookupGet {
    my ( $Self, %Param ) = @_;

    # if we have generated lookup before
    return $Self->{LinkTypeLookup} if $Self->{LinkTypeLookup};

    my %LinkTypeLookup;
    my %TypeList = $Kernel::OM->Get('Kernel::System::LinkObject')->TypeList();
    for my $TypeNameInternal ( sort keys %TypeList ) {
        my $SourceName = $TypeList{$TypeNameInternal}->{SourceName};
        my $TargetName = $TypeList{$TypeNameInternal}->{TargetName};
        $LinkTypeLookup{$SourceName} = {
            Name      => $TypeNameInternal,
            Direction => {
                ITSMConfigItem => $SourceName eq $TargetName ? 'Both' : 'Target',
                Ticket         => $SourceName eq $TargetName ? 'Both' : 'Source',
            },
        };
        $LinkTypeLookup{$TargetName} = {
            Name      => $TypeNameInternal,
            Direction => {
                ITSMConfigItem => $SourceName eq $TargetName ? 'Both' : 'Source',
                Ticket         => $SourceName eq $TargetName ? 'Both' : 'Target',
            },
        };
    }

    # remember result
    $Self->{LinkTypeLookup} = \%LinkTypeLookup;

    return \%LinkTypeLookup;
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::Ticket::Event::TicketStatusLink;

use strict;
use warnings;

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

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::Log',
    'Kernel::System::Ticket',
    'Kernel::System::User',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::LinkObject',
);

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

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

    return $Self;
}

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

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

    # check if functionality is enabled
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');
    return 1 if !$ConfigObject->Get('ITSMConfigItem::SetIncidentStateOnLink');

    # get ticket id depending on event
    my $TicketID;
    if ( $Param{Event} eq 'LinkAdd' || $Param{Event} eq 'LinkDelete' ) {
        for my $Needed (qw(ConfigItemID Comment)) {
            if ( !$Param{Data}->{$Needed} ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  => "Need $Needed in Data!",
                );
                return;
            }
        }

        if ( $Param{Event} eq 'LinkAdd' ) {
            if ( !$Param{Data}->{Type} ) {
                $Kernel::OM->Get('Kernel::System::Log')->Log(
                    Priority => 'error',
                    Message  => "Need Type in Data!",
                );
                return;
            }
        }

        # check if link event concerns a ticket
        return 1 if $Param{Data}->{Comment} !~ m{ \A ( \d+ ) %%Ticket \z }xms;
        $TicketID = $1;
    }
    else {
        if ( !$Param{Data}->{TicketID} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need TicketID in Data!",
            );
            return;
        }
        $TicketID = $Param{Data}->{TicketID};
    }

    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');
    my %Ticket       = $TicketObject->TicketGet(
        TicketID => $TicketID,
        UserID   => 1,
    );
    return 1 if !%Ticket;

    # check if ticket type is relevant (optional functionality)
    my $OldTicketTypeRelevant;
    my $NewTicketTypeRelevant;
    my $TicketTypes = $ConfigObject->Get('ITSMConfigItem::LinkStatus::TicketTypes');
    if ( $Param{Event} eq 'TicketTypeUpdate' ) {

        # ticket types are not used, changes don't affect CI incident states
        return 1 if !IsArrayRefWithData($TicketTypes);

        # determine type of ticket before last update
        my $OldTicketType;
        my @HistoryLines = $TicketObject->HistoryGet(
            TicketID => $TicketID,
            UserID   => 1,
        );
        LINE:
        for my $Line ( reverse @HistoryLines ) {
            next LINE if $Line->{HistoryType} ne 'TypeUpdate';
            my @CommentParts = split '%%', $Line->{Name};
            $OldTicketType = $CommentParts[3];
            last LINE;
        }
        if ( !$OldTicketType ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Found no previous ticket type for ticket id $TicketID!",
            );
            return;
        }

        # only changes relevant->irrelevant and irrelevant->relevant are of interest
        $OldTicketTypeRelevant = 1 if grep { $_ eq $OldTicketType } @{$TicketTypes};
        $NewTicketTypeRelevant = 1 if grep { $_ eq $Ticket{Type} } @{$TicketTypes};
        return 1 if ( $OldTicketTypeRelevant || '' ) eq ( $NewTicketTypeRelevant || '' );
    }

    # shortcut for other events (check if ticket type is in scope)
    elsif ( IsArrayRefWithData($TicketTypes) ) {
        return 1 if !$Ticket{Type};
        return 1 if !grep { $_ eq $Ticket{Type} } @{$TicketTypes};
    }

    # check if ticket state is relevant
    my $OldTicketStateRelevant;
    my $NewTicketStateRelevant;
    my @OpenStateTypes = ( 'new', 'open', 'pending reminder', 'pending auto' );
    if ( $Param{Event} eq 'TicketStateUpdate' ) {
        if ( !$Param{Data}->{OldTicketData} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need TicketID in OldTicketData!",
            );
            return;
        }

        # only changes open->close and close->open are of interest
        $NewTicketStateRelevant = 1 if grep { $_ eq $Ticket{StateType} } @OpenStateTypes;
        $OldTicketStateRelevant = 1 if grep { $_ eq $Param{Data}->{OldTicketData}->{StateType} } @OpenStateTypes;
        return 1 if ( $NewTicketStateRelevant || '' ) eq ( $OldTicketStateRelevant || '' );
    }

    # shortcut for other events (check if ticket state is in scope)
    else {
        return 1 if !grep { $_ eq $Ticket{StateType} } @OpenStateTypes;
    }

    # shortcut for ticket events (check if ticket has any potentially relevant links)
    if ( $Param{Event} eq 'TicketStateUpdate' || $Param{Event} eq 'TicketTypeUpdate' ) {
        my %LinkKeyList = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkKeyList(
            Object1 => 'Ticket',
            Key1    => $TicketID,
            Object2 => 'ITSMConfigItem',
            State   => 'Valid',
            UserID  => 1,
        );
        return 1 if !IsHashRefWithData( \%LinkKeyList );
    }

    # prepare incident states and relevant link types per incident state
    my $IncidentStates = $ConfigObject->Get('ITSMConfigItem::LinkStatus::IncidentStates');
    my $LinkTypes      = $ConfigObject->Get('ITSMConfigItem::LinkStatus::LinkTypes');
    if ( !IsHashRefWithData($LinkTypes) ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need configuration for 'ITSMConfigItem::LinkStatus::LinkTypes'!",
        );
        return;
    }
    my %LinkTypesByIncidentState;

    if ( $Param{Event} eq 'LinkAdd' ) {
        for my $LinkType ( sort keys %{$LinkTypes} ) {
            if ( $LinkType eq $Param{Data}->{Type} ) {
                push @{ $LinkTypesByIncidentState{ $LinkTypes->{$LinkType} } }, $LinkType;
            }
        }
        return 1 if !IsHashRefWithData( \%LinkTypesByIncidentState );
    }
    else {
        for my $LinkType ( sort keys %{$LinkTypes} ) {
            push @{ $LinkTypesByIncidentState{ $LinkTypes->{$LinkType} } }, $LinkType;
        }
    }

    # handle added or removed links
    if ( $Param{Event} eq 'LinkAdd' || $Param{Event} eq 'LinkDelete' ) {
        my $ConfigItemID = $Param{Data}->{ConfigItemID};
        my $Version      = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
            ConfigItemID => $ConfigItemID,
        );

        # optional: check if CI deployment state is relevant
        my $DeploymentStates = $ConfigObject->Get('ITSMConfigItem::LinkStatus::DeploymentStates');
        if ( IsArrayRefWithData($DeploymentStates) ) {
            return 1 if !grep { $_ eq $Version->{DeplState} } @{$DeploymentStates};
        }

        # raise incident state if necessary
        if ( $Param{Event} eq 'LinkAdd' ) {
            return $Self->_CheckRaiseIncidentState(
                Version                  => $Version,
                IncidentStates           => $IncidentStates,
                LinkTypesByIncidentState => \%LinkTypesByIncidentState,
                TicketID                 => $TicketID,
            );
        }

        # lower incident state if necessary
        return $Self->_CheckLowerIncidentState(
            Version                  => $Version,
            IncidentStates           => $IncidentStates,
            LinkTypesByIncidentState => \%LinkTypesByIncidentState,
            TicketID                 => $TicketID,
        );
    }

    # handle ticket closure/re-open and relevant/irrelevant type change
    elsif ( $Param{Event} eq 'TicketStateUpdate' || $Param{Event} eq 'TicketTypeUpdate' ) {

        # relevant ticket state and type updates are handled the same way
        # both mean a ticket is relevant for CI incident state changes or not
        # ticket re-open or change to relevant type = might have to raise incident state of CIs
        # ticket closure or change to irrelevant type = might have to lower incident state of CIs
        my $RelevantNow;
        if (
            ( $Param{Event} eq 'TicketStateUpdate' && $NewTicketStateRelevant )
            || ( $Param{Event} eq 'TicketTypeUpdate' && $NewTicketTypeRelevant )
            )
        {
            $RelevantNow = 1;
        }

        # loop through all link types and check links to see if there are possibly affected CIs
        my $DeploymentStates = $ConfigObject->Get('ITSMConfigItem::LinkStatus::DeploymentStates');
        INCIDENTSTATE:
        for my $IncidentState ( @{$IncidentStates} ) {
            LINKTYPE:
            for my $LinkType ( @{ $LinkTypesByIncidentState{$IncidentState} } ) {
                my @LinkedCIs = $Self->_CheckTicketLinks(
                    TicketID => $TicketID,
                    Type     => $LinkType,
                );
                next LINKTYPE if !@LinkedCIs;

                CONFIGITEMID:
                for my $ConfigItemID (@LinkedCIs) {
                    my $Version = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
                        ConfigItemID => $ConfigItemID,
                    );

                    # optional: check if CI deployment state is relevant
                    if ( IsArrayRefWithData($DeploymentStates) ) {
                        next CONFIGITEMID if !grep { $_ eq $Version->{DeplState} } @{$DeploymentStates};
                    }

                    # check current incident state vs state caused by link

                    # current incident state is lower than caused by link -> set
                    if ($RelevantNow) {

                        # shortcut: nothing to do if CI is already in highest incident state
                        next CONFIGITEMID if $Version->{InciState} eq $IncidentStates->[0];

                        # shortcut: nothing to do if CI is already in same incident state
                        next CONFIGITEMID if $Version->{InciState} eq $IncidentState;

                        # check if CI is in higher incident state already
                        TMPINCIDENTSTATE:
                        for my $TmpIncidentState ( @{$IncidentStates} ) {

                            # all further incident states are lower than current one -> set new state
                            last TMPINCIDENTSTATE if $TmpIncidentState eq $IncidentState;

                            next CONFIGITEMID if $Version->{InciState} eq $TmpIncidentState;
                        }

                        # set new incident state
                        $Self->_SetCIStatus(
                            Version       => $Version,
                            IncidentState => $IncidentState,
                            TicketID      => $TicketID,
                        );
                    }

                    # current incident state is same as caused by link -> re-calculate
                    else {

                        # shortcut: nothing to do if CI is already in lowest incident state
                        next CONFIGITEMID if $Version->{InciState} eq $IncidentStates->[-1];

                        # shortcut: nothing to do if CI is not in same incident state
                        next CONFIGITEMID if $Version->{InciState} ne $IncidentState;

                        # recalculate incident state exactly
                        $Self->_CheckLowerIncidentState(
                            Version                  => $Version,
                            IncidentStates           => $IncidentStates,
                            LinkTypesByIncidentState => \%LinkTypesByIncidentState,
                            TicketID                 => $TicketID,
                        );
                    }
                }
            }
        }
    }

    # unknown/irrelevant event
    else {
        return 1;
    }

    return 1;
}

# Rationale:
# Start at highest (worst) incident state and loop down to current incident state,
#   checking for links on all types configured for the corresponding incident state.
# If any link for a higher incident state than the current one is found,
#   set this state and exit.
# Due to top-to-bottom method, no higher incident state can occur later.
sub _CheckRaiseIncidentState {
    my ( $Self, %Param ) = @_;

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

    # loop through target states and check if it needs to be changed
    INCIDENTSTATE:
    for my $IncidentState ( @{ $Param{IncidentStates} } ) {

        # CI already in target state - nothing to do
        return 1 if $Param{Version}->{InciState} eq $IncidentState;

        # check if we have linked tickets that lead to incident state
        LINKTYPE:
        for my $LinkType ( @{ $Param{LinkTypesByIncidentState}->{$IncidentState} } ) {
            next LINKTYPE if !$Self->_CheckConfigItemLinks(
                ConfigItemID => $Param{Version}->{ConfigItemID},
                Type         => $LinkType,
            );

            # we have at least one linked ticket - set incident state and exit
            return $Self->_SetCIStatus(
                Version       => $Param{Version},
                IncidentState => $IncidentState,
                TicketID      => $Param{TicketID},
            );
        }
    }

    return 1;
}

# Rationale:
# Shortcut if CI is alredy in lowest (best) incident state = can't have changed after link delete.
# Start at highest (worst) incident state and loop down to current incident state,
#   checking for links on all types configured for the corresponding incident states.
# If a link for the current incident state is found, exit immediately (no change).
# Otherwise set incident state for first occurance of a link, then exit.
# If no link is found, set the lowest state.
# Due to top-to-bottom method, first remembered incident state is the highest possible.
sub _CheckLowerIncidentState {
    my ( $Self, %Param ) = @_;

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

    # shortcut for lowest level
    return 1 if $Param{Version}->{InciState} eq $Param{IncidentStates}->[-1];

    # loop through target states and check if it needs to be changed
    my $LowerIncidentState;
    INCIDENTSTATE:
    for my $IncidentState ( @{ $Param{IncidentStates} } ) {

        # skip level if current CI incident state is not on this or any higher level
        next INCIDENTSTATE if $Param{Version}->{InciState} ne $IncidentState && !$LowerIncidentState;

        # check if we have linked tickets that keep incident state
        LINKTYPE:
        for my $LinkType ( @{ $Param{LinkTypesByIncidentState}->{$IncidentState} } ) {
            next LINKTYPE if !$Self->_CheckConfigItemLinks(
                ConfigItemID => $Param{Version}->{ConfigItemID},
                Type         => $LinkType,
            );

            # we have at least one linked ticket

            # keep incident state
            return 1 if $Param{Version}->{InciState} eq $IncidentState;

            # otherwise set new (lower) incident state and exit
            return $Self->_SetCIStatus(
                Version       => $Param{Version},
                IncidentState => $IncidentState,
                TicketID      => $Param{TicketID},
            );
        }

        # remember that state needs to be changed (= current level isn'T justified by links anymore)
        $LowerIncidentState = 1;
    }

    # no relevant links have been found, but CI wasn't in lowest incident state before - set it now and exit
    return $Self->_SetCIStatus(
        Version       => $Param{Version},
        IncidentState => $Param{IncidentStates}->[-1],
        TicketID      => $Param{TicketID},
    );
}

# set CI to a specified incident state and log change in corresponding ticket
sub _SetCIStatus {
    my ( $Self, %Param ) = @_;

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

            return;
        }
    }

    # get incident state list
    my $IncidentStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::Core::IncidentState',
    );
    return if !IsHashRefWithData($IncidentStateList);

    # check if incident state is valid
    my %ReverseIncidentStateList = reverse %{$IncidentStateList};
    if ( !$ReverseIncidentStateList{ $Param{IncidentState} } ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Have invalid incident state '$Param{IncidentState}'!",
        );
        return;
    }

    # add a new version with the new incident state
    my $VersionID = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionAdd(
        %{ $Param{Version} },
        InciStateID => $ReverseIncidentStateList{ $Param{IncidentState} },
        UserID      => 1,
    );
    return if !$VersionID;

    # log change in ticket
    $Kernel::OM->Get('Kernel::System::Ticket')->HistoryAdd(
        TicketID     => $Param{TicketID},
        HistoryType  => 'Misc',
        Name         => "Updated incident state of config item '$Param{Version}->{Number}' to '$Param{IncidentState}'.",
        CreateUserID => 1,
    );

    return 1;
}

# check if relevant CIs are linked to a ticket
sub _CheckTicketLinks {
    my ( $Self, %Param ) = @_;

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

    # get link type lookup
    my $LinkTypeLookup = $Self->_LinkTypeLookupGet();

    my %LinkKeyList = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkKeyList(
        Object1   => 'Ticket',
        Key1      => $Param{TicketID},
        Object2   => 'ITSMConfigItem',
        State     => 'Valid',
        Type      => $LinkTypeLookup->{ $Param{Type} }->{Name},
        Direction => $LinkTypeLookup->{ $Param{Type} }->{Direction}->{Ticket},
        UserID    => 1,
    );
    return if !IsHashRefWithData( \%LinkKeyList );
    return ( sort keys %LinkKeyList );
}

# check if relevant open tickets are linked to a CI
sub _CheckConfigItemLinks {
    my ( $Self, %Param ) = @_;

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

    # get link type lookup
    my $LinkTypeLookup = $Self->_LinkTypeLookupGet();

    my %LinkKeyList = $Kernel::OM->Get('Kernel::System::LinkObject')->LinkKeyList(
        Object1   => 'ITSMConfigItem',
        Key1      => $Param{ConfigItemID},
        Object2   => 'Ticket',
        State     => 'Valid',
        Type      => $LinkTypeLookup->{ $Param{Type} }->{Name},
        Direction => $LinkTypeLookup->{ $Param{Type} }->{Direction}->{ITSMConfigItem},
        UserID    => 1,
    );
    return if !IsHashRefWithData( \%LinkKeyList );

    # list of relevant state types
    my @OpenStateTypes = ( 'new', 'open', 'pending reminder', 'pending auto' );

    # use ticket type check?
    my $TicketTypes = $Kernel::OM->Get('Kernel::Config')->Get('ITSMConfigItem::LinkStatus::TicketTypes');
    my $CheckTicketTypes;
    $CheckTicketTypes = 1 if IsArrayRefWithData($TicketTypes);

    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');
    TICKETID:
    for my $TicketID ( sort keys %LinkKeyList ) {
        my %Ticket = $TicketObject->TicketGet(
            TicketID => $TicketID,
            UserID   => 1,
        );
        next TICKETID if !grep { $_ eq $Ticket{StateType} } @OpenStateTypes;
        if ($CheckTicketTypes) {
            next TICKETID if !$Ticket{Type};
            next TICKETID if !grep { $_ eq $Ticket{Type} } @{$TicketTypes};
        }
        return 1;
    }

    return;
}

# compile list of internal link type names and directions to use for LinkKeyList
sub _LinkTypeLookupGet {
    my ( $Self, %Param ) = @_;

    # if we have generated lookup before
    return $Self->{LinkTypeLookup} if $Self->{LinkTypeLookup};

    my %LinkTypeLookup;
    my %TypeList = $Kernel::OM->Get('Kernel::System::LinkObject')->TypeList();
    for my $TypeNameInternal ( sort keys %TypeList ) {
        my $SourceName = $TypeList{$TypeNameInternal}->{SourceName};
        my $TargetName = $TypeList{$TypeNameInternal}->{TargetName};
        $LinkTypeLookup{$SourceName} = {
            Name      => $TypeNameInternal,
            Direction => {
                ITSMConfigItem => $SourceName eq $TargetName ? 'Both' : 'Target',
                Ticket         => $SourceName eq $TargetName ? 'Both' : 'Source',
            },
        };
        $LinkTypeLookup{$TargetName} = {
            Name      => $TypeNameInternal,
            Direction => {
                ITSMConfigItem => $SourceName eq $TargetName ? 'Both' : 'Source',
                Ticket         => $SourceName eq $TargetName ? 'Both' : 'Target',
            },
        };
    }

    # remember result
    $Self->{LinkTypeLookup} = \%LinkTypeLookup;

    return \%LinkTypeLookup;
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

## no critic (Modules::RequireExplicitPackage)
use strict;
use warnings;
use utf8;

use vars (qw($Self));

my $CommandObject = $Kernel::OM->Get('Kernel::System::Console::Command::Admin::ITSM::Configitem::Delete');

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

my $ExitCode = $CommandObject->Execute();

$Self->Is(
    $ExitCode,
    1,
    "Admin::ITSM::Configitem::Delete exit code without options",
);

# check command with option --all and argument --accept n ( cancel deleting all config item)
$ExitCode = $CommandObject->Execute( '--all', 'n' );

$Self->Is(
    $ExitCode,
    0,
    "Option '--all' n",
);

# check command with class options (invalid class)
my $RandomClass = 'TestClass' . $Helper->GetRandomID();
$ExitCode = $CommandObject->Execute( '--class', $RandomClass );

$Self->Is(
    $ExitCode,
    1,
    "Option 'class' (but class $RandomClass doesn't exist) ",
);

# get general catalog object
my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

# add test general catalog item
my $GeneralCatalogItemID = $GeneralCatalogObject->ItemAdd(
    Class   => 'ITSM::ConfigItem::Class',
    Name    => $RandomClass,
    ValidID => 1,
    Comment => 'Comment',
    UserID  => 1,
);

$GeneralCatalogObject->GeneralCatalogPreferencesSet(
    ItemID => $GeneralCatalogItemID,
    Key    => 'Permission',
    Value  => 5,
);

$Self->True(
    $GeneralCatalogItemID,
    "Test general catalog item is created - $GeneralCatalogItemID ",
);

# get ConfigItem object
my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

# get list of all deployment states
my $DeplStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %ReverseDeplStateList = reverse %{$DeplStateList};

# get list of all incident states
my $InciStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %ReverseInciStateList = reverse %{$InciStateList};

my @ConfigItemNumbers;
my $ConfigItemID;

for ( 1 .. 10 ) {

    # create ConfigItem number
    my $ConfigItemNumber = $ConfigItemObject->ConfigItemNumberCreate(
        Type    => $Kernel::OM->Get('Kernel::Config')->Get('ITSMConfigItem::NumberGenerator'),
        ClassID => $GeneralCatalogItemID,
    );

    # add test ConfigItem
    $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        Number  => $ConfigItemNumber,
        ClassID => $GeneralCatalogItemID,
        UserID  => 1,
    );

    push @ConfigItemNumbers, $ConfigItemNumber;

    # add new versions for the last added in previous loop
    my $ConfigItemName = 'TestConfigItem' . $Helper->GetRandomID();

    COUNT:
    for my $Count ( 1 .. 50 ) {

        my $VersionID = $ConfigItemObject->VersionAdd(
            Name         => $ConfigItemName . '-' . $Count,
            DefinitionID => 1,
            DeplStateID  => $ReverseDeplStateList{Planned},
            InciStateID  => $ReverseInciStateList{Operational},
            UserID       => 1,
            ConfigItemID => $ConfigItemID,
        );

        $Self->True(
            $VersionID,
            "Version $Count for config item $ConfigItemID is created - $ConfigItemName",
        );

        # change the date into past for the first 20 versions
        next COUNT if $Count > 10;

        my $VersionTestCreateTime = '2010-01-01 00:00:00';

        # insert new version
        my $Success = $Kernel::OM->Get('Kernel::System::DB')->Do(
            SQL => 'UPDATE configitem_version
                SET create_time = ?
                WHERE id = ?',
            Bind => [
                \$VersionTestCreateTime,
                \$VersionID,
            ],
        );
    }

    # check command with all-older-than-days-versions options (delete all versions older than one day)
    $ExitCode = $CommandObject->Execute( '--all-older-than-days-versions', 1 );

    $Self->Is(
        $ExitCode,
        0,
        "Exit code: Options --all-older-than-days-versions 1",
    );

    # get the list of remaining versions of this config item
    my $VersionList = $ConfigItemObject->VersionList(
        ConfigItemID => $ConfigItemID,
    );

    # result should only be 40 versions now
    $Self->Is(
        scalar @{$VersionList},
        40,
        "Number of remaining versions after running command with Options --all-older-than-days-versions 1",
    );

    # check command with all-but-keep-last-versions options (delete all versions but keep the last 30 versions)
    $ExitCode = $CommandObject->Execute( '--all-but-keep-last-versions', 30 );

    $Self->Is(
        $ExitCode,
        0,
        "Exit code: Options --all-but-keep-last-versions 30",
    );

    # get the list of remaining versions of this config item
    $VersionList = $ConfigItemObject->VersionList(
        ConfigItemID => $ConfigItemID,
    );

    # result should only be 40 versions now
    $Self->Is(
        scalar @{$VersionList},
        30,
        "Number of remaining versions after running command with Options --all-but-keep-last-versions 30",
    );

    # check command with all-old-versions options (delete all old versions except the last one)
    $ExitCode = $CommandObject->Execute('--all-old-versions');

    $Self->Is(
        $ExitCode,
        0,
        "Exit code: Options --all-old-versions",
    );

    # get the list of remaining versions of this config item
    $VersionList = $ConfigItemObject->VersionList(
        ConfigItemID => $ConfigItemID,
    );

    # result should only be 40 versions now
    $Self->Is(
        scalar @{$VersionList},
        1,
        "Number of remaining versions after running command with Options --all-old-versions",
    );
}

# check command with class options ($RandomClass class) and deployment-state 'Planned'
$ExitCode = $CommandObject->Execute( '--class', $RandomClass, '--deployment-state', 'Planned' );

$Self->Is(
    $ExitCode,
    0,
    "Exit code: Options --class $RandomClass --deployment-state' Planned",
);

# check command with configitem-number options
$ExitCode = $CommandObject->Execute(
    '--configitem-number', $ConfigItemNumbers[0], '--configitem-number',
    $ConfigItemNumbers[1]
);

$Self->Is(
    $ExitCode,
    0,
    "Exit code: Options --configitem-number",
);

# check command with class options ($RandomClass class)
# three config Items of ten created in test were deleted with the previous commands
# with the next command other seven will be deleted
$ExitCode = $CommandObject->Execute( '--class', $RandomClass );

$Self->Is(
    $ExitCode,
    0,
    "Exit code: Option --class $RandomClass",
);

# check command with configitem-number options and not allowed additional class option
$ExitCode = $CommandObject->Execute( '--configitem-number', $ConfigItemNumbers[0], '--class', $RandomClass );

$Self->Is(
    $ExitCode,
    1,
    "Exit code: Options --configitem-number --class",
);

# check command with configitem-number options and not allowed additional deployment state option
$ExitCode = $CommandObject->Execute( '--configitem-number', $ConfigItemNumbers[0], '--deployment-state', 'Planned' );

$Self->Is(
    $ExitCode,
    1,
    "Exit code: Options --configitem-number --deployment-state",
);

# check command with --all, --all-old-versions, --all-but-keep-last-versions, --all-older-than-days-versions mixed together.
$ExitCode = $CommandObject->Execute(
    '--all', '--all-old-versions',             '--all-but-keep-last-versions',
    2,       '--all-older-than-days-versions', '10'
);

$Self->Is(
    $ExitCode,
    1,
    "Exit code: Options --all --all-old-versions --all-but-keep-last-versions 2 --all-older-than-days-versions 10",
);

# cleanup is done by RestoreDatabase

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

## no critic (Modules::RequireExplicitPackage)
use strict;
use warnings;
use utf8;

use vars (qw($Self));

my $CommandObject = $Kernel::OM->Get('Kernel::System::Console::Command::Admin::ITSM::Configitem::Delete');

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

my $ExitCode = $CommandObject->Execute();

$Self->Is(
    $ExitCode,
    1,
    "Admin::ITSM::Configitem::Delete exit code without options",
);

# check command with option --all and argument --accept n ( cancel deleting all config item)
$ExitCode = $CommandObject->Execute( '--all', 'n' );

$Self->Is(
    $ExitCode,
    0,
    "Option '--all' n",
);

# check command with class options (invalid class)
my $RandomClass = 'TestClass' . $Helper->GetRandomID();
$ExitCode = $CommandObject->Execute( '--class', $RandomClass );

$Self->Is(
    $ExitCode,
    1,
    "Option 'class' (but class $RandomClass doesn't exist) ",
);

# get general catalog object
my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

# add test general catalog item
my $GeneralCatalogItemID = $GeneralCatalogObject->ItemAdd(
    Class   => 'ITSM::ConfigItem::Class',
    Name    => $RandomClass,
    ValidID => 1,
    Comment => 'Comment',
    UserID  => 1,
);

$GeneralCatalogObject->GeneralCatalogPreferencesSet(
    ItemID => $GeneralCatalogItemID,
    Key    => 'Permission',
    Value  => 5,
);

$Self->True(
    $GeneralCatalogItemID,
    "Test general catalog item is created - $GeneralCatalogItemID ",
);

# get ConfigItem object
my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

# get list of all deployment states
my $DeplStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %ReverseDeplStateList = reverse %{$DeplStateList};

# get list of all incident states
my $InciStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %ReverseInciStateList = reverse %{$InciStateList};

my @ConfigItemNumbers;
my $ConfigItemID;

for ( 1 .. 10 ) {

    # create ConfigItem number
    my $ConfigItemNumber = $ConfigItemObject->ConfigItemNumberCreate(
        Type    => $Kernel::OM->Get('Kernel::Config')->Get('ITSMConfigItem::NumberGenerator'),
        ClassID => $GeneralCatalogItemID,
    );

    # add test ConfigItem
    $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        Number  => $ConfigItemNumber,
        ClassID => $GeneralCatalogItemID,
        UserID  => 1,
    );

    push @ConfigItemNumbers, $ConfigItemNumber;

    # add new versions for the last added in previous loop
    my $ConfigItemName = 'TestConfigItem' . $Helper->GetRandomID();

    COUNT:
    for my $Count ( 1 .. 50 ) {

        my $VersionID = $ConfigItemObject->VersionAdd(
            Name         => $ConfigItemName . '-' . $Count,
            DefinitionID => 1,
            DeplStateID  => $ReverseDeplStateList{Planned},
            InciStateID  => $ReverseInciStateList{Operational},
            UserID       => 1,
            ConfigItemID => $ConfigItemID,
        );

        $Self->True(
            $VersionID,
            "Version $Count for config item $ConfigItemID is created - $ConfigItemName",
        );

        # change the date into past for the first 20 versions
        next COUNT if $Count > 10;

        my $VersionTestCreateTime = '2010-01-01 00:00:00';

        # insert new version
        my $Success = $Kernel::OM->Get('Kernel::System::DB')->Do(
            SQL => 'UPDATE configitem_version
                SET create_time = ?
                WHERE id = ?',
            Bind => [
                \$VersionTestCreateTime,
                \$VersionID,
            ],
        );
    }

    # check command with all-older-than-days-versions options (delete all versions older than one day)
    $ExitCode = $CommandObject->Execute( '--all-older-than-days-versions', 1 );

    $Self->Is(
        $ExitCode,
        0,
        "Exit code: Options --all-older-than-days-versions 1",
    );

    # get the list of remaining versions of this config item
    my $VersionList = $ConfigItemObject->VersionList(
        ConfigItemID => $ConfigItemID,
    );

    # result should only be 40 versions now
    $Self->Is(
        scalar @{$VersionList},
        40,
        "Number of remaining versions after running command with Options --all-older-than-days-versions 1",
    );

    # check command with all-but-keep-last-versions options (delete all versions but keep the last 30 versions)
    $ExitCode = $CommandObject->Execute( '--all-but-keep-last-versions', 30 );

    $Self->Is(
        $ExitCode,
        0,
        "Exit code: Options --all-but-keep-last-versions 30",
    );

    # get the list of remaining versions of this config item
    $VersionList = $ConfigItemObject->VersionList(
        ConfigItemID => $ConfigItemID,
    );

    # result should only be 40 versions now
    $Self->Is(
        scalar @{$VersionList},
        30,
        "Number of remaining versions after running command with Options --all-but-keep-last-versions 30",
    );

    # check command with all-old-versions options (delete all old versions except the last one)
    $ExitCode = $CommandObject->Execute('--all-old-versions');

    $Self->Is(
        $ExitCode,
        0,
        "Exit code: Options --all-old-versions",
    );

    # get the list of remaining versions of this config item
    $VersionList = $ConfigItemObject->VersionList(
        ConfigItemID => $ConfigItemID,
    );

    # result should only be 40 versions now
    $Self->Is(
        scalar @{$VersionList},
        1,
        "Number of remaining versions after running command with Options --all-old-versions",
    );
}

# check command with class options ($RandomClass class) and deployment-state 'Planned'
$ExitCode = $CommandObject->Execute( '--class', $RandomClass, '--deployment-state', 'Planned' );

$Self->Is(
    $ExitCode,
    0,
    "Exit code: Options --class $RandomClass --deployment-state' Planned",
);

# check command with configitem-number options
$ExitCode = $CommandObject->Execute(
    '--configitem-number', $ConfigItemNumbers[0], '--configitem-number',
    $ConfigItemNumbers[1]
);

$Self->Is(
    $ExitCode,
    0,
    "Exit code: Options --configitem-number",
);

# check command with class options ($RandomClass class)
# three config Items of ten created in test were deleted with the previous commands
# with the next command other seven will be deleted
$ExitCode = $CommandObject->Execute( '--class', $RandomClass );

$Self->Is(
    $ExitCode,
    0,
    "Exit code: Option --class $RandomClass",
);

# check command with configitem-number options and not allowed additional class option
$ExitCode = $CommandObject->Execute( '--configitem-number', $ConfigItemNumbers[0], '--class', $RandomClass );

$Self->Is(
    $ExitCode,
    1,
    "Exit code: Options --configitem-number --class",
);

# check command with configitem-number options and not allowed additional deployment state option
$ExitCode = $CommandObject->Execute( '--configitem-number', $ConfigItemNumbers[0], '--deployment-state', 'Planned' );

$Self->Is(
    $ExitCode,
    1,
    "Exit code: Options --configitem-number --deployment-state",
);

# check command with --all, --all-old-versions, --all-but-keep-last-versions, --all-older-than-days-versions mixed together.
$ExitCode = $CommandObject->Execute(
    '--all', '--all-old-versions',             '--all-but-keep-last-versions',
    2,       '--all-older-than-days-versions', '10'
);

$Self->Is(
    $ExitCode,
    1,
    "Exit code: Options --all --all-old-versions --all-but-keep-last-versions 2 --all-older-than-days-versions 10",
);

# cleanup is done by RestoreDatabase

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

## no critic (Modules::RequireExplicitPackage)
use strict;
use warnings;
use utf8;

use vars (qw($Self));

my $CommandObject = $Kernel::OM->Get('Kernel::System::Console::Command::Admin::ITSM::Configitem::Delete');

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

my $ExitCode = $CommandObject->Execute();

$Self->Is(
    $ExitCode,
    1,
    "Admin::ITSM::Configitem::Delete exit code without options",
);

# check command with option --all and argument --accept n ( cancel deleting all config item)
$ExitCode = $CommandObject->Execute( '--all', 'n' );

$Self->Is(
    $ExitCode,
    0,
    "Option '--all' n",
);

# check command with class options (invalid class)
my $RandomClass = 'TestClass' . $Helper->GetRandomID();
$ExitCode = $CommandObject->Execute( '--class', $RandomClass );

$Self->Is(
    $ExitCode,
    1,
    "Option 'class' (but class $RandomClass doesn't exist) ",
);

# get general catalog object
my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

# add test general catalog item
my $GeneralCatalogItemID = $GeneralCatalogObject->ItemAdd(
    Class   => 'ITSM::ConfigItem::Class',
    Name    => $RandomClass,
    ValidID => 1,
    Comment => 'Comment',
    UserID  => 1,
);

$GeneralCatalogObject->GeneralCatalogPreferencesSet(
    ItemID => $GeneralCatalogItemID,
    Key    => 'Permission',
    Value  => 5,
);

$Self->True(
    $GeneralCatalogItemID,
    "Test general catalog item is created - $GeneralCatalogItemID ",
);

# get ConfigItem object
my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

# get list of all deployment states
my $DeplStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %ReverseDeplStateList = reverse %{$DeplStateList};

# get list of all incident states
my $InciStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %ReverseInciStateList = reverse %{$InciStateList};

my @ConfigItemNumbers;
my $ConfigItemID;

for ( 1 .. 10 ) {

    # create ConfigItem number
    my $ConfigItemNumber = $ConfigItemObject->ConfigItemNumberCreate(
        Type    => $Kernel::OM->Get('Kernel::Config')->Get('ITSMConfigItem::NumberGenerator'),
        ClassID => $GeneralCatalogItemID,
    );

    # add test ConfigItem
    $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        Number  => $ConfigItemNumber,
        ClassID => $GeneralCatalogItemID,
        UserID  => 1,
    );

    push @ConfigItemNumbers, $ConfigItemNumber;

    # add new versions for the last added in previous loop
    my $ConfigItemName = 'TestConfigItem' . $Helper->GetRandomID();

    COUNT:
    for my $Count ( 1 .. 50 ) {

        my $VersionID = $ConfigItemObject->VersionAdd(
            Name         => $ConfigItemName . '-' . $Count,
            DefinitionID => 1,
            DeplStateID  => $ReverseDeplStateList{Planned},
            InciStateID  => $ReverseInciStateList{Operational},
            UserID       => 1,
            ConfigItemID => $ConfigItemID,
        );

        $Self->True(
            $VersionID,
            "Version $Count for config item $ConfigItemID is created - $ConfigItemName",
        );

        # change the date into past for the first 20 versions
        next COUNT if $Count > 10;

        my $VersionTestCreateTime = '2010-01-01 00:00:00';

        # insert new version
        my $Success = $Kernel::OM->Get('Kernel::System::DB')->Do(
            SQL => 'UPDATE configitem_version
                SET create_time = ?
                WHERE id = ?',
            Bind => [
                \$VersionTestCreateTime,
                \$VersionID,
            ],
        );
    }

    # check command with all-older-than-days-versions options (delete all versions older than one day)
    $ExitCode = $CommandObject->Execute( '--all-older-than-days-versions', 1 );

    $Self->Is(
        $ExitCode,
        0,
        "Exit code: Options --all-older-than-days-versions 1",
    );

    # get the list of remaining versions of this config item
    my $VersionList = $ConfigItemObject->VersionList(
        ConfigItemID => $ConfigItemID,
    );

    # result should only be 40 versions now
    $Self->Is(
        scalar @{$VersionList},
        40,
        "Number of remaining versions after running command with Options --all-older-than-days-versions 1",
    );

    # check command with all-but-keep-last-versions options (delete all versions but keep the last 30 versions)
    $ExitCode = $CommandObject->Execute( '--all-but-keep-last-versions', 30 );

    $Self->Is(
        $ExitCode,
        0,
        "Exit code: Options --all-but-keep-last-versions 30",
    );

    # get the list of remaining versions of this config item
    $VersionList = $ConfigItemObject->VersionList(
        ConfigItemID => $ConfigItemID,
    );

    # result should only be 40 versions now
    $Self->Is(
        scalar @{$VersionList},
        30,
        "Number of remaining versions after running command with Options --all-but-keep-last-versions 30",
    );

    # check command with all-old-versions options (delete all old versions except the last one)
    $ExitCode = $CommandObject->Execute('--all-old-versions');

    $Self->Is(
        $ExitCode,
        0,
        "Exit code: Options --all-old-versions",
    );

    # get the list of remaining versions of this config item
    $VersionList = $ConfigItemObject->VersionList(
        ConfigItemID => $ConfigItemID,
    );

    # result should only be 40 versions now
    $Self->Is(
        scalar @{$VersionList},
        1,
        "Number of remaining versions after running command with Options --all-old-versions",
    );
}

# check command with class options ($RandomClass class) and deployment-state 'Planned'
$ExitCode = $CommandObject->Execute( '--class', $RandomClass, '--deployment-state', 'Planned' );

$Self->Is(
    $ExitCode,
    0,
    "Exit code: Options --class $RandomClass --deployment-state' Planned",
);

# check command with configitem-number options
$ExitCode = $CommandObject->Execute(
    '--configitem-number', $ConfigItemNumbers[0], '--configitem-number',
    $ConfigItemNumbers[1]
);

$Self->Is(
    $ExitCode,
    0,
    "Exit code: Options --configitem-number",
);

# check command with class options ($RandomClass class)
# three config Items of ten created in test were deleted with the previous commands
# with the next command other seven will be deleted
$ExitCode = $CommandObject->Execute( '--class', $RandomClass );

$Self->Is(
    $ExitCode,
    0,
    "Exit code: Option --class $RandomClass",
);

# check command with configitem-number options and not allowed additional class option
$ExitCode = $CommandObject->Execute( '--configitem-number', $ConfigItemNumbers[0], '--class', $RandomClass );

$Self->Is(
    $ExitCode,
    1,
    "Exit code: Options --configitem-number --class",
);

# check command with configitem-number options and not allowed additional deployment state option
$ExitCode = $CommandObject->Execute( '--configitem-number', $ConfigItemNumbers[0], '--deployment-state', 'Planned' );

$Self->Is(
    $ExitCode,
    1,
    "Exit code: Options --configitem-number --deployment-state",
);

# check command with --all, --all-old-versions, --all-but-keep-last-versions, --all-older-than-days-versions mixed together.
$ExitCode = $CommandObject->Execute(
    '--all', '--all-old-versions',             '--all-but-keep-last-versions',
    2,       '--all-older-than-days-versions', '10'
);

$Self->Is(
    $ExitCode,
    1,
    "Exit code: Options --all --all-old-versions --all-but-keep-last-versions 2 --all-older-than-days-versions 10",
);

# cleanup is done by RestoreDatabase

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgKHF3KCRTZWxmKSk7CgojIGdldCBjb21tYW5kIG9iamVjdApteSAkQ29tbWFuZE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpDb25zb2xlOjpDb21tYW5kOjpBZG1pbjo6SVRTTTo6Q29uZmlnaXRlbTo6TGlzdER1cGxpY2F0ZXMnKTsKCiRLZXJuZWw6Ok9NLT5PYmplY3RQYXJhbUFkZCgKICAgICdLZXJuZWw6OlN5c3RlbTo6VW5pdFRlc3Q6OkhlbHBlcicgPT4gewogICAgICAgIFJlc3RvcmVEYXRhYmFzZSA9PiAxLAogICAgfSwKKTsKbXkgJEhlbHBlck9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyk7CgojIGNoZWNrIGNvbW1hbmQgd2l0aG91dCBhbnkgb3B0aW9ucwpteSAkRXhpdENvZGUgPSAkQ29tbWFuZE9iamVjdC0+RXhlY3V0ZSgpOwoKJFNlbGYtPklzKAogICAgJEV4aXRDb2RlLAogICAgMCwKICAgICJObyBvcHRpb25zIC0gbGlzdCBhbGwgY29uZmlnIGl0ZW1zIGluIHByb2R1Y3RpdmUgc3RhdGVzIiwKKTsKCiMgY2hlY2sgY29tbWFuZCB3aXRoIC0tY2xhc3Mgb3B0aW9ucyAoaW52YWxpZCBjbGFzcykKbXkgJFJhbmRvbUNsYXNzID0gJ05vbkV4aXN0aW5nQ2xhc3MnIC4gJEhlbHBlck9iamVjdC0+R2V0UmFuZG9tSUQoKTsKJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoICctLWNsYXNzJywgJFJhbmRvbUNsYXNzICk7CgokU2VsZi0+SXMoCiAgICAkRXhpdENvZGUsCiAgICAxLAogICAgIk9wdGlvbiAnY2xhc3MnIChidXQgY2xhc3MgJFJhbmRvbUNsYXNzIGRvZXNuJ3QgZXhpc3QpICIsCik7CgojIGFkZCB0ZXN0IGNvbmZpZyBpdGVtCm15ICRHZW5lcmFsQ2F0YWxvZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpOwoKIyBnZXQgJ0hhcmR3YXJlJyBjYXRhbG9nIGNsYXNzIElEcwpteSAkQ29uZmlnSXRlbURhdGFSZWYgPSAkR2VuZXJhbENhdGFsb2dPYmplY3QtPkl0ZW1HZXQoCiAgICBDbGFzcyA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgPT4gJ0hhcmR3YXJlJywKKTsKbXkgJEhhcmR3YXJlQ29uZmlnSXRlbUlEID0gJENvbmZpZ0l0ZW1EYXRhUmVmLT57SXRlbUlEfTsKCiMgZ2V0ICdQcm9kdWN0aW9uJyBkZXBsb3ltZW50IHN0YXRlIElEcwpteSAkUHJvZHVjdGlvbkRlcGxTdGF0ZURhdGFSZWYgPSAkR2VuZXJhbENhdGFsb2dPYmplY3QtPkl0ZW1HZXQoCiAgICBDbGFzcyA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6RGVwbG95bWVudFN0YXRlJywKICAgIE5hbWUgID0+ICdQcm9kdWN0aW9uJywKKTsKbXkgJFByb2R1Y3Rpb25EZXBsU3RhdGVJRCA9ICRQcm9kdWN0aW9uRGVwbFN0YXRlRGF0YVJlZi0+e0l0ZW1JRH07CgojIGdldCBDb25maWdJdGVtIG9iamVjdApteSAkQ29uZmlnSXRlbU9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbScpOwoKIyBjcmVhdGUgQ29uZmlnSXRlbSBudW1iZXIKbXkgJENvbmZpZ0l0ZW1OdW1iZXIgPSAkQ29uZmlnSXRlbU9iamVjdC0+Q29uZmlnSXRlbU51bWJlckNyZWF0ZSgKICAgIFR5cGUgICAgPT4gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpDb25maWcnKS0+R2V0KCdJVFNNQ29uZmlnSXRlbTo6TnVtYmVyR2VuZXJhdG9yJyksCiAgICBDbGFzc0lEID0+ICRIYXJkd2FyZUNvbmZpZ0l0ZW1JRCwKKTsKCm15IEBDb25maWdJdGVtSUQ7CgojIGFkZCB0aGUgbmV3IENvbmZpZ0l0ZW0KbXkgJENvbmZpZ0l0ZW1JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5Db25maWdJdGVtQWRkKAogICAgTnVtYmVyICA9PiAkQ29uZmlnSXRlbU51bWJlciwKICAgIENsYXNzSUQgPT4gJEhhcmR3YXJlQ29uZmlnSXRlbUlELAogICAgVXNlcklEICA9PiAxLAopOwpwdXNoIEBDb25maWdJdGVtSUQsICRDb25maWdJdGVtSUQ7CgpteSAkQ29uZmlnSXRlbU5hbWUgPSAnVGVzdENvbmZpZ0l0ZW0nIC4gJEhlbHBlck9iamVjdC0+R2V0UmFuZG9tSUQoKTsKbXkgJFZlcnNpb25JRCAgICAgID0gJENvbmZpZ0l0ZW1PYmplY3QtPlZlcnNpb25BZGQoCiAgICBOYW1lICAgICAgICAgPT4gJENvbmZpZ0l0ZW1OYW1lLAogICAgRGVmaW5pdGlvbklEID0+IDEsCiAgICBEZXBsU3RhdGVJRCAgPT4gJFByb2R1Y3Rpb25EZXBsU3RhdGVJRCwKICAgIEluY2lTdGF0ZUlEICA9PiAxLAogICAgVXNlcklEICAgICAgID0+IDEsCiAgICBDb25maWdJdGVtSUQgPT4gJENvbmZpZ0l0ZW1JRCwKKTsKCiMgYWRkIHRoZSBuZXcgZHVwbGljYXRlIENvbmZpZ0l0ZW0KJENvbmZpZ0l0ZW1JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5Db25maWdJdGVtQWRkKAogICAgQ2xhc3NJRCA9PiAkSGFyZHdhcmVDb25maWdJdGVtSUQsCiAgICBVc2VySUQgID0+IDEsCik7CnB1c2ggQENvbmZpZ0l0ZW1JRCwgJENvbmZpZ0l0ZW1JRDsKCiRWZXJzaW9uSUQgPSAkQ29uZmlnSXRlbU9iamVjdC0+VmVyc2lvbkFkZCgKICAgIE5hbWUgICAgICAgICA9PiAkQ29uZmlnSXRlbU5hbWUsCiAgICBEZWZpbml0aW9uSUQgPT4gMSwKICAgIERlcGxTdGF0ZUlEICA9PiAkUHJvZHVjdGlvbkRlcGxTdGF0ZUlELAogICAgSW5jaVN0YXRlSUQgID0+IDEsCiAgICBVc2VySUQgICAgICAgPT4gMSwKICAgIENvbmZpZ0l0ZW1JRCA9PiAkQ29uZmlnSXRlbUlELAopOwoKIyBhZGQgdGhlIG5ldyBkdXBsaWNhdGUgQ29uZmlnSXRlbSBpbiBTb2Z0d2FyZSBjYXRhbG9nIGNsYXNzCiMgZ2V0ICdTb2Z0d2FyZScgY2F0YWxvZyBjbGFzcyBJRHMKJENvbmZpZ0l0ZW1EYXRhUmVmID0gJEdlbmVyYWxDYXRhbG9nT2JqZWN0LT5JdGVtR2V0KAogICAgQ2xhc3MgPT4gJ0lUU006OkNvbmZpZ0l0ZW06OkNsYXNzJywKICAgIE5hbWUgID0+ICdTb2Z0d2FyZScsCik7Cm15ICRTb2Z0d2FyZUNvbmZpZ0l0ZW1JRCA9ICRDb25maWdJdGVtRGF0YVJlZi0+e0l0ZW1JRH07CgokQ29uZmlnSXRlbUlEID0gJENvbmZpZ0l0ZW1PYmplY3QtPkNvbmZpZ0l0ZW1BZGQoCiAgICBDbGFzc0lEID0+ICRTb2Z0d2FyZUNvbmZpZ0l0ZW1JRCwKICAgIFVzZXJJRCAgPT4gMSwKKTsKcHVzaCBAQ29uZmlnSXRlbUlELCAkQ29uZmlnSXRlbUlEOwoKJFZlcnNpb25JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5WZXJzaW9uQWRkKAogICAgTmFtZSAgICAgICAgID0+ICRDb25maWdJdGVtTmFtZSwKICAgIERlZmluaXRpb25JRCA9PiAxLAogICAgRGVwbFN0YXRlSUQgID0+ICRQcm9kdWN0aW9uRGVwbFN0YXRlSUQsCiAgICBJbmNpU3RhdGVJRCAgPT4gMSwKICAgIFVzZXJJRCAgICAgICA9PiAxLAogICAgQ29uZmlnSXRlbUlEID0+ICRDb25maWdJdGVtSUQsCik7CgojIGNoZWNrIGNvbW1hbmQgd2l0aCAtLWNsYXNzIEhhcmR3YXJlIG9wdGlvbnMKJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoICctLWNsYXNzJywgIkhhcmR3YXJlIiApOwoKJFNlbGYtPklzKAogICAgJEV4aXRDb2RlLAogICAgMCwKICAgICJPcHRpb24gJ2NsYXNzJyAoSGFyZHdhcmUpICIsCik7CgojIGNoZWNrIGNvbW1hbmQgd2l0aCAtLXNjb3BlIG9wdGlvbnMgKGludmFsaWQgc2NvcGUpCm15ICRSYW5kb21TY29wZSA9ICdzY29wZScgLiAkSGVscGVyT2JqZWN0LT5HZXRSYW5kb21JRCgpOwokRXhpdENvZGUgPSAkQ29tbWFuZE9iamVjdC0+RXhlY3V0ZSggJy0tc2NvcGUnLCAkUmFuZG9tU2NvcGUgKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDEsCiAgICAiT3B0aW9uICdzY29wZScgKGJ1dCBwcm92aWRlZCBpbnZhbGlkIHZhbHVlIGZvciBvcHRpb24gJy0tc2NvcGUnIC0gJFJhbmRvbVNjb3BlICkgIiwKKTsKCiMgY2hlY2sgY29tbWFuZCB3aXRoIC0tc2NvcGUgY2xhc3Mgb3B0aW9ucwokRXhpdENvZGUgPSAkQ29tbWFuZE9iamVjdC0+RXhlY3V0ZSggJy0tc2NvcGUnLCAnY2xhc3MnICk7CgokU2VsZi0+SXMoCiAgICAkRXhpdENvZGUsCiAgICAwLAogICAgIk9wdGlvbiAnc2NvcGUnIChjbGFzcykgIiwKKTsKCiMgY2hlY2sgY29tbWFuZCB3aXRoIC0tc2NvcGUgZ2xvYmFsIG9wdGlvbnMKJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoICctLXNjb3BlJywgJ2dsb2JhbCcgKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDAsCiAgICAiT3B0aW9uICdzY29wZScgKGdsb2JhbCkgIiwKKTsKCiMgY2hlY2sgY29tbWFuZCB3aXRoIC0tYWxsLXN0YXRlcyBvcHRpb25zCiRFeGl0Q29kZSA9ICRDb21tYW5kT2JqZWN0LT5FeGVjdXRlKCctLWFsbC1zdGF0ZXMnKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDAsCiAgICAiT3B0aW9uICdzY29wZScgKGFsbC1zdGF0ZXMpICIsCik7CgojIGNsZWFudXAgaXMgZG9uZSBieSBSZXN0b3JlRGF0YWJzZQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgKHF3KCRTZWxmKSk7CgojIGdldCBjb21tYW5kIG9iamVjdApteSAkQ29tbWFuZE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpDb25zb2xlOjpDb21tYW5kOjpBZG1pbjo6SVRTTTo6Q29uZmlnaXRlbTo6TGlzdER1cGxpY2F0ZXMnKTsKCiRLZXJuZWw6Ok9NLT5PYmplY3RQYXJhbUFkZCgKICAgICdLZXJuZWw6OlN5c3RlbTo6VW5pdFRlc3Q6OkhlbHBlcicgPT4gewogICAgICAgIFJlc3RvcmVEYXRhYmFzZSA9PiAxLAogICAgfSwKKTsKbXkgJEhlbHBlck9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyk7CgojIGNoZWNrIGNvbW1hbmQgd2l0aG91dCBhbnkgb3B0aW9ucwpteSAkRXhpdENvZGUgPSAkQ29tbWFuZE9iamVjdC0+RXhlY3V0ZSgpOwoKJFNlbGYtPklzKAogICAgJEV4aXRDb2RlLAogICAgMCwKICAgICJObyBvcHRpb25zIC0gbGlzdCBhbGwgY29uZmlnIGl0ZW1zIGluIHByb2R1Y3RpdmUgc3RhdGVzIiwKKTsKCiMgY2hlY2sgY29tbWFuZCB3aXRoIC0tY2xhc3Mgb3B0aW9ucyAoaW52YWxpZCBjbGFzcykKbXkgJFJhbmRvbUNsYXNzID0gJ05vbkV4aXN0aW5nQ2xhc3MnIC4gJEhlbHBlck9iamVjdC0+R2V0UmFuZG9tSUQoKTsKJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoICctLWNsYXNzJywgJFJhbmRvbUNsYXNzICk7CgokU2VsZi0+SXMoCiAgICAkRXhpdENvZGUsCiAgICAxLAogICAgIk9wdGlvbiAnY2xhc3MnIChidXQgY2xhc3MgJFJhbmRvbUNsYXNzIGRvZXNuJ3QgZXhpc3QpICIsCik7CgojIGFkZCB0ZXN0IGNvbmZpZyBpdGVtCm15ICRHZW5lcmFsQ2F0YWxvZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpOwoKIyBnZXQgJ0hhcmR3YXJlJyBjYXRhbG9nIGNsYXNzIElEcwpteSAkQ29uZmlnSXRlbURhdGFSZWYgPSAkR2VuZXJhbENhdGFsb2dPYmplY3QtPkl0ZW1HZXQoCiAgICBDbGFzcyA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgPT4gJ0hhcmR3YXJlJywKKTsKbXkgJEhhcmR3YXJlQ29uZmlnSXRlbUlEID0gJENvbmZpZ0l0ZW1EYXRhUmVmLT57SXRlbUlEfTsKCiMgZ2V0ICdQcm9kdWN0aW9uJyBkZXBsb3ltZW50IHN0YXRlIElEcwpteSAkUHJvZHVjdGlvbkRlcGxTdGF0ZURhdGFSZWYgPSAkR2VuZXJhbENhdGFsb2dPYmplY3QtPkl0ZW1HZXQoCiAgICBDbGFzcyA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6RGVwbG95bWVudFN0YXRlJywKICAgIE5hbWUgID0+ICdQcm9kdWN0aW9uJywKKTsKbXkgJFByb2R1Y3Rpb25EZXBsU3RhdGVJRCA9ICRQcm9kdWN0aW9uRGVwbFN0YXRlRGF0YVJlZi0+e0l0ZW1JRH07CgojIGdldCBDb25maWdJdGVtIG9iamVjdApteSAkQ29uZmlnSXRlbU9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbScpOwoKIyBjcmVhdGUgQ29uZmlnSXRlbSBudW1iZXIKbXkgJENvbmZpZ0l0ZW1OdW1iZXIgPSAkQ29uZmlnSXRlbU9iamVjdC0+Q29uZmlnSXRlbU51bWJlckNyZWF0ZSgKICAgIFR5cGUgICAgPT4gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpDb25maWcnKS0+R2V0KCdJVFNNQ29uZmlnSXRlbTo6TnVtYmVyR2VuZXJhdG9yJyksCiAgICBDbGFzc0lEID0+ICRIYXJkd2FyZUNvbmZpZ0l0ZW1JRCwKKTsKCm15IEBDb25maWdJdGVtSUQ7CgojIGFkZCB0aGUgbmV3IENvbmZpZ0l0ZW0KbXkgJENvbmZpZ0l0ZW1JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5Db25maWdJdGVtQWRkKAogICAgTnVtYmVyICA9PiAkQ29uZmlnSXRlbU51bWJlciwKICAgIENsYXNzSUQgPT4gJEhhcmR3YXJlQ29uZmlnSXRlbUlELAogICAgVXNlcklEICA9PiAxLAopOwpwdXNoIEBDb25maWdJdGVtSUQsICRDb25maWdJdGVtSUQ7CgpteSAkQ29uZmlnSXRlbU5hbWUgPSAnVGVzdENvbmZpZ0l0ZW0nIC4gJEhlbHBlck9iamVjdC0+R2V0UmFuZG9tSUQoKTsKbXkgJFZlcnNpb25JRCAgICAgID0gJENvbmZpZ0l0ZW1PYmplY3QtPlZlcnNpb25BZGQoCiAgICBOYW1lICAgICAgICAgPT4gJENvbmZpZ0l0ZW1OYW1lLAogICAgRGVmaW5pdGlvbklEID0+IDEsCiAgICBEZXBsU3RhdGVJRCAgPT4gJFByb2R1Y3Rpb25EZXBsU3RhdGVJRCwKICAgIEluY2lTdGF0ZUlEICA9PiAxLAogICAgVXNlcklEICAgICAgID0+IDEsCiAgICBDb25maWdJdGVtSUQgPT4gJENvbmZpZ0l0ZW1JRCwKKTsKCiMgYWRkIHRoZSBuZXcgZHVwbGljYXRlIENvbmZpZ0l0ZW0KJENvbmZpZ0l0ZW1JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5Db25maWdJdGVtQWRkKAogICAgQ2xhc3NJRCA9PiAkSGFyZHdhcmVDb25maWdJdGVtSUQsCiAgICBVc2VySUQgID0+IDEsCik7CnB1c2ggQENvbmZpZ0l0ZW1JRCwgJENvbmZpZ0l0ZW1JRDsKCiRWZXJzaW9uSUQgPSAkQ29uZmlnSXRlbU9iamVjdC0+VmVyc2lvbkFkZCgKICAgIE5hbWUgICAgICAgICA9PiAkQ29uZmlnSXRlbU5hbWUsCiAgICBEZWZpbml0aW9uSUQgPT4gMSwKICAgIERlcGxTdGF0ZUlEICA9PiAkUHJvZHVjdGlvbkRlcGxTdGF0ZUlELAogICAgSW5jaVN0YXRlSUQgID0+IDEsCiAgICBVc2VySUQgICAgICAgPT4gMSwKICAgIENvbmZpZ0l0ZW1JRCA9PiAkQ29uZmlnSXRlbUlELAopOwoKIyBhZGQgdGhlIG5ldyBkdXBsaWNhdGUgQ29uZmlnSXRlbSBpbiBTb2Z0d2FyZSBjYXRhbG9nIGNsYXNzCiMgZ2V0ICdTb2Z0d2FyZScgY2F0YWxvZyBjbGFzcyBJRHMKJENvbmZpZ0l0ZW1EYXRhUmVmID0gJEdlbmVyYWxDYXRhbG9nT2JqZWN0LT5JdGVtR2V0KAogICAgQ2xhc3MgPT4gJ0lUU006OkNvbmZpZ0l0ZW06OkNsYXNzJywKICAgIE5hbWUgID0+ICdTb2Z0d2FyZScsCik7Cm15ICRTb2Z0d2FyZUNvbmZpZ0l0ZW1JRCA9ICRDb25maWdJdGVtRGF0YVJlZi0+e0l0ZW1JRH07CgokQ29uZmlnSXRlbUlEID0gJENvbmZpZ0l0ZW1PYmplY3QtPkNvbmZpZ0l0ZW1BZGQoCiAgICBDbGFzc0lEID0+ICRTb2Z0d2FyZUNvbmZpZ0l0ZW1JRCwKICAgIFVzZXJJRCAgPT4gMSwKKTsKcHVzaCBAQ29uZmlnSXRlbUlELCAkQ29uZmlnSXRlbUlEOwoKJFZlcnNpb25JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5WZXJzaW9uQWRkKAogICAgTmFtZSAgICAgICAgID0+ICRDb25maWdJdGVtTmFtZSwKICAgIERlZmluaXRpb25JRCA9PiAxLAogICAgRGVwbFN0YXRlSUQgID0+ICRQcm9kdWN0aW9uRGVwbFN0YXRlSUQsCiAgICBJbmNpU3RhdGVJRCAgPT4gMSwKICAgIFVzZXJJRCAgICAgICA9PiAxLAogICAgQ29uZmlnSXRlbUlEID0+ICRDb25maWdJdGVtSUQsCik7CgojIGNoZWNrIGNvbW1hbmQgd2l0aCAtLWNsYXNzIEhhcmR3YXJlIG9wdGlvbnMKJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoICctLWNsYXNzJywgIkhhcmR3YXJlIiApOwoKJFNlbGYtPklzKAogICAgJEV4aXRDb2RlLAogICAgMCwKICAgICJPcHRpb24gJ2NsYXNzJyAoSGFyZHdhcmUpICIsCik7CgojIGNoZWNrIGNvbW1hbmQgd2l0aCAtLXNjb3BlIG9wdGlvbnMgKGludmFsaWQgc2NvcGUpCm15ICRSYW5kb21TY29wZSA9ICdzY29wZScgLiAkSGVscGVyT2JqZWN0LT5HZXRSYW5kb21JRCgpOwokRXhpdENvZGUgPSAkQ29tbWFuZE9iamVjdC0+RXhlY3V0ZSggJy0tc2NvcGUnLCAkUmFuZG9tU2NvcGUgKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDEsCiAgICAiT3B0aW9uICdzY29wZScgKGJ1dCBwcm92aWRlZCBpbnZhbGlkIHZhbHVlIGZvciBvcHRpb24gJy0tc2NvcGUnIC0gJFJhbmRvbVNjb3BlICkgIiwKKTsKCiMgY2hlY2sgY29tbWFuZCB3aXRoIC0tc2NvcGUgY2xhc3Mgb3B0aW9ucwokRXhpdENvZGUgPSAkQ29tbWFuZE9iamVjdC0+RXhlY3V0ZSggJy0tc2NvcGUnLCAnY2xhc3MnICk7CgokU2VsZi0+SXMoCiAgICAkRXhpdENvZGUsCiAgICAwLAogICAgIk9wdGlvbiAnc2NvcGUnIChjbGFzcykgIiwKKTsKCiMgY2hlY2sgY29tbWFuZCB3aXRoIC0tc2NvcGUgZ2xvYmFsIG9wdGlvbnMKJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoICctLXNjb3BlJywgJ2dsb2JhbCcgKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDAsCiAgICAiT3B0aW9uICdzY29wZScgKGdsb2JhbCkgIiwKKTsKCiMgY2hlY2sgY29tbWFuZCB3aXRoIC0tYWxsLXN0YXRlcyBvcHRpb25zCiRFeGl0Q29kZSA9ICRDb21tYW5kT2JqZWN0LT5FeGVjdXRlKCctLWFsbC1zdGF0ZXMnKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDAsCiAgICAiT3B0aW9uICdzY29wZScgKGFsbC1zdGF0ZXMpICIsCik7CgojIGNsZWFudXAgaXMgZG9uZSBieSBSZXN0b3JlRGF0YWJzZQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgKHF3KCRTZWxmKSk7CgojIGdldCBjb21tYW5kIG9iamVjdApteSAkQ29tbWFuZE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpDb25zb2xlOjpDb21tYW5kOjpBZG1pbjo6SVRTTTo6Q29uZmlnaXRlbTo6TGlzdER1cGxpY2F0ZXMnKTsKCiRLZXJuZWw6Ok9NLT5PYmplY3RQYXJhbUFkZCgKICAgICdLZXJuZWw6OlN5c3RlbTo6VW5pdFRlc3Q6OkhlbHBlcicgPT4gewogICAgICAgIFJlc3RvcmVEYXRhYmFzZSA9PiAxLAogICAgfSwKKTsKbXkgJEhlbHBlck9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyk7CgojIGNoZWNrIGNvbW1hbmQgd2l0aG91dCBhbnkgb3B0aW9ucwpteSAkRXhpdENvZGUgPSAkQ29tbWFuZE9iamVjdC0+RXhlY3V0ZSgpOwoKJFNlbGYtPklzKAogICAgJEV4aXRDb2RlLAogICAgMCwKICAgICJObyBvcHRpb25zIC0gbGlzdCBhbGwgY29uZmlnIGl0ZW1zIGluIHByb2R1Y3RpdmUgc3RhdGVzIiwKKTsKCiMgY2hlY2sgY29tbWFuZCB3aXRoIC0tY2xhc3Mgb3B0aW9ucyAoaW52YWxpZCBjbGFzcykKbXkgJFJhbmRvbUNsYXNzID0gJ05vbkV4aXN0aW5nQ2xhc3MnIC4gJEhlbHBlck9iamVjdC0+R2V0UmFuZG9tSUQoKTsKJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoICctLWNsYXNzJywgJFJhbmRvbUNsYXNzICk7CgokU2VsZi0+SXMoCiAgICAkRXhpdENvZGUsCiAgICAxLAogICAgIk9wdGlvbiAnY2xhc3MnIChidXQgY2xhc3MgJFJhbmRvbUNsYXNzIGRvZXNuJ3QgZXhpc3QpICIsCik7CgojIGFkZCB0ZXN0IGNvbmZpZyBpdGVtCm15ICRHZW5lcmFsQ2F0YWxvZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpOwoKIyBnZXQgJ0hhcmR3YXJlJyBjYXRhbG9nIGNsYXNzIElEcwpteSAkQ29uZmlnSXRlbURhdGFSZWYgPSAkR2VuZXJhbENhdGFsb2dPYmplY3QtPkl0ZW1HZXQoCiAgICBDbGFzcyA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgPT4gJ0hhcmR3YXJlJywKKTsKbXkgJEhhcmR3YXJlQ29uZmlnSXRlbUlEID0gJENvbmZpZ0l0ZW1EYXRhUmVmLT57SXRlbUlEfTsKCiMgZ2V0ICdQcm9kdWN0aW9uJyBkZXBsb3ltZW50IHN0YXRlIElEcwpteSAkUHJvZHVjdGlvbkRlcGxTdGF0ZURhdGFSZWYgPSAkR2VuZXJhbENhdGFsb2dPYmplY3QtPkl0ZW1HZXQoCiAgICBDbGFzcyA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6RGVwbG95bWVudFN0YXRlJywKICAgIE5hbWUgID0+ICdQcm9kdWN0aW9uJywKKTsKbXkgJFByb2R1Y3Rpb25EZXBsU3RhdGVJRCA9ICRQcm9kdWN0aW9uRGVwbFN0YXRlRGF0YVJlZi0+e0l0ZW1JRH07CgojIGdldCBDb25maWdJdGVtIG9iamVjdApteSAkQ29uZmlnSXRlbU9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbScpOwoKIyBjcmVhdGUgQ29uZmlnSXRlbSBudW1iZXIKbXkgJENvbmZpZ0l0ZW1OdW1iZXIgPSAkQ29uZmlnSXRlbU9iamVjdC0+Q29uZmlnSXRlbU51bWJlckNyZWF0ZSgKICAgIFR5cGUgICAgPT4gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpDb25maWcnKS0+R2V0KCdJVFNNQ29uZmlnSXRlbTo6TnVtYmVyR2VuZXJhdG9yJyksCiAgICBDbGFzc0lEID0+ICRIYXJkd2FyZUNvbmZpZ0l0ZW1JRCwKKTsKCm15IEBDb25maWdJdGVtSUQ7CgojIGFkZCB0aGUgbmV3IENvbmZpZ0l0ZW0KbXkgJENvbmZpZ0l0ZW1JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5Db25maWdJdGVtQWRkKAogICAgTnVtYmVyICA9PiAkQ29uZmlnSXRlbU51bWJlciwKICAgIENsYXNzSUQgPT4gJEhhcmR3YXJlQ29uZmlnSXRlbUlELAogICAgVXNlcklEICA9PiAxLAopOwpwdXNoIEBDb25maWdJdGVtSUQsICRDb25maWdJdGVtSUQ7CgpteSAkQ29uZmlnSXRlbU5hbWUgPSAnVGVzdENvbmZpZ0l0ZW0nIC4gJEhlbHBlck9iamVjdC0+R2V0UmFuZG9tSUQoKTsKbXkgJFZlcnNpb25JRCAgICAgID0gJENvbmZpZ0l0ZW1PYmplY3QtPlZlcnNpb25BZGQoCiAgICBOYW1lICAgICAgICAgPT4gJENvbmZpZ0l0ZW1OYW1lLAogICAgRGVmaW5pdGlvbklEID0+IDEsCiAgICBEZXBsU3RhdGVJRCAgPT4gJFByb2R1Y3Rpb25EZXBsU3RhdGVJRCwKICAgIEluY2lTdGF0ZUlEICA9PiAxLAogICAgVXNlcklEICAgICAgID0+IDEsCiAgICBDb25maWdJdGVtSUQgPT4gJENvbmZpZ0l0ZW1JRCwKKTsKCiMgYWRkIHRoZSBuZXcgZHVwbGljYXRlIENvbmZpZ0l0ZW0KJENvbmZpZ0l0ZW1JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5Db25maWdJdGVtQWRkKAogICAgQ2xhc3NJRCA9PiAkSGFyZHdhcmVDb25maWdJdGVtSUQsCiAgICBVc2VySUQgID0+IDEsCik7CnB1c2ggQENvbmZpZ0l0ZW1JRCwgJENvbmZpZ0l0ZW1JRDsKCiRWZXJzaW9uSUQgPSAkQ29uZmlnSXRlbU9iamVjdC0+VmVyc2lvbkFkZCgKICAgIE5hbWUgICAgICAgICA9PiAkQ29uZmlnSXRlbU5hbWUsCiAgICBEZWZpbml0aW9uSUQgPT4gMSwKICAgIERlcGxTdGF0ZUlEICA9PiAkUHJvZHVjdGlvbkRlcGxTdGF0ZUlELAogICAgSW5jaVN0YXRlSUQgID0+IDEsCiAgICBVc2VySUQgICAgICAgPT4gMSwKICAgIENvbmZpZ0l0ZW1JRCA9PiAkQ29uZmlnSXRlbUlELAopOwoKIyBhZGQgdGhlIG5ldyBkdXBsaWNhdGUgQ29uZmlnSXRlbSBpbiBTb2Z0d2FyZSBjYXRhbG9nIGNsYXNzCiMgZ2V0ICdTb2Z0d2FyZScgY2F0YWxvZyBjbGFzcyBJRHMKJENvbmZpZ0l0ZW1EYXRhUmVmID0gJEdlbmVyYWxDYXRhbG9nT2JqZWN0LT5JdGVtR2V0KAogICAgQ2xhc3MgPT4gJ0lUU006OkNvbmZpZ0l0ZW06OkNsYXNzJywKICAgIE5hbWUgID0+ICdTb2Z0d2FyZScsCik7Cm15ICRTb2Z0d2FyZUNvbmZpZ0l0ZW1JRCA9ICRDb25maWdJdGVtRGF0YVJlZi0+e0l0ZW1JRH07CgokQ29uZmlnSXRlbUlEID0gJENvbmZpZ0l0ZW1PYmplY3QtPkNvbmZpZ0l0ZW1BZGQoCiAgICBDbGFzc0lEID0+ICRTb2Z0d2FyZUNvbmZpZ0l0ZW1JRCwKICAgIFVzZXJJRCAgPT4gMSwKKTsKcHVzaCBAQ29uZmlnSXRlbUlELCAkQ29uZmlnSXRlbUlEOwoKJFZlcnNpb25JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5WZXJzaW9uQWRkKAogICAgTmFtZSAgICAgICAgID0+ICRDb25maWdJdGVtTmFtZSwKICAgIERlZmluaXRpb25JRCA9PiAxLAogICAgRGVwbFN0YXRlSUQgID0+ICRQcm9kdWN0aW9uRGVwbFN0YXRlSUQsCiAgICBJbmNpU3RhdGVJRCAgPT4gMSwKICAgIFVzZXJJRCAgICAgICA9PiAxLAogICAgQ29uZmlnSXRlbUlEID0+ICRDb25maWdJdGVtSUQsCik7CgojIGNoZWNrIGNvbW1hbmQgd2l0aCAtLWNsYXNzIEhhcmR3YXJlIG9wdGlvbnMKJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoICctLWNsYXNzJywgIkhhcmR3YXJlIiApOwoKJFNlbGYtPklzKAogICAgJEV4aXRDb2RlLAogICAgMCwKICAgICJPcHRpb24gJ2NsYXNzJyAoSGFyZHdhcmUpICIsCik7CgojIGNoZWNrIGNvbW1hbmQgd2l0aCAtLXNjb3BlIG9wdGlvbnMgKGludmFsaWQgc2NvcGUpCm15ICRSYW5kb21TY29wZSA9ICdzY29wZScgLiAkSGVscGVyT2JqZWN0LT5HZXRSYW5kb21JRCgpOwokRXhpdENvZGUgPSAkQ29tbWFuZE9iamVjdC0+RXhlY3V0ZSggJy0tc2NvcGUnLCAkUmFuZG9tU2NvcGUgKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDEsCiAgICAiT3B0aW9uICdzY29wZScgKGJ1dCBwcm92aWRlZCBpbnZhbGlkIHZhbHVlIGZvciBvcHRpb24gJy0tc2NvcGUnIC0gJFJhbmRvbVNjb3BlICkgIiwKKTsKCiMgY2hlY2sgY29tbWFuZCB3aXRoIC0tc2NvcGUgY2xhc3Mgb3B0aW9ucwokRXhpdENvZGUgPSAkQ29tbWFuZE9iamVjdC0+RXhlY3V0ZSggJy0tc2NvcGUnLCAnY2xhc3MnICk7CgokU2VsZi0+SXMoCiAgICAkRXhpdENvZGUsCiAgICAwLAogICAgIk9wdGlvbiAnc2NvcGUnIChjbGFzcykgIiwKKTsKCiMgY2hlY2sgY29tbWFuZCB3aXRoIC0tc2NvcGUgZ2xvYmFsIG9wdGlvbnMKJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoICctLXNjb3BlJywgJ2dsb2JhbCcgKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDAsCiAgICAiT3B0aW9uICdzY29wZScgKGdsb2JhbCkgIiwKKTsKCiMgY2hlY2sgY29tbWFuZCB3aXRoIC0tYWxsLXN0YXRlcyBvcHRpb25zCiRFeGl0Q29kZSA9ICRDb21tYW5kT2JqZWN0LT5FeGVjdXRlKCctLWFsbC1zdGF0ZXMnKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDAsCiAgICAiT3B0aW9uICdzY29wZScgKGFsbC1zdGF0ZXMpICIsCik7CgojIGNsZWFudXAgaXMgZG9uZSBieSBSZXN0b3JlRGF0YWJzZQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgKHF3KCRTZWxmKSk7CgpteSAkQ29tbWFuZE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpDb25zb2xlOjpDb21tYW5kOjpBZG1pbjo6SVRTTTo6SW5jaWRlbnRTdGF0ZTo6UmVjYWxjdWxhdGUnKTsKCm15ICRFeGl0Q29kZSA9ICRDb21tYW5kT2JqZWN0LT5FeGVjdXRlKCk7CgokU2VsZi0+SXMoCiAgICAkRXhpdENvZGUsCiAgICAwLAogICAgIkFkbWluOjpJVFNNOjpJbmNpZGVudFN0YXRlOjpSZWNhbGN1bGF0ZSBleGl0IGNvZGUiLAopOwoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgKHF3KCRTZWxmKSk7CgpteSAkQ29tbWFuZE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpDb25zb2xlOjpDb21tYW5kOjpBZG1pbjo6SVRTTTo6SW5jaWRlbnRTdGF0ZTo6UmVjYWxjdWxhdGUnKTsKCm15ICRFeGl0Q29kZSA9ICRDb21tYW5kT2JqZWN0LT5FeGVjdXRlKCk7CgokU2VsZi0+SXMoCiAgICAkRXhpdENvZGUsCiAgICAwLAogICAgIkFkbWluOjpJVFNNOjpJbmNpZGVudFN0YXRlOjpSZWNhbGN1bGF0ZSBleGl0IGNvZGUiLAopOwoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgKHF3KCRTZWxmKSk7CgpteSAkQ29tbWFuZE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpDb25zb2xlOjpDb21tYW5kOjpBZG1pbjo6SVRTTTo6SW5jaWRlbnRTdGF0ZTo6UmVjYWxjdWxhdGUnKTsKCm15ICRFeGl0Q29kZSA9ICRDb21tYW5kT2JqZWN0LT5FeGVjdXRlKCk7CgokU2VsZi0+SXMoCiAgICAkRXhpdENvZGUsCiAgICAwLAogICAgIkFkbWluOjpJVFNNOjpJbmNpZGVudFN0YXRlOjpSZWNhbGN1bGF0ZSBleGl0IGNvZGUiLAopOwoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgKHF3KCRTZWxmKSk7Cgp1c2UgS2VybmVsOjpMYW5ndWFnZSBxdyhUcmFuc2xhdGFibGUpOwoKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCm15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAndXNlcnMnIF0sCik7Cm15ICRVc2VySUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpLT5Vc2VyTG9va3VwKCBVc2VyTG9naW4gPT4gJFRlc3RVc2VyTG9naW4gKTsKCm15ICRSYW5kb21JRCA9ICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CgpteSAkQ2xhc3NJRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtQWRkKAogICAgQ2xhc3MgICA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgICA9PiAkUmFuZG9tSUQsCiAgICBWYWxpZElEID0+IDEsCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CiRTZWxmLT5UcnVlKAogICAgJENsYXNzSUQsCiAgICAiQ2xhc3MgYWRkZWQgdG8gR2VuZXJhbENhdGFsb2ciLAopOwoKbXkgJFZlcnNpb24gPSAxOwoKbXkgJFBlcmxEZWZpbml0aW9uID0gPDwgJ0VPRic7ClsKICAgIHsKICAgICAgICBLZXkgICAgICAgID0+ICdWZW5kb3InLAogICAgICAgIE5hbWUgICAgICAgPT4gVHJhbnNsYXRhYmxlKCdWZW5kb3InKSwKICAgICAgICBTZWFyY2hhYmxlID0+IDEsCiAgICAgICAgSW5wdXQgICAgICA9PiB7CiAgICAgICAgICAgIFR5cGUgICAgICA9PiAnVGV4dCcsCiAgICAgICAgICAgIFNpemUgICAgICA9PiA1MCwKICAgICAgICAgICAgTWF4TGVuZ3RoID0+IDUwLAogICAgICAgIH0sCiAgICAgICAgQ291bnRNaW4gICAgID0+IDEsCiAgICAgICAgQ291bnRNYXggICAgID0+IDEsCiAgICAgICAgQ291bnREZWZhdWx0ID0+IDEsCiAgICB9LApdCkVPRgoKbXkgJERCT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkRCJyk7CgpyZXR1cm4gaWYgISREQk9iamVjdC0+RG8oCiAgICBTUUwgPT4gJwogICAgICAgIElOU0VSVCBJTlRPIGNvbmZpZ2l0ZW1fZGVmaW5pdGlvbgogICAgICAgICAgICAoY2xhc3NfaWQsIGNvbmZpZ2l0ZW1fZGVmaW5pdGlvbiwgdmVyc2lvbiwgY3JlYXRlX3RpbWUsIGNyZWF0ZV9ieSkKICAgICAgICBWQUxVRVMKICAgICAgICAgICAgKD8sID8sID8sIGN1cnJlbnRfdGltZXN0YW1wLCA/KScsCiAgICBCaW5kID0+IFsgXCRDbGFzc0lELCBcJFBlcmxEZWZpbml0aW9uLCBcJFZlcnNpb24sIFwkVXNlcklEIF0sCik7CgpyZXR1cm4gaWYgISREQk9iamVjdC0+UHJlcGFyZSgKICAgIFNRTCA9PiAnCiAgICAgICAgU0VMRUNUIGlkCiAgICAgICAgRlJPTSBjb25maWdpdGVtX2RlZmluaXRpb24KICAgICAgICBXSEVSRSBjbGFzc19pZCA9ID8KICAgICAgICAgICAgQU5EIHZlcnNpb24gPSA/CiAgICAgICAgT1JERVIgQlkgdmVyc2lvbiBERVNDJywKICAgIEJpbmQgID0+IFsgXCRDbGFzc0lELCBcJFZlcnNpb24gXSwKICAgIExpbWl0ID0+IDEsCik7CgpteSAkRGVmaW5pdGlvbklEOwp3aGlsZSAoIG15IEBSb3cgPSAkREJPYmplY3QtPkZldGNocm93QXJyYXkoKSApIHsKICAgICREZWZpbml0aW9uSUQgPSAkUm93WzBdOwp9CgpteSAkR2V0RGVmaW5pdGlvbiA9IHN1YiB7CiAgICBteSAlUGFyYW0gPSBAXzsKCiAgICByZXR1cm4gaWYgISREQk9iamVjdC0+UHJlcGFyZSgKICAgICAgICBTUUwgPT4gJwogICAgICAgICAgICBTRUxFQ1QgY29uZmlnaXRlbV9kZWZpbml0aW9uCiAgICAgICAgICAgIEZST00gY29uZmlnaXRlbV9kZWZpbml0aW9uCiAgICAgICAgICAgIFdIRVJFIGlkID0gPwogICAgICAgICAgICBPUkRFUiBCWSB2ZXJzaW9uIERFU0MnLAogICAgICAgIEJpbmQgID0+IFsgXCRQYXJhbXtEZWZpbml0aW9uSUR9IF0sCiAgICAgICAgTGltaXQgPT4gMSwKICAgICk7CgogICAgbXkgJERlZmluaXRpb247CiAgICB3aGlsZSAoIG15IEBSb3cgPSAkREJPYmplY3QtPkZldGNocm93QXJyYXkoKSApIHsKICAgICAgICAkRGVmaW5pdGlvbiA9ICRSb3dbMF07CiAgICB9CiAgICByZXR1cm4gJERlZmluaXRpb247Cn07CgpteSAkRGVmaW5pdGlvbiA9ICRHZXREZWZpbml0aW9uLT4oIERlZmluaXRpb25JRCA9PiAkRGVmaW5pdGlvbklEICk7CgokU2VsZi0+SXNEZWVwbHkoCiAgICAkRGVmaW5pdGlvbiwKICAgICRQZXJsRGVmaW5pdGlvbiwKICAgICJEZWZpbml0aW9uIGJlZm9yZSBydW4gKFBlcmwpIiwKKTsKCm15ICRDb21tYW5kT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkNvbnNvbGU6OkNvbW1hbmQ6Ok1haW50OjpJVFNNOjpDb25maWdpdGVtOjpEZWZpbml0aW9uUGVybDJZQU1MJyk7CgpteSAoICRSZXN1bHQsICRFeGl0Q29kZSApOwoKewogICAgbG9jYWwgKlNURE9VVDsKICAgIG9wZW4gU1RET1VULCAnPjp1dGY4JywgXCRSZXN1bHQ7ICAgICMjIG5vIGNyaXRpYwogICAgJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoKTsKfQoKJFNlbGYtPklzKAogICAgJEV4aXRDb2RlLAogICAgMCwKICAgICJDb21tYW5kIGV4aXQgY29kZSIsCik7CgokRGVmaW5pdGlvbiA9ICRHZXREZWZpbml0aW9uLT4oIERlZmluaXRpb25JRCA9PiAkRGVmaW5pdGlvbklEICk7CgpteSAkWUFNTERlZmluaXRpb24gPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6WUFNTCcpLT5EdW1wKAogICAgRGF0YSA9PiBldmFsICRQZXJsRGVmaW5pdGlvbiwgICAgIyMgbm8gY3JpdGljCik7CgokU2VsZi0+SXNEZWVwbHkoCiAgICAkRGVmaW5pdGlvbiwKICAgICRZQU1MRGVmaW5pdGlvbiwKICAgICJEZWZpbml0aW9uIGJlZm9yZSBydW4gKFlBTUwpIiwKKTsKCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgKHF3KCRTZWxmKSk7Cgp1c2UgS2VybmVsOjpMYW5ndWFnZSBxdyhUcmFuc2xhdGFibGUpOwoKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCm15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAndXNlcnMnIF0sCik7Cm15ICRVc2VySUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpLT5Vc2VyTG9va3VwKCBVc2VyTG9naW4gPT4gJFRlc3RVc2VyTG9naW4gKTsKCm15ICRSYW5kb21JRCA9ICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CgpteSAkQ2xhc3NJRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtQWRkKAogICAgQ2xhc3MgICA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgICA9PiAkUmFuZG9tSUQsCiAgICBWYWxpZElEID0+IDEsCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CiRTZWxmLT5UcnVlKAogICAgJENsYXNzSUQsCiAgICAiQ2xhc3MgYWRkZWQgdG8gR2VuZXJhbENhdGFsb2ciLAopOwoKbXkgJFZlcnNpb24gPSAxOwoKbXkgJFBlcmxEZWZpbml0aW9uID0gPDwgJ0VPRic7ClsKICAgIHsKICAgICAgICBLZXkgICAgICAgID0+ICdWZW5kb3InLAogICAgICAgIE5hbWUgICAgICAgPT4gVHJhbnNsYXRhYmxlKCdWZW5kb3InKSwKICAgICAgICBTZWFyY2hhYmxlID0+IDEsCiAgICAgICAgSW5wdXQgICAgICA9PiB7CiAgICAgICAgICAgIFR5cGUgICAgICA9PiAnVGV4dCcsCiAgICAgICAgICAgIFNpemUgICAgICA9PiA1MCwKICAgICAgICAgICAgTWF4TGVuZ3RoID0+IDUwLAogICAgICAgIH0sCiAgICAgICAgQ291bnRNaW4gICAgID0+IDEsCiAgICAgICAgQ291bnRNYXggICAgID0+IDEsCiAgICAgICAgQ291bnREZWZhdWx0ID0+IDEsCiAgICB9LApdCkVPRgoKbXkgJERCT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkRCJyk7CgpyZXR1cm4gaWYgISREQk9iamVjdC0+RG8oCiAgICBTUUwgPT4gJwogICAgICAgIElOU0VSVCBJTlRPIGNvbmZpZ2l0ZW1fZGVmaW5pdGlvbgogICAgICAgICAgICAoY2xhc3NfaWQsIGNvbmZpZ2l0ZW1fZGVmaW5pdGlvbiwgdmVyc2lvbiwgY3JlYXRlX3RpbWUsIGNyZWF0ZV9ieSkKICAgICAgICBWQUxVRVMKICAgICAgICAgICAgKD8sID8sID8sIGN1cnJlbnRfdGltZXN0YW1wLCA/KScsCiAgICBCaW5kID0+IFsgXCRDbGFzc0lELCBcJFBlcmxEZWZpbml0aW9uLCBcJFZlcnNpb24sIFwkVXNlcklEIF0sCik7CgpyZXR1cm4gaWYgISREQk9iamVjdC0+UHJlcGFyZSgKICAgIFNRTCA9PiAnCiAgICAgICAgU0VMRUNUIGlkCiAgICAgICAgRlJPTSBjb25maWdpdGVtX2RlZmluaXRpb24KICAgICAgICBXSEVSRSBjbGFzc19pZCA9ID8KICAgICAgICAgICAgQU5EIHZlcnNpb24gPSA/CiAgICAgICAgT1JERVIgQlkgdmVyc2lvbiBERVNDJywKICAgIEJpbmQgID0+IFsgXCRDbGFzc0lELCBcJFZlcnNpb24gXSwKICAgIExpbWl0ID0+IDEsCik7CgpteSAkRGVmaW5pdGlvbklEOwp3aGlsZSAoIG15IEBSb3cgPSAkREJPYmplY3QtPkZldGNocm93QXJyYXkoKSApIHsKICAgICREZWZpbml0aW9uSUQgPSAkUm93WzBdOwp9CgpteSAkR2V0RGVmaW5pdGlvbiA9IHN1YiB7CiAgICBteSAlUGFyYW0gPSBAXzsKCiAgICByZXR1cm4gaWYgISREQk9iamVjdC0+UHJlcGFyZSgKICAgICAgICBTUUwgPT4gJwogICAgICAgICAgICBTRUxFQ1QgY29uZmlnaXRlbV9kZWZpbml0aW9uCiAgICAgICAgICAgIEZST00gY29uZmlnaXRlbV9kZWZpbml0aW9uCiAgICAgICAgICAgIFdIRVJFIGlkID0gPwogICAgICAgICAgICBPUkRFUiBCWSB2ZXJzaW9uIERFU0MnLAogICAgICAgIEJpbmQgID0+IFsgXCRQYXJhbXtEZWZpbml0aW9uSUR9IF0sCiAgICAgICAgTGltaXQgPT4gMSwKICAgICk7CgogICAgbXkgJERlZmluaXRpb247CiAgICB3aGlsZSAoIG15IEBSb3cgPSAkREJPYmplY3QtPkZldGNocm93QXJyYXkoKSApIHsKICAgICAgICAkRGVmaW5pdGlvbiA9ICRSb3dbMF07CiAgICB9CiAgICByZXR1cm4gJERlZmluaXRpb247Cn07CgpteSAkRGVmaW5pdGlvbiA9ICRHZXREZWZpbml0aW9uLT4oIERlZmluaXRpb25JRCA9PiAkRGVmaW5pdGlvbklEICk7CgokU2VsZi0+SXNEZWVwbHkoCiAgICAkRGVmaW5pdGlvbiwKICAgICRQZXJsRGVmaW5pdGlvbiwKICAgICJEZWZpbml0aW9uIGJlZm9yZSBydW4gKFBlcmwpIiwKKTsKCm15ICRDb21tYW5kT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkNvbnNvbGU6OkNvbW1hbmQ6Ok1haW50OjpJVFNNOjpDb25maWdpdGVtOjpEZWZpbml0aW9uUGVybDJZQU1MJyk7CgpteSAoICRSZXN1bHQsICRFeGl0Q29kZSApOwoKewogICAgbG9jYWwgKlNURE9VVDsKICAgIG9wZW4gU1RET1VULCAnPjp1dGY4JywgXCRSZXN1bHQ7ICAgICMjIG5vIGNyaXRpYwogICAgJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoKTsKfQoKJFNlbGYtPklzKAogICAgJEV4aXRDb2RlLAogICAgMCwKICAgICJDb21tYW5kIGV4aXQgY29kZSIsCik7CgokRGVmaW5pdGlvbiA9ICRHZXREZWZpbml0aW9uLT4oIERlZmluaXRpb25JRCA9PiAkRGVmaW5pdGlvbklEICk7CgpteSAkWUFNTERlZmluaXRpb24gPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6WUFNTCcpLT5EdW1wKAogICAgRGF0YSA9PiBldmFsICRQZXJsRGVmaW5pdGlvbiwgICAgIyMgbm8gY3JpdGljCik7CgokU2VsZi0+SXNEZWVwbHkoCiAgICAkRGVmaW5pdGlvbiwKICAgICRZQU1MRGVmaW5pdGlvbiwKICAgICJEZWZpbml0aW9uIGJlZm9yZSBydW4gKFlBTUwpIiwKKTsKCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgKHF3KCRTZWxmKSk7Cgp1c2UgS2VybmVsOjpMYW5ndWFnZSBxdyhUcmFuc2xhdGFibGUpOwoKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCm15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAndXNlcnMnIF0sCik7Cm15ICRVc2VySUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpLT5Vc2VyTG9va3VwKCBVc2VyTG9naW4gPT4gJFRlc3RVc2VyTG9naW4gKTsKCm15ICRSYW5kb21JRCA9ICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CgpteSAkQ2xhc3NJRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtQWRkKAogICAgQ2xhc3MgICA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgICA9PiAkUmFuZG9tSUQsCiAgICBWYWxpZElEID0+IDEsCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CiRTZWxmLT5UcnVlKAogICAgJENsYXNzSUQsCiAgICAiQ2xhc3MgYWRkZWQgdG8gR2VuZXJhbENhdGFsb2ciLAopOwoKbXkgJFZlcnNpb24gPSAxOwoKbXkgJFBlcmxEZWZpbml0aW9uID0gPDwgJ0VPRic7ClsKICAgIHsKICAgICAgICBLZXkgICAgICAgID0+ICdWZW5kb3InLAogICAgICAgIE5hbWUgICAgICAgPT4gVHJhbnNsYXRhYmxlKCdWZW5kb3InKSwKICAgICAgICBTZWFyY2hhYmxlID0+IDEsCiAgICAgICAgSW5wdXQgICAgICA9PiB7CiAgICAgICAgICAgIFR5cGUgICAgICA9PiAnVGV4dCcsCiAgICAgICAgICAgIFNpemUgICAgICA9PiA1MCwKICAgICAgICAgICAgTWF4TGVuZ3RoID0+IDUwLAogICAgICAgIH0sCiAgICAgICAgQ291bnRNaW4gICAgID0+IDEsCiAgICAgICAgQ291bnRNYXggICAgID0+IDEsCiAgICAgICAgQ291bnREZWZhdWx0ID0+IDEsCiAgICB9LApdCkVPRgoKbXkgJERCT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkRCJyk7CgpyZXR1cm4gaWYgISREQk9iamVjdC0+RG8oCiAgICBTUUwgPT4gJwogICAgICAgIElOU0VSVCBJTlRPIGNvbmZpZ2l0ZW1fZGVmaW5pdGlvbgogICAgICAgICAgICAoY2xhc3NfaWQsIGNvbmZpZ2l0ZW1fZGVmaW5pdGlvbiwgdmVyc2lvbiwgY3JlYXRlX3RpbWUsIGNyZWF0ZV9ieSkKICAgICAgICBWQUxVRVMKICAgICAgICAgICAgKD8sID8sID8sIGN1cnJlbnRfdGltZXN0YW1wLCA/KScsCiAgICBCaW5kID0+IFsgXCRDbGFzc0lELCBcJFBlcmxEZWZpbml0aW9uLCBcJFZlcnNpb24sIFwkVXNlcklEIF0sCik7CgpyZXR1cm4gaWYgISREQk9iamVjdC0+UHJlcGFyZSgKICAgIFNRTCA9PiAnCiAgICAgICAgU0VMRUNUIGlkCiAgICAgICAgRlJPTSBjb25maWdpdGVtX2RlZmluaXRpb24KICAgICAgICBXSEVSRSBjbGFzc19pZCA9ID8KICAgICAgICAgICAgQU5EIHZlcnNpb24gPSA/CiAgICAgICAgT1JERVIgQlkgdmVyc2lvbiBERVNDJywKICAgIEJpbmQgID0+IFsgXCRDbGFzc0lELCBcJFZlcnNpb24gXSwKICAgIExpbWl0ID0+IDEsCik7CgpteSAkRGVmaW5pdGlvbklEOwp3aGlsZSAoIG15IEBSb3cgPSAkREJPYmplY3QtPkZldGNocm93QXJyYXkoKSApIHsKICAgICREZWZpbml0aW9uSUQgPSAkUm93WzBdOwp9CgpteSAkR2V0RGVmaW5pdGlvbiA9IHN1YiB7CiAgICBteSAlUGFyYW0gPSBAXzsKCiAgICByZXR1cm4gaWYgISREQk9iamVjdC0+UHJlcGFyZSgKICAgICAgICBTUUwgPT4gJwogICAgICAgICAgICBTRUxFQ1QgY29uZmlnaXRlbV9kZWZpbml0aW9uCiAgICAgICAgICAgIEZST00gY29uZmlnaXRlbV9kZWZpbml0aW9uCiAgICAgICAgICAgIFdIRVJFIGlkID0gPwogICAgICAgICAgICBPUkRFUiBCWSB2ZXJzaW9uIERFU0MnLAogICAgICAgIEJpbmQgID0+IFsgXCRQYXJhbXtEZWZpbml0aW9uSUR9IF0sCiAgICAgICAgTGltaXQgPT4gMSwKICAgICk7CgogICAgbXkgJERlZmluaXRpb247CiAgICB3aGlsZSAoIG15IEBSb3cgPSAkREJPYmplY3QtPkZldGNocm93QXJyYXkoKSApIHsKICAgICAgICAkRGVmaW5pdGlvbiA9ICRSb3dbMF07CiAgICB9CiAgICByZXR1cm4gJERlZmluaXRpb247Cn07CgpteSAkRGVmaW5pdGlvbiA9ICRHZXREZWZpbml0aW9uLT4oIERlZmluaXRpb25JRCA9PiAkRGVmaW5pdGlvbklEICk7CgokU2VsZi0+SXNEZWVwbHkoCiAgICAkRGVmaW5pdGlvbiwKICAgICRQZXJsRGVmaW5pdGlvbiwKICAgICJEZWZpbml0aW9uIGJlZm9yZSBydW4gKFBlcmwpIiwKKTsKCm15ICRDb21tYW5kT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkNvbnNvbGU6OkNvbW1hbmQ6Ok1haW50OjpJVFNNOjpDb25maWdpdGVtOjpEZWZpbml0aW9uUGVybDJZQU1MJyk7CgpteSAoICRSZXN1bHQsICRFeGl0Q29kZSApOwoKewogICAgbG9jYWwgKlNURE9VVDsKICAgIG9wZW4gU1RET1VULCAnPjp1dGY4JywgXCRSZXN1bHQ7ICAgICMjIG5vIGNyaXRpYwogICAgJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoKTsKfQoKJFNlbGYtPklzKAogICAgJEV4aXRDb2RlLAogICAgMCwKICAgICJDb21tYW5kIGV4aXQgY29kZSIsCik7CgokRGVmaW5pdGlvbiA9ICRHZXREZWZpbml0aW9uLT4oIERlZmluaXRpb25JRCA9PiAkRGVmaW5pdGlvbklEICk7CgpteSAkWUFNTERlZmluaXRpb24gPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6WUFNTCcpLT5EdW1wKAogICAgRGF0YSA9PiBldmFsICRQZXJsRGVmaW5pdGlvbiwgICAgIyMgbm8gY3JpdGljCik7CgokU2VsZi0+SXNEZWVwbHkoCiAgICAkRGVmaW5pdGlvbiwKICAgICRZQU1MRGVmaW5pdGlvbiwKICAgICJEZWZpbml0aW9uIGJlZm9yZSBydW4gKFlBTUwpIiwKKTsKCjE7Cg==
# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

use Kernel::System::ObjectManager;

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

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);

my $HelperObject                 = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
my $ConfigItemObject             = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
my $GeneralCatalogObject         = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $DynamicFieldObject           = $Kernel::OM->Get('Kernel::System::DynamicField');
my $ValidObject                  = $Kernel::OM->Get('Kernel::System::Valid');
my $TicketObject                 = $Kernel::OM->Get('Kernel::System::Ticket');
my $ZnunyHelperObject            = $Kernel::OM->Get('Kernel::System::ZnunyHelper');
my $DynamicFieldConfigItemObject = $Kernel::OM->Get('Kernel::System::DynamicField::ConfigItem');

my $ValidID = $ValidObject->ValidLookup(
    Valid => 'valid',
);

my $UserID = 1;

#
# Prepare config item
#
my $ClassListRef = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::Class',
    Valid => $ValidID,
);
my %ClassList = reverse %{ $ClassListRef || {} };

my $YesNoRef = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::YesNo',
    Valid => $ValidID,
);
my %YesNoList = reverse %{ $YesNoRef || {} };

my $ConfigItem1ID = $ConfigItemObject->ConfigItemAdd(
    ClassID => $ClassList{Computer},
    UserID  => $UserID,
);

$Self->True(
    scalar $ConfigItem1ID,
    'Config item must have been created successfully.',
);

my $ConfigItem1VersionID = $ZnunyHelperObject->_ITSMConfigItemVersionAdd(
    ConfigItemID  => $ConfigItem1ID,
    Name          => 'Unit test computer 1',
    ClassName     => 'Computer',
    DeplStateName => 'Production',
    InciStateName => 'Operational',
    XMLData       => {
        OtherEquipment         => 'Mouse',
        Note                   => 'Unit test computer 1 note.',
        WarrantyExpirationDate => '2040-01-01',
        InstallDate            => '2040-01-01',
        CPU                    => [
            {
                Content => 'CPU 1',
            },
            {
                Content => 'CPU 2',
            },
        ],
        NIC => [
            {
                Content    => 'NIC',
                IPoverDHCP => [
                    {
                        Content => $YesNoList{Yes},
                    },
                ],
                IPAddress => [
                    {
                        Content => '127.0.0.1'
                    },
                ],
            },
        ],
    },
);

$Self->True(
    scalar $ConfigItem1VersionID,
    'Config item version must have been created successfully.',
);

my $ConfigItem2ID = $ConfigItemObject->ConfigItemAdd(
    ClassID => $ClassList{Computer},
    UserID  => $UserID,
);

$Self->True(
    scalar $ConfigItem2ID,
    'Config item must have been created successfully.',
);

my $ConfigItem2VersionID = $ZnunyHelperObject->_ITSMConfigItemVersionAdd(
    ConfigItemID  => $ConfigItem2ID,
    Name          => 'Unit test computer 2',
    ClassName     => 'Computer',
    DeplStateName => 'Production',
    InciStateName => 'Operational',
    XMLData       => {
        OtherEquipment         => 'Mouse',
        Note                   => 'Unit test computer 2 note.',
        WarrantyExpirationDate => '2041-01-01',
        InstallDate            => '2041-01-01',
        CPU                    => [
            {
                Content => 'CPU 1',
            },
            {
                Content => 'CPU 4',
            },
        ],
        NIC => [
            {
                Content    => 'NIC',
                IPoverDHCP => [
                    {
                        Content => $YesNoList{No},
                    },
                ],
                IPAddress => [
                    {
                        Content => '127.0.0.2'
                    },
                ],
            },
        ],
    },
);

$Self->True(
    scalar $ConfigItem2VersionID,
    'Config item version must have been created successfully.',
);

#
# Prepare dynamic fields.
#
my @DynamicFields = (
    {
        Name          => 'DynamicFieldConfigItemUnitTestText1',
        Label         => 'DynamicFieldConfigItemUnitTestText1',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Text',
        Config        => {
            DefaultValue => '',
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestText2',
        Label         => 'DynamicFieldConfigItemUnitTestText2',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Text',
        Config        => {
            DefaultValue => '',
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestText3',
        Label         => 'DynamicFieldConfigItemUnitTestText3',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Text',
        Config        => {
            DefaultValue => '',
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestText4',
        Label         => 'DynamicFieldConfigItemUnitTestText4',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Text',
        Config        => {
            DefaultValue => '',
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestText5',
        Label         => 'DynamicFieldConfigItemUnitTestText5',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Text',
        Config        => {
            DefaultValue => '',
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestText6',
        Label         => 'DynamicFieldConfigItemUnitTestText6',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Text',
        Config        => {
            DefaultValue => '',
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestDateTime',
        Label         => 'DynamicFieldConfigItemUnitTestDateTime',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'DateTime',
        Config        => {
            DefaultValue => '',
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestDate',
        Label         => 'DynamicFieldConfigItemUnitTestDate',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Date',
        Config        => {
            DefaultValue => '',
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestMultiselect1',
        Label         => 'DynamicFieldConfigItemUnitTestMultiselect1',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Multiselect',
        Config        => {
            DefaultValue   => '',
            PossibleNone   => 1,
            PossibleValues => {
                'CPU 1' => 'CPU 1',
                'CPU 2' => 'CPU 2',
                'CPU 3' => 'CPU 3',
            },
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestMultiselect2',
        Label         => 'DynamicFieldConfigItemUnitTestMultiselect2',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Multiselect',
        Config        => {
            DefaultValue   => '',
            PossibleNone   => 1,
            PossibleValues => {
                'CPU 1' => 'CPU 1',
                'CPU 2' => 'CPU 2',
                'CPU 3' => 'CPU 3',
            },
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestMultiselect3',
        Label         => 'DynamicFieldConfigItemUnitTestMultiselect3',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Multiselect',
        Config        => {
            DefaultValue   => '',
            PossibleNone   => 1,
            PossibleValues => {
                'CPU 1' => 'CPU 1',
                'CPU 2' => 'CPU 2',
                'CPU 3' => 'CPU 3',
            },
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestDropdown1',
        Label         => 'DynamicFieldConfigItemUnitTestDropdown1',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Dropdown',
        Config        => {
            DefaultValue   => '',
            PossibleNone   => 1,
            PossibleValues => {
                'CPU 1' => 'CPU 1',
                'CPU 2' => 'CPU 2',
                'CPU 3' => 'CPU 3',
            },
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestDropdown2',
        Label         => 'DynamicFieldConfigItemUnitTestDropdown2',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Dropdown',
        Config        => {
            DefaultValue   => '',
            PossibleNone   => 1,
            PossibleValues => {
                'CPU 1' => 'CPU 1',
                'CPU 2' => 'CPU 2',
                'CPU 3' => 'CPU 3',
            },
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestDropdown3',
        Label         => 'DynamicFieldConfigItemUnitTestDropdown3',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Dropdown',
        Config        => {
            DefaultValue   => '',
            PossibleNone   => 1,
            PossibleValues => {
                'CPU 1' => 'CPU 1',
                'CPU 2' => 'CPU 2',
                'CPU 3' => 'CPU 3',
            },
        },
    },

    # ConfigItemDropdown (single select)
    {
        Name          => 'DynamicFieldConfigItemUnitTestConfigItem1',
        Label         => 'DynamicFieldConfigItemUnitTestConfigItem1',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'ConfigItemDropdown',
        Config        => {
            ConfigItemClass       => 'Computer',
            ConfigItemLinkType    => undef,
            ConfigItemLinkSource  => undef,
            ConfigItemLinkRemoval => 0,
            AdditionalDFStorage   => [
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText1',
                    ConfigItemKey => 'CPU',
                    Type          => 'Frontend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText2',
                    ConfigItemKey => 'CPU::2',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText3',
                    ConfigItemKey => 'Name',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText4',
                    ConfigItemKey => 'NIC::1::IPAddress::1',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText5',
                    ConfigItemKey => 'ClassID',
                    Type          => 'Backend',                               # Notice that this one is backend only
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText6',
                    ConfigItemKey => 'NIC::1::IPoverDHCP::1_Value',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDate',
                    ConfigItemKey => 'WarrantyExpirationDate',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDateTime',
                    ConfigItemKey => 'WarrantyExpirationDate',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDropdown1',
                    ConfigItemKey => 'CPU::1',
                    Type          => 'Frontend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDropdown2',
                    ConfigItemKey => 'CPU',
                    Type          => 'Frontend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestMultiselect1',
                    ConfigItemKey => 'CPU::2',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestMultiselect2',
                    ConfigItemKey => 'CPU',
                    Type          => 'FrontendBackend',
                },
            ],
        },
    },

    # ConfigItemMultiselect
    {
        Name          => 'DynamicFieldConfigItemUnitTestConfigItem2',
        Label         => 'DynamicFieldConfigItemUnitTestConfigItem2',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'ConfigItemMultiselect',
        Config        => {
            ConfigItemClass       => 'Computer',
            ConfigItemLinkType    => undef,
            ConfigItemLinkSource  => undef,
            ConfigItemLinkRemoval => 0,
            AdditionalDFStorage   => [
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText1',
                    ConfigItemKey => 'CPU',
                    Type          => 'Backend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText2',
                    ConfigItemKey => 'CPU::2',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText3',
                    ConfigItemKey => 'Name',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText4',
                    ConfigItemKey => 'NIC::1::IPAddress::1',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText5',
                    ConfigItemKey => 'ClassID',
                    Type          => 'Frontend',                              # Notice that this one is frontend only
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText6',
                    ConfigItemKey => 'NIC::1::IPoverDHCP::1_Value',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDate',
                    ConfigItemKey => 'WarrantyExpirationDate',
                    Type          => 'Backend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDateTime',
                    ConfigItemKey => 'WarrantyExpirationDate',
                    Type          => 'Backend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDropdown1',
                    ConfigItemKey => 'CPU::1',
                    Type          => 'Backend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDropdown2',
                    ConfigItemKey => 'CPU',
                    Type          => 'Backend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestMultiselect1',
                    ConfigItemKey => 'CPU::2',
                    Type          => 'Backend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestMultiselect2',
                    ConfigItemKey => 'CPU',
                    Type          => 'Backend',
                },
            ],
        },
    },
);

my $DynamicFieldsCreated = $ZnunyHelperObject->_DynamicFieldsCreate(@DynamicFields);

$Self->True(
    scalar $DynamicFieldsCreated,
    'Dynamic fields must have been created successfully.',
);

#
# Tests for GetAdditionalDFStorageData() and StoreDynamicFieldValues()
#

# ConfigItemDropdown (DynamicFieldConfigItemUnitTestConfigItem1)
my $DynamicFieldData = $DynamicFieldConfigItemObject->GetAdditionalDFStorageData(
    SourceDynamicFieldName => 'DynamicFieldConfigItemUnitTestConfigItem1',
    SelectedConfigItemIDs  => [ $ConfigItem1ID, ],
    StorageType            => 'Frontend',
    UserID                 => $UserID,
);

my %ExpectedDynamicFieldData = (
    DynamicFieldConfigItemUnitTestText1        => 'CPU 1, CPU 2',
    DynamicFieldConfigItemUnitTestText2        => 'CPU 2',
    DynamicFieldConfigItemUnitTestText3        => 'Unit test computer 1',
    DynamicFieldConfigItemUnitTestText4        => '127.0.0.1',
    DynamicFieldConfigItemUnitTestText6        => 'Yes',
    DynamicFieldConfigItemUnitTestDate         => '2040-01-01',
    DynamicFieldConfigItemUnitTestDateTime     => '2040-01-01 00:00:00',
    DynamicFieldConfigItemUnitTestDropdown1    => 'CPU 1',
    DynamicFieldConfigItemUnitTestDropdown2    => 'CPU 1',
    DynamicFieldConfigItemUnitTestMultiselect1 => [
        'CPU 2',
    ],
    DynamicFieldConfigItemUnitTestMultiselect2 => [
        'CPU 1',
        'CPU 2',
    ],
);

$Self->IsDeeply(
    $DynamicFieldData,
    \%ExpectedDynamicFieldData,
    'ConfigItemDropdown - GetAdditionalDFStorageData() must return expected dynamic field data.',
);

# Store dynamic field values in ticket.
my $TicketID = $HelperObject->TicketCreate();

my $StoredDynamicFields = $DynamicFieldConfigItemObject->StoreDynamicFieldValues(
    TicketID                => $TicketID,
    AdditionalDFStorageData => $DynamicFieldData,
    UserID                  => $UserID,
);

my @ExpectedStoredDynamicFields = sort keys %ExpectedDynamicFieldData;

$Self->IsDeeply(
    $StoredDynamicFields,
    \@ExpectedStoredDynamicFields,
    'ConfigItemDropdown - Stored dynamic fields must match expected ones.',
);

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

for my $DynamicFieldName ( sort keys %ExpectedDynamicFieldData ) {
    my $ExpectedValue = $ExpectedDynamicFieldData{$DynamicFieldName};

    if ( $DynamicFieldName eq 'DynamicFieldConfigItemUnitTestDate' ) {
        $ExpectedValue .= ' 00:00:00';
    }

    $Self->IsDeeply(
        $Ticket{ 'DynamicField_' . $DynamicFieldName },
        $ExpectedValue,
        "ConfigItemDropdown - Content of dynamic field $DynamicFieldName must match expected one.",
    );
}

# ConfigItemMultiselect (DynamicFieldConfigItemUnitTestConfigItem2)
$DynamicFieldData = $DynamicFieldConfigItemObject->GetAdditionalDFStorageData(
    SourceDynamicFieldName => 'DynamicFieldConfigItemUnitTestConfigItem2',
    SelectedConfigItemIDs  => [ $ConfigItem1ID, $ConfigItem2ID, ],
    StorageType            => 'Backend',
    UserID                 => $UserID,
);

%ExpectedDynamicFieldData = (
    DynamicFieldConfigItemUnitTestText1        => 'CPU 1, CPU 2, CPU 1, CPU 4',
    DynamicFieldConfigItemUnitTestText2        => 'CPU 2, CPU 4',
    DynamicFieldConfigItemUnitTestText3        => 'Unit test computer 1, Unit test computer 2',
    DynamicFieldConfigItemUnitTestText4        => '127.0.0.1, 127.0.0.2',
    DynamicFieldConfigItemUnitTestText6        => 'Yes, No',
    DynamicFieldConfigItemUnitTestDate         => '2040-01-01',
    DynamicFieldConfigItemUnitTestDateTime     => '2040-01-01 00:00:00',
    DynamicFieldConfigItemUnitTestDropdown1    => 'CPU 1',
    DynamicFieldConfigItemUnitTestDropdown2    => 'CPU 1',
    DynamicFieldConfigItemUnitTestMultiselect1 => [
        'CPU 2',
        'CPU 4',
    ],
    DynamicFieldConfigItemUnitTestMultiselect2 => [
        'CPU 1',
        'CPU 2',
        'CPU 1',
        'CPU 4',
    ],
);

$Self->IsDeeply(
    $DynamicFieldData,
    \%ExpectedDynamicFieldData,
    'ConfigItemMultiselect - GetAdditionalDFStorageData() must return expected dynamic field data.',
);

# Store dynamic field values in ticket.
$TicketID = $HelperObject->TicketCreate();

$StoredDynamicFields = $DynamicFieldConfigItemObject->StoreDynamicFieldValues(
    TicketID                => $TicketID,
    AdditionalDFStorageData => $DynamicFieldData,
    UserID                  => $UserID,
);

@ExpectedStoredDynamicFields = sort keys %ExpectedDynamicFieldData;

$Self->IsDeeply(
    $StoredDynamicFields,
    \@ExpectedStoredDynamicFields,
    'ConfigItemMultiselect - Stored dynamic fields must match expected ones.',
);

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

for my $DynamicFieldName ( sort keys %ExpectedDynamicFieldData ) {
    my $ExpectedValue = $ExpectedDynamicFieldData{$DynamicFieldName};

    if ( $DynamicFieldName eq 'DynamicFieldConfigItemUnitTestDate' ) {
        $ExpectedValue .= ' 00:00:00';
    }

    $Self->IsDeeply(
        $Ticket{ 'DynamicField_' . $DynamicFieldName },
        $ExpectedValue,
        "ConfigItemMultiselect 1 - Content of dynamic field $DynamicFieldName must match expected one.",
    );
}

# ConfigItemMultiselect (DynamicFieldConfigItemUnitTestConfigItem2)
$DynamicFieldData = $DynamicFieldConfigItemObject->GetAdditionalDFStorageData(
    SourceDynamicFieldName => 'DynamicFieldConfigItemUnitTestConfigItem2',

    # changed order of config items must  have an effect because values for dynamic fields
    # will be sorted.
    SelectedConfigItemIDs => [ $ConfigItem2ID, $ConfigItem1ID, ],
    StorageType           => 'Backend',
    UserID                => $UserID,
);

%ExpectedDynamicFieldData = (
    DynamicFieldConfigItemUnitTestText1        => 'CPU 1, CPU 4, CPU 1, CPU 2',
    DynamicFieldConfigItemUnitTestText2        => 'CPU 4, CPU 2',
    DynamicFieldConfigItemUnitTestText3        => 'Unit test computer 2, Unit test computer 1',
    DynamicFieldConfigItemUnitTestText4        => '127.0.0.2, 127.0.0.1',
    DynamicFieldConfigItemUnitTestText6        => 'No, Yes',
    DynamicFieldConfigItemUnitTestDate         => '2041-01-01',
    DynamicFieldConfigItemUnitTestDateTime     => '2041-01-01 00:00:00',
    DynamicFieldConfigItemUnitTestDropdown1    => 'CPU 1',
    DynamicFieldConfigItemUnitTestDropdown2    => 'CPU 1',
    DynamicFieldConfigItemUnitTestMultiselect1 => [
        'CPU 4',
        'CPU 2',
    ],
    DynamicFieldConfigItemUnitTestMultiselect2 => [
        'CPU 1',
        'CPU 4',
        'CPU 1',
        'CPU 2',
    ],
);

$Self->IsDeeply(
    $DynamicFieldData,
    \%ExpectedDynamicFieldData,
    'ConfigItemMultiselect - GetAdditionalDFStorageData() must return expected dynamic field data.',
);

# Store dynamic field values in ticket.
$TicketID = $HelperObject->TicketCreate();

$StoredDynamicFields = $DynamicFieldConfigItemObject->StoreDynamicFieldValues(
    TicketID                => $TicketID,
    AdditionalDFStorageData => $DynamicFieldData,
    UserID                  => $UserID,
);

@ExpectedStoredDynamicFields = sort keys %ExpectedDynamicFieldData;

$Self->IsDeeply(
    $StoredDynamicFields,
    \@ExpectedStoredDynamicFields,
    'Stored dynamic fields must match expected ones.',
);

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

for my $DynamicFieldName ( sort keys %ExpectedDynamicFieldData ) {
    my $ExpectedValue = $ExpectedDynamicFieldData{$DynamicFieldName};

    if ( $DynamicFieldName eq 'DynamicFieldConfigItemUnitTestDate' ) {
        $ExpectedValue .= ' 00:00:00';
    }

    $Self->IsDeeply(
        $Ticket{ 'DynamicField_' . $DynamicFieldName },
        $ExpectedValue,
        "ConfigItemMultiselect 2 - Content of dynamic field $DynamicFieldName must match expected one.",
    );
}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

use Kernel::System::ObjectManager;

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

use Kernel::GenericInterface::Invoker::ITSMConfigItem::Generic;

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);

my $HelperObject         = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
my $ZnunyHelperObject    = $Kernel::OM->Get('Kernel::System::ZnunyHelper');
my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');
my $ValidObject          = $Kernel::OM->Get('Kernel::System::Valid');
my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

my $GenericInvokerObject = Kernel::GenericInterface::Invoker::ITSMConfigItem::Generic->new(
    DebuggerObject => 1,    # this is a mock-up, the debugger object will not be used within this test.
);

my $UserID = 1;

#
# Prepare a config item
#
my $ValidID = $ValidObject->ValidLookup(
    Valid => 'valid',
);

my $Classes = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::Class',
    Valid => $ValidID,
);
my %Classes = reverse %{ $Classes // {} };

$Self->True(
    scalar $Classes{Computer},
    'Class "Computer" must be present.',
);

my $YesNo = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::YesNo',
    Valid => $ValidID,
);
my %YesNo = reverse %{ $YesNo // {} };

my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
    ClassID => $Classes{Computer},
    UserID  => $UserID,
);

$Self->True(
    scalar $ConfigItemID,
    'Config item must have been created successfully.',
) || return;

my $ComputerNameFirstVersion  = 'UnitTestComputer' . $HelperObject->GetRandomID();
my $ComputerNameSecondVersion = 'UnitTestComputer' . $HelperObject->GetRandomID();
my $Class                     = 'Computer';
my $ClassID                   = $Classes{Computer};

my $DeplStateProduction = $GeneralCatalogObject->ItemGet(
    Class => 'ITSM::ConfigItem::DeploymentState',
    Name  => 'Production',
);

my $DeplStateRepair = $GeneralCatalogObject->ItemGet(
    Class => 'ITSM::ConfigItem::DeploymentState',
    Name  => 'Repair',
);

my $InciStateOperational = $GeneralCatalogObject->ItemGet(
    Class => 'ITSM::Core::IncidentState',
    Name  => 'Operational',
);

my $InciStateIncident = $GeneralCatalogObject->ItemGet(
    Class => 'ITSM::Core::IncidentState',
    Name  => 'Incident',
);

my $Definition = $ConfigItemObject->DefinitionGet(
    ClassID => $ClassID,
);

my $FirstVersionID = $ZnunyHelperObject->_ITSMConfigItemVersionAdd(
    ConfigItemID  => $ConfigItemID,
    Name          => $ComputerNameFirstVersion,
    ClassName     => $Class,
    DeplStateName => 'Production',
    InciStateName => 'Operational',
    XMLData       => {
        CPU => [
            {
                Content => 'AMD Ryzen',
            },
        ],
        WarrantyExpirationDate => '2021-07-16',
        InstallDate            => '2021-07-16',
        NIC                    => [
            {
                Content    => 'NIC',
                IPoverDHCP => [
                    {
                        Content => $YesNo{Yes},
                    },
                ],
                IPAddress => [
                    {
                        Content => '127.0.0.1'
                    },
                ],
            },
        ],
    },
);

#
# Invalid config item ID
#
my $RequestData = $GenericInvokerObject->PrepareRequest(
    Data => {
        ConfigItemID => 999999999999,
        Event        => undef,
    },
);

my %ExpectedRequestData = (
    Success      => 0,
    ErrorMessage => 'Data of config item with ID 999999999999 could not be retrieved.',
);

$Self->IsDeeply(
    $RequestData,
    \%ExpectedRequestData,
    'PrepareRequest(): Request data must match expected one for invalid config item ID.',
);

#
# Valid config item ID without event
#
$RequestData = $GenericInvokerObject->PrepareRequest(
    Data => {
        ConfigItemID => $ConfigItemID,
        Event        => undef,
    },
);

my %ExpectedRequestDataFirstVersion = (
    XMLData => {
        NIC => [
            {
                IPAddress => [
                    {
                        Content => '127.0.0.1',
                    }
                ],
                IPoverDHCP => [
                    {
                        ReadableValue => 'Yes',
                        Content       => $YesNo{Yes},
                    }
                ],
                Content => 'NIC',
            },
        ],
        WarrantyExpirationDate => [
            {
                Content => '2021-07-16',
            },
        ],
        CPU => [
            {
                Content => 'AMD Ryzen',
            },
        ],
        InstallDate => [
            {
                Content => '2021-07-16',
            },
        ],
    },
    ConfigItemID => $ConfigItemID,
    DefinitionID => $Definition->{DefinitionID},
    InciStateID  => $InciStateOperational->{ItemID},
    Name         => $ComputerNameFirstVersion,
    DeplStateID  => $DeplStateProduction->{ItemID},
    DeplState    => 'Production',
    ClassID      => $ClassID,
    InciState    => 'Operational',
    Class        => $Class,
);

%ExpectedRequestData = (
    Success => 1,
    Data    => {
        ConfigItem => {
            %ExpectedRequestDataFirstVersion,
        },
        Event => undef,
    },
);

$Self->IsDeeply(
    $RequestData,
    \%ExpectedRequestData,
    'PrepareRequest(): Request data must match expected one for valid config item ID (first version) without event.',
);

my $SecondVersionID = $ZnunyHelperObject->_ITSMConfigItemVersionAdd(
    ConfigItemID  => $ConfigItemID,
    Name          => $ComputerNameSecondVersion,
    ClassName     => $Class,
    DeplStateName => 'Repair',
    InciStateName => 'Incident',
    XMLData       => {
        CPU => [
            {
                Content => 'Intel Core',
            },
        ],
        WarrantyExpirationDate => '2023-07-15',
        InstallDate            => '2021-07-16',
        NIC                    => [
            {
                Content    => 'NIC1',
                IPoverDHCP => [
                    {
                        Content => $YesNo{Yes},
                    },
                ],
                IPAddress => [
                    {
                        Content => '127.0.0.1'
                    },
                ],
            },
            {
                Content    => 'NIC2',
                IPoverDHCP => [
                    {
                        Content => $YesNo{No},
                    },
                ],
                IPAddress => [
                    {
                        Content => '127.1.0.1'
                    },
                ],
            },
        ],
    },
);

$RequestData = $GenericInvokerObject->PrepareRequest(
    Data => {
        ConfigItemID => $ConfigItemID,
        Event        => undef,
    },
);

my %ExpectedRequestDataSecondVersion = (
    XMLData => {
        NIC => [
            {
                IPAddress => [
                    {
                        Content => '127.0.0.1',
                    }
                ],
                IPoverDHCP => [
                    {
                        ReadableValue => 'Yes',
                        Content       => $YesNo{Yes},
                    }
                ],
                Content => 'NIC1',
            },
            {
                IPAddress => [
                    {
                        Content => '127.1.0.1',
                    }
                ],
                IPoverDHCP => [
                    {
                        ReadableValue => 'No',
                        Content       => $YesNo{No},
                    }
                ],
                Content => 'NIC2',
            },
        ],
        WarrantyExpirationDate => [
            {
                Content => '2023-07-15',
            },
        ],
        CPU => [
            {
                Content => 'Intel Core',
            },
        ],
        InstallDate => [
            {
                Content => '2021-07-16',
            },
        ],
    },
    ConfigItemID => $ConfigItemID,
    DefinitionID => $Definition->{DefinitionID},
    InciStateID  => $InciStateIncident->{ItemID},
    Name         => $ComputerNameSecondVersion,
    DeplStateID  => $DeplStateRepair->{ItemID},
    DeplState    => 'Repair',
    ClassID      => $ClassID,
    InciState    => 'Incident',
    Class        => $Class,
);

%ExpectedRequestData = (
    Success => 1,
    Data    => {
        ConfigItem => {
            %ExpectedRequestDataSecondVersion,
        },
        Event => undef,
    },
);

$Self->IsDeeply(
    $RequestData,
    \%ExpectedRequestData,
    'PrepareRequest(): Request data must match expected one for valid config item ID (second version) without event.',
);

#
# Valid config item ID with event ConfigItemCreate
# This should not return the data of the previous config item version.
#
$RequestData = $GenericInvokerObject->PrepareRequest(
    Data => {
        ConfigItemID => $ConfigItemID,
        Event        => 'ConfigItemCreate',
    },
);

%ExpectedRequestData = (
    Success => 1,
    Data    => {
        ConfigItem => {
            %ExpectedRequestDataSecondVersion,
        },
        Event => 'ConfigItemCreate',
    },
);

$Self->IsDeeply(
    $RequestData,
    \%ExpectedRequestData,
    'PrepareRequest(): Request data must match expected one for valid config item ID (second version) with event ConfigItemCreate.',
);

#
# Valid config item ID with event ConfigItemDelete
# This should not return the data of the previous config item version.
#
$RequestData = $GenericInvokerObject->PrepareRequest(
    Data => {
        ConfigItemID => $ConfigItemID,
        Event        => 'ConfigItemDelete',
    },
);

%ExpectedRequestData = (
    Success => 1,
    Data    => {
        ConfigItem => {
            %ExpectedRequestDataSecondVersion,
        },
        Event => 'ConfigItemDelete',
    },
);

$Self->IsDeeply(
    $RequestData,
    \%ExpectedRequestData,
    'PrepareRequest(): Request data must match expected one for valid config item ID (second version) with event ConfigItemDelete.',
);

#
# Valid config item ID with event DeploymentStateUpdate
# This should return the data of the previous config item version and as separate information
# the changed deployment status.
#
$RequestData = $GenericInvokerObject->PrepareRequest(
    Data => {
        ConfigItemID => $ConfigItemID,
        Event        => 'DeploymentStateUpdate',
    },
);

%ExpectedRequestData = (
    Success => 1,
    Data    => {
        ConfigItem => {
            %ExpectedRequestDataSecondVersion,
            PreviousConfigItemVersion => {
                %ExpectedRequestDataFirstVersion,
            },
            DeploymentStateUpdate => {
                Old => {
                    ID      => $ExpectedRequestDataFirstVersion{DeplStateID},
                    Content => $ExpectedRequestDataFirstVersion{DeplState},
                },
                New => {
                    ID      => $ExpectedRequestDataSecondVersion{DeplStateID},
                    Content => $ExpectedRequestDataSecondVersion{DeplState},
                },
            },
        },
        Event => 'DeploymentStateUpdate',
    },
);

$Self->IsDeeply(
    $RequestData,
    \%ExpectedRequestData,
    'PrepareRequest(): Request data must match expected one for valid config item ID (second version) with event DeploymentStateUpdate.',
);

#
# Valid config item ID with event IncidentStateUpdate
# This should return the data of the previous config item version and as separate information
# the changed incident status.
#
$RequestData = $GenericInvokerObject->PrepareRequest(
    Data => {
        ConfigItemID => $ConfigItemID,
        Event        => 'IncidentStateUpdate',
    },
);

%ExpectedRequestData = (
    Success => 1,
    Data    => {
        ConfigItem => {
            %ExpectedRequestDataSecondVersion,
            PreviousConfigItemVersion => {
                %ExpectedRequestDataFirstVersion,
            },
            IncidentStateUpdate => {
                Old => {
                    ID      => $ExpectedRequestDataFirstVersion{InciStateID},
                    Content => $ExpectedRequestDataFirstVersion{InciState},
                },
                New => {
                    ID      => $ExpectedRequestDataSecondVersion{InciStateID},
                    Content => $ExpectedRequestDataSecondVersion{InciState},
                },
            },
        },
        Event => 'IncidentStateUpdate',
    },
);

$Self->IsDeeply(
    $RequestData,
    \%ExpectedRequestData,
    'PrepareRequest(): Request data must match expected one for valid config item ID (second version) with event IncidentStateUpdate.',
);

#
# Valid config item ID with event NameUpdate
# This should return the data of the previous config item version and as separate information
# the changed name.
#
$RequestData = $GenericInvokerObject->PrepareRequest(
    Data => {
        ConfigItemID => $ConfigItemID,
        Event        => 'NameUpdate',
    },
);

%ExpectedRequestData = (
    Success => 1,
    Data    => {
        ConfigItem => {
            %ExpectedRequestDataSecondVersion,
            PreviousConfigItemVersion => {
                %ExpectedRequestDataFirstVersion,
            },
            NameUpdate => {
                Old => {
                    Content => $ExpectedRequestDataFirstVersion{Name},
                },
                New => {
                    Content => $ExpectedRequestDataSecondVersion{Name},
                },
            },
        },
        Event => 'NameUpdate',
    },
);

$Self->IsDeeply(
    $RequestData,
    \%ExpectedRequestData,
    'PrepareRequest(): Request data must match expected one for valid config item ID (second version) with event NameUpdate.',
);

#
# Valid config item ID with event ValueUpdate
# This should return the data of the previous config item version and as separate information
# the changed values of field CPU.
#
$RequestData = $GenericInvokerObject->PrepareRequest(
    Data => {
        ConfigItemID => $ConfigItemID,
        Event        => 'ValueUpdate',
    },
);

%ExpectedRequestData = (
    Success => 1,
    Data    => {
        ConfigItem => {
            %ExpectedRequestDataSecondVersion,
            PreviousConfigItemVersion => {
                %ExpectedRequestDataFirstVersion,
            },
            ValueUpdate => {
                'CPU::1' => {
                    New => {
                        Content => 'Intel Core',
                    },
                    Old => {
                        Content => 'AMD Ryzen',
                    },
                },
                'NIC::1' => {
                    New => {
                        Content => 'NIC1',
                    },
                    Old => {
                        Content => 'NIC',
                    },
                },
                'NIC::2' => {
                    New => {
                        Content => 'NIC2',
                    },
                    Old => {
                        Content => '',
                    },
                },
                'NIC::2::IPAddress::1' => {
                    New => {
                        Content => '127.1.0.1',
                    },
                    Old => {
                        Content => '',
                    },
                },
                'NIC::2::IPoverDHCP::1' => {
                    New => {
                        Content       => $YesNo{No},
                        ReadableValue => 'No',
                    },
                    Old => {
                        Content => '',
                    },
                },
                'WarrantyExpirationDate::1' => {
                    New => {
                        Content => '2023-07-15',
                    },
                    Old => {
                        Content => '2021-07-16',
                    },
                },
            },
        },
        Event => 'ValueUpdate',
    },
);

$Self->IsDeeply(
    $RequestData,
    \%ExpectedRequestData,
    'PrepareRequest(): Request data must match expected one for valid config item ID (second version) with event ValueUpdate.',
);

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

use MIME::Base64;

use Kernel::GenericInterface::Debugger;
use Kernel::GenericInterface::Operation::ConfigItem::ConfigItemCreate;
use Kernel::System::VariableCheck qw(:all);

# Set UserID to root.
$Self->{UserID} = 1;

# Skip SSL certificate verification.
$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        SkipSSLVerify => 1,
    },
);
my $HelperObject     = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
my $ConfigObject     = $Kernel::OM->Get('Kernel::Config');
my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

my $RandomID = $HelperObject->GetRandomID();

# Check if SSL Certificate verification is disabled.
$Self->Is(
    $ENV{PERL_LWP_SSL_VERIFY_HOSTNAME},
    0,
    'Disabled SSL certiticates verification in environment',
);

my $TestCustomerUserLogin = $HelperObject->TestCustomerUserCreate();

# Create webservice object.
my $WebserviceObject = $Kernel::OM->Get('Kernel::System::GenericInterface::Webservice');
$Self->Is(
    'Kernel::System::GenericInterface::Webservice',
    ref $WebserviceObject,
    "Create webservice object",
);

# Set webservice name.
my $WebserviceName = '-Test-' . $RandomID;

my $WebserviceID = $WebserviceObject->WebserviceAdd(
    Name   => $WebserviceName,
    Config => {
        Debugger => {
            DebugThreshold => 'debug',
        },
        Provider => {
            Transport => {
                Type => '',
            },
        },
    },
    ValidID => 1,
    UserID  => 1,
);
$Self->True(
    $WebserviceID,
    "Added Webservice",
);

# Get remote host with some precautions for certain unit test systems.
my $Host;
my $FQDN = $ConfigObject->Get('FQDN');

# Try to resolve fqdn host.
if ( $FQDN ne 'yourhost.example.com' && gethostbyname($FQDN) ) {
    $Host = $FQDN;
}

# Try to resolve localhost instead.
if ( !$Host && gethostbyname('localhost') ) {
    $Host = 'localhost';
}

# Use hardcoded localhost ip address.
if ( !$Host ) {
    $Host = '127.0.0.1';
}

# Prepare webservice config.
my $RemoteSystem =
    $ConfigObject->Get('HttpType')
    . '://'
    . $Host
    . '/'
    . $ConfigObject->Get('ScriptAlias')
    . '/nph-genericinterface.pl/WebserviceID/'
    . $WebserviceID;

my $WebserviceConfig = {
    Description =>
        'Test for ConfigItem Connector using SOAP transport backend.',
    Debugger => {
        DebugThreshold => 'debug',
        TestMode       => 1,
    },
    Provider => {
        Transport => {
            Type   => 'HTTP::SOAP',
            Config => {
                MaxLength => 10_000_000,
                NameSpace => 'http://otrs.org/SoapTestInterface/',
                Endpoint  => $RemoteSystem,
            },
        },
        Operation => {
            ConfigItemCreate => {
                Type => 'ConfigItem::ConfigItemCreate',
            },
            SessionCreate => {
                Type => 'Session::SessionCreate',
            },
        },
    },
    Requester => {
        Transport => {
            Type   => 'HTTP::SOAP',
            Config => {
                NameSpace => 'http://otrs.org/SoapTestInterface/',
                Encoding  => 'UTF-8',
                Endpoint  => $RemoteSystem,
            },
        },
        Invoker => {
            ConfigItemCreate => {
                Type => 'Test::TestSimple',
            },
            SessionCreate => {
                Type => 'Test::TestSimple',
            },
        },
    },
};

# Update webservice with real config.
my $WebserviceUpdate = $WebserviceObject->WebserviceUpdate(
    ID      => $WebserviceID,
    Name    => $WebserviceName,
    Config  => $WebserviceConfig,
    ValidID => 1,
    UserID  => 1,
);
$Self->True(
    $WebserviceUpdate,
    "Updated Webservice $WebserviceID - $WebserviceName",
);

# Debugger object.
my $DebuggerObject = Kernel::GenericInterface::Debugger->new(
    DebuggerConfig => {
        DebugThreshold => 'debug',
        TestMode       => 1,
    },
    WebserviceID      => $WebserviceID,
    CommunicationType => 'Provider',
);
$Self->Is(
    ref $DebuggerObject,
    'Kernel::GenericInterface::Debugger',
    'DebuggerObject instanciated correctly',
);

# Get SessionID.
# Create requester object.
my $RequesterSessionObject = $Kernel::OM->Get('Kernel::GenericInterface::Requester');
$Self->Is(
    'Kernel::GenericInterface::Requester',
    ref $RequesterSessionObject,
    "SessionID - Create requester object",
);

# Create a new user for current test.
my $UserLogin = $HelperObject->TestUserCreate(
    Groups => [ 'admin', 'users', 'itsm-configitem' ],
);
my $Password = $UserLogin;

# Start requester with our webservice.
my $RequesterSessionResult = $RequesterSessionObject->Run(
    WebserviceID => $WebserviceID,
    Invoker      => 'SessionCreate',
    Data         => {
        UserLogin => $UserLogin,
        Password  => $Password,
    },
);

my $NewSessionID = $RequesterSessionResult->{Data}->{SessionID};

# Actual tests.
my @Tests = (
    {
        Name           => 'Empty Request',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {},
        ExpectedData   => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.MissingParameter',
                    ErrorMessage => 'ConfigItemCreate: ConfigItem parameter is missing or not valid!',
                }
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Invalid ConfigItem',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => 1,
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.MissingParameter',
                    ErrorMessage => 'ConfigItemCreate: ConfigItem parameter is missing or not valid!',
                }
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Missing CIXMLData',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'Test' . $RandomID,
                DeplState => 'Production',
                InciState => 'Operational',
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.MissingParameter',
                    ErrorMessage => 'ConfigItemCreate: ConfigItem->CIXMLData->NIC->IPoverDHCP parameter is missing!',
                }
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Invalid CIXMLData',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'Test' . $RandomID,
                DeplState => 'Production',
                InciState => 'Operational',
                CIXMLData => 0,
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.MissingParameter',
                    ErrorMessage => 'ConfigItemCreate: ConfigItem->CIXMLData is missing or invalid!',
                }
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Missing Class',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                CIXMLData => {
                    Test => 1,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.MissingParameter',
                    ErrorMessage => 'ConfigItemCreate: ConfigItem->Class parameter is missing!',
                }
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Missing Name',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                CIXMLData => {
                    Test => 1,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.MissingParameter',
                    ErrorMessage => 'ConfigItemCreate: ConfigItem->Name parameter is missing!',
                }
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Missing DeplState',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                CIXMLData => {
                    Test => 1,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.MissingParameter',
                    ErrorMessage => 'ConfigItemCreate: ConfigItem->DeplState parameter is missing!',
                }
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Missing InciState',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                CIXMLData => {
                    Test => 1,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.MissingParameter',
                    ErrorMessage => 'ConfigItemCreate: ConfigItem->InciState parameter is missing!',
                }
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Wrong Class',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'NotExisitng' . $RandomID,
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Test => 1,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.InvalidParameter',
                    ErrorMessage => 'ConfigItemCreate: ConfigItem->Class parameter is invalid!',
                }
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Wrong DeplState',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production' . $RandomID,
                InciState => 'Incident',
                CIXMLData => {
                    Test => 1,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.InvalidParameter',
                    ErrorMessage => 'ConfigItemCreate: ConfigItem->DeplState parameter is invalid!',
                }
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Wrong InciState',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident' . $RandomID,
                CIXMLData => {
                    Test => 1,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.InvalidParameter',
                    ErrorMessage => 'ConfigItemCreate: ConfigItem->InciState parameter is invalid!',
                }
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Missing NIC->NIC',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    NIC => {
                        Test => 1,
                    }
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemCreate.MissingParameter',
                    ErrorMessage =>
                        'ConfigItemCreate: ConfigItem->CIXMLData->NIC parameter value is required and is missing!',
                }
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Missing NIC->IpOverDHCP',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    NIC => {
                        NIC => 'Eth0',
                    }
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.MissingParameter',
                    ErrorMessage => 'ConfigItemCreate: ConfigItem->CIXMLData->NIC->IPoverDHCP parameter is missing!',
                }
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Missing NIC->NIC in array',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    NIC => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            IPoverDHCP => 'No',
                        },
                    ],
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemCreate.MissingParameter',
                    ErrorMessage =>
                        'ConfigItemCreate: ConfigItem->CIXMLData->NIC parameter value is required and is missing!',
                },
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Missing NIC->IpOverDHCP in array',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    NIC => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC => 'Eth0',
                        },
                    ],
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.MissingParameter',
                    ErrorMessage => 'ConfigItemCreate: ConfigItem->CIXMLData->NIC[1]->IPoverDHCP parameter is missing!',
                },
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Wrong NIC->IpOverDHCP General Catalog in Hash',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    NIC => {
                        NIC        => 'Eth0',
                        IPoverDHCP => 'No' . $RandomID,
                    },
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemCreate.InvalidParameter',
                    ErrorMessage =>
                        'ConfigItemCreate: ConfigItem->CIXMLData->NIC->IPoverDHCP parameter value is not a valid for General Catalog \'ITSM::ConfigItem::YesNo\'!',
                },
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Wrong NIC->IpOverDHCP General Catalog in Array Hash',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    NIC => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No' . $RandomID,
                        },
                    ],
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemCreate.InvalidParameter',
                    ErrorMessage =>
                        'ConfigItemCreate: ConfigItem->CIXMLData->NIC[1]->IPoverDHCP parameter value is not a valid for General Catalog \'ITSM::ConfigItem::YesNo\'!',
                },
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Wrong Vendor Long Text ',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'a' x 51,
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemCreate.InvalidParameter',
                    ErrorMessage =>
                        'ConfigItemCreate: ConfigItem->CIXMLData->Vendor parameter value excedes the maxium length!',
                },
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Wrong WarrantyExpirationDate Date',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'Torero Chips',
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                    WarrantyExpirationDate => '1930-30-30',
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemCreate.InvalidParameter',
                    ErrorMessage =>
                        'ConfigItemCreate: ConfigItem->CIXMLData->WarrantyExpirationDate parameter value is not a valid Date format!',
                },
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Wrong Owner Customer',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'Torero Chips',
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                    WarrantyExpirationDate => '1977-12-12',
                    Owner                  => $TestCustomerUserLogin . $RandomID,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemCreate.InvalidParameter',
                    ErrorMessage =>
                        'ConfigItemCreate: ConfigItem->CIXMLData->Owner parameter value is not a valid customer!',
                },
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Wrong Ram Too Many',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'Torero Chips',
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                    WarrantyExpirationDate => '1977-12-12',
                    Owner                  => $TestCustomerUserLogin,
                    Ram                    => [
                        1,
                        2,
                        3,
                        4,
                        5,
                        6,
                        7,
                        8,
                        9,
                        10,
                        11,
                    ],
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemCreate.InvalidParameter',
                    ErrorMessage =>
                        'ConfigItemCreate: ConfigItem->CIXMLData->Ram parameter repetitions is higher than the maxium value!',
                },
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Wrong Attachment',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'Torero Chips',
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                    WarrantyExpirationDate => '1977-12-12',
                    Owner                  => $TestCustomerUserLogin,
                    Ram                    => [
                        4000,
                        4000,
                    ],
                },
                Attachment => 1,
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.InvalidParameter',
                    ErrorMessage => 'ConfigItemCreate: ConfigItem->Attachment parameter is invalid!',
                },
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Missing Attachment->Content',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'Torero Chips',
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                    WarrantyExpirationDate => '1977-12-12',
                    Owner                  => $TestCustomerUserLogin,
                    Ram                    => [
                        4000,
                        4000,
                    ],
                },
                Attachment => [
                    {
                        Test => 1,
                    },
                ],
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.MissingParameter',
                    ErrorMessage => 'ConfigItemCreate: Attachment->Content parameter is missing!',
                },
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Missing Attachment->ContentType',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'Torero Chips',
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                    WarrantyExpirationDate => '1977-12-12',
                    Owner                  => $TestCustomerUserLogin,
                    Ram                    => [
                        4000,
                        4000,
                    ],
                },
                Attachment => [
                    {
                        Content => 'VGhpcyBpcyBhbiBlbmNvZGVkIHRleHQ=',
                    },
                ],
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.MissingParameter',
                    ErrorMessage => 'ConfigItemCreate: Attachment->ContentType parameter is missing!',
                },
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Missing Attachment->Filename',
        SuccessRequest => 1,
        SuccessCreate  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'Torero Chips',
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                    WarrantyExpirationDate => '1977-12-12',
                    Owner                  => $TestCustomerUserLogin,
                    Ram                    => [
                        4000,
                        4000,
                    ],
                },
                Attachment => [
                    {
                        Content     => 'VGhpcyBpcyBhbiBlbmNvZGVkIHRleHQ=',
                        ContentType => 'text/plain',
                    },
                ],
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode    => 'ConfigItemCreate.MissingParameter',
                    ErrorMessage => 'ConfigItemCreate: Attachment->Filename parameter is missing!',
                },
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Correct ConfigItem',
        SuccessRequest => 1,
        SuccessCreate  => 1,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                Name      => 'Test' . $RandomID,
                DeplState => 'Production',
                InciState => 'Operational',
                CIXMLData => {
                    Vendor          => 'Lenovo',
                    Model           => 'Thinkpad',
                    Description     => 'Thinkpad X300',
                    Type            => 'Desktop',
                    Owner           => $TestCustomerUserLogin,
                    SerialNumber    => 'abc12345abc',
                    OperatingSystem => 'CentOS 6.0',
                    CPU             => 'Intel Core i3',
                    Ram             => [
                        '4000',
                        '2000',
                    ],
                    HardDisk => {
                        HardDisk => '/dev',
                        Capacity => '50000',
                    },
                    FQDN => 'hots.example.com',
                    NIC  => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                            IPAddress  => '192.168.30.1',

                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                            IPAddress  => '200.34.56.78',
                        },
                    ],
                    GraphicAdapter         => 'ATI Radeon 300',
                    WarrantyExpirationDate => '1977-12-12',
                    InstallDate            => '1977-12-12',
                    Note                   => 'This is a Demo CI',
                },
                Attachment => [
                    {
                        Content     => 'VGhpcyBpcyBhbiBlbmNvZGVkIHRleHQ=',
                        ContentType => 'text/plain',
                        Filename    => 'My Text.txt',
                    },
                    {
                        Content     => 'VGhpcyBpcyBhbiBlbmNvZGVkIHRleHQ=',
                        ContentType => 'text/plain; charset=iso-8859-1',
                        Filename    => 'My Text2.txt',
                    },
                ],
            },
        },
        Operation => 'ConfigItemCreate',
    },
    {
        Name           => 'Correct ConfigItem without XSLData',
        SuccessRequest => 1,
        SuccessCreate  => 1,
        RequestData    => {
            ConfigItem => {
                Class     => 'Location',
                Name      => 'Test' . $RandomID,
                DeplState => 'Production',
                InciState => 'Operational',
            },
        },
        Operation => 'ConfigItemCreate',
    },
);

# Start testing.
for my $Test (@Tests) {

    # Create local object.
    my $LocalObject = "Kernel::GenericInterface::Operation::ConfigItem::$Test->{Operation}"->new(
        DebuggerObject => $DebuggerObject,
        WebserviceID   => $WebserviceID,
    );

    $Self->Is(
        "Kernel::GenericInterface::Operation::ConfigItem::$Test->{Operation}",
        ref $LocalObject,
        "$Test->{Name} - Create local object",
    );

    # Make a deep copy to avoid changing the definition.
    my $ClonedRequestData = Storable::dclone( $Test->{RequestData} );

    # Start requester with our webservice.
    my $LocalResult = $LocalObject->Run(
        WebserviceID => $WebserviceID,
        Invoker      => $Test->{Operation},
        Data         => {
            UserLogin => $UserLogin,
            Password  => $Password,
            %{ $Test->{RequestData} },
        },
    );

    # Restore cloned data.
    $Test->{RequestData} = $ClonedRequestData;

    # Check result.
    $Self->Is(
        'HASH',
        ref $LocalResult,
        "$Test->{Name} - Local result structure is valid",
    );

    # Create requester object.
    my $RequesterObject = $Kernel::OM->Get('Kernel::GenericInterface::Requester');
    $Self->Is(
        'Kernel::GenericInterface::Requester',
        ref $RequesterObject,
        "$Test->{Name} - Create requester object",
    );

    # Start requester with our webservice.
    my $RequesterResult = $RequesterObject->Run(
        WebserviceID => $WebserviceID,
        Invoker      => $Test->{Operation},
        Data         => {
            SessionID => $NewSessionID,
            %{ $Test->{RequestData} },
        },
    );

    # Check result.
    $Self->Is(
        'HASH',
        ref $RequesterResult,
        "$Test->{Name} - Requester result structure is valid",
    );

    $Self->Is(
        $RequesterResult->{Success},
        $Test->{SuccessRequest},
        "$Test->{Name} - Requester successful result",
    );

    # Tests supposed to succeed.
    if ( $Test->{SuccessCreate} ) {

        # Local results.
        $Self->True(
            $LocalResult->{Data}->{ConfigItemID},
            "$Test->{Name} - Local result ConfigItemID with True.",
        );
        $Self->True(
            $LocalResult->{Data}->{Number},
            "$Test->{Name} - Local result Number with True.",
        );
        $Self->Is(
            $LocalResult->{Data}->{Error},
            undef,
            "$Test->{Name} - Local result Error is undefined.",
        );

        # Requester results.
        $Self->True(
            $RequesterResult->{Data}->{ConfigItemID},
            "$Test->{Name} - Requester result ConfigItemID with True.",
        );
        $Self->True(
            $RequesterResult->{Data}->{Number},
            "$Test->{Name} - Requester result Number with True.",
        );
        $Self->Is(
            $RequesterResult->{Data}->{Error},
            undef,
            "$Test->{Name} - Requester result Error is undefined.",
        );

        # Get the ConfigItem entry (from local result).
        my $LocalVersionData = $ConfigItemObject->VersionGet(
            ConfigItemID => $LocalResult->{Data}->{ConfigItemID},
            UserID       => 1,
        );

        $Self->True(
            IsHashRefWithData($LocalVersionData),
            "$Test->{Name} - created local version strcture with True.",
        );

        # Get the config item entry (from requester result).
        my $RequesterVersionData = $ConfigItemObject->VersionGet(
            ConfigItemID => $RequesterResult->{Data}->{ConfigItemID},
            UserID       => 1,
        );

        $Self->True(
            IsHashRefWithData($RequesterVersionData),
            "$Test->{Name} - created requester config item strcture with True.",
        );

        # Check config item attributes as defined in the test.
        for my $Attribute (qw(Number Class Name InciState DeplState DeplStateType)) {
            if ( $Test->{RequestData}->{ConfigItem}->{$Attribute} ) {
                $Self->Is(
                    $LocalVersionData->{$Attribute},
                    $Test->{RequestData}->{ConfigItem}->{$Attribute},
                    "$Test->{Name} - local ConfigItem->$Attribute" . " match test definition.",
                );
            }
        }

        if ( $Test->{RequestData}->{ConfigItem}->{CIXMLData} ) {

            # Transform XML data to a comparable format.
            my $Definition = $LocalVersionData->{XMLDefinition};

            # Make a deep copy to avoid changing the result.
            my $ClonedXMLData = Storable::dclone( $LocalVersionData->{XMLData} );

            my $FormatedXMLData = $LocalObject->InvertFormatXMLData(
                XMLData => $ClonedXMLData->[1]->{Version},
            );

            my $ReplacedXMLData = $LocalObject->InvertReplaceXMLData(
                XMLData    => $FormatedXMLData,
                Definition => $Definition,
            );

            # Compare XML data.
            $Self->IsDeeply(
                $ReplacedXMLData,
                $Test->{RequestData}->{ConfigItem}->{CIXMLData},
                "$Test->{Name} - local ConfigItem->CIXMLData match test definition.",
            );
        }

        if ( $Test->{RequestData}->{ConfigItem}->{Attachment} ) {

            # Check attachments.
            my @AttachmentList = $ConfigItemObject->ConfigItemAttachmentList(
                ConfigItemID => $RequesterResult->{Data}->{ConfigItemID},
            );

            my @Attachments;
            ATTACHMENT:
            for my $FileName (@AttachmentList) {
                next ATTACHMENT if !$FileName;

                my $Attachment = $ConfigItemObject->ConfigItemAttachmentGet(
                    ConfigItemID => $RequesterResult->{Data}->{ConfigItemID},
                    Filename     => $FileName,
                );

                # Next if not attachment.
                next ATTACHMENT if !IsHashRefWithData($Attachment);

                # Convert content to base64.
                $Attachment->{Content} = encode_base64( $Attachment->{Content}, '' );

                # Delete not needed attibutes.
                for my $Attribute (qw(Preferences Filesize Type)) {
                    delete $Attachment->{$Attribute};
                }
                push @Attachments, $Attachment;
            }

            my @RequestedAttachments;
            if ( ref $Test->{RequestData}->{Attachment} eq 'HASH' ) {
                push @RequestedAttachments, $Test->{RequestData}->{ConfigItem}->{Attachment};
            }
            else {
                @RequestedAttachments = @{ $Test->{RequestData}->{ConfigItem}->{Attachment} };
            }

            $Self->IsDeeply(
                \@Attachments,
                \@RequestedAttachments,
                "$Test->{Name} - local ConfigItem->Attachment match test definition.",
            );
        }

        # Remove attributes that might be different from local and requester responses.
        for my $Attribute (
            qw(ConfigItemID Number CreateTime VersionID LastVersionID)
            )
        {
            delete $LocalVersionData->{$Attribute};
            delete $RequesterVersionData->{$Attribute};
        }

        $Self->IsDeeply(
            $LocalVersionData,
            $RequesterVersionData,
            "$Test->{Name} - Local config item result matched with remote result.",
        );

        # Delete the config items.
        for my $ConfigItemID (
            $LocalResult->{Data}->{ConfigItemID},
            $RequesterResult->{Data}->{ConfigItemID}
            )
        {

            my $ConfigItemDelete = $ConfigItemObject->ConfigItemDelete(
                ConfigItemID => $ConfigItemID,
                UserID       => 1,
            );

            # Sanity check.
            $Self->True(
                $ConfigItemDelete,
                "ConfigItemDelete() successful for ConfigItem ID $ConfigItemID",
            );
        }
    }

    # Tests supposed to fail.
    else {
        $Self->False(
            $LocalResult->{ConfigItemID},
            "$Test->{Name} - Local result ConfigItemID with false.",
        );
        $Self->False(
            $LocalResult->{Number},
            "$Test->{Name} - Local result Number with false.",
        );
        $Self->Is(
            $LocalResult->{Data}->{Error}->{ErrorCode},
            $Test->{ExpectedData}->{Data}->{Error}->{ErrorCode},
            "$Test->{Name} - Local result ErrorCode matched with expected local call result.",
        );
        $Self->Is(
            $LocalResult->{Data}->{Error}->{ErrorMessage},
            $Test->{ExpectedData}->{Data}->{Error}->{ErrorMessage},
            "$Test->{Name} - Local result ErrorMessage matched with expected local call result.",
        );
        $Self->Is(
            $LocalResult->{ErrorMessage},
            $LocalResult->{Data}->{Error}->{ErrorCode}
                . ': '
                . $LocalResult->{Data}->{Error}->{ErrorMessage},
            "$Test->{Name} - Local result ErrorMessage (outside Data hash) matched with concatenation"
                . " of ErrorCode and ErrorMessage within Data hash.",
        );

        # Remove ErrorMessage parameter from direct call.
        # Result to be consistent with SOAP call result.
        if ( $LocalResult->{ErrorMessage} ) {
            delete $LocalResult->{ErrorMessage};
        }

        # Sanity check.
        $Self->False(
            $LocalResult->{ErrorMessage},
            "$Test->{Name} - Local result ErroMessage (outsise Data hash) got removed to compare"
                . " local and remote tests.",
        );

        $Self->IsDeeply(
            $LocalResult,
            $RequesterResult,
            "$Test->{Name} - Local result matched with remote result.",
        );
    }
}

# Clean up webservice.
my $WebserviceDelete = $WebserviceObject->WebserviceDelete(
    ID     => $WebserviceID,
    UserID => 1,
);
$Self->True(
    $WebserviceDelete,
    "Deleted Webservice $WebserviceID",
);

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

use Kernel::GenericInterface::Debugger;
use Kernel::GenericInterface::Operation::ConfigItem::ConfigItemCreate;
use Kernel::GenericInterface::Operation::ConfigItem::ConfigItemGet;
use Kernel::GenericInterface::Operation::ConfigItem::ConfigItemDelete;
use Kernel::System::VariableCheck qw(:all);

# set UserID to root
$Self->{UserID} = 1;

# helper object
# skip SSL certiciate verification
$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        SkipSSLVerify => 1,
    },
);
my $HelperObject = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

my $RandomID = $HelperObject->GetRandomID();

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

# check if SSL Certificate verification is disabled
$Self->Is(
    $ENV{PERL_LWP_SSL_VERIFY_HOSTNAME},
    0,
    'Disabled SSL certiticates verification in environment',
);

# create ConfigItem object
my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

my $TestCustomerUserLogin = $HelperObject->TestCustomerUserCreate();

# create webservice object
my $WebserviceObject = $Kernel::OM->Get('Kernel::System::GenericInterface::Webservice');
$Self->Is(
    'Kernel::System::GenericInterface::Webservice',
    ref $WebserviceObject,
    "Create webservice object",
);

# set webservice name
my $WebserviceName = '-Test-' . $RandomID;

my $WebserviceID = $WebserviceObject->WebserviceAdd(
    Name   => $WebserviceName,
    Config => {
        Debugger => {
            DebugThreshold => 'debug',
        },
        Provider => {
            Transport => {
                Type => '',
            },
        },
    },
    ValidID => 1,
    UserID  => 1,
);
$Self->True(
    $WebserviceID,
    "Added Webservice",
);

# get remote host with some precautions for certain unit test systems
my $Host;
my $FQDN = $ConfigObject->Get('FQDN');

# try to resolve fqdn host
if ( $FQDN ne 'yourhost.example.com' && gethostbyname($FQDN) ) {
    $Host = $FQDN;
}

# try to resolve localhost instead
if ( !$Host && gethostbyname('localhost') ) {
    $Host = 'localhost';
}

# use hardcoded localhost ip address
if ( !$Host ) {
    $Host = '127.0.0.1';
}

# prepare webservice config
my $RemoteSystem =
    $ConfigObject->Get('HttpType')
    . '://'
    . $Host
    . '/'
    . $ConfigObject->Get('ScriptAlias')
    . '/nph-genericinterface.pl/WebserviceID/'
    . $WebserviceID;

my $WebserviceConfig = {

    #    Name => '',
    Description =>
        'Test for ConfigItem Connector using SOAP transport backend.',
    Debugger => {
        DebugThreshold => 'debug',
        TestMode       => 1,
    },
    Provider => {
        Transport => {
            Type   => 'HTTP::SOAP',
            Config => {
                MaxLength => 10_000_000,
                NameSpace => 'http://otrs.org/SoapTestInterface/',
                Endpoint  => $RemoteSystem,
            },
        },
        Operation => {
            ConfigItemDelete => {
                Type => 'ConfigItem::ConfigItemDelete',
            },
            ConfigItemGet => {
                Type => 'ConfigItem::ConfigItemGet',
            },
            ConfigItemCreate => {
                Type => 'ConfigItem::ConfigItemCreate',
            },
            SessionCreate => {
                Type => 'Session::SessionCreate',
            },
        },
    },
    Requester => {
        Transport => {
            Type   => 'HTTP::SOAP',
            Config => {
                NameSpace => 'http://otrs.org/SoapTestInterface/',
                Encoding  => 'UTF-8',
                Endpoint  => $RemoteSystem,
            },
        },
        Invoker => {
            ConfigItemDelete => {
                Type => 'Test::TestSimple',
            },
            ConfigItemGet => {
                Type => 'Test::TestSimple',
            },
            ConfigItemCreate => {
                Type => 'Test::TestSimple',
            },
            SessionCreate => {
                Type => 'Test::TestSimple',
            },
        },
    },
};

# update webservice with real config
my $WebserviceUpdate = $WebserviceObject->WebserviceUpdate(
    ID      => $WebserviceID,
    Name    => $WebserviceName,
    Config  => $WebserviceConfig,
    ValidID => 1,
    UserID  => 1,
);
$Self->True(
    $WebserviceUpdate,
    "Updated Webservice $WebserviceID - $WebserviceName",
);

# debugger object
my $DebuggerObject = Kernel::GenericInterface::Debugger->new(
    DebuggerConfig => {
        DebugThreshold => 'debug',
        TestMode       => 1,
    },
    WebserviceID      => $WebserviceID,
    CommunicationType => 'Provider',
);
$Self->Is(
    ref $DebuggerObject,
    'Kernel::GenericInterface::Debugger',
    'DebuggerObject instanciate correctly',
);

# Get SessionID
# create requester object
my $RequesterSessionObject = $Kernel::OM->Get('Kernel::GenericInterface::Requester');
$Self->Is(
    'Kernel::GenericInterface::Requester',
    ref $RequesterSessionObject,
    "SessionID - Create requester object",
);

# create a new user for current test
my $UserLogin = $HelperObject->TestUserCreate(
    Groups => [ 'admin', 'users', 'itsm-configitem' ],
);

my $Password = $UserLogin;

# start requester with our webservice
my $RequesterSessionResult = $RequesterSessionObject->Run(
    WebserviceID => $WebserviceID,
    Invoker      => 'SessionCreate',
    Data         => {
        UserLogin => $UserLogin,
        Password  => $Password,
    },
);

my $NewSessionID = $RequesterSessionResult->{Data}->{SessionID};

#ConfigItemts container (usefull for lookup compare)
my %AddedConfigItems;
my @AddedConfigItemIDs;

# ConfigItem settings
my @ConfigItems = (
    {
        Class     => 'Computer',
        Name      => 'TestCI-1-' . $RandomID,
        DeplState => 'Production',
        InciState => 'Incident',
        CIXMLData => {
            NIC => {
                NIC        => 'Lo0',
                IPoverDHCP => 'No',
            },
        },
    },
    {
        Class     => 'Computer',
        Name      => 'TestCI-2-' . $RandomID,
        DeplState => 'Repair',
        InciState => 'Warning',
        CIXMLData => {
            Vendor          => 'Lenovo',
            Model           => 'Thinkpad',
            Description     => 'Thinkpad X300',
            Type            => 'Desktop',
            Owner           => $TestCustomerUserLogin,
            SerialNumber    => 'abc12345abc',
            OperatingSystem => 'CentOS 6.0',
            CPU             => 'Intel Core i3',
            Ram             => [
                4000,
                2000,
            ],
            HardDisk => {
                HardDisk => '/dev',
                Capacity => 50000,
            },
            FQDN => 'hots.example.com',
            NIC  => [
                {
                    NIC        => 'Eth0',
                    IPoverDHCP => 'No',
                    IPAddress  => '192.168.30.1',
                },
                {
                    NIC        => 'Eth1',
                    IPoverDHCP => 'Yes',
                    IPAddress  => '200.34.56.78',
                },
            ],
            GraphicAdapter => 'ATI Radeon 300',
            InstallDate    => '1977-12-12',
            Note           => 'This is a Demo CI',
        },
        Attachment => [
            {
                Content     => 'VGhpcyBpcyBhbiBlbmNvZGVkIHRleHQ=',
                ContentType => 'text/plain',
                Filename    => 'My Text.txt',
            },
            {
                Content     => 'VGhpcyBpcyBhbiBlbmNvZGVkIHRleHQ=',
                ContentType => 'text/plain; charset=ISO-8859-1',
                Filename    => 'My Text2.txt',
            },
        ],
    },
    {
        Class     => 'Hardware',
        Name      => 'TestCI-3-' . $RandomID,
        DeplState => 'Production',
        InciState => 'Operational',
        CIXMLData => {
            Vendor       => 'Hardware Vendor',
            Model        => 'Hardware Model',
            Description  => 'Tesitng Hanrdware',
            Type         => 'Camera',
            Owner        => $TestCustomerUserLogin,
            SerialNumber => '123amc456',
        },
    },
    {
        Class     => 'Location',
        Name      => 'TestCI-4-' . $RandomID,
        DeplState => 'Production',
        InciState => 'Operational',
        CIXMLData => {
            Type     => 'Building',
            Phone1   => '123',
            Phone2   => '456',
            Fax      => '789',
            'E-Mail' => 'soemone@example.com',
            Address  => 'Some address',
            Note     => 'This is a test location',
        },
    },
    {
        Class     => 'Network',
        Name      => 'TestCI-5-' . $RandomID,
        DeplState => 'Production',
        InciState => 'Operational',
        CIXMLData => {
            Description    => 'Network CI for testing',
            Type           => 'LAN',
            NetworkAddress => [
                {
                    NetworkAddress => '192.1.1.1',
                    SubnetMask     => '255.255.255.0',
                    Gateway        => '192.1.1.254',
                },
                {
                    NetworkAddress => '192.1.50.1',
                    SubnetMask     => '255.255.255.0',
                    Gateway        => '192.1.50.254',
                },
            ],
            Note => 'This is a test CI',
        },
    },
    {
        Class     => 'Software',
        Name      => 'TestCI-6-' . $RandomID,
        DeplState => 'Production',
        InciState => 'Operational',
        CIXMLData => {
            Vendor       => 'Some software vendor',
            Version      => '1.1.1',
            Description  => 'Some software description',
            Type         => 'Admin Tool',
            Owner        => $TestCustomerUserLogin,
            SerialNumber => '123ABC456',
            LicenceType  => 'Open Source',
            LicenceKey   => [
                {
                    LicenceKey     => '1234',
                    Quantity       => '1',
                    ExpirationDate => '1977-12-12',
                },
                {
                    LicenceKey     => '4567',
                    Quantity       => '3',
                    ExpirationDate => '1977-12-25',
                },
            ],
            Media => 'Download',
            Note  => 'This is a test CI',
        },
    },
    {
        Class     => 'Software',
        Name      => 'TestCI-7-' . $RandomID,
        DeplState => 'Production',
        InciState => 'Operational',
        CIXMLData => {
            Vendor       => 'Some software vendor',
            Version      => '1.1.1',
            Description  => 'Some software description',
            Type         => 'Admin Tool',
            Owner        => $TestCustomerUserLogin,
            SerialNumber => '123ABC456',
            LicenceType  => 'Open Source',
            LicenceKey   => [
                {
                    LicenceKey     => '1234',
                    Quantity       => '1',
                    ExpirationDate => '1977-12-12',
                },
                {
                    LicenceKey     => '4567',
                    Quantity       => '3',
                    ExpirationDate => '1977-12-25',
                },
            ],
            Media => 'Download',
            Note  => 'This is a test CI',
        },
    },
    {
        Class     => 'Software',
        Name      => 'TestCI-8-' . $RandomID,
        DeplState => 'Production',
        InciState => 'Operational',
        CIXMLData => {
            Vendor       => 'Some software vendor',
            Version      => '1.1.1',
            Description  => 'Some software description',
            Type         => 'Admin Tool',
            Owner        => $TestCustomerUserLogin,
            SerialNumber => '123ABC456',
            LicenceType  => 'Open Source',
            LicenceKey   => [
                {
                    LicenceKey     => '1234',
                    Quantity       => '1',
                    ExpirationDate => '1977-12-12',
                },
                {
                    LicenceKey     => '4567',
                    Quantity       => '3',
                    ExpirationDate => '1977-12-25',
                },
            ],
            Media => 'Download',
            Note  => 'This is a test CI',
        },
    },
);

my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $ClassList            = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::Class',
);
my %ReverseClassList = reverse %{$ClassList};
my $InciStateList    = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %ReverseInciStateList = reverse %{$InciStateList};
my $DeplStateList        = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %ReverseDeplStateList = reverse %{$DeplStateList};
for my $ConfigItem (@ConfigItems) {

    # make a deep copy to avoid changing the definition
    my $ClonedConfigItem = Storable::dclone($ConfigItem);

    # create new config item
    my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        ClassID => $ReverseClassList{ $ConfigItem->{Class} },
        UserID  => $Self->{UserID},
    );

    # sanity checks
    $Self->True(
        $ConfigItemID,
        "Added ConfigItem $ConfigItemID",
    );

    my $DefinitionData = $ConfigItemObject->DefinitionGet(
        ClassID => $ReverseClassList{ $ConfigItem->{Class} },
    );

    my $LocalObject = Kernel::GenericInterface::Operation::ConfigItem::ConfigItemCreate->new(
        DebuggerObject => $DebuggerObject,
        WebserviceID   => $WebserviceID,
    );

    my $ReplacedXMLData = $LocalObject->ReplaceXMLData(
        XMLData    => $ConfigItem->{CIXMLData},
        Definition => $DefinitionData->{DefinitionRef},
    );

    my $XMLData = $LocalObject->FormatXMLData(
        XMLData => $ReplacedXMLData,
    );

    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ConfigItemID,
        Name         => $ConfigItem->{Name},
        DefinitionID => $DefinitionData->{DefinitionID},
        DeplStateID  => $ReverseDeplStateList{ $ConfigItem->{DeplState} },
        InciStateID  => $ReverseInciStateList{ $ConfigItem->{InciState} },
        XMLData      => $XMLData,
        UserID       => $Self->{UserID},
    );

    # sanity checks
    $Self->True(
        $ConfigItemID,
        "Added ConfigItem $ConfigItemID",
    );

    my $VersionInfo = $ConfigItemObject->VersionGet(
        ConfigItemID => $ConfigItemID,
    );

    $Self->IsNotDeeply(
        $VersionInfo,
        {},
        "Added ConfigItem $ConfigItemID - Version is not an empty hash",
    );

    # remember the config item
    if ($ConfigItemID) {
        $AddedConfigItems{$ConfigItemID} = $ClonedConfigItem;
        push @AddedConfigItemIDs, $ConfigItemID;
    }
}

# actual tests for ConfigItemDelete operation
my @Tests = (
    {
        Name           => 'Empty Request',
        SuccessRequest => 1,
        SuccessDelete  => 0,
        RequestData    => {},
        ExpectedData   => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemDelete.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemDelete',
    },
    {
        Name           => 'Wrong ConfigItemID format',
        SuccessRequest => 1,
        SuccessDelete  => 0,
        RequestData    => {
            ConfigItemID => {
                Test1 => 1,
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemDelete.WrongStructure',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemDelete',
    },
    {
        Name           => 'Wrong ConfigItemID',
        SuccessRequest => 1,
        SuccessDelete  => 0,
        RequestData    => {
            ConfigItemID => 'NotExistent' . $RandomID,
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemDelete.AccessDenied',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemDelete',
    },
    {
        Name           => 'Wrong ConfigItemID Array',
        SuccessRequest => 1,
        SuccessDelete  => 0,
        RequestData    => {
            ConfigItemID => [ 'NotExistent' . $RandomID ],
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemDelete.AccessDenied',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemDelete',
    },
    {
        Name           => 'Correct ConfigItemID 1',
        SuccessRequest => 1,
        SuccessDelete  => 1,
        RequestData    => {
            ConfigItemID => $AddedConfigItemIDs[0],
        },
        ExpectedData => {
            ConfigItemID => [ $AddedConfigItemIDs[0] ],
        },
        Operation => 'ConfigItemDelete',
    },
    {
        Name           => 'Wrong ConfigItemID 1 (was already deleted before)',
        SuccessRequest => 1,
        SuccessDelete  => 0,
        RequestData    => {
            ConfigItemID => $AddedConfigItemIDs[0],
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemDelete.AccessDenied',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemDelete',
    },
    {
        Name           => 'Correct ConfigItemID 2',
        SuccessRequest => 1,
        SuccessDelete  => 1,
        RequestData    => {
            ConfigItemID => $AddedConfigItemIDs[1],
        },
        ExpectedData => {
            ConfigItemID => [ $AddedConfigItemIDs[1] ],
        },
        Operation => 'ConfigItemDelete',
    },
    {
        Name           => 'Correct ConfigItemID 3',
        SuccessRequest => 1,
        SuccessDelete  => 1,
        RequestData    => {
            ConfigItemID => $AddedConfigItemIDs[2],
        },
        ExpectedData => {},
        Operation    => 'ConfigItemDelete',
    },
    {
        Name           => 'Correct ConfigItemID 4',
        SuccessRequest => 1,
        SuccessDelete  => 1,
        RequestData    => {
            ConfigItemID => $AddedConfigItemIDs[3],
        },
        ExpectedData => {
            ConfigItemID => [ $AddedConfigItemIDs[4] ],
        },
        Operation => 'ConfigItemDelete',
    },
    {
        Name           => 'Correct ConfigItemID 5',
        SuccessRequest => 1,
        SuccessDelete  => 1,
        RequestData    => {
            ConfigItemID => $AddedConfigItemIDs[4],
        },
        ExpectedData => {
            ConfigItemID => [ $AddedConfigItemIDs[4] ],
        },
        Operation => 'ConfigItemDelete',
    },
    {
        Name           => 'Correct ConfigItemID 6',
        SuccessRequest => 1,
        SuccessDelete  => 1,
        RequestData    => {
            ConfigItemID => $AddedConfigItemIDs[5],
        },
        ExpectedData => {
            ConfigItemID => [ $AddedConfigItemIDs[5] ],
        },
        Operation => 'ConfigItemDelete',
    },
    {
        Name           => 'Correct ConfigItemID 7-8',
        SuccessRequest => 1,
        SuccessDelete  => 1,
        RequestData    => {
            ConfigItemID => [
                $AddedConfigItemIDs[6],
                $AddedConfigItemIDs[7],
            ],
        },
        ExpectedData => {
            ConfigItemID => [
                $AddedConfigItemIDs[6],
                $AddedConfigItemIDs[7],
            ],
        },
        Operation => 'ConfigItemDelete',
    },
);

for my $Test (@Tests) {

    # create local object
    my $LocalObject = "Kernel::GenericInterface::Operation::ConfigItem::$Test->{Operation}"->new(
        DebuggerObject => $DebuggerObject,
        WebserviceID   => $WebserviceID,
    );

    $Self->Is(
        "Kernel::GenericInterface::Operation::ConfigItem::$Test->{Operation}",
        ref $LocalObject,
        "$Test->{Name} - Create local object",
    );

    # create requester object
    my $RequesterObject = $Kernel::OM->Get('Kernel::GenericInterface::Requester');
    $Self->Is(
        'Kernel::GenericInterface::Requester',
        ref $RequesterObject,
        "$Test->{Name} - Create requester object",
    );

    # start requester with our webservice
    my $RequesterResult = $RequesterObject->Run(
        WebserviceID => $WebserviceID,
        Invoker      => $Test->{Operation},
        Data         => {
            SessionID => $NewSessionID,
            %{ $Test->{RequestData} },
        },
    );

    # check result
    $Self->Is(
        'HASH',
        ref $RequesterResult,
        "$Test->{Name} - Requester result structure is valid",
    );

    $Self->Is(
        $RequesterResult->{Success},
        $Test->{SuccessRequest},
        "$Test->{Name} - Requester successful result",
    );

    if ( $Test->{SuccessDelete} ) {

        # requester results
        $Self->IsNot(
            $RequesterResult->{Data}->{ConfigItemID},
            undef,
            "$Test->{Name} - Requester result ConfigItemID is not undefined.",
        );

        my $ExpectedResultType = ref $Test->{RequestData}->{ConfigItemID};

        $Self->Is(
            ref $RequesterResult->{Data}->{ConfigItemID},
            $ExpectedResultType,
            "$Test->{Name} - Requester result ConfigItemID reference.",
        );

        if ( $ExpectedResultType eq 'ARRAY' ) {
            $Self->True(
                IsArrayRefWithData( $RequesterResult->{Data}->{ConfigItemID} ),
                "$Test->{Name} - Requester result ConfigItemID reference is not empty.",
            );
        }
    }

    # tests supposed to fail
    else {
        $Self->False(
            $RequesterResult->{ConfigItemID},
            "$Test->{Name} - Local result ConfigItemID with false.",
        );
        $Self->Is(
            $RequesterResult->{Data}->{Error}->{ErrorCode},
            $Test->{ExpectedData}->{Data}->{Error}->{ErrorCode},
            "$Test->{Name} - Requester result ErrorCode matched with expected call result.",
        );
        $Self->True(
            $RequesterResult->{Data}->{Error}->{ErrorMessage},
            "$Test->{Name} - Requester result ErrorMessage with true.",
        );
        $Self->IsNot(
            $RequesterResult->{Data}->{Error}->{ErrorMessage},
            '',
            "$Test->{Name} - Requester result ErrorMessage is not empty.",
        );
    }
}    #end loop

# clean up

# clean up webservice
my $WebserviceDelete = $WebserviceObject->WebserviceDelete(
    ID     => $WebserviceID,
    UserID => $Self->{UserID},
);
$Self->True(
    $WebserviceDelete,
    "Deleted Webservice $WebserviceID",
);

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

use Kernel::GenericInterface::Debugger;
use Kernel::GenericInterface::Operation::ConfigItem::ConfigItemCreate;
use Kernel::GenericInterface::Operation::ConfigItem::ConfigItemGet;
use Kernel::System::VariableCheck qw(:all);

# set UserID to root
$Self->{UserID} = 1;

# helper object
# skip SSL certiciate verification
$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        SkipSSLVerify => 1,
    },
);
my $HelperObject = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

my $RandomID = $HelperObject->GetRandomID();

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

# check if SSL Certificate verification is disabled
$Self->Is(
    $ENV{PERL_LWP_SSL_VERIFY_HOSTNAME},
    0,
    'Disabled SSL certiticates verification in environment',
);

# create ConfigItem object
my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

my $TestCustomerUserLogin = $HelperObject->TestCustomerUserCreate();

# create webservice object
my $WebserviceObject = $Kernel::OM->Get('Kernel::System::GenericInterface::Webservice');
$Self->Is(
    'Kernel::System::GenericInterface::Webservice',
    ref $WebserviceObject,
    "Create webservice object",
);

# set webservice name
my $WebserviceName = '-Test-' . $RandomID;

my $WebserviceID = $WebserviceObject->WebserviceAdd(
    Name   => $WebserviceName,
    Config => {
        Debugger => {
            DebugThreshold => 'debug',
        },
        Provider => {
            Transport => {
                Type => '',
            },
        },
    },
    ValidID => 1,
    UserID  => 1,
);
$Self->True(
    $WebserviceID,
    "Added Webservice",
);

# get remote host with some precautions for certain unit test systems
my $Host;
my $FQDN = $ConfigObject->Get('FQDN');

# try to resolve fqdn host
if ( $FQDN ne 'yourhost.example.com' && gethostbyname($FQDN) ) {
    $Host = $FQDN;
}

# try to resolve localhost instead
if ( !$Host && gethostbyname('localhost') ) {
    $Host = 'localhost';
}

# use hardcoded localhost ip address
if ( !$Host ) {
    $Host = '127.0.0.1';
}

# prepare webservice config
my $RemoteSystem =
    $ConfigObject->Get('HttpType')
    . '://'
    . $Host
    . '/'
    . $ConfigObject->Get('ScriptAlias')
    . '/nph-genericinterface.pl/WebserviceID/'
    . $WebserviceID;

my $WebserviceConfig = {

    #    Name => '',
    Description =>
        'Test for ConfigItem Connector using SOAP transport backend.',
    Debugger => {
        DebugThreshold => 'debug',
        TestMode       => 1,
    },
    Provider => {
        Transport => {
            Type   => 'HTTP::SOAP',
            Config => {
                MaxLength => 10_000_000,
                NameSpace => 'http://otrs.org/SoapTestInterface/',
                Endpoint  => $RemoteSystem,
            },
        },
        Operation => {
            ConfigItemGet => {
                Type => 'ConfigItem::ConfigItemGet',
            },
            ConfigItemCreate => {
                Type => 'ConfigItem::ConfigItemCreate',
            },
            SessionCreate => {
                Type => 'Session::SessionCreate',
            },
        },
    },
    Requester => {
        Transport => {
            Type   => 'HTTP::SOAP',
            Config => {
                NameSpace => 'http://otrs.org/SoapTestInterface/',
                Encoding  => 'UTF-8',
                Endpoint  => $RemoteSystem,
            },
        },
        Invoker => {
            ConfigItemGet => {
                Type => 'Test::TestSimple',
            },
            ConfigItemCreate => {
                Type => 'Test::TestSimple',
            },
            SessionCreate => {
                Type => 'Test::TestSimple',
            },
        },
    },
};

# update webservice with real config
my $WebserviceUpdate = $WebserviceObject->WebserviceUpdate(
    ID      => $WebserviceID,
    Name    => $WebserviceName,
    Config  => $WebserviceConfig,
    ValidID => 1,
    UserID  => 1,
);
$Self->True(
    $WebserviceUpdate,
    "Updated Webservice $WebserviceID - $WebserviceName",
);

# debugger object
my $DebuggerObject = Kernel::GenericInterface::Debugger->new(
    DebuggerConfig => {
        DebugThreshold => 'debug',
        TestMode       => 1,
    },
    WebserviceID      => $WebserviceID,
    CommunicationType => 'Provider',
);
$Self->Is(
    ref $DebuggerObject,
    'Kernel::GenericInterface::Debugger',
    'DebuggerObject instanciate correctly',
);

# Get SessionID
# create requester object
my $RequesterSessionObject = $Kernel::OM->Get('Kernel::GenericInterface::Requester');
$Self->Is(
    'Kernel::GenericInterface::Requester',
    ref $RequesterSessionObject,
    "SessionID - Create requester object",
);

# create a new user for current test
my $UserLogin = $HelperObject->TestUserCreate(
    Groups => [ 'admin', 'users', 'itsm-configitem' ],
);

my $Password = $UserLogin;

# start requester with our webservice
my $RequesterSessionResult = $RequesterSessionObject->Run(
    WebserviceID => $WebserviceID,
    Invoker      => 'SessionCreate',
    Data         => {
        UserLogin => $UserLogin,
        Password  => $Password,
    },
);

my $NewSessionID = $RequesterSessionResult->{Data}->{SessionID};

#ConfigItemts container (usefull for lookup compare)
my %AddedConfigItems;
my @AddedConfigItemIDs;

# ConfigItem settings
my @ConfigItems = (
    {
        Class     => 'Computer',
        Name      => 'TestCI-1-' . $RandomID,
        DeplState => 'Production',
        InciState => 'Incident',
        CIXMLData => {
            NIC => {
                NIC        => 'Lo0',
                IPoverDHCP => 'No',
            },
        },
    },
    {
        Class     => 'Computer',
        Name      => 'TestCI-2-' . $RandomID,
        DeplState => 'Repair',
        InciState => 'Warning',
        CIXMLData => {
            Vendor          => 'Lenovo',
            Model           => 'Thinkpad',
            Description     => 'Thinkpad X300',
            Type            => 'Desktop',
            Owner           => $TestCustomerUserLogin,
            SerialNumber    => 'abc12345abc',
            OperatingSystem => 'CentOS 6.0',
            CPU             => 'Intel Core i3',
            Ram             => [
                4000,
                2000,
            ],
            HardDisk => {
                HardDisk => '/dev',
                Capacity => 50000,
            },
            FQDN => 'hots.example.com',
            NIC  => [
                {
                    NIC        => 'Eth0',
                    IPoverDHCP => 'No',
                    IPAddress  => '192.168.30.1',
                },
                {
                    NIC        => 'Eth1',
                    IPoverDHCP => 'Yes',
                    IPAddress  => '200.34.56.78',
                },
            ],
            GraphicAdapter => 'ATI Radeon 300',
            InstallDate    => '1977-12-12',
            Note           => 'This is a Demo CI',
        },
        Attachment => [
            {
                Content     => 'VGhpcyBpcyBhbiBlbmNvZGVkIHRleHQ=',
                ContentType => 'text/plain',
                Filename    => 'My Text.txt',
            },
            {
                Content     => 'VGhpcyBpcyBhbiBlbmNvZGVkIHRleHQ=',
                ContentType => 'text/plain; charset=ISO-8859-1',
                Filename    => 'My Text2.txt',
            },
        ],
    },
    {
        Class     => 'Hardware',
        Name      => 'TestCI-3-' . $RandomID,
        DeplState => 'Production',
        InciState => 'Operational',
        CIXMLData => {
            Vendor       => 'Hardware Vendor',
            Model        => 'Hardware Model',
            Description  => 'Tesitng Hanrdware',
            Type         => 'Camera',
            Owner        => $TestCustomerUserLogin,
            SerialNumber => '123amc456',
        },
    },
    {
        Class     => 'Location',
        Name      => 'TestCI-4-' . $RandomID,
        DeplState => 'Production',
        InciState => 'Operational',
        CIXMLData => {
            Type     => 'Building',
            Phone1   => '123',
            Phone2   => '456',
            Fax      => '789',
            'E-Mail' => 'soemone@example.com',
            Address  => 'Some address',
            Note     => 'This is a test location',
        },
    },
    {
        Class     => 'Network',
        Name      => 'TestCI-5-' . $RandomID,
        DeplState => 'Production',
        InciState => 'Operational',
        CIXMLData => {
            Description    => 'Network CI for testing',
            Type           => 'LAN',
            NetworkAddress => [
                {
                    NetworkAddress => '192.1.1.1',
                    SubnetMask     => '255.255.255.0',
                    Gateway        => '192.1.1.254',
                },
                {
                    NetworkAddress => '192.1.50.1',
                    SubnetMask     => '255.255.255.0',
                    Gateway        => '192.1.50.254',
                },
            ],
            Note => 'This is a test CI',
        },
    },
    {
        Class     => 'Software',
        Name      => 'TestCI-6-' . $RandomID,
        DeplState => 'Production',
        InciState => 'Operational',
        CIXMLData => {
            Vendor       => 'Some software vendor',
            Version      => '1.1.1',
            Description  => 'Some software description',
            Type         => 'Admin Tool',
            Owner        => $TestCustomerUserLogin,
            SerialNumber => '123ABC456',
            LicenceType  => 'Open Source',
            LicenceKey   => [
                {
                    LicenceKey     => '1234',
                    Quantity       => '1',
                    ExpirationDate => '1977-12-12',
                },
                {
                    LicenceKey     => '4567',
                    Quantity       => '3',
                    ExpirationDate => '1977-12-25',
                },
            ],
            Media => 'Download',
            Note  => 'This is a test CI',
        },
    },
);

my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $ClassList            = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::Class',
);
my %ReverseClassList = reverse %{$ClassList};
my $InciStateList    = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %ReverseInciStateList = reverse %{$InciStateList};
my $DeplStateList        = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %ReverseDeplStateList = reverse %{$DeplStateList};
for my $ConfigItem (@ConfigItems) {

    # make a deep copy to avoid changing the definition
    my $ClonedConfigItem = Storable::dclone($ConfigItem);

    # create new config item
    my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        ClassID => $ReverseClassList{ $ConfigItem->{Class} },
        UserID  => $Self->{UserID},
    );

    # sanity checks
    $Self->True(
        $ConfigItemID,
        "Added ConfigItem $ConfigItemID",
    );

    my $DefinitionData = $ConfigItemObject->DefinitionGet(
        ClassID => $ReverseClassList{ $ConfigItem->{Class} },
    );

    my $LocalObject = Kernel::GenericInterface::Operation::ConfigItem::ConfigItemCreate->new(
        DebuggerObject => $DebuggerObject,
        WebserviceID   => $WebserviceID,
    );

    my $ReplacedXMLData = $LocalObject->ReplaceXMLData(
        XMLData    => $ConfigItem->{CIXMLData},
        Definition => $DefinitionData->{DefinitionRef},
    );

    my $XMLData = $LocalObject->FormatXMLData(
        XMLData => $ReplacedXMLData,
    );

    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ConfigItemID,
        Name         => $ConfigItem->{Name},
        DefinitionID => $DefinitionData->{DefinitionID},
        DeplStateID  => $ReverseDeplStateList{ $ConfigItem->{DeplState} },
        InciStateID  => $ReverseInciStateList{ $ConfigItem->{InciState} },
        XMLData      => $XMLData,
        UserID       => $Self->{UserID},
    );

    # sanity checks
    $Self->True(
        $ConfigItemID,
        "Added ConfigItem $ConfigItemID",
    );

    my $VersionInfo = $ConfigItemObject->VersionGet(
        ConfigItemID => $ConfigItemID,
    );

    $Self->IsNotDeeply(
        $VersionInfo,
        {},
        "Added ConfigItem $ConfigItemID - Version is not an empty hash",
    );

    # remember the config item
    if ($ConfigItemID) {
        $AddedConfigItems{$ConfigItemID} = $ClonedConfigItem;
        push @AddedConfigItemIDs, $ConfigItemID;
    }
}

# actual tests for ConfigItemGet operation
my @Tests = (
    {
        Name           => 'Empty Request',
        SuccessRequest => 1,
        SuccessGet     => 0,
        RequestData    => {},
        ExpectedData   => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemGet.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemGet',
    },
    {
        Name           => 'Wrong ConfigItemID format',
        SuccessRequest => 1,
        SuccessGet     => 0,
        RequestData    => {
            ConfigItemID => {
                Test1 => 1,
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemGet.WrongStructure',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemGet',
    },
    {
        Name           => 'Wrong ConfigItemID',
        SuccessRequest => 1,
        SuccessGet     => 0,
        RequestData    => {
            ConfigItemID => 'NotExistent' . $RandomID,
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemGet.AccessDenied',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemGet',
    },
    {
        Name           => 'Wrong ConfigItemID Array',
        SuccessRequest => 1,
        SuccessGet     => 0,
        RequestData    => {
            ConfigItemID => [ 'NotExistent' . $RandomID, ],
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemGet.AccessDenied',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemGet',
    },
    {
        Name           => 'Correct ConfigItemID 1',
        SuccessRequest => 1,
        SuccessGet     => 1,
        RequestData    => {
            ConfigItemID => $AddedConfigItemIDs[0],
        },
        ExpectedData => {},
        Operation    => 'ConfigItemGet',
    },
    {
        Name           => 'Correct ConfigItemID 2',
        SuccessRequest => 1,
        SuccessGet     => 1,
        RequestData    => {
            ConfigItemID => $AddedConfigItemIDs[1],
        },
        ExpectedData => {},
        Operation    => 'ConfigItemGet',
    },
    {
        Name           => 'Correct ConfigItemID 3',
        SuccessRequest => 1,
        SuccessGet     => 1,
        RequestData    => {
            ConfigItemID => $AddedConfigItemIDs[2],
        },
        ExpectedData => {},
        Operation    => 'ConfigItemGet',
    },
    {
        Name           => 'Correct ConfigItemID 4',
        SuccessRequest => 1,
        SuccessGet     => 1,
        RequestData    => {
            ConfigItemID => $AddedConfigItemIDs[3],
        },
        ExpectedData => {},
        Operation    => 'ConfigItemGet',
    },
    {
        Name           => 'Correct ConfigItemID 5',
        SuccessRequest => 1,
        SuccessGet     => 1,
        RequestData    => {
            ConfigItemID => $AddedConfigItemIDs[4],
        },
        ExpectedData => {},
        Operation    => 'ConfigItemGet',
    },
    {
        Name           => 'Correct ConfigItemID 6',
        SuccessRequest => 1,
        SuccessGet     => 1,
        RequestData    => {
            ConfigItemID => $AddedConfigItemIDs[5],
        },
        ExpectedData => {},
        Operation    => 'ConfigItemGet',
    },
    {
        Name           => 'Correct ConfigItemID 1-6',
        SuccessRequest => 1,
        SuccessGet     => 1,
        RequestData    => {
            ConfigItemID => \@AddedConfigItemIDs,
        },
        ExpectedData => {},
        Operation    => 'ConfigItemGet',
    },
);

for my $Test (@Tests) {

    # create local object
    my $LocalObject = "Kernel::GenericInterface::Operation::ConfigItem::$Test->{Operation}"->new(
        DebuggerObject => $DebuggerObject,
        WebserviceID   => $WebserviceID,
    );

    $Self->Is(
        "Kernel::GenericInterface::Operation::ConfigItem::$Test->{Operation}",
        ref $LocalObject,
        "$Test->{Name} - Create local object",
    );

    # start requester with our webservice
    my $LocalResult = $LocalObject->Run(
        WebserviceID => $WebserviceID,
        Invoker      => $Test->{Operation},
        Data         => {
            UserLogin => $UserLogin,
            Password  => $Password,
            %{ $Test->{RequestData} },
        },
    );

    # check result
    $Self->Is(
        'HASH',
        ref $LocalResult,
        "$Test->{Name} - Local result structure is valid",
    );

    # create requester object
    my $RequesterObject = $Kernel::OM->Get('Kernel::GenericInterface::Requester');
    $Self->Is(
        'Kernel::GenericInterface::Requester',
        ref $RequesterObject,
        "$Test->{Name} - Create requester object",
    );

    # start requester with our webservice
    my $RequesterResult = $RequesterObject->Run(
        WebserviceID => $WebserviceID,
        Invoker      => $Test->{Operation},
        Data         => {
            SessionID => $NewSessionID,
            %{ $Test->{RequestData} },
        },
    );

    # check result
    $Self->Is(
        'HASH',
        ref $RequesterResult,
        "$Test->{Name} - Requester result structure is valid",
    );

    $Self->Is(
        $RequesterResult->{Success},
        $Test->{SuccessRequest},
        "$Test->{Name} - Requester successful result",
    );

    if ( $Test->{SuccessGet} ) {

        # local results
        $Self->IsNot(
            $LocalResult->{Data}->{ConfigItem},
            undef,
            "$Test->{Name} - Local result ConfigItem is not undefined.",
        );
        $Self->Is(
            ref $LocalResult->{Data}->{ConfigItem},
            'ARRAY',
            "$Test->{Name} - Local result ConfigItem reference.",
        );
        $Self->True(
            IsArrayRefWithData( $LocalResult->{Data}->{ConfigItem} ),
            "$Test->{Name} - Local result ConfigItem reference is not empty.",
        );
        $Self->Is(
            $LocalResult->{Data}->{Error},
            undef,
            "$Test->{Name} - Local result Error is undefined.",
        );

        # requester results
        $Self->IsNot(
            $RequesterResult->{Data}->{ConfigItem},
            undef,
            "$Test->{Name} - Requester result ConfigItem is not undefined.",
        );

        my $ExpectedResultType = ref $Test->{RequestData}->{ConfigItemID} || 'HASH';

        $Self->Is(
            ref $RequesterResult->{Data}->{ConfigItem},
            $ExpectedResultType,
            "$Test->{Name} - Requester result ConfigItem reference.",
        );

        if ( $ExpectedResultType eq 'ARRAY' ) {
            $Self->True(
                IsArrayRefWithData( $RequesterResult->{Data}->{ConfigItem} ),
                "$Test->{Name} - Requester result ConfigItem reference is not empty.",
            );
        }
        else {
            $Self->True(
                IsHashRefWithData( $RequesterResult->{Data}->{ConfigItem} ),
                "$Test->{Name} - Requester result ConfigItem reference is not empty.",
            );
        }
        $Self->Is(
            $RequesterResult->{Data}->{Error},
            undef,
            "$Test->{Name} - Requester result Error is undefined.",
        );

        if ( $ExpectedResultType eq 'ARRAY' ) {

            # loop all ConfigItemIDs in the response
            for my $Counter ( 0 .. scalar @{ $Test->{RequestData}->{ConfigItemID} } - 1 ) {

                # check specific attributes
                for my $Attribute (qw(CIXMLData Class DeplState InciState Name)) {
                    $Self->IsDeeply(
                        $LocalResult->{Data}->{ConfigItem}->[$Counter]->{$Attribute},
                        $AddedConfigItems{ $Test->{RequestData}->{ConfigItemID}->[$Counter] }
                            ->{$Attribute},
                        "$Test->{Name} - Local result match ConfigItem->[$Counter] $Attribute create"
                            . " data."
                    );
                }
            }
        }
        else {

            # check specific attributes for the only 1 result
            for my $Attribute (qw(CIXMLData Class DeplState InciState Name)) {
                $Self->IsDeeply(
                    $LocalResult->{Data}->{ConfigItem}->[0]->{$Attribute},
                    $AddedConfigItems{ $Test->{RequestData}->{ConfigItemID} }->{$Attribute},
                    "$Test->{Name} - Local result match ConfigItem $Attribute create data."
                );
            }

            # make Requester result an array for easy compare
            my $RequesterConfigItem = delete $RequesterResult->{Data}->{ConfigItem};
            $RequesterResult->{Data}->{ConfigItem} = [$RequesterConfigItem];
        }

        $Self->IsDeeply(
            $LocalResult,
            $RequesterResult,
            "$Test->{Name} - Local config item result matched with remote result.",
        );
    }

    # tests supposed to fail
    else {
        $Self->False(
            $LocalResult->{ConfigItem},
            "$Test->{Name} - Local result ConfigItemID with false.",
        );
        $Self->False(
            $LocalResult->{Number},
            "$Test->{Name} - Local result Number with false.",
        );
        $Self->Is(
            $LocalResult->{Data}->{Error}->{ErrorCode},
            $Test->{ExpectedData}->{Data}->{Error}->{ErrorCode},
            "$Test->{Name} - Local result ErrorCode matched with expected local call result.",
        );
        $Self->True(
            $LocalResult->{Data}->{Error}->{ErrorMessage},
            "$Test->{Name} - Local result ErrorMessage with true.",
        );
        $Self->IsNot(
            $LocalResult->{Data}->{Error}->{ErrorMessage},
            '',
            "$Test->{Name} - Local result ErrorMessage is not empty.",
        );
        $Self->Is(
            $LocalResult->{ErrorMessage},
            $LocalResult->{Data}->{Error}->{ErrorCode}
                . ': '
                . $LocalResult->{Data}->{Error}->{ErrorMessage},
            "$Test->{Name} - Local result ErrorMessage (outside Data hash) matched with concatenation"
                . " of ErrorCode and ErrorMessage within Data hash.",
        );

        # remove ErrorMessage parameter from direct call
        # result to be consistent with SOAP call result
        if ( $LocalResult->{ErrorMessage} ) {
            delete $LocalResult->{ErrorMessage};
        }

        # sanity check
        $Self->False(
            $LocalResult->{ErrorMessage},
            "$Test->{Name} - Local result ErroMessage (outsise Data hash) got removed to compare"
                . " local and remote tests.",
        );

        $Self->IsDeeply(
            $LocalResult,
            $RequesterResult,
            "$Test->{Name} - Local result matched with remote result.",
        );
    }
}    #end loop

# clean up

# clean up webservice
my $WebserviceDelete = $WebserviceObject->WebserviceDelete(
    ID     => $WebserviceID,
    UserID => $Self->{UserID},
);
$Self->True(
    $WebserviceDelete,
    "Deleted Webservice $WebserviceID",
);

for my $ConfigItemID ( sort keys %AddedConfigItems ) {

    # delete the ConfigItems
    my $SuccessDelete = $ConfigItemObject->ConfigItemDelete(
        ConfigItemID => $ConfigItemID,
        UserID       => $Self->{UserID},
    );

    # sanity check
    $Self->True(
        $SuccessDelete,
        "SuccessDelete() successful for ConfigItem ID $ConfigItemID",
    );
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

use Kernel::GenericInterface::Debugger;
use Kernel::GenericInterface::Operation::ConfigItem::ConfigItemCreate;
use Kernel::GenericInterface::Operation::ConfigItem::ConfigItemSearch;
use Kernel::System::VariableCheck qw(:all);

# set UserID to root
$Self->{UserID} = 1;

# helper object
# skip SSL certiciate verification
$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        SkipSSLVerify => 1,
    },
);
my $HelperObject = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

my $RandomID = $HelperObject->GetRandomID();

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

# check if SSL Certificate verification is disabled
$Self->Is(
    $ENV{PERL_LWP_SSL_VERIFY_HOSTNAME},
    0,
    'Disabled SSL certiticates verification in environment',
);

# create ConfigItem object
my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

my $TestCustomerUserLogin = $HelperObject->TestCustomerUserCreate();

# create webservice object
my $WebserviceObject = $Kernel::OM->Get('Kernel::System::GenericInterface::Webservice');
$Self->Is(
    'Kernel::System::GenericInterface::Webservice',
    ref $WebserviceObject,
    "Create webservice object",
);

# set webservice name
my $WebserviceName = '-Test-' . $RandomID;

my $WebserviceID = $WebserviceObject->WebserviceAdd(
    Name   => $WebserviceName,
    Config => {
        Debugger => {
            DebugThreshold => 'debug',
        },
        Provider => {
            Transport => {
                Type => '',
            },
        },
    },
    ValidID => 1,
    UserID  => 1,
);
$Self->True(
    $WebserviceID,
    "Added Webservice",
);

# get remote host with some precautions for certain unit test systems
my $Host;
my $FQDN = $ConfigObject->Get('FQDN');

# try to resolve fqdn host
if ( $FQDN ne 'yourhost.example.com' && gethostbyname($FQDN) ) {
    $Host = $FQDN;
}

# try to resolve localhost instead
if ( !$Host && gethostbyname('localhost') ) {
    $Host = 'localhost';
}

# use hardcoded localhost ip address
if ( !$Host ) {
    $Host = '127.0.0.1';
}

# prepare webservice config
my $RemoteSystem =
    $ConfigObject->Get('HttpType')
    . '://'
    . $Host
    . '/'
    . $ConfigObject->Get('ScriptAlias')
    . '/nph-genericinterface.pl/WebserviceID/'
    . $WebserviceID;

my $WebserviceConfig = {

    #    Name => '',
    Description =>
        'Test for ConfigItem Connector using SOAP transport backend.',
    Debugger => {
        DebugThreshold => 'debug',
        TestMode       => 1,
    },
    Provider => {
        Transport => {
            Type   => 'HTTP::SOAP',
            Config => {
                MaxLength => 10_000_000,
                NameSpace => 'http://otrs.org/SoapTestInterface/',
                Endpoint  => $RemoteSystem,
            },
        },
        Operation => {
            ConfigItemSearch => {
                Type => 'ConfigItem::ConfigItemSearch',
            },
            ConfigItemCreate => {
                Type => 'ConfigItem::ConfigItemCreate',
            },
            SessionCreate => {
                Type => 'Session::SessionCreate',
            },
        },
    },
    Requester => {
        Transport => {
            Type   => 'HTTP::SOAP',
            Config => {
                NameSpace => 'http://otrs.org/SoapTestInterface/',
                Encoding  => 'UTF-8',
                Endpoint  => $RemoteSystem,
            },
        },
        Invoker => {
            ConfigItemSearch => {
                Type => 'Test::TestSimple',
            },
            ConfigItemCreate => {
                Type => 'Test::TestSimple',
            },
            SessionCreate => {
                Type => 'Test::TestSimple',
            },
        },
    },
};

# update webservice with real config
my $WebserviceUpdate = $WebserviceObject->WebserviceUpdate(
    ID      => $WebserviceID,
    Name    => $WebserviceName,
    Config  => $WebserviceConfig,
    ValidID => 1,
    UserID  => 1,
);
$Self->True(
    $WebserviceUpdate,
    "Updated Webservice $WebserviceID - $WebserviceName",
);

# debugger object
my $DebuggerObject = Kernel::GenericInterface::Debugger->new(
    DebuggerConfig => {
        DebugThreshold => 'debug',
        TestMode       => 1,
    },
    WebserviceID      => $WebserviceID,
    CommunicationType => 'Provider',
);
$Self->Is(
    ref $DebuggerObject,
    'Kernel::GenericInterface::Debugger',
    'DebuggerObject instanciate correctly',
);

# Get SessionID
# create requester object
my $RequesterSessionObject = $Kernel::OM->Get('Kernel::GenericInterface::Requester');
$Self->Is(
    'Kernel::GenericInterface::Requester',
    ref $RequesterSessionObject,
    "SessionID - Create requester object",
);

# create a new user for current test
my $UserLogin = $HelperObject->TestUserCreate(
    Groups => [ 'admin', 'users', 'itsm-configitem' ],
);
my $Password = $UserLogin;

# start requester with our webservice
my $RequesterSessionResult = $RequesterSessionObject->Run(
    WebserviceID => $WebserviceID,
    Invoker      => 'SessionCreate',
    Data         => {
        UserLogin => $UserLogin,
        Password  => $Password,
    },
);

my $NewSessionID = $RequesterSessionResult->{Data}->{SessionID};

#ConfigItemts container (usefull for lookup compare)
my @AddedConfigItemIDs;

# ConfigItem settings
my @ConfigItems = (
    {
        Class     => 'Computer',
        Name      => 'TestCI-1-' . $RandomID,
        DeplState => 'Repair',
        InciState => 'Warning',
        CIXMLData => {
            Vendor          => 'Lenovo',
            Model           => 'Thinkpad',
            Description     => 'Thinkpad X300',
            Type            => 'Desktop',
            Owner           => $TestCustomerUserLogin,
            SerialNumber    => 'abc12345abc',
            OperatingSystem => 'CentOS 6.0',
            CPU             => 'Intel Core i3' . $RandomID,
            Ram             => [
                4000,
                2000,
            ],
            HardDisk => {
                HardDisk => '/dev',
                Capacity => 50000,
            },
            FQDN => 'hots.example.com',
            NIC  => [
                {
                    NIC        => 'Eth0',
                    IPoverDHCP => 'No',
                    IPAddress  => '192.168.30.1',
                },
                {
                    NIC        => 'Eth1',
                    IPoverDHCP => 'Yes',
                    IPAddress  => '200.34.56.78',
                },
            ],
            GraphicAdapter => 'ATI Radeon 300',
            InstallDate    => '1977-12-12',
            Note           => 'This is a Demo CI',
        },
        Attachment => [
            {
                Content     => 'VGhpcyBpcyBhbiBlbmNvZGVkIHRleHQ=',
                ContentType => 'text/plain',
                Filename    => 'My Text.txt',
            },
            {
                Content     => 'VGhpcyBpcyBhbiBlbmNvZGVkIHRleHQ=',
                ContentType => 'text/plain; charset=ISO-8859-1',
                Filename    => 'My Text2.txt',
            },
        ],
    },
    {
        Class     => 'Computer',
        Name      => 'TestCI-2-' . $RandomID,
        DeplState => 'Repair',
        InciState => 'Incident',
        CIXMLData => {
            Vendor          => 'Apple',
            Model           => 'iMac',
            Description     => 'Apple Computer',
            Type            => 'Desktop',
            Owner           => $TestCustomerUserLogin,
            SerialNumber    => 'abc12345abc',
            OperatingSystem => 'OSX 10.8.3',
            CPU             => 'Intel Core i3' . $RandomID,
            Ram             => [
                4000,
                2000,
            ],
            HardDisk => {
                HardDisk => '/dev',
                Capacity => 80000,
            },
            FQDN => 'hots.example.com',
            NIC  => [
                {
                    NIC        => 'En0',
                    IPoverDHCP => 'Yes',
                    IPAddress  => '192.168.30.1',
                },
                {
                    NIC        => 'En1',
                    IPoverDHCP => 'Yes',
                    IPAddress  => '200.34.56.78',
                },
            ],
            GraphicAdapter => 'nvidia',
            InstallDate    => '2012-12-12',
            Note           => 'This is a Demo CI',
        },
        Attachment => [
            {
                Content     => 'VGhpcyBpcyBhbiBlbmNvZGVkIHRleHQ=',
                ContentType => 'text/plain',
                Filename    => 'My Text.txt',
            },
            {
                Content     => 'VGhpcyBpcyBhbiBlbmNvZGVkIHRleHQ=',
                ContentType => 'text/plain; charset=ISO-8859-1',
                Filename    => 'My Text2.txt',
            },
        ],
    },
    {
        Class     => 'Software',
        Name      => 'TestCI-3-' . $RandomID,
        DeplState => 'Production',
        InciState => 'Operational',
        CIXMLData => {
            Vendor       => 'Some software vendor',
            Version      => '1.1.1',
            Description  => 'Some software description',
            Type         => 'Admin Tool',
            Owner        => $TestCustomerUserLogin,
            SerialNumber => '123ABC456',
            LicenceType  => 'Open Source',
            LicenceKey   => [
                {
                    LicenceKey     => '1234',
                    Quantity       => '1',
                    ExpirationDate => '1977-12-12',
                },
                {
                    LicenceKey     => '4567',
                    Quantity       => '3',
                    ExpirationDate => '1977-12-25',
                },
            ],
            Media => 'Download',
            Note  => 'This is a test CI',
        },
    },
    {
        Class     => 'Software',
        Name      => 'TestCI-4-' . $RandomID,
        DeplState => 'Production',
        InciState => 'Operational',
        CIXMLData => {
            Vendor       => 'Some software vendor',
            Version      => '1.1.1',
            Description  => 'Some software description',
            Type         => 'Admin Tool',
            Owner        => $TestCustomerUserLogin,
            SerialNumber => '123ABC456',
            LicenceType  => 'Open Source',
            LicenceKey   => [
                {
                    LicenceKey     => '1234',
                    Quantity       => '4',
                    ExpirationDate => '1977-12-12',
                },
                {
                    LicenceKey     => '4567',
                    Quantity       => '3',
                    ExpirationDate => '1977-12-25',
                },
            ],
            Media => 'CD-ROM',
            Note  => 'This is a test CI',
        },
    },
);

# create local object
my $ConfigItemCreateObject = Kernel::GenericInterface::Operation::ConfigItem::ConfigItemCreate->new(
    DebuggerObject => $DebuggerObject,
    WebserviceID   => $WebserviceID,
);

for my $ConfigItem (@ConfigItems) {

    # make a deep copy to avoid changing the definition
    my $ClonedConfigItem = Storable::dclone($ConfigItem);

    # start requester with our webservice
    # add new ConfigItems
    my $CreateResult = $ConfigItemCreateObject->Run(
        WebserviceID => $WebserviceID,
        Invoker      => 'ConfigItemCreate',
        Data         => {
            UserLogin  => $UserLogin,
            Password   => $Password,
            ConfigItem => $ConfigItem,
        },
    );

    my $ConfigItemID = $CreateResult->{Data}->{ConfigItemID} || '';

    # sanity checks
    $Self->True(
        $ConfigItemID,
        "Added ConfigItem $ConfigItemID - $CreateResult->{Data}->{Number}",
    );

    my $VersionInfo = $ConfigItemObject->VersionGet(
        ConfigItemID => $ConfigItemID,
    );

    $Self->IsNotDeeply(
        $VersionInfo,
        {},
        "Added ConfigItem $ConfigItemID - Version is not an empty hash",
    );

    # remember the config item
    if ($ConfigItemID) {
        push @AddedConfigItemIDs, $ConfigItemID;
    }
}

# actual tests for ConfigItemSearch operation
my @Tests = (
    {
        Name           => 'Empty Request',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {},
        ExpectedData   => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Missing ConfigItem',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            Test => 1
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Wrong ConfigItem format',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => 1
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Missing ConfigItem->Class',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Test => 1,
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid ConfigItem->Class',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class => 'NonExisting' . $RandomID,
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid ConfigItem->CIXMLData',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class     => 'Computer',
                CIXMLData => 1,
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid ConfigItem->InciStates',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class      => 'Computer',
                InciStates => 'NotExistsning' . $RandomID,
                CIXMLData  => {
                    'NotExistsning' . $RandomID => 1
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid ConfigItem->InciStates HASH',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class      => 'Computer',
                InciStates => {
                    Test => 1,
                },
                CIXMLData => {
                    'NotExistsning' . $RandomID => 1
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.WrongStructure',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid ConfigItem->InciStates ARRAY',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class      => 'Computer',
                InciStates => [ 'Incident', 'NotExistsning' . $RandomID ],
                CIXMLData  => {
                    'NotExistsning' . $RandomID => 1
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid ConfigItem->DeplStates',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class      => 'Computer',
                InciStates => [ 'Incident', 'Warning' ],
                DeplStates => 'NotExistsning' . $RandomID,
                CIXMLData  => {
                    'NotExistsning' . $RandomID => 1,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid ConfigItem->DeplStates HASH',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class      => 'Computer',
                InciStates => [ 'Incident', 'Warning' ],
                DeplStates => {
                    Test => 1,
                },
                CIXMLData => {
                    'NotExistsning' . $RandomID => 1
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.WrongStructure',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid ConfigItem->DeplStates ARRAY',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class      => 'Computer',
                InciStates => [ 'Incident', 'Warning' ],
                DeplStates => [ 'Production', 'NotExistsning' . $RandomID ],
                CIXMLData  => {
                    'NotExistsning' . $RandomID => 1
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid ConfigItem->OrderBy HASH',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class      => 'Computer',
                InciStates => [ 'Incident', 'Warning' ],
                DeplStates => [ 'Production', 'Retired' ],
                OrderBy    => {
                    Test => 1,
                },
                CIXMLData => {
                    'NotExistsning' . $RandomID => 1
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.WrongStructure',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid ConfigItem->OrderByDirection HASH',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class            => 'Computer',
                InciStates       => [ 'Incident', 'Warning' ],
                DeplStates       => [ 'Production', 'Retired' ],
                OrderBy          => ['Number'],
                OrderByDirection => {
                    Test => 1,
                },
                CIXMLData => {
                    'NotExistsning' . $RandomID => 1
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.WrongStructure',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid ConfigItem->CreateTimeNewerDate',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class                         => 'Computer',
                InciStates                    => [ 'Incident', 'Warning' ],
                DeplStates                    => [ 'Production', 'Retired' ],
                OrderBy                       => ['Number'],
                OrderByDirection              => ['Down'],
                ConfigItemCreateTimeNewerDate => '2012-02-30',
                CIXMLData                     => {
                    'NotExistsning' . $RandomID => 1
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid ConfigItem->CreateTimeOlderDate',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class                         => 'Computer',
                InciStates                    => [ 'Incident', 'Warning' ],
                DeplStates                    => [ 'Production', 'Retired' ],
                OrderBy                       => ['Number'],
                OrderByDirection              => ['Down'],
                ConfigItemCreateTimeNewerDate => '2012-12-12',
                ConfigItemCreateTimeOlderDate => '2012-02-30',
                CIXMLData                     => {
                    'NotExistsning' . $RandomID => 1
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid ConfigItem->ChangeTimeNewerDate',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class                         => 'Computer',
                InciStates                    => [ 'Incident', 'Warning' ],
                DeplStates                    => [ 'Production', 'Retired' ],
                OrderBy                       => ['Number'],
                OrderByDirection              => ['Down'],
                ConfigItemCreateTimeNewerDate => '2012-12-12',
                ConfigItemCreateTimeOlderDate => '2012-12-12',
                ConfigItemChangeTimeNewerDate => '2012-02-30',
                CIXMLData                     => {
                    'NotExistsning' . $RandomID => 1
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid ConfigItem->ChangeTimeOlderDate',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class                         => 'Computer',
                InciStates                    => [ 'Incident', 'Warning' ],
                DeplStates                    => [ 'Production', 'Retired' ],
                OrderBy                       => ['Number'],
                OrderByDirection              => ['Down'],
                ConfigItemCreateTimeNewerDate => '2012-12-12',
                ConfigItemCreateTimeOlderDate => '2012-12-12',
                ConfigItemChangeTimeNewerDate => '2012-12-12',
                ConfigItemChangeTimeOlderDate => '2012-02-30',
                CIXMLData                     => {
                    'NotExistsning' . $RandomID => 1
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid ConfigItem->Limit',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class                         => 'Computer',
                InciStates                    => [ 'Incident', 'Warning' ],
                DeplStates                    => [ 'Production', 'Retired' ],
                OrderBy                       => ['Number'],
                OrderByDirection              => ['Down'],
                ConfigItemCreateTimeNewerDate => '2012-12-12',
                ConfigItemCreateTimeOlderDate => '2012-12-12',
                ConfigItemChangeTimeNewerDate => '2012-12-12',
                ConfigItemChangeTimeOlderDate => '2012-12-12',
                Limit                         => 'Unlimited',
                CIXMLData                     => {
                    'NotExistsning' . $RandomID => 1
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid CIXMLData General Catalog',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class                         => 'Computer',
                InciStates                    => [ 'Incident', 'Warning' ],
                DeplStates                    => [ 'Production', 'Retired' ],
                OrderBy                       => ['Number'],
                OrderByDirection              => ['Down'],
                ConfigItemCreateTimeNewerDate => '2012-12-12',
                ConfigItemCreateTimeOlderDate => '2012-12-12',
                ConfigItemChangeTimeNewerDate => '2012-12-12',
                ConfigItemChangeTimeOlderDate => '2012-12-12',
                Limit                         => 1,
                CIXMLData                     => {
                    NIC => {
                        IPoverDHCP => 1,
                    },
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid CIXMLData Date',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class                         => 'Computer',
                InciStates                    => [ 'Incident', 'Warning' ],
                DeplStates                    => [ 'Production', 'Retired' ],
                OrderBy                       => ['Number'],
                OrderByDirection              => ['Down'],
                ConfigItemCreateTimeNewerDate => '2012-12-12',
                ConfigItemCreateTimeOlderDate => '2012-12-12',
                ConfigItemChangeTimeNewerDate => '2012-12-12',
                ConfigItemChangeTimeOlderDate => '2012-12-12',
                Limit                         => 1,
                CIXMLData                     => {
                    NIC => {
                        IPoverDHCP => 'Yes',
                    },
                    InstallDate => '2012-02-30',
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid CIXMLData Text',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class                         => 'Software',
                InciStates                    => [ 'Incident', 'Warning' ],
                DeplStates                    => [ 'Production', 'Retired' ],
                OrderBy                       => ['Number'],
                OrderByDirection              => ['Down'],
                ConfigItemCreateTimeNewerDate => '2012-12-12',
                ConfigItemCreateTimeOlderDate => '2012-12-12',
                ConfigItemChangeTimeNewerDate => '2012-12-12',
                ConfigItemChangeTimeOlderDate => '2012-12-12',
                Limit                         => 1,
                CIXMLData                     => {
                    Vendor => 'a' x 51,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Invalid CIXMLData Number',
        SuccessRequest => 1,
        SuccessSearch  => 0,
        RequestData    => {
            ConfigItem => {
                Class                         => 'Software',
                InciStates                    => [ 'Incident', 'Warning' ],
                DeplStates                    => [ 'Production', 'Retired' ],
                OrderBy                       => ['Number'],
                OrderByDirection              => ['Down'],
                ConfigItemCreateTimeNewerDate => '2012-12-12',
                ConfigItemCreateTimeOlderDate => '2012-12-12',
                ConfigItemChangeTimeNewerDate => '2012-12-12',
                ConfigItemChangeTimeOlderDate => '2012-12-12',
                Limit                         => 1,
                CIXMLData                     => {
                    Vendor     => 'a' x 50,
                    LicenceKey => {
                        Quantity => 'NotExisting' . $RandomID,
                    },
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemSearch.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Correct Computer No Limit',
        SuccessRequest => 1,
        SuccessSearch  => 1,
        RequestData    => {
            ConfigItem => {
                Class            => 'Computer',
                InciStates       => [ 'Incident', 'Warning' ],
                DeplStates       => [ 'Production', 'Retired', 'Repair' ],
                OrderBy          => ['Number'],
                OrderByDirection => ['Up'],
                CIXMLData        => {
                    CPU => 'Intel Core i3' . $RandomID,
                    NIC => {
                        IPoverDHCP => 'Yes',
                    },
                },
            },
        },
        ExpectedData => {
            Data => {
                ConfigItemIDs => [ $AddedConfigItemIDs[0], $AddedConfigItemIDs[1] ],
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Correct Computer Limit',
        SuccessRequest => 1,
        SuccessSearch  => 1,
        RequestData    => {
            ConfigItem => {
                Class            => 'Computer',
                InciStates       => [ 'Incident', 'Warning' ],
                DeplStates       => [ 'Production', 'Retired', 'Repair' ],
                OrderBy          => ['Number'],
                OrderByDirection => ['Up'],
                Limit            => 1,
                CIXMLData        => {
                    CPU => 'Intel Core i3' . $RandomID,
                    NIC => {
                        IPoverDHCP => 'Yes',
                    },
                },
            },
        },
        ExpectedData => {
            Data => {
                ConfigItemIDs => [ $AddedConfigItemIDs[0] ],
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
    {
        Name           => 'Correct Software',
        SuccessRequest => 1,
        SuccessSearch  => 1,
        RequestData    => {
            ConfigItem => {
                Class            => 'Software',
                InciStates       => 'Operational',
                DeplStates       => 'Production',
                OrderBy          => ['Number'],
                OrderByDirection => ['Up'],
                CIXMLData        => {
                    Owner      => $TestCustomerUserLogin,
                    LicenceKey => {
                        Quantity => [ '3', '4' ],
                    },
                },
            },
        },
        ExpectedData => {
            Data => {
                ConfigItemIDs => [ $AddedConfigItemIDs[2], $AddedConfigItemIDs[3] ],
            },
            Success => 1,
        },
        Operation => 'ConfigItemSearch',
    },
);

for my $Test (@Tests) {

    # create local object
    my $LocalObject = "Kernel::GenericInterface::Operation::ConfigItem::$Test->{Operation}"->new(
        DebuggerObject => $DebuggerObject,
        WebserviceID   => $WebserviceID,
    );

    $Self->Is(
        "Kernel::GenericInterface::Operation::ConfigItem::$Test->{Operation}",
        ref $LocalObject,
        "$Test->{Name} - Create local object",
    );

    # make a deep copy to avoid changing the definition
    my $ClonedRequestData = Storable::dclone( $Test->{RequestData} );

    # start requester with our webservice
    my $LocalResult = $LocalObject->Run(
        WebserviceID => $WebserviceID,
        Invoker      => $Test->{Operation},
        Data         => {
            UserLogin => $UserLogin,
            Password  => $Password,
            %{ $Test->{RequestData} },
        },
    );

    # restore cloned data
    $Test->{RequestData} = $ClonedRequestData;

    # check result
    $Self->Is(
        'HASH',
        ref $LocalResult,
        "$Test->{Name} - Local result structure is valid",
    );

    # create requester object
    my $RequesterObject = $Kernel::OM->Get('Kernel::GenericInterface::Requester');
    $Self->Is(
        'Kernel::GenericInterface::Requester',
        ref $RequesterObject,
        "$Test->{Name} - Create requester object",
    );

    # start requester with our webservice
    my $RequesterResult = $RequesterObject->Run(
        WebserviceID => $WebserviceID,
        Invoker      => $Test->{Operation},
        Data         => {
            SessionID => $NewSessionID,
            %{ $Test->{RequestData} },
        },
    );

    # check result
    $Self->Is(
        'HASH',
        ref $RequesterResult,
        "$Test->{Name} - Requester result structure is valid",
    );

    $Self->Is(
        $RequesterResult->{Success},
        $Test->{SuccessRequest},
        "$Test->{Name} - Requester successful result",
    );

    if ( $Test->{SuccessSearch} ) {

        # local results
        $Self->IsNot(
            $LocalResult->{Data}->{ConfigItemIDs},
            undef,
            "$Test->{Name} - Local result ConfigItemIDs is not undefined.",
        );
        $Self->Is(
            ref $LocalResult->{Data}->{ConfigItemIDs},
            'ARRAY',
            "$Test->{Name} - Local result ConfigItemIDs reference.",
        );
        $Self->True(
            IsArrayRefWithData( $LocalResult->{Data}->{ConfigItemIDs} ),
            "$Test->{Name} - Local result ConfigItem is a non emptry array with true.",
        );
        $Self->Is(
            $LocalResult->{Data}->{Error},
            undef,
            "$Test->{Name} - Local result Error is undefined.",
        );

        # requester results
        $Self->IsNot(
            $RequesterResult->{Data}->{ConfigItemIDs},
            undef,
            "$Test->{Name} - Requester result ConfigItemIDs is not undefined.",
        );

        if ( ref $RequesterResult->{Data}->{ConfigItemIDs} eq 'ARRAY' ) {
            $Self->True(
                IsArrayRefWithData( $RequesterResult->{Data}->{ConfigItemIDs} ),
                "$Test->{Name} - Requester result ConfigItemIDs is a non emptry array with true.",
            );
        }
        else {
            $Self->True(
                IsNumber( $RequesterResult->{Data}->{ConfigItemIDs} ),
                "$Test->{Name} - Requester result ConfigItemIDs is number with true.",
            );

            # make Requester result an array for easy compare
            my $RequesterConfigItemIDs = delete $RequesterResult->{Data}->{ConfigItemIDs};
            $RequesterResult->{Data}->{ConfigItemIDs} = [$RequesterConfigItemIDs];
        }

        $Self->IsDeeply(
            $LocalResult,
            $Test->{ExpectedData},
            "$Test->{Name} - Local result match expected data.",
        );

        $Self->IsDeeply(
            $LocalResult,
            $RequesterResult,
            "$Test->{Name} - Local config item result matched with remote result.",
        );
    }

    # tests supposed to fail
    else {
        $Self->False(
            $LocalResult->{ConfigItemID},
            "$Test->{Name} - Local result ConfigItemID with false.",
        );
        $Self->False(
            $LocalResult->{Number},
            "$Test->{Name} - Local result Number with false.",
        );
        $Self->Is(
            $LocalResult->{Data}->{Error}->{ErrorCode},
            $Test->{ExpectedData}->{Data}->{Error}->{ErrorCode},
            "$Test->{Name} - Local result ErrorCode matched with expected local call result.",
        );
        $Self->True(
            $LocalResult->{Data}->{Error}->{ErrorMessage},
            "$Test->{Name} - Local result ErrorMessage with true.",
        );
        $Self->IsNot(
            $LocalResult->{Data}->{Error}->{ErrorMessage},
            '',
            "$Test->{Name} - Local result ErrorMessage is not empty.",
        );
        $Self->Is(
            $LocalResult->{ErrorMessage},
            $LocalResult->{Data}->{Error}->{ErrorCode}
                . ': '
                . $LocalResult->{Data}->{Error}->{ErrorMessage},
            "$Test->{Name} - Local result ErrorMessage (outside Data hash) matched with concatenation"
                . " of ErrorCode and ErrorMessage within Data hash.",
        );

        # remove ErrorMessage parameter from direct call
        # result to be consistent with SOAP call result
        if ( $LocalResult->{ErrorMessage} ) {
            delete $LocalResult->{ErrorMessage};
        }

        # sanity check
        $Self->False(
            $LocalResult->{ErrorMessage},
            "$Test->{Name} - Local result ErroMessage (outsise Data hash) got removed to compare"
                . " local and remote tests.",
        );

        $Self->IsDeeply(
            $LocalResult,
            $RequesterResult,
            "$Test->{Name} - Local result matched with remote result.",
        );
    }
}    #end loop

# clean up

# clean up webservice
my $WebserviceDelete = $WebserviceObject->WebserviceDelete(
    ID     => $WebserviceID,
    UserID => $Self->{UserID},
);
$Self->True(
    $WebserviceDelete,
    "Deleted Webservice $WebserviceID",
);

for my $ConfigItemID ( sort @AddedConfigItemIDs ) {

    # delete the ConfigItems
    my $SuccessDelete = $ConfigItemObject->ConfigItemDelete(
        ConfigItemID => $ConfigItemID,
        UserID       => $Self->{UserID},
    );

    # sanity check
    $Self->True(
        $SuccessDelete,
        "SuccessDelete() successful for ConfigItem ID $ConfigItemID",
    );
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

use MIME::Base64;

use Kernel::GenericInterface::Debugger;
use Kernel::GenericInterface::Operation::ConfigItem::ConfigItemCreate;
use Kernel::GenericInterface::Operation::ConfigItem::ConfigItemUpdate;
use Kernel::System::VariableCheck qw(:all);

# set UserID to root
$Self->{UserID} = 1;

# helper object
# skip SSL certiciate verification
$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        SkipSSLVerify => 1,
    },
);
my $HelperObject = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

my $RandomID = $HelperObject->GetRandomID();

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

# check if SSL Certificate verification is disabled
$Self->Is(
    $ENV{PERL_LWP_SSL_VERIFY_HOSTNAME},
    0,
    'Disabled SSL certiticates verification in environment',
);

# create ConfigItem object
my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

my $TestCustomerUserLogin = $HelperObject->TestCustomerUserCreate();

# create webservice object
my $WebserviceObject = $Kernel::OM->Get('Kernel::System::GenericInterface::Webservice');
$Self->Is(
    'Kernel::System::GenericInterface::Webservice',
    ref $WebserviceObject,
    "Create webservice object",
);

# set webservice name
my $WebserviceName = '-Test-' . $RandomID;

my $WebserviceID = $WebserviceObject->WebserviceAdd(
    Name   => $WebserviceName,
    Config => {
        Debugger => {
            DebugThreshold => 'debug',
        },
        Provider => {
            Transport => {
                Type => '',
            },
        },
    },
    ValidID => 1,
    UserID  => 1,
);
$Self->True(
    $WebserviceID,
    "Added Webservice",
);

# get remote host with some precautions for certain unit test systems
my $Host;
my $FQDN = $ConfigObject->Get('FQDN');

# try to resolve fqdn host
if ( $FQDN ne 'yourhost.example.com' && gethostbyname($FQDN) ) {
    $Host = $FQDN;
}

# try to resolve localhost instead
if ( !$Host && gethostbyname('localhost') ) {
    $Host = 'localhost';
}

# use hardcoded localhost ip address
if ( !$Host ) {
    $Host = '127.0.0.1';
}

# prepare webservice config
my $RemoteSystem =
    $ConfigObject->Get('HttpType')
    . '://'
    . $Host
    . '/'
    . $ConfigObject->Get('ScriptAlias')
    . '/nph-genericinterface.pl/WebserviceID/'
    . $WebserviceID;

my $WebserviceConfig = {

    #    Name => '',
    Description =>
        'Test for ConfigItem Connector using SOAP transport backend.',
    Debugger => {
        DebugThreshold => 'debug',
        TestMode       => 1,
    },
    Provider => {
        Transport => {
            Type   => 'HTTP::SOAP',
            Config => {
                MaxLength => 10_000_000,
                NameSpace => 'http://otrs.org/SoapTestInterface/',
                Endpoint  => $RemoteSystem,
            },
        },
        Operation => {
            ConfigItemUpdate => {
                Type => 'ConfigItem::ConfigItemUpdate',
            },
            ConfigItemCreate => {
                Type => 'ConfigItem::ConfigItemCreate',
            },
            SessionCreate => {
                Type => 'Session::SessionCreate',
            },
        },
    },
    Requester => {
        Transport => {
            Type   => 'HTTP::SOAP',
            Config => {
                NameSpace => 'http://otrs.org/SoapTestInterface/',
                Encoding  => 'UTF-8',
                Endpoint  => $RemoteSystem,
            },
        },
        Invoker => {
            ConfigItemCreate => {
                Type => 'Test::TestSimple',
            },
            ConfigItemUpdate => {
                Type => 'Test::TestSimple',
            },
            SessionCreate => {
                Type => 'Test::TestSimple',
            },
        },
    },
};

# update webservice with real config
my $WebserviceUpdate = $WebserviceObject->WebserviceUpdate(
    ID      => $WebserviceID,
    Name    => $WebserviceName,
    Config  => $WebserviceConfig,
    ValidID => 1,
    UserID  => 1,
);
$Self->True(
    $WebserviceUpdate,
    "Updated Webservice $WebserviceID - $WebserviceName",
);

# debugger object
my $DebuggerObject = Kernel::GenericInterface::Debugger->new(
    DebuggerConfig => {
        DebugThreshold => 'debug',
        TestMode       => 1,
    },
    WebserviceID      => $WebserviceID,
    CommunicationType => 'Provider',
);
$Self->Is(
    ref $DebuggerObject,
    'Kernel::GenericInterface::Debugger',
    'DebuggerObject instanciate correctly',
);

# Get SessionID
# create requester object
my $RequesterSessionObject = $Kernel::OM->Get('Kernel::GenericInterface::Requester');
$Self->Is(
    'Kernel::GenericInterface::Requester',
    ref $RequesterSessionObject,
    "SessionID - Create requester object",
);

# create a new user for current test
my $UserLogin = $HelperObject->TestUserCreate(
    Groups => [ 'admin', 'users', 'itsm-configitem' ],
);
my $Password = $UserLogin;

# start requester with our webservice
my $RequesterSessionResult = $RequesterSessionObject->Run(
    WebserviceID => $WebserviceID,
    Invoker      => 'SessionCreate',
    Data         => {
        UserLogin => $UserLogin,
        Password  => $Password,
    },
);

my $NewSessionID = $RequesterSessionResult->{Data}->{SessionID};

# create a new config item with the basic options
my $ConfigItem = {
    Class     => 'Computer',
    Name      => 'TestCI' . $RandomID,
    DeplState => 'Production',
    InciState => 'Incident',
    CIXMLData => {
        NIC => {
            NIC        => 'Lo0',
            IPoverDHCP => 'No',
        },
    },
};

# create local object
my $ConfigItemCreateObject = Kernel::GenericInterface::Operation::ConfigItem::ConfigItemCreate->new(
    DebuggerObject => $DebuggerObject,
    WebserviceID   => $WebserviceID,
);

# start requester with our webservice
my $CreateResult = $ConfigItemCreateObject->Run(
    WebserviceID => $WebserviceID,
    Invoker      => 'ConfigItemCreate',
    Data         => {
        UserLogin  => $UserLogin,
        Password   => $Password,
        ConfigItem => $ConfigItem,
    },
);

my $ConfigItemID = $CreateResult->{Data}->{ConfigItemID} || '';

$Self->True(
    $ConfigItemID,
    "Added ConfigItem $ConfigItemID - $CreateResult->{Data}->{Number}",
);

# actual tests
my @Tests = (
    {
        Name           => 'Empty Request',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {},
        ExpectedData   => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Invalid ConfigItem',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItem => 1,
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Missing ConfigItemID',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItem => {
                Test => 1,
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Invalid ConfigItemID',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => 'NotExisitng' . $RandomID,
            ConfigItem   => {
                Test => 1,
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Missing CIXMLData',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Test => 1,
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Invalid CIXMLData',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                CIXMLData => 1,
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Missing Class',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                CIXMLData => {
                    Test => 1,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Missing Name',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                CIXMLData => {
                    Test => 1,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Missing DeplState',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                CIXMLData => {
                    Test => 1,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Missing InciState',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                CIXMLData => {
                    Test => 1,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Wrong Class',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'NotExisitng' . $RandomID,
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Test => 1,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Wrong DeplState',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production' . $RandomID,
                InciState => 'Incident',
                CIXMLData => {
                    Test => 1,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Wrong InciState',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident' . $RandomID,
                CIXMLData => {
                    Test => 1,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Missing NIC->NIC',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    NIC => {
                        Test => 1,
                    }
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Missing NIC->IpOverDHCP',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    NIC => {
                        NIC => 'Eth0',
                    }
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Missing NIC->NIC in array',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    NIC => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            IPoverDHCP => 'No',
                        },
                    ],
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Missing NIC->IpOverDHCP in array',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    NIC => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC => 'Eth0',
                        },
                    ],
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Wrong NIC->IpOverDHCP General Catalog in Hash',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    NIC => {
                        NIC        => 'Eth0',
                        IPoverDHCP => 'No' . $RandomID,
                    },
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Wrong NIC->IpOverDHCP General Catalog in Array Hash',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    NIC => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No' . $RandomID,
                        },
                    ],
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Wrong Vendor Long Text ',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'a' x 51,
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Wrong WarrantyExpirationDate Date',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'Torero Chips',
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                    WarrantyExpirationDate => '1930-30-30',
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Wrong Owner Customer',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'Torero Chips',
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                    WarrantyExpirationDate => '1977-12-12',
                    Owner                  => $TestCustomerUserLogin . $RandomID,
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Wrong Ram Too Many',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'Torero Chips',
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                    WarrantyExpirationDate => '1977-12-12',
                    Owner                  => $TestCustomerUserLogin,
                    Ram                    => [
                        1,
                        2,
                        3,
                        4,
                        5,
                        6,
                        7,
                        8,
                        9,
                        10,
                        11,
                    ],
                },
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Wrong Attachment',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'Torero Chips',
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                    WarrantyExpirationDate => '1977-12-12',
                    Owner                  => $TestCustomerUserLogin,
                    Ram                    => [
                        4000,
                        4000,
                    ],
                },
                Attachment => 1,
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.InvalidParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Missing Attachment->Content',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'Torero Chips',
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                    WarrantyExpirationDate => '1977-12-12',
                    Owner                  => $TestCustomerUserLogin,
                    Ram                    => [
                        4000,
                        4000,
                    ],
                },
                Attachment => [
                    {
                        Test => 1,
                    },
                ],
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Missing Attachment->ContentType',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'Torero Chips',
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                    WarrantyExpirationDate => '1977-12-12',
                    Owner                  => $TestCustomerUserLogin,
                    Ram                    => [
                        4000,
                        4000,
                    ],
                },
                Attachment => [
                    {
                        Content => 'VGhpcyBpcyBhbiBlbmNvZGVkIHRleHQ=',
                    },
                ],
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Missing Attachment->Filename',
        SuccessRequest => 1,
        SuccessUpdate  => 0,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'TestCI' . $RandomID,
                DeplState => 'Production',
                InciState => 'Incident',
                CIXMLData => {
                    Vendor => 'Torero Chips',
                    NIC    => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                        },
                    ],
                    WarrantyExpirationDate => '1977-12-12',
                    Owner                  => $TestCustomerUserLogin,
                    Ram                    => [
                        4000,
                        4000,
                    ],
                },
                Attachment => [
                    {
                        Content     => 'VGhpcyBpcyBhbiBlbmNvZGVkIHRleHQ=',
                        ContentType => 'text/plain',
                    },
                ],
            },
        },
        ExpectedData => {
            Data => {
                Error => {
                    ErrorCode => 'ConfigItemUpdate.MissingParameter',
                },
            },
            Success => 1,
        },
        Operation => 'ConfigItemUpdate',
    },
    {
        Name           => 'Correct ConfigItem',
        SuccessRequest => 1,
        SuccessUpdate  => 1,
        RequestData    => {
            ConfigItemID => $ConfigItemID,
            ConfigItem   => {
                Class     => 'Computer',
                Name      => 'Test' . $RandomID,
                DeplState => 'Production',
                InciState => 'Operational',
                CIXMLData => {
                    Vendor          => 'Lenovo',
                    Model           => 'Thinkpad',
                    Description     => 'Thinkpad X300',
                    Type            => 'Desktop',
                    Owner           => $TestCustomerUserLogin,
                    SerialNumber    => 'abc12345abc',
                    OperatingSystem => 'CentOS 6.0',
                    CPU             => 'Intel Core i3',
                    Ram             => [
                        '4000',
                        '2000',
                    ],
                    HardDisk => {
                        HardDisk => '/dev',
                        Capacity => '50000',
                    },
                    FQDN => 'hots.example.com',
                    NIC  => [
                        {
                            NIC        => 'Eth0',
                            IPoverDHCP => 'No',
                            IPAddress  => '192.168.30.1',

                        },
                        {
                            NIC        => 'Eth1',
                            IPoverDHCP => 'Yes',
                            IPAddress  => '200.34.56.78',
                        },
                    ],
                    GraphicAdapter         => 'ATI Radeon 300',
                    WarrantyExpirationDate => '1977-12-12',
                    InstallDate            => '1977-12-12',
                    Note                   => 'This is a Demo CI',
                },
            },
        },
        ExpectedData => {},
        Operation    => 'ConfigItemUpdate',
    },
);

# start testing
for my $Test (@Tests) {

    # create local object
    my $LocalObject = "Kernel::GenericInterface::Operation::ConfigItem::$Test->{Operation}"->new(
        DebuggerObject => $DebuggerObject,
        WebserviceID   => $WebserviceID,
    );

    $Self->Is(
        "Kernel::GenericInterface::Operation::ConfigItem::$Test->{Operation}",
        ref $LocalObject,
        "$Test->{Name} - Create local object",
    );

    # make a deep copy to avoid changing the definition
    my $ClonedRequestData = Storable::dclone( $Test->{RequestData} );

    # start requester with our webservice
    my $LocalResult = $LocalObject->Run(
        WebserviceID => $WebserviceID,
        Invoker      => $Test->{Operation},
        Data         => {
            UserLogin => $UserLogin,
            Password  => $Password,
            %{ $Test->{RequestData} },
        },
    );

    # restore cloned data
    $Test->{RequestData} = $ClonedRequestData;

    # check result
    $Self->Is(
        'HASH',
        ref $LocalResult,
        "$Test->{Name} - Local result structure is valid",
    );

    # create requester object
    my $RequesterObject = $Kernel::OM->Get('Kernel::GenericInterface::Requester');
    $Self->Is(
        'Kernel::GenericInterface::Requester',
        ref $RequesterObject,
        "$Test->{Name} - Update requester object",
    );

    # start requester with our webservice
    my $RequesterResult = $RequesterObject->Run(
        WebserviceID => $WebserviceID,
        Invoker      => $Test->{Operation},
        Data         => {
            SessionID => $NewSessionID,
            %{ $Test->{RequestData} },
        },
    );

    # check result
    $Self->Is(
        'HASH',
        ref $RequesterResult,
        "$Test->{Name} - Requester result structure is valid",
    );

    $Self->Is(
        $RequesterResult->{Success},
        $Test->{SuccessRequest},
        "$Test->{Name} - Requester successful result",
    );

    # tests supposed to succeed
    if ( $Test->{SuccessUpdate} ) {

        # local results
        $Self->True(
            $LocalResult->{Data}->{ConfigItemID},
            "$Test->{Name} - Local result ConfigItemID with True.",
        );
        $Self->True(
            $LocalResult->{Data}->{Number},
            "$Test->{Name} - Local result Number with True.",
        );
        $Self->Is(
            $LocalResult->{Data}->{Error},
            undef,
            "$Test->{Name} - Local result Error is undefined.",
        );

        # requester results
        $Self->True(
            $RequesterResult->{Data}->{ConfigItemID},
            "$Test->{Name} - Requester result ConfigItemID with True.",
        );
        $Self->True(
            $RequesterResult->{Data}->{Number},
            "$Test->{Name} - Requester result Number with True.",
        );
        $Self->Is(
            $RequesterResult->{Data}->{Error},
            undef,
            "$Test->{Name} - Requester result Error is undefined.",
        );

        # get the ConfigItem entry (from local result)
        my $LocalVersionData = $ConfigItemObject->VersionGet(
            ConfigItemID => $LocalResult->{Data}->{ConfigItemID},
            UserID       => 1,
        );

        $Self->True(
            IsHashRefWithData($LocalVersionData),
            "$Test->{Name} - created local version strcture with True.",
        );

        # get the config item entry (from requester result)
        my $RequesterVersionData = $ConfigItemObject->VersionGet(
            ConfigItemID => $RequesterResult->{Data}->{ConfigItemID},
            UserID       => 1,
        );

        $Self->True(
            IsHashRefWithData($RequesterVersionData),
            "$Test->{Name} - created requester config item strcture with True.",
        );

        # check config item attributes as defined in the test
        for my $Attribute (qw(Number Class Name InciState DeplState DeplStateType)) {
            if ( $Test->{RequestData}->{ConfigItem}->{$Attribute} ) {
                $Self->Is(
                    $LocalVersionData->{$Attribute},
                    $Test->{RequestData}->{ConfigItem}->{$Attribute},
                    "$Test->{Name} - local ConfigItem->$Attribute" . " match test definition.",
                );
            }
        }

        # transform XML data to a comparable format
        my $Definition = $LocalVersionData->{XMLDefinition};

        # make a deep copy to avoid changing the result
        my $ClonedXMLData = Storable::dclone( $LocalVersionData->{XMLData} );

        my $FormatedXMLData = $LocalObject->InvertFormatXMLData(
            XMLData => $ClonedXMLData->[1]->{Version},
        );

        my $ReplacedXMLData = $LocalObject->InvertReplaceXMLData(
            XMLData    => $FormatedXMLData,
            Definition => $Definition,
        );

        # compare XML data
        $Self->IsDeeply(
            $ReplacedXMLData,
            $Test->{RequestData}->{ConfigItem}->{CIXMLData},
            "$Test->{Name} - local ConfigItem->CIXMLData match test definition.",
        );

        # check attachments
        if ( defined $Test->{RequestData}->{Attachment} ) {
            my @AttachmentList = $ConfigItemObject->ConfigItemAttachmentList(
                ConfigItemID => $RequesterResult->{Data}->{ConfigItemID},
            );

            my @Attachments;
            ATTACHMENT:
            for my $FileName (@AttachmentList) {
                next ATTACHMENT if !$FileName;

                my $Attachment = $ConfigItemObject->ConfigItemAttachmentGet(
                    ConfigItemID => $RequesterResult->{Data}->{ConfigItemID},
                    Filename     => $FileName,
                );

                # next if not attachment
                next ATTACHMENT if !IsHashRefWithData($Attachment);

                # convert content to base64
                $Attachment->{Content} = encode_base64( $Attachment->{Content}, '' );

                # delete not needed attibutes
                for my $Attribute (qw(Preferences Filesize Type)) {
                    delete $Attachment->{$Attribute};
                }
                push @Attachments, $Attachment;
            }

            my @RequestedAttachments;
            if ( ref $Test->{RequestData}->{Attachment} eq 'HASH' ) {
                push @RequestedAttachments, $Test->{RequestData}->{ConfigItem}->{Attachment};
            }
            else {
                @RequestedAttachments = @{ $Test->{RequestData}->{ConfigItem}->{Attachment} };
            }

            $Self->IsDeeply(
                \@Attachments,
                \@RequestedAttachments,
                "$Test->{Name} - local ConfigItem->Attachment match test definition.",
            );
        }

        $Self->IsDeeply(
            $LocalVersionData,
            $RequesterVersionData,
            "$Test->{Name} - Local config item result matched with remote result.",
        );
    }

    # tests supposed to fail
    else {
        $Self->False(
            $LocalResult->{ConfigItemID},
            "$Test->{Name} - Local result ConfigItemID with false.",
        );
        $Self->False(
            $LocalResult->{Number},
            "$Test->{Name} - Local result Number with false.",
        );
        $Self->Is(
            $LocalResult->{Data}->{Error}->{ErrorCode},
            $Test->{ExpectedData}->{Data}->{Error}->{ErrorCode},
            "$Test->{Name} - Local result ErrorCode matched with expected local call result.",
        );
        $Self->True(
            $LocalResult->{Data}->{Error}->{ErrorMessage},
            "$Test->{Name} - Local result ErrorMessage with true.",
        );
        $Self->IsNot(
            $LocalResult->{Data}->{Error}->{ErrorMessage},
            '',
            "$Test->{Name} - Local result ErrorMessage is not empty.",
        );
        $Self->Is(
            $LocalResult->{ErrorMessage},
            $LocalResult->{Data}->{Error}->{ErrorCode}
                . ': '
                . $LocalResult->{Data}->{Error}->{ErrorMessage},
            "$Test->{Name} - Local result ErrorMessage (outside Data hash) matched with concatenation"
                . " of ErrorCode and ErrorMessage within Data hash.",
        );

        # remove ErrorMessage parameter from direct call
        # result to be consistent with SOAP call result
        if ( $LocalResult->{ErrorMessage} ) {
            delete $LocalResult->{ErrorMessage};
        }

        # sanity check
        $Self->False(
            $LocalResult->{ErrorMessage},
            "$Test->{Name} - Local result ErroMessage (outsise Data hash) got removed to compare"
                . " local and remote tests.",
        );

        $Self->IsDeeply(
            $LocalResult,
            $RequesterResult,
            "$Test->{Name} - Local result matched with remote result.",
        );
    }
}

# clean up config items
my $ConfigItemDelete = $ConfigItemObject->ConfigItemDelete(
    ConfigItemID => $ConfigItemID,
    UserID       => 1,
);

# sanity check
$Self->True(
    $ConfigItemDelete,
    "ConfigItemDelete() successful for ConfigItem ID $ConfigItemID",
);

# clean up webservice
my $WebserviceDelete = $WebserviceObject->WebserviceDelete(
    ID     => $WebserviceID,
    UserID => 1,
);
$Self->True(
    $WebserviceDelete,
    "Deleted Webservice $WebserviceID",
);

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;

use utf8;

use vars qw($Self);

my $DBObject             = $Kernel::OM->Get('Kernel::System::DB');
my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');
my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $LinkObject           = $Kernel::OM->Get('Kernel::System::LinkObject');
my $UserObject           = $Kernel::OM->Get('Kernel::System::User');

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# define needed variable
my $RandomID = $Helper->GetRandomID();

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# perform ConfigItemCount to fill the empty fields
$ConfigItemObject->ConfigItemCount();

# create needed users
my @UserIDs;
{

    # disable email checks to create new user
    my $CheckEmailAddressesOrg = $ConfigObject->Get('CheckEmailAddresses') || 1;
    $ConfigObject->Set(
        Key   => 'CheckEmailAddresses',
        Value => 0,
    );

    for my $Counter ( 1 .. 3 ) {

        # create new users for the tests
        my $UserID = $UserObject->UserAdd(
            UserFirstname => 'ITSMConfigItem' . $Counter,
            UserLastname  => 'UnitTest',
            UserLogin     => 'UnitTest-ITSMConfigItem-' . $Counter . $RandomID,
            UserEmail     => 'UnitTest-ITSMConfigItem-' . $Counter . '@localhost',
            ValidID       => 1,
            ChangeUserID  => 1,
        );

        push @UserIDs, $UserID;
    }

    # restore original email check param
    $ConfigObject->Set(
        Key   => 'CheckEmailAddresses',
        Value => $CheckEmailAddressesOrg,
    );
}

my $GeneralCatalogClass = 'UnitTest' . $RandomID;

# add a general catalog test list
for my $Name (qw(Test1 Test2 Test3 Test4)) {

    # add a new item
    my $ItemID = $GeneralCatalogObject->ItemAdd(
        Class   => $GeneralCatalogClass,
        Name    => $Name,
        ValidID => 1,
        UserID  => 1,
    );

    # check item id
    if ( !$ItemID ) {

        $Self->True(
            0,
            "Can't add new general catalog item.",
        );
    }
}

# define the first test definition (all provided data types)
my @ConfigItemPerlDefinitions;
$ConfigItemPerlDefinitions[0] = " [
    {
        Key        => 'Customer1',
        Name       => 'Customer 1',
        Searchable => 1,
        Input      => {
            Type => 'Customer',
        },
    },
    {
        Key        => 'Date1',
        Name       => 'Date 1',
        Searchable => 1,
        Input      => {
            Type => 'Date',
        },
    },
    {
        Key        => 'DateTime1',
        Name       => 'Date Time 1',
        Searchable => 1,
        Input      => {
            Type => 'DateTime',
        },
    },
    {
        Key   => 'Dummy1',
        Name  => 'Dummy 1',
        Input => {
            Type => 'Dummy',
        },
    },
    {
        Key        => 'GeneralCatalog1',
        Name       => 'GeneralCatalog 1',
        Searchable => 1,
        Input      => {
            Type  => 'GeneralCatalog',
            Class => '$GeneralCatalogClass',
        },
    },
    {
        Key        => 'Integer1',
        Name       => 'Integer 1',
        Searchable => 1,
        Input      => {
            Type => 'Integer',
        },
    },
    {
        Key        => 'Text1',
        Name       => 'Text 1',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 50,
        },
    },
    {
        Key        => 'TextArea1',
        Name       => 'TextArea 1',
        Searchable => 1,
        Input      => {
            Type => 'TextArea',
        },
    },
] ";

# define the second test definition (sub data types)
$ConfigItemPerlDefinitions[1] = " [
    {
        Key        => 'Main1',
        Name       => 'Main 1',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 50,
        },
        CountMax => 10,
        Sub => [
            {
                Key        => 'Main1Sub1',
                Name       => 'Main 1 Sub 1',
                Searchable => 1,
                Input      => {
                    Type      => 'Text',
                    Size      => 50,
                    MaxLength => 50,
                },
                CountMax => 10,
                Sub => [
                    {
                        Key        => 'Main1Sub1SubSub1',
                        Name       => 'Main 1 Sub 1 SubSub 1',
                        Searchable => 1,
                        Input      => {
                            Type      => 'Text',
                            Size      => 50,
                            MaxLength => 50,
                        },
                        CountMax => 10,
                    },
                    {
                        Key        => 'Main1Sub1SubSub2',
                        Name       => 'Main 1 Sub 1 SubSub 2',
                        Searchable => 1,
                        Input      => {
                            Type => 'TextArea',
                        },
                        CountMax => 10,
                    },
                ],
            },
            {
                Key        => 'Main1Sub2',
                Name       => 'Main 1 Sub 2',
                Searchable => 1,
                Input      => {
                    Type => 'TextArea',
                },
                CountMax => 10,
            },
        ],
    },
    {
        Key        => 'Main2',
        Name       => 'Main 2',
        Searchable => 1,
        Input      => {
            Type => 'TextArea',
        },
        CountMax => 10,
        Sub => [
            {
                Key        => 'Main2Sub1',
                Name       => 'Main 2 Sub 1',
                Searchable => 1,
                Input      => {
                    Type      => 'Text',
                    Size      => 50,
                    MaxLength => 50,
                },
                CountMax => 10,
            },
            {
                Key        => 'Main2Sub2',
                Name       => 'Main 2 Sub 2',
                Searchable => 1,
                Input      => {
                    Type => 'TextArea',
                },
                CountMax => 10,
            },
        ],
    },
] ";

# define the third test definition (especially for search tests with XMLData)
$ConfigItemPerlDefinitions[2] = " [
    {
        Key        => 'Customer1',
        Name       => 'Customer 1',
        Searchable => 1,
        Input      => {
            Type => 'Customer',
        },
    },
    {
        Key        => 'Date1',
        Name       => 'Date 1',
        Searchable => 1,
        Input      => {
            Type => 'Date',
        },
    },
    {
        Key        => 'DateTime1',
        Name       => 'Date Time 1',
        Searchable => 1,
        Input      => {
            Type => 'DateTime',
        },
    },
] ";

# define the fourth test definition (only for search tests)
$ConfigItemPerlDefinitions[3] = " [
    {
        Key        => 'Customer1',
        Name       => 'Customer 1',
        Searchable => 1,
        Input      => {
            Type => 'Customer',
        },
    },
] ";

my $YAMLObject = $Kernel::OM->Get('Kernel::System::YAML');

my @ConfigItemDefinitions;
for my $PerlDefinition (@ConfigItemPerlDefinitions) {
    my $YAMLDefinition = $YAMLObject->Dump(
        Data => eval $PerlDefinition,    ## no critic
    );
    push @ConfigItemDefinitions, $YAMLDefinition;
}

# add the test classes
my @ConfigItemClassIDs;
my @ConfigItemClasses;
my @ConfigItemDefinitionIDs;
for my $Definition (@ConfigItemDefinitions) {

    # generate a random name
    my $ClassName = 'UnitTest' . $Helper->GetRandomID();

    # add an unittest config item class
    my $ClassID = $GeneralCatalogObject->ItemAdd(
        Class   => 'ITSM::ConfigItem::Class',
        Name    => $ClassName,
        ValidID => 1,
        UserID  => 1,
    );

    # check class id
    if ( !$ClassID ) {

        $Self->True(
            0,
            "Can't add new config item class.",
        );
    }

    push @ConfigItemClassIDs, $ClassID;
    push @ConfigItemClasses,  $ClassName;

    # add a definition to the class
    my $DefinitionID = $ConfigItemObject->DefinitionAdd(
        ClassID    => $ClassID,
        Definition => $Definition,
        UserID     => 1,
    );

    # check definition id
    if ( !$DefinitionID ) {

        $Self->True(
            0,
            "Can't add new config item definition.",
        );
    }

    push @ConfigItemDefinitionIDs, $DefinitionID;
}

# test DefinitionList for those simple cases
my $Counter = 0;
for my $ClassID (@ConfigItemClassIDs) {
    my $DefinitionListRef = $ConfigItemObject->DefinitionList(
        ClassID => $ClassID,
    );

    # expect a single definition per config item class
    $Self->Is(
        scalar @{$DefinitionListRef},
        1,
        "DefinitionList() for class id $ClassID: got a single result",
    );

    # expect the remembered definition id in the first definition
    $Self->Is(
        $DefinitionListRef->[0]->{DefinitionID},
        $ConfigItemDefinitionIDs[$Counter],
        "DefinitionList() for class id $ClassID: got expected definition id",
    );

    $Counter++;
}

# create some random numbers
my @ConfigItemNumbers;
for ( 1 .. 100 ) {
    push @ConfigItemNumbers, $Helper->GetRandomNumber();
}

# get class list
my $ClassList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::Class',
);
my %ClassListReverse = reverse %{$ClassList};

# get deployment state list
my $DeplStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %DeplStateListReverse = reverse %{$DeplStateList};

# get incident state list
my $InciStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %InciStateListReverse = reverse %{$InciStateList};

# get general catalog test list
my $GeneralCatalogList = $GeneralCatalogObject->ItemList(
    Class => $GeneralCatalogClass,
);
my %GeneralCatalogListReverse = reverse %{$GeneralCatalogList};

# ------------------------------------------------------------ #
# define general config item tests
# ------------------------------------------------------------ #

my $ConfigItemTests = [

    # ConfigItemAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                UserID => 1,
            },
        },
    },

    # ConfigItemAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                ClassID => $ConfigItemClassIDs[0],
            },
        },
    },

    # invalid class id is given (check return false)
    {
        SourceData => {
            ConfigItemAdd => {
                ClassID => $ConfigItemClassIDs[-1] + 1,
                UserID  => 1,
            },
        },
    },

    # all required config item values are given (check returned config item values)
    {
        SourceData => {
            ConfigItemAdd => {
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
        },
        ReferenceData => {
            ConfigItemGet => {
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # all required config item values are given (check number attribute)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[0],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => $UserIDs[1],
            },
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[0],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => $UserIDs[1],
                ChangeBy         => $UserIDs[1],
            },
        },
    },

    # config item with this number already exists (check return false)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[0],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
        },
    },

    # VersionAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[1],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[1],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # VersionAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[2],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name        => 'UnitTest - Class 1 ConfigItem 2 Version 1',
                    DeplStateID => $DeplStateListReverse{Production},
                    InciStateID => $InciStateListReverse{Operational},
                    UserID      => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[2],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # VersionAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[3],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 3 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[3],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # VersionAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[4],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 4 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[4],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # VersionAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[5],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => $UserIDs[1],
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 5 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[5],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => $UserIDs[1],
                ChangeBy         => $UserIDs[1],
            },
        },
    },

    # invalid deployment state id is given (check returned values)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[6],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 6 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => 9999999,
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[6],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # invalid incident state id is given (check returned values)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[7],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 7 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => 9999999,
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[7],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # all required values are given (check returned values)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[8],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 8 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[8],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Operational},
                CurInciState     => 'Operational',
                CurInciStateType => 'operational',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[8],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Class 1 ConfigItem 8 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
        },
    },

    # all required values are given (general check with two versions)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[9],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 9 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 9 Version 2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => $UserIDs[1],
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[9],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Operational},
                CurInciState     => 'Operational',
                CurInciStateType => 'operational',
                CreateBy         => 1,
                ChangeBy         => $UserIDs[1],
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[9],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Class 1 ConfigItem 9 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => 1,
                },
                {
                    Number           => $ConfigItemNumbers[9],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Class 1 ConfigItem 9 Version 2',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => $UserIDs[1],
                },
            ],
        },
    },

    # all required values are given (check the calculation of deployment and incident state)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[10],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 10 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Planned},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 10 Version 2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[10],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Maintenance},
                CurDeplState     => 'Maintenance',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[10],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Class 1 ConfigItem 10 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Planned},
                    DeplState        => 'Planned',
                    DeplStateType    => 'preproductive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
                {
                    Number           => $ConfigItemNumbers[10],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Class 1 ConfigItem 10 Version 2',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Maintenance},
                    DeplState        => 'Maintenance',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
        },
    },

    # add config item only for later search tests
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[50],
                ClassID => $ConfigItemClassIDs[2],
                UserID  => $UserIDs[2],
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 3 ConfigItem 1 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[2],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => $UserIDs[2],
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[50],
                ClassID          => $ConfigItemClassIDs[2],
                Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => $UserIDs[2],
                ChangeBy         => $UserIDs[2],
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[50],
                    ClassID          => $ConfigItemClassIDs[2],
                    Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                    Name             => 'UnitTest - Class 3 ConfigItem 1 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[2],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => $UserIDs[2],
                },
            ],
        },
    },

    # add config item only for later search tests
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[51],
                ClassID => $ConfigItemClassIDs[2],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 3 ConfigItem 2 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[2],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Class 3 ConfigItem 2 Version 2',
                    DefinitionID => $ConfigItemDefinitionIDs[2],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => $UserIDs[1],
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[51],
                ClassID          => $ConfigItemClassIDs[2],
                Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                CurDeplStateID   => $DeplStateListReverse{Maintenance},
                CurDeplState     => 'Maintenance',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => 1,
                ChangeBy         => $UserIDs[1],
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[51],
                    ClassID          => $ConfigItemClassIDs[2],
                    Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                    Name             => 'UnitTest - Class 3 ConfigItem 2 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[2],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
                {
                    Number           => $ConfigItemNumbers[51],
                    ClassID          => $ConfigItemClassIDs[2],
                    Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                    Name             => 'UnitTest - Class 3 ConfigItem 2 Version 2',
                    DefinitionID     => $ConfigItemDefinitionIDs[2],
                    DeplStateID      => $DeplStateListReverse{Maintenance},
                    DeplState        => 'Maintenance',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => $UserIDs[1],
                },
            ],
        },
    },

    # add config item only for later search tests, including XMLData
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[52],
                ClassID => $ConfigItemClassIDs[2],
                UserID  => $UserIDs[2],
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 3 ConfigItem 3 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[2],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => $UserIDs[2],
                    XMLData      => [
                        undef,
                        {
                            Version => [
                                undef,
                                {
                                    Customer1 => [
                                        undef,
                                        {
                                            Content => 'dummy_customer_for_unitest',
                                        },
                                    ],
                                    Date1 => [
                                        undef,
                                        {
                                            Content => '2010-02-12',
                                        },
                                    ],
                                    DateTime1 => [
                                        undef,
                                        {
                                            Content => '2010-02-12 09:14',
                                        },
                                    ],
                                },
                            ],
                        },
                    ],
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[52],
                ClassID          => $ConfigItemClassIDs[2],
                Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => $UserIDs[2],
                ChangeBy         => $UserIDs[2],
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[52],
                    ClassID          => $ConfigItemClassIDs[2],
                    Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                    Name             => 'UnitTest - Class 3 ConfigItem 3 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[2],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [
                        undef,
                        {
                            'TagKey'  => '[1]',
                            'Version' => [
                                undef,
                                {
                                    'Customer1' => [
                                        undef,
                                        {
                                            'Content' => 'dummy_customer_for_unitest',
                                            'TagKey'  => '[1]{\'Version\'}[1]{\'Customer1\'}[1]',
                                        },
                                    ],
                                    'Date1' => [
                                        undef,
                                        {
                                            'Content' => '2010-02-12',
                                            'TagKey'  => '[1]{\'Version\'}[1]{\'Date1\'}[1]',
                                        },
                                    ],
                                    'DateTime1' => [
                                        undef,
                                        {
                                            'Content' => '2010-02-12 09:14',
                                            'TagKey'  => '[1]{\'Version\'}[1]{\'DateTime1\'}[1]',
                                        },
                                    ],
                                    'TagKey' => '[1]{\'Version\'}[1]',
                                },
                            ],
                        },
                    ],
                    CreateBy => $UserIDs[2],
                },
            ],
        },
    },

    # add config item only for later search tests
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[60],
                ClassID => $ConfigItemClassIDs[3],
                UserID  => $UserIDs[1],
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 4 ConfigItem 1 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[2],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => $UserIDs[1],
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[60],
                ClassID          => $ConfigItemClassIDs[3],
                Class            => $ClassList->{ $ConfigItemClassIDs[3] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Operational},
                CurInciState     => 'Operational',
                CurInciStateType => 'operational',
                CreateBy         => $UserIDs[1],
                ChangeBy         => $UserIDs[1],
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[60],
                    ClassID          => $ConfigItemClassIDs[3],
                    Class            => $ClassList->{ $ConfigItemClassIDs[3] },
                    Name             => 'UnitTest - Class 4 ConfigItem 1 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[2],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => $UserIDs[1],
                },
            ],
        },
    },

    # added to check history functions
    # all required values are given (check the calculation of deployment and incident state)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[70],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - HistoryTest',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Planned},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - HistoryTest Version 2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[70],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Maintenance},
                CurDeplState     => 'Maintenance',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[70],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - HistoryTest',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Planned},
                    DeplState        => 'Planned',
                    DeplStateType    => 'preproductive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
                {
                    Number           => $ConfigItemNumbers[70],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - HistoryTest Version 2',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Maintenance},
                    DeplState        => 'Maintenance',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
            HistoryGet => [
                {
                    HistoryType   => 'ConfigItemCreate',
                    HistoryTypeID => 1,
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'VersionCreate',
                    HistoryTypeID => 6,
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'DefinitionUpdate',
                    HistoryTypeID => 8,
                    Comment       => $ConfigItemDefinitionIDs[0],
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'NameUpdate',
                    HistoryTypeID => 5,
                    Comment       => 'UnitTest - HistoryTest%%',
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'IncidentStateUpdate',
                    HistoryTypeID => 9,
                    Comment       => $InciStateListReverse{Operational} . '%%',
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'DeploymentStateUpdate',
                    HistoryTypeID => 10,
                    Comment       => $DeplStateListReverse{Planned} . '%%',
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'VersionCreate',
                    HistoryTypeID => 6,
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'NameUpdate',
                    HistoryTypeID => 5,
                    Comment =>
                        'UnitTest - HistoryTest Version 2%%UnitTest - HistoryTest',
                    CreateBy => 1,
                },
                {
                    HistoryType   => 'IncidentStateUpdate',
                    HistoryTypeID => 9,
                    Comment       => $InciStateListReverse{Incident} . '%%'
                        . $InciStateListReverse{Operational},
                    CreateBy => 1,
                },
                {
                    HistoryType   => 'DeploymentStateUpdate',
                    HistoryTypeID => 10,
                    Comment       => $DeplStateListReverse{Maintenance} . '%%'
                        . $DeplStateListReverse{Planned},
                    CreateBy => 1,
                },
            ],
        },
    },

    # added for Bug4196
    # all required values are given (check the calculation of deployment and incident state)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[71],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Bugfix4196',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Planned},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Bugfix4196 V2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Bugfix4196 V2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Bugfix4196 V2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[71],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Maintenance},
                CurDeplState     => 'Maintenance',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[71],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Bugfix4196',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Planned},
                    DeplState        => 'Planned',
                    DeplStateType    => 'preproductive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
                {
                    Number           => $ConfigItemNumbers[71],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Bugfix4196 V2',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Maintenance},
                    DeplState        => 'Maintenance',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
        },
    },

    # added for Bug 4377 - CI-A
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[72],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Bugfix4377 - CI-A',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[72],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Operational},
                CurInciState     => 'Operational',
                CurInciStateType => 'operational',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[72],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Bugfix4377 - CI-A',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
        },
    },

    # added for Bug 4377 - CI-B
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[73],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Bugfix4377 - CI-B',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[73],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Operational},
                CurInciState     => 'Operational',
                CurInciStateType => 'operational',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[73],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Bugfix4377 - CI-B',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
        },
    },

];

# ------------------------------------------------------------ #
# run general config item tests
# ------------------------------------------------------------ #

my $TestCount = 1;
my @ConfigItemIDs;

TEST:
for my $Test ( @{$ConfigItemTests} ) {

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

        $Self->True(
            0,
            "Test $TestCount: No SourceData found for this test.",
        );

        next TEST;
    }

    # extract source data
    my $SourceData = $Test->{SourceData};

    # add a new config item
    my $ConfigItemID;
    if ( $SourceData->{ConfigItemAdd} ) {

        # add the new config item
        $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
            %{ $SourceData->{ConfigItemAdd} },
        );

        if ($ConfigItemID) {
            push @ConfigItemIDs, $ConfigItemID;
        }
    }

    # check the config item
    if ( $Test->{ReferenceData} && $Test->{ReferenceData}->{ConfigItemGet} ) {

        $Self->True(
            $ConfigItemID,
            "Test $TestCount: ConfigItemAdd() - Add new config item. Insert success.",
        );

        next TEST if !$ConfigItemID;
    }
    else {

        $Self->False(
            $ConfigItemID,
            "Test $TestCount: ConfigItemAdd() - Add new config item. Return false.",
        );
    }

    # add all defined versions
    my @VersionIDs;
    my %VersionIDsSeen;
    if ( $SourceData->{VersionAdd} ) {

        for my $Version ( @{ $SourceData->{VersionAdd} } ) {

            if ($ConfigItemID) {
                $Version->{ConfigItemID} = $ConfigItemID;
            }

            # add a new version
            my $VersionID = $ConfigItemObject->VersionAdd(
                %{$Version},
            );

            if ($VersionID) {
                push @VersionIDs, $VersionID if !$VersionIDsSeen{$VersionID}++;
            }
        }
    }

    # check the config item
    my $ConfigItemData;
    if ( $Test->{ReferenceData} && $Test->{ReferenceData}->{ConfigItemGet} ) {

        # get the config item data
        $ConfigItemData = $ConfigItemObject->ConfigItemGet(
            ConfigItemID => $ConfigItemID,
        );

        if ( !$ConfigItemData ) {

            $Self->True(
                0,
                "Test $TestCount: ConfigItemGet() - get config item data."
            );
        }

        # check all config item attributes
        my $Counter = 0;
        for my $Attribute ( sort keys %{ $Test->{ReferenceData}->{ConfigItemGet} } ) {

            # set content if values are undef
            if ( !defined $ConfigItemData->{$Attribute} ) {
                $ConfigItemData->{$Attribute} = 'UNDEF-unittest';
            }
            if ( !defined $Test->{ReferenceData}->{ConfigItemGet}->{$Attribute} ) {
                $Test->{ReferenceData}->{ConfigItemGet}->{$Attribute} = 'UNDEF-unittest';
            }

            # check attributes
            $Self->Is(
                $ConfigItemData->{$Attribute},
                $Test->{ReferenceData}->{ConfigItemGet}->{$Attribute},
                "Test $TestCount: ConfigItemGet() - $Attribute",
            );

            $Counter++;
        }
    }

    # check the versions
    if (
        $Test->{ReferenceData}
        && $Test->{ReferenceData}->{VersionGet}
        && @{ $Test->{ReferenceData}->{VersionGet} }
        )
    {

        $Self->Is(
            scalar @VersionIDs,
            scalar @{ $Test->{ReferenceData}->{VersionGet} },
            "Test $TestCount: VersionAdd() - correct number of versions",
        );

        next TEST if !$ConfigItemID;
    }
    else {

        $Self->False(
            scalar @VersionIDs,
            "Test $TestCount: VersionAdd() - no versions exits",
        );
    }

    next TEST if !$Test->{ReferenceData};
    next TEST if !$Test->{ReferenceData}->{VersionGet};

    my $Counter           = 0;
    my $LastVersionIDMust = 'UNDEF-unittest';

    # Refresh ITSMConfigurationManagement cache to avoid issues with previously cached values.
    $Kernel::OM->Get('Kernel::System::Cache')->CleanUp(
        Type => 'ITSMConfigurationManagement',
    );

    VERSIONID:
    for my $VersionID (@VersionIDs) {

        # get this version
        my $VersionData = $ConfigItemObject->VersionGet(
            VersionID  => $VersionID,
            XMLDataGet => 1,
        );

        if ( !$VersionData ) {

            $Self->True(
                0,
                "Test $TestCount: VersionGet() - get version data."
            );

            next VERSIONID;
        }

        # save last version id
        $LastVersionIDMust = $VersionData->{VersionID};

        # check all version attributes
        for my $Attribute ( sort keys %{ $Test->{ReferenceData}->{VersionGet}->[$Counter] } ) {

            # extract the needed attributes
            my $VersionAttribute   = $VersionData->{$Attribute};
            my $ReferenceAttribute = $Test->{ReferenceData}->{VersionGet}->[$Counter]->{$Attribute};

            # set content if values are undef
            if ( !defined $VersionAttribute ) {
                $VersionAttribute = 'UNDEF-unittest';
            }
            if ( !defined $ReferenceAttribute ) {
                $ReferenceAttribute = 'UNDEF-unittest';
            }

            # check attributes
            $Self->IsDeeply(
                $VersionAttribute,
                $ReferenceAttribute,
                "Test $TestCount: VersionGet() - $Attribute",
            );
        }

        $Counter++;
    }

    # prepare last version id
    my $LastVersionIDActual = 'UNDEF-unittest';
    if ( $ConfigItemData->{LastVersionID} ) {
        $LastVersionIDActual = $ConfigItemData->{LastVersionID};
    }

    # check last version id
    $Self->Is(
        $ConfigItemData->{LastVersionID},
        $LastVersionIDMust,
        "Test $TestCount: last version id identical",
    );

    # check history entries
    if (
        $Test->{ReferenceData}
        && $Test->{ReferenceData}->{HistoryGet}
        && @{ $Test->{ReferenceData}->{HistoryGet} }
        )
    {
        my $CompleteHistory = $ConfigItemObject->HistoryGet(
            ConfigItemID => $ConfigItemID,
        );

        # check nr of history entries
        $Self->Is(
            scalar @{ $Test->{ReferenceData}->{HistoryGet} },
            scalar @{$CompleteHistory},
            "Test $TestCount: nr of history entries",
        );

        CHECKNR: for my $CheckNr ( 0 .. $#{$CompleteHistory} ) {
            my $Check = $Test->{ReferenceData}->{HistoryGet}->[$CheckNr];
            my $Data  = $CompleteHistory->[$CheckNr];

            next CHECKNR if !( $Check && $Data );

            for my $Key ( sort keys %{$Check} ) {

                # check history data
                $Self->Is(
                    $Check->{$Key},
                    $Data->{$Key},
                    "Test $TestCount: $Key",
                );
            }
        }
    }
}
continue {
    $TestCount++;
}

# ------------------------------------------------------------ #
# test for bugfix 4377
# ------------------------------------------------------------ #

{

    my $CI1 = $ConfigItemObject->ConfigItemLookup(
        ConfigItemNumber => $ConfigItemNumbers[72],
    );

    my $CI2 = $ConfigItemObject->ConfigItemLookup(
        ConfigItemNumber => $ConfigItemNumbers[73],
    );

    # link the CI with a CI
    my $LinkResult = $LinkObject->LinkAdd(
        SourceObject => 'ITSMConfigItem',
        SourceKey    => $CI1,
        TargetObject => 'ITSMConfigItem',
        TargetKey    => $CI2,
        Type         => 'DependsOn',
        State        => 'Valid',
        UserID       => 1,
    );

    # update incident state of CI1
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $CI1,
        Name         => 'UnitTest - Bugfix4377 - CI-A',
        DefinitionID => $ConfigItemDefinitionIDs[0],
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Incident},
        UserID       => 1,
    );

    # check if version could be added
    $Self->True(
        $VersionID,
        "Test $TestCount: VersionAdd() for $CI1 - Set to 'Incident'",
    );

    # get the latest version for CI1
    my $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI1,
    );

    # check if incident state of CI1 is 'Incident'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Incident',
        "Test $TestCount: Current incident state of CI $CI1",
    );

    # get the latest version for CI2
    $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI2,
    );

    # check if incident state of CI2 is 'Warning'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Warning',
        "Test $TestCount: Current incident state of CI $CI2",
    );

    # update incident state of CI2 to 'Incident'
    $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $CI2,
        Name         => 'UnitTest - Bugfix4377 - CI-B',
        DefinitionID => $ConfigItemDefinitionIDs[0],
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Incident},
        UserID       => 1,
    );

    # check if version could be added
    $Self->True(
        $VersionID,
        "Test $TestCount: VersionAdd() for CI $CI2 - Set to 'Incident'",
    );

    # get the latest version for CI2
    $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI2,
    );

    # check if incident state of CI2 is 'Incident'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Incident',
        "Test $TestCount: Current incident state of CI $CI2",
    );

    # update incident state of CI1 to 'Operational'
    $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $CI1,
        Name         => 'UnitTest - Bugfix4377 - CI-A',
        DefinitionID => $ConfigItemDefinitionIDs[0],
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Operational},
        UserID       => 1,
    );

    # check if version could be added
    $Self->True(
        $VersionID,
        "Test $TestCount: VersionAdd() for CI $CI1 - Set to 'Operational'",
    );

    # get the latest version for CI1
    $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI1,
    );

    # check if incident state of CI1 is 'Warning' (because of linked CI2 in state 'incident')
    $Self->Is(
        $VersionRef->{CurInciState},
        'Warning',
        "Test $TestCount: Current incident state of CI $CI1",
    );

    # update incident state of CI2 to 'Operational'
    $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $CI2,
        Name         => 'UnitTest - Bugfix4377 - CI-B',
        DefinitionID => $ConfigItemDefinitionIDs[0],
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Operational},
        UserID       => 1,
    );

    # check if version could be added
    $Self->True(
        $VersionID,
        "Test $TestCount: VersionAdd() for CI $CI2 - Set to 'Operational'",
    );

    # get the latest version for CI1
    $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI1,
    );

    # check if incident state of CI1 is 'Operational'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Operational',
        "Test $TestCount: Current incident state of CI $CI1",
    );

    # get the latest version for CI2
    $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI2,
    );

    # check if incident state of CI2 is 'Warning'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Operational',
        "Test $TestCount: Current incident state of CI $CI2",
    );

    # increase the test counter
    $TestCount++;
}

# ------------------------------------------------------------ #
# test for bugfix 10356
# ------------------------------------------------------------ #

{

    my $CI1 = $ConfigItemObject->ConfigItemLookup(
        ConfigItemNumber => $ConfigItemNumbers[72],
    );

    # add a version, set incident state to incident
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $CI1,
        Name         => 'UnitTest - Bugfix10356',
        DefinitionID => $ConfigItemDefinitionIDs[0],
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Incident},
        UserID       => 1,
    );

    # check if version could be added
    $Self->True(
        $VersionID,
        "Test $TestCount: VersionAdd() for $CI1 - Set to 'Incident'",
    );

    # get the latest version for CI1
    my $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI1,
    );

    # check if incident state of CI1 is 'Incident'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Incident',
        "Test $TestCount: Current incident state of CI $CI1",
    );

    # delete the last version
    my $VersionDeleteSuccess = $ConfigItemObject->VersionDelete(
        VersionID => $VersionRef->{VersionID},
        UserID    => 1,
    );

    # check if version could be deleted
    $Self->True(
        $VersionDeleteSuccess,
        "Test $TestCount: VersionDelete() for $CI1'",
    );

    # get the history
    my $HistoryRef = $ConfigItemObject->HistoryGet(
        ConfigItemID => $CI1,
    );

    my $LastHistoryEntry = pop @{$HistoryRef};

    # check if last history entry has the correct history type
    $Self->Is(
        $LastHistoryEntry->{HistoryType},
        'VersionDelete',
        "Test $TestCount: HistoryType of last version of CI $CI1",
    );

    # increase the test counter
    $TestCount++;
}

# ------------------------------------------------------------ #
# define general config item search tests
# ------------------------------------------------------------ #

my @SearchTests = (

    # search ALL config items in the two test classes
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the number param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            Number => $ConfigItemNumbers[50],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
        ],
    },

    # test the number param with wildcards
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            Number => '*' . $ConfigItemNumbers[50] . '*',
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
        ],
    },

    # test the number param with wildcards but with deactivated wildcard feature
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            Number         => '*' . $ConfigItemNumbers[50] . '*',
            UsingWildcards => 0,
        },
        ReferenceData => [],
    },

    # test the deployment state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [ $DeplStateListReverse{Production} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the deployment state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [ $DeplStateListReverse{Maintenance} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
        ],
    },

    # test the deployment state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [
                $DeplStateListReverse{Production},
                $DeplStateListReverse{Maintenance},
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [ $InciStateListReverse{Operational} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [ $InciStateListReverse{Incident} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test the incident state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [
                $InciStateListReverse{Incident},
                $InciStateListReverse{Operational},
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the order by param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs         => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            OrderBy          => ['CreateBy'],
            OrderByDirection => ['Up'],
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[60],
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
        ],
    },

    # test the limit param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 100,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the limit param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 3,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test the limit param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 2,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
        ],
    },

    # test the limit param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
        ],
    },

    # test the limit param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 0,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the create by param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            CreateBy => [ $UserIDs[2] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
        ],
    },

    # test the create by param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            CreateBy => [ $UserIDs[1],            $UserIDs[2] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the change by param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ChangeBy => [ $UserIDs[2] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
        ],
    },

    # test the change by param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            ChangeBy => [1],
        },
        ReferenceData => [],
    },

    # test the change by param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            ChangeBy => [ $UserIDs[1] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[60],
        ],
    },

    # test the change by param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            ChangeBy => [ $UserIDs[2] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
        ],
    },

    # search ALL config items in the two test classes using the version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the name param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name     => 'UnitTest - Class 3 ConfigItem 1 Version 1',
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
        ],
    },

    # test the name param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name     => 'UnitTest - Class 3 ConfigItem 3 Version 1',
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test the name param with an wildcard
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name     => 'UnitTest - * 1',
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the name param with an wildcard and a previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name                  => 'UnitTest - * 1',
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the name param with wildcards
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name     => 'UnitTest - Class 3*',
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test the name param with wildcards but with deactivated wildcard feature
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name           => 'UnitTest - Class 3*',
            ClassIDs       => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            UsingWildcards => 0,
        },
        ReferenceData => [],
    },

    # test the last version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name     => 'UnitTest - Class 3 ConfigItem 2 Version 1',
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [],
    },

    # test the PreviousVersionSearch param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name                  => 'UnitTest - Class 3 ConfigItem 2 Version 1',
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
        ],
    },

    # test the limit param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 100,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the limit param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 3,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test the limit param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 2,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
        ],
    },

    # test the limit param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
        ],
    },

    # test the deployment state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [
                $DeplStateListReverse{Production},
                $DeplStateListReverse{Maintenance},
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the deployment state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [ $DeplStateListReverse{Production} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the deployment state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [ $DeplStateListReverse{Maintenance} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
        ],
    },

    # test the deployment state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [
                $DeplStateListReverse{Production},
                $DeplStateListReverse{Maintenance},
            ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the deployment state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs          => [ $DeplStateListReverse{Production} ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the deployment state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs          => [ $DeplStateListReverse{Maintenance} ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
        ],
    },

    # test the incident state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [
                $InciStateListReverse{Operational},
                $InciStateListReverse{Incident},
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [ $InciStateListReverse{Operational} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [ $InciStateListReverse{Incident} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test the incident state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [
                $InciStateListReverse{Operational},
                $InciStateListReverse{Incident},
            ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs          => [ $InciStateListReverse{Operational} ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs          => [ $InciStateListReverse{Incident} ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Customer1)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            ChangeBy => [ $UserIDs[2] ],
            What     => [
                {
                    "[1]{'Version'}[1]{'Customer1'}[1]{'Content'}" => 'dummy_customer_for_unitest',
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}" => '2010-02-12',
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (DateTime1)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'DateTime1'}[1]{'Content'}" => '2010-02-12 09:14',
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Customer1, Date1, DateTime1)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Customer1'}[1]{'Content'}" => 'dummy_customer_for_unitest',
                },
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}" => '2010-02-12',
                },
                {
                    "[1]{'Version'}[1]{'DateTime1'}[1]{'Content'}" => '2010-02-12 09:14',
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, <, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '<' => '2010-02-12',
                        }
                },
            ],

        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, <, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '<' => '2010-02-13',
                        }
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, <=, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '<=' => '2010-02-11',
                        }
                },
            ],

        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, <=, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '<=' => '2010-02-12',
                        }
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, <=, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '<=' => '2010-02-13',
                        }
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, =, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '=' => '2010-02-11',
                        }
                },
            ],
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, =, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '=' => '2010-02-12',
                        }
                },
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, !=, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '!=' => '2010-02-12',
                        }
                },
            ],
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, !=, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '!=' => '2010-02-13',
                        }
                },
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, >=, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '>=' => '2010-02-13',
                        }
                },
            ],
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, >=, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '>=' => '2010-02-12',
                        }
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, >=, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '>=' => '2010-02-11',
                        }
                },
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, >, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '>' => '2010-02-12',
                        }
                },
            ],
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, >, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '>' => '2010-02-11',
                        }
                },
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, -between, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => { '-between' => [ '2010-01-01', '2010-01-31' ] }
                },
            ],
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, -between, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => { '-between' => [ '2010-02-01', '2010-02-31' ] }
                },
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with Name = 0 without any wildcards
    # should return no results
    # Bugfix# 8881
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            Name     => 0,
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with Number = 0 without any wildcards
    # should return no results
    # Bugfix# 8881
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            Number   => 0,
        },
        ReferenceData => [
        ],
    },
);

# ------------------------------------------------------------ #
# run general config item search tests
# ------------------------------------------------------------ #

# $SearchTestCount provides grouping of test cases
my $SearchTestCount = 1;

TEST:
for my $Test (@SearchTests) {

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

        $Self->True(
            0,
            "SearchTest $SearchTestCount: No SearchData found for this test.",
        );

        next TEST;
    }

    if ( !$Test->{Function} || ref $Test->{Function} ne 'ARRAY' || !@{ $Test->{Function} } ) {
        $Test->{Function} = ['ConfigItemSearchExtended'];
    }

    for my $Function ( @{ $Test->{Function} } ) {

        # start search
        my $ConfigItemList = $ConfigItemObject->$Function(
            %{ $Test->{SearchData} },
        );

        # check the config item list
        if ( $Test->{ReferenceData} ) {

            $Self->True(
                $ConfigItemList && ref $ConfigItemList eq 'ARRAY',
                "SearchTest $SearchTestCount: $Function() - List is an array reference.",
            );

            next TEST if !$ConfigItemList;
        }
        else {

            $Self->False(
                $ConfigItemList,
                "SearchTest $SearchTestCount: $Function() - Return false.",
            );

            next TEST if !$ConfigItemList;
        }

        # check number of found config items
        $Self->Is(
            scalar @{$ConfigItemList},
            scalar @{ $Test->{ReferenceData} },
            "SearchTest $SearchTestCount: $Function() - correct number of found config items",
        );

        my @ReferenceList;
        for my $Number ( @{ $Test->{ReferenceData} } ) {

            # find id of the item
            $DBObject->Prepare(
                SQL => "SELECT id FROM configitem WHERE "
                    . "configitem_number = '$Number' "
                    . "ORDER BY id DESC",
                Limit => 1,
            );

            # fetch the result
            my $ConfigItemID;
            while ( my @Row = $DBObject->FetchrowArray() ) {
                $ConfigItemID = $Row[0];
            }

            push @ReferenceList, $ConfigItemID;
        }

        # check arrays
        $Self->IsDeeply(
            $ConfigItemList,
            \@ReferenceList,
            "SearchTest $SearchTestCount: $Function() - List",
        );
    }
}
continue {
    $SearchTestCount++;
}

# ------------------------------------------------------------ #
# testing adding of invalid definitions
# ------------------------------------------------------------ #

{
    # define some invalid definitions

    my @InvalidConfigItemDefinitions;

    # Data sctructure is no array reference.
    push @InvalidConfigItemDefinitions, "
        {
            Key        => 'Customer1',
            Name       => 'Customer 1',
            Searchable => 1,
            Input      => {
                Type => 'Customer',
            },
        }
    ";

    # Data sctructure hash elements contain no data.
    push @InvalidConfigItemDefinitions, " [
        {
        },
        {
        },
    ]";

    # Missing comma between the 2 elements
    push @InvalidConfigItemDefinitions, " [
        {
            Key        => 'Customer1',
            Name       => 'Customer 1',
            Searchable => 1,
            Input      => {
                Type => 'Customer',
            },
        }
        {
            Key        => 'Date1',
            Name       => 'Date 1',
            Searchable => 1,
            Input      => {
                Type => 'Date',
            },
        },
    ]";

    # Key "Customer 1" containing a space.
    push @InvalidConfigItemDefinitions, " [
        {
            Key        => 'Customer 1',   # Key containing a space!
            Name       => 'Customer 1',
            Searchable => 1,
            Input      => {
                Type => 'Customer',
            },
        },
    ]";

    # Key "Customer1ΣՎέ" containing non-ascii characters.
    push @InvalidConfigItemDefinitions, " [
        {
            Key        => 'Customer1ΣՎέ',
            Name       => 'Customer 1',
            Searchable => 1,
            Input      => {
                Type => 'Customer',
            },
        },
    ]";

    # Invalid empty sub element.
    push @InvalidConfigItemDefinitions, " [
        {
            Key        => 'Customer1',
            Name       => 'Customer 1',
            Searchable => 1,
            Input      => {
                Type => 'Customer',
            },
            Sub => [
                {},
            ],
        },
    ]";

    # generate a random name
    my $ClassName = 'UnitTest' . $Helper->GetRandomID();

    # add an unittest config item class
    my $ClassID = $GeneralCatalogObject->ItemAdd(
        Class   => 'ITSM::ConfigItem::Class',
        Name    => $ClassName,
        ValidID => 1,
        UserID  => 1,
    );

    # check class id
    if ( !$ClassID ) {
        $Self->True(
            0,
            "Can't add new config item class.",
        );
    }

    for my $Definition (@InvalidConfigItemDefinitions) {

        # add a definition to the class
        my $DefinitionID = $ConfigItemObject->DefinitionAdd(
            ClassID    => $ClassID,
            Definition => $Definition,
            UserID     => 1,
        );

        # check definition id, must be false, because all definitions have errors
        $Self->False(
            $DefinitionID,
            "Can't add new config item definition.",
        );
    }

}

# ------------------------------------------------------------ #
# testing support for attachments
# ------------------------------------------------------------ #

my $AttachmentTestConfigItemID = $ConfigItemIDs[0];

# verify that initially no attachment exists
my @AttachmentList = $ConfigItemObject->ConfigItemAttachmentList(
    ConfigItemID => $AttachmentTestConfigItemID,
);

$Self->Is(
    scalar @AttachmentList,
    0,
    'No attachments initially',
);

my @TestFileList = (
    {
        Filename    => 'first attachment',
        Content     => 'First attachment from ITSMConfigItem.t',
        ContentType => 'text/plain',
    },
    {
        Filename    => 'second attachment',
        Content     => 'Second attachment from ITSMConfigItem.t',
        ContentType => 'text/plain',
    },
);

my $FileCount;
for my $TestFile (@TestFileList) {

    $FileCount++;

    my $AddOk = $ConfigItemObject->ConfigItemAttachmentAdd(
        %{$TestFile},
        ConfigItemID => $AttachmentTestConfigItemID,
        UserID       => 1,
    );
    $Self->True(
        $AddOk,
        "Attachment $FileCount: attachment added",
    );

    my @AttachmentList = $ConfigItemObject->ConfigItemAttachmentList(
        ConfigItemID => $AttachmentTestConfigItemID,
        UserID       => 1,
    );
    $Self->Is(
        scalar @AttachmentList,
        $FileCount,
        "Attachment $FileCount: number of attachments after adding",
    );

    # check whether the last added attachment is in the list
    my %AttachmentLookup = map { $_ => 1 } @AttachmentList;
    $Self->True(
        $AttachmentLookup{ $TestFile->{Filename} },
        "Attachment $FileCount: filename from ConfigItemAttachmentList()",
    );

    # get the attachment
    my $Attachment = $ConfigItemObject->ConfigItemAttachmentGet(
        ConfigItemID => $AttachmentTestConfigItemID,
        Filename     => $TestFile->{Filename},
    );
    $Self->True(
        $Attachment,
        "Attachment $FileCount: ConfigItemAttachmentGet() returned true",
    );

    # check attachment file attributes
    for my $Attribute (qw(Filename Content ContentType)) {
        $Self->Is(
            $Attachment->{$Attribute},
            $TestFile->{$Attribute},
            "Attachment $FileCount: $Attribute from ConfigItemAttachmentGet",
        );
    }

    # check existence of attachment
    my $AttachmentExists = $ConfigItemObject->ConfigItemAttachmentExists(
        ConfigItemID => $AttachmentTestConfigItemID,
        Filename     => $TestFile->{Filename},
        UserID       => 1,
    );
    $Self->True(
        $AttachmentExists,
        "Attachment $FileCount: attachment exists",
    );

}

# now delete the attachments
$FileCount = 0;
my $MaxTestFiles = scalar @TestFileList;
for my $TestFile (@TestFileList) {

    $FileCount++;

    my $DeleteOk = $ConfigItemObject->ConfigItemAttachmentDelete(
        ConfigItemID => $AttachmentTestConfigItemID,
        Filename     => $TestFile->{Filename},
        UserID       => 1,
    );
    $Self->True(
        $DeleteOk,
        "Attachment $FileCount: attachment deleted",
    );

    my @AttachmentList = $ConfigItemObject->ConfigItemAttachmentList(
        ConfigItemID => $AttachmentTestConfigItemID,
        UserID       => 1,
    );

    $Self->Is(
        scalar @AttachmentList,
        $MaxTestFiles - $FileCount,
        "Attachment $FileCount: number of attachments after deletion",
    );

    my $AttachmentExists = $ConfigItemObject->ConfigItemAttachmentExists(
        Filename     => $TestFile->{Filename},
        ConfigItemID => $AttachmentTestConfigItemID,
        UserID       => 1,
    );
    $Self->False(
        $AttachmentExists,
        "Attachment $FileCount: attachment is gone",
    );
}

# check config item delete
my $DeleteTestCount = 1;
for my $ConfigItemID (@ConfigItemIDs) {
    my $DeleteOk = $ConfigItemObject->ConfigItemDelete(
        ConfigItemID => $ConfigItemID,
        UserID       => 1,
    );
    $Self->True(
        $DeleteOk,
        "DeleteTest $DeleteTestCount - ConfigItemDelete() (ConfigItemID=$ConfigItemID)"
    );

    # double check if config item is really deleted
    my $ConfigItemData = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => $ConfigItemID,
        UserID       => 1,
        Cache        => 0,
    );
    $Self->False(
        $ConfigItemData->{ConfigItemID},
        "DeleteTest $DeleteTestCount - double check (ConfigItemID=$ConfigItemID)",
    );

    $DeleteTestCount++;
}

# cleanup is done by RestoreDatabase

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;

use utf8;

use vars qw($Self);

my $DBObject             = $Kernel::OM->Get('Kernel::System::DB');
my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');
my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $LinkObject           = $Kernel::OM->Get('Kernel::System::LinkObject');
my $UserObject           = $Kernel::OM->Get('Kernel::System::User');

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# define needed variable
my $RandomID = $Helper->GetRandomID();

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# perform ConfigItemCount to fill the empty fields
$ConfigItemObject->ConfigItemCount();

# create needed users
my @UserIDs;
{

    # disable email checks to create new user
    my $CheckEmailAddressesOrg = $ConfigObject->Get('CheckEmailAddresses') || 1;
    $ConfigObject->Set(
        Key   => 'CheckEmailAddresses',
        Value => 0,
    );

    for my $Counter ( 1 .. 3 ) {

        # create new users for the tests
        my $UserID = $UserObject->UserAdd(
            UserFirstname => 'ITSMConfigItem' . $Counter,
            UserLastname  => 'UnitTest',
            UserLogin     => 'UnitTest-ITSMConfigItem-' . $Counter . $RandomID,
            UserEmail     => 'UnitTest-ITSMConfigItem-' . $Counter . '@localhost',
            ValidID       => 1,
            ChangeUserID  => 1,
        );

        push @UserIDs, $UserID;
    }

    # restore original email check param
    $ConfigObject->Set(
        Key   => 'CheckEmailAddresses',
        Value => $CheckEmailAddressesOrg,
    );
}

my $GeneralCatalogClass = 'UnitTest' . $RandomID;

# add a general catalog test list
for my $Name (qw(Test1 Test2 Test3 Test4)) {

    # add a new item
    my $ItemID = $GeneralCatalogObject->ItemAdd(
        Class   => $GeneralCatalogClass,
        Name    => $Name,
        ValidID => 1,
        UserID  => 1,
    );

    # check item id
    if ( !$ItemID ) {

        $Self->True(
            0,
            "Can't add new general catalog item.",
        );
    }
}

# define the first test definition (all provided data types)
my @ConfigItemPerlDefinitions;
$ConfigItemPerlDefinitions[0] = " [
    {
        Key        => 'Customer1',
        Name       => 'Customer 1',
        Searchable => 1,
        Input      => {
            Type => 'Customer',
        },
    },
    {
        Key        => 'Date1',
        Name       => 'Date 1',
        Searchable => 1,
        Input      => {
            Type => 'Date',
        },
    },
    {
        Key        => 'DateTime1',
        Name       => 'Date Time 1',
        Searchable => 1,
        Input      => {
            Type => 'DateTime',
        },
    },
    {
        Key   => 'Dummy1',
        Name  => 'Dummy 1',
        Input => {
            Type => 'Dummy',
        },
    },
    {
        Key        => 'GeneralCatalog1',
        Name       => 'GeneralCatalog 1',
        Searchable => 1,
        Input      => {
            Type  => 'GeneralCatalog',
            Class => '$GeneralCatalogClass',
        },
    },
    {
        Key        => 'Integer1',
        Name       => 'Integer 1',
        Searchable => 1,
        Input      => {
            Type => 'Integer',
        },
    },
    {
        Key        => 'Text1',
        Name       => 'Text 1',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 50,
        },
    },
    {
        Key        => 'TextArea1',
        Name       => 'TextArea 1',
        Searchable => 1,
        Input      => {
            Type => 'TextArea',
        },
    },
] ";

# define the second test definition (sub data types)
$ConfigItemPerlDefinitions[1] = " [
    {
        Key        => 'Main1',
        Name       => 'Main 1',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 50,
        },
        CountMax => 10,
        Sub => [
            {
                Key        => 'Main1Sub1',
                Name       => 'Main 1 Sub 1',
                Searchable => 1,
                Input      => {
                    Type      => 'Text',
                    Size      => 50,
                    MaxLength => 50,
                },
                CountMax => 10,
                Sub => [
                    {
                        Key        => 'Main1Sub1SubSub1',
                        Name       => 'Main 1 Sub 1 SubSub 1',
                        Searchable => 1,
                        Input      => {
                            Type      => 'Text',
                            Size      => 50,
                            MaxLength => 50,
                        },
                        CountMax => 10,
                    },
                    {
                        Key        => 'Main1Sub1SubSub2',
                        Name       => 'Main 1 Sub 1 SubSub 2',
                        Searchable => 1,
                        Input      => {
                            Type => 'TextArea',
                        },
                        CountMax => 10,
                    },
                ],
            },
            {
                Key        => 'Main1Sub2',
                Name       => 'Main 1 Sub 2',
                Searchable => 1,
                Input      => {
                    Type => 'TextArea',
                },
                CountMax => 10,
            },
        ],
    },
    {
        Key        => 'Main2',
        Name       => 'Main 2',
        Searchable => 1,
        Input      => {
            Type => 'TextArea',
        },
        CountMax => 10,
        Sub => [
            {
                Key        => 'Main2Sub1',
                Name       => 'Main 2 Sub 1',
                Searchable => 1,
                Input      => {
                    Type      => 'Text',
                    Size      => 50,
                    MaxLength => 50,
                },
                CountMax => 10,
            },
            {
                Key        => 'Main2Sub2',
                Name       => 'Main 2 Sub 2',
                Searchable => 1,
                Input      => {
                    Type => 'TextArea',
                },
                CountMax => 10,
            },
        ],
    },
] ";

# define the third test definition (especially for search tests with XMLData)
$ConfigItemPerlDefinitions[2] = " [
    {
        Key        => 'Customer1',
        Name       => 'Customer 1',
        Searchable => 1,
        Input      => {
            Type => 'Customer',
        },
    },
    {
        Key        => 'Date1',
        Name       => 'Date 1',
        Searchable => 1,
        Input      => {
            Type => 'Date',
        },
    },
    {
        Key        => 'DateTime1',
        Name       => 'Date Time 1',
        Searchable => 1,
        Input      => {
            Type => 'DateTime',
        },
    },
] ";

# define the fourth test definition (only for search tests)
$ConfigItemPerlDefinitions[3] = " [
    {
        Key        => 'Customer1',
        Name       => 'Customer 1',
        Searchable => 1,
        Input      => {
            Type => 'Customer',
        },
    },
] ";

my $YAMLObject = $Kernel::OM->Get('Kernel::System::YAML');

my @ConfigItemDefinitions;
for my $PerlDefinition (@ConfigItemPerlDefinitions) {
    my $YAMLDefinition = $YAMLObject->Dump(
        Data => eval $PerlDefinition,    ## no critic
    );
    push @ConfigItemDefinitions, $YAMLDefinition;
}

# add the test classes
my @ConfigItemClassIDs;
my @ConfigItemClasses;
my @ConfigItemDefinitionIDs;
for my $Definition (@ConfigItemDefinitions) {

    # generate a random name
    my $ClassName = 'UnitTest' . $Helper->GetRandomID();

    # add an unittest config item class
    my $ClassID = $GeneralCatalogObject->ItemAdd(
        Class   => 'ITSM::ConfigItem::Class',
        Name    => $ClassName,
        ValidID => 1,
        UserID  => 1,
    );

    # check class id
    if ( !$ClassID ) {

        $Self->True(
            0,
            "Can't add new config item class.",
        );
    }

    push @ConfigItemClassIDs, $ClassID;
    push @ConfigItemClasses,  $ClassName;

    # add a definition to the class
    my $DefinitionID = $ConfigItemObject->DefinitionAdd(
        ClassID    => $ClassID,
        Definition => $Definition,
        UserID     => 1,
    );

    # check definition id
    if ( !$DefinitionID ) {

        $Self->True(
            0,
            "Can't add new config item definition.",
        );
    }

    push @ConfigItemDefinitionIDs, $DefinitionID;
}

# test DefinitionList for those simple cases
my $Counter = 0;
for my $ClassID (@ConfigItemClassIDs) {
    my $DefinitionListRef = $ConfigItemObject->DefinitionList(
        ClassID => $ClassID,
    );

    # expect a single definition per config item class
    $Self->Is(
        scalar @{$DefinitionListRef},
        1,
        "DefinitionList() for class id $ClassID: got a single result",
    );

    # expect the remembered definition id in the first definition
    $Self->Is(
        $DefinitionListRef->[0]->{DefinitionID},
        $ConfigItemDefinitionIDs[$Counter],
        "DefinitionList() for class id $ClassID: got expected definition id",
    );

    $Counter++;
}

# create some random numbers
my @ConfigItemNumbers;
for ( 1 .. 100 ) {
    push @ConfigItemNumbers, $Helper->GetRandomNumber();
}

# get class list
my $ClassList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::Class',
);
my %ClassListReverse = reverse %{$ClassList};

# get deployment state list
my $DeplStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %DeplStateListReverse = reverse %{$DeplStateList};

# get incident state list
my $InciStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %InciStateListReverse = reverse %{$InciStateList};

# get general catalog test list
my $GeneralCatalogList = $GeneralCatalogObject->ItemList(
    Class => $GeneralCatalogClass,
);
my %GeneralCatalogListReverse = reverse %{$GeneralCatalogList};

# ------------------------------------------------------------ #
# define general config item tests
# ------------------------------------------------------------ #

my $ConfigItemTests = [

    # ConfigItemAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                UserID => 1,
            },
        },
    },

    # ConfigItemAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                ClassID => $ConfigItemClassIDs[0],
            },
        },
    },

    # invalid class id is given (check return false)
    {
        SourceData => {
            ConfigItemAdd => {
                ClassID => $ConfigItemClassIDs[-1] + 1,
                UserID  => 1,
            },
        },
    },

    # all required config item values are given (check returned config item values)
    {
        SourceData => {
            ConfigItemAdd => {
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
        },
        ReferenceData => {
            ConfigItemGet => {
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # all required config item values are given (check number attribute)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[0],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => $UserIDs[1],
            },
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[0],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => $UserIDs[1],
                ChangeBy         => $UserIDs[1],
            },
        },
    },

    # config item with this number already exists (check return false)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[0],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
        },
    },

    # VersionAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[1],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[1],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # VersionAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[2],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name        => 'UnitTest - Class 1 ConfigItem 2 Version 1',
                    DeplStateID => $DeplStateListReverse{Production},
                    InciStateID => $InciStateListReverse{Operational},
                    UserID      => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[2],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # VersionAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[3],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 3 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[3],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # VersionAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[4],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 4 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[4],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # VersionAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[5],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => $UserIDs[1],
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 5 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[5],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => $UserIDs[1],
                ChangeBy         => $UserIDs[1],
            },
        },
    },

    # invalid deployment state id is given (check returned values)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[6],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 6 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => 9999999,
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[6],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # invalid incident state id is given (check returned values)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[7],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 7 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => 9999999,
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[7],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # all required values are given (check returned values)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[8],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 8 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[8],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Operational},
                CurInciState     => 'Operational',
                CurInciStateType => 'operational',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[8],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Class 1 ConfigItem 8 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
        },
    },

    # all required values are given (general check with two versions)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[9],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 9 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 9 Version 2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => $UserIDs[1],
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[9],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Operational},
                CurInciState     => 'Operational',
                CurInciStateType => 'operational',
                CreateBy         => 1,
                ChangeBy         => $UserIDs[1],
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[9],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Class 1 ConfigItem 9 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => 1,
                },
                {
                    Number           => $ConfigItemNumbers[9],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Class 1 ConfigItem 9 Version 2',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => $UserIDs[1],
                },
            ],
        },
    },

    # all required values are given (check the calculation of deployment and incident state)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[10],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 10 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Planned},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 10 Version 2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[10],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Maintenance},
                CurDeplState     => 'Maintenance',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[10],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Class 1 ConfigItem 10 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Planned},
                    DeplState        => 'Planned',
                    DeplStateType    => 'preproductive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
                {
                    Number           => $ConfigItemNumbers[10],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Class 1 ConfigItem 10 Version 2',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Maintenance},
                    DeplState        => 'Maintenance',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
        },
    },

    # add config item only for later search tests
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[50],
                ClassID => $ConfigItemClassIDs[2],
                UserID  => $UserIDs[2],
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 3 ConfigItem 1 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[2],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => $UserIDs[2],
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[50],
                ClassID          => $ConfigItemClassIDs[2],
                Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => $UserIDs[2],
                ChangeBy         => $UserIDs[2],
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[50],
                    ClassID          => $ConfigItemClassIDs[2],
                    Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                    Name             => 'UnitTest - Class 3 ConfigItem 1 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[2],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => $UserIDs[2],
                },
            ],
        },
    },

    # add config item only for later search tests
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[51],
                ClassID => $ConfigItemClassIDs[2],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 3 ConfigItem 2 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[2],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Class 3 ConfigItem 2 Version 2',
                    DefinitionID => $ConfigItemDefinitionIDs[2],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => $UserIDs[1],
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[51],
                ClassID          => $ConfigItemClassIDs[2],
                Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                CurDeplStateID   => $DeplStateListReverse{Maintenance},
                CurDeplState     => 'Maintenance',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => 1,
                ChangeBy         => $UserIDs[1],
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[51],
                    ClassID          => $ConfigItemClassIDs[2],
                    Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                    Name             => 'UnitTest - Class 3 ConfigItem 2 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[2],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
                {
                    Number           => $ConfigItemNumbers[51],
                    ClassID          => $ConfigItemClassIDs[2],
                    Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                    Name             => 'UnitTest - Class 3 ConfigItem 2 Version 2',
                    DefinitionID     => $ConfigItemDefinitionIDs[2],
                    DeplStateID      => $DeplStateListReverse{Maintenance},
                    DeplState        => 'Maintenance',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => $UserIDs[1],
                },
            ],
        },
    },

    # add config item only for later search tests, including XMLData
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[52],
                ClassID => $ConfigItemClassIDs[2],
                UserID  => $UserIDs[2],
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 3 ConfigItem 3 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[2],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => $UserIDs[2],
                    XMLData      => [
                        undef,
                        {
                            Version => [
                                undef,
                                {
                                    Customer1 => [
                                        undef,
                                        {
                                            Content => 'dummy_customer_for_unitest',
                                        },
                                    ],
                                    Date1 => [
                                        undef,
                                        {
                                            Content => '2010-02-12',
                                        },
                                    ],
                                    DateTime1 => [
                                        undef,
                                        {
                                            Content => '2010-02-12 09:14',
                                        },
                                    ],
                                },
                            ],
                        },
                    ],
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[52],
                ClassID          => $ConfigItemClassIDs[2],
                Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => $UserIDs[2],
                ChangeBy         => $UserIDs[2],
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[52],
                    ClassID          => $ConfigItemClassIDs[2],
                    Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                    Name             => 'UnitTest - Class 3 ConfigItem 3 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[2],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [
                        undef,
                        {
                            'TagKey'  => '[1]',
                            'Version' => [
                                undef,
                                {
                                    'Customer1' => [
                                        undef,
                                        {
                                            'Content' => 'dummy_customer_for_unitest',
                                            'TagKey'  => '[1]{\'Version\'}[1]{\'Customer1\'}[1]',
                                        },
                                    ],
                                    'Date1' => [
                                        undef,
                                        {
                                            'Content' => '2010-02-12',
                                            'TagKey'  => '[1]{\'Version\'}[1]{\'Date1\'}[1]',
                                        },
                                    ],
                                    'DateTime1' => [
                                        undef,
                                        {
                                            'Content' => '2010-02-12 09:14',
                                            'TagKey'  => '[1]{\'Version\'}[1]{\'DateTime1\'}[1]',
                                        },
                                    ],
                                    'TagKey' => '[1]{\'Version\'}[1]',
                                },
                            ],
                        },
                    ],
                    CreateBy => $UserIDs[2],
                },
            ],
        },
    },

    # add config item only for later search tests
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[60],
                ClassID => $ConfigItemClassIDs[3],
                UserID  => $UserIDs[1],
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 4 ConfigItem 1 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[2],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => $UserIDs[1],
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[60],
                ClassID          => $ConfigItemClassIDs[3],
                Class            => $ClassList->{ $ConfigItemClassIDs[3] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Operational},
                CurInciState     => 'Operational',
                CurInciStateType => 'operational',
                CreateBy         => $UserIDs[1],
                ChangeBy         => $UserIDs[1],
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[60],
                    ClassID          => $ConfigItemClassIDs[3],
                    Class            => $ClassList->{ $ConfigItemClassIDs[3] },
                    Name             => 'UnitTest - Class 4 ConfigItem 1 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[2],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => $UserIDs[1],
                },
            ],
        },
    },

    # added to check history functions
    # all required values are given (check the calculation of deployment and incident state)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[70],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - HistoryTest',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Planned},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - HistoryTest Version 2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[70],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Maintenance},
                CurDeplState     => 'Maintenance',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[70],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - HistoryTest',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Planned},
                    DeplState        => 'Planned',
                    DeplStateType    => 'preproductive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
                {
                    Number           => $ConfigItemNumbers[70],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - HistoryTest Version 2',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Maintenance},
                    DeplState        => 'Maintenance',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
            HistoryGet => [
                {
                    HistoryType   => 'ConfigItemCreate',
                    HistoryTypeID => 1,
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'VersionCreate',
                    HistoryTypeID => 6,
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'DefinitionUpdate',
                    HistoryTypeID => 8,
                    Comment       => $ConfigItemDefinitionIDs[0],
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'NameUpdate',
                    HistoryTypeID => 5,
                    Comment       => 'UnitTest - HistoryTest%%',
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'IncidentStateUpdate',
                    HistoryTypeID => 9,
                    Comment       => $InciStateListReverse{Operational} . '%%',
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'DeploymentStateUpdate',
                    HistoryTypeID => 10,
                    Comment       => $DeplStateListReverse{Planned} . '%%',
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'VersionCreate',
                    HistoryTypeID => 6,
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'NameUpdate',
                    HistoryTypeID => 5,
                    Comment =>
                        'UnitTest - HistoryTest Version 2%%UnitTest - HistoryTest',
                    CreateBy => 1,
                },
                {
                    HistoryType   => 'IncidentStateUpdate',
                    HistoryTypeID => 9,
                    Comment       => $InciStateListReverse{Incident} . '%%'
                        . $InciStateListReverse{Operational},
                    CreateBy => 1,
                },
                {
                    HistoryType   => 'DeploymentStateUpdate',
                    HistoryTypeID => 10,
                    Comment       => $DeplStateListReverse{Maintenance} . '%%'
                        . $DeplStateListReverse{Planned},
                    CreateBy => 1,
                },
            ],
        },
    },

    # added for Bug4196
    # all required values are given (check the calculation of deployment and incident state)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[71],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Bugfix4196',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Planned},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Bugfix4196 V2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Bugfix4196 V2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Bugfix4196 V2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[71],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Maintenance},
                CurDeplState     => 'Maintenance',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[71],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Bugfix4196',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Planned},
                    DeplState        => 'Planned',
                    DeplStateType    => 'preproductive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
                {
                    Number           => $ConfigItemNumbers[71],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Bugfix4196 V2',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Maintenance},
                    DeplState        => 'Maintenance',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
        },
    },

    # added for Bug 4377 - CI-A
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[72],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Bugfix4377 - CI-A',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[72],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Operational},
                CurInciState     => 'Operational',
                CurInciStateType => 'operational',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[72],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Bugfix4377 - CI-A',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
        },
    },

    # added for Bug 4377 - CI-B
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[73],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Bugfix4377 - CI-B',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[73],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Operational},
                CurInciState     => 'Operational',
                CurInciStateType => 'operational',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[73],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Bugfix4377 - CI-B',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
        },
    },

];

# ------------------------------------------------------------ #
# run general config item tests
# ------------------------------------------------------------ #

my $TestCount = 1;
my @ConfigItemIDs;

TEST:
for my $Test ( @{$ConfigItemTests} ) {

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

        $Self->True(
            0,
            "Test $TestCount: No SourceData found for this test.",
        );

        next TEST;
    }

    # extract source data
    my $SourceData = $Test->{SourceData};

    # add a new config item
    my $ConfigItemID;
    if ( $SourceData->{ConfigItemAdd} ) {

        # add the new config item
        $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
            %{ $SourceData->{ConfigItemAdd} },
        );

        if ($ConfigItemID) {
            push @ConfigItemIDs, $ConfigItemID;
        }
    }

    # check the config item
    if ( $Test->{ReferenceData} && $Test->{ReferenceData}->{ConfigItemGet} ) {

        $Self->True(
            $ConfigItemID,
            "Test $TestCount: ConfigItemAdd() - Add new config item. Insert success.",
        );

        next TEST if !$ConfigItemID;
    }
    else {

        $Self->False(
            $ConfigItemID,
            "Test $TestCount: ConfigItemAdd() - Add new config item. Return false.",
        );
    }

    # add all defined versions
    my @VersionIDs;
    my %VersionIDsSeen;
    if ( $SourceData->{VersionAdd} ) {

        for my $Version ( @{ $SourceData->{VersionAdd} } ) {

            if ($ConfigItemID) {
                $Version->{ConfigItemID} = $ConfigItemID;
            }

            # add a new version
            my $VersionID = $ConfigItemObject->VersionAdd(
                %{$Version},
            );

            if ($VersionID) {
                push @VersionIDs, $VersionID if !$VersionIDsSeen{$VersionID}++;
            }
        }
    }

    # check the config item
    my $ConfigItemData;
    if ( $Test->{ReferenceData} && $Test->{ReferenceData}->{ConfigItemGet} ) {

        # get the config item data
        $ConfigItemData = $ConfigItemObject->ConfigItemGet(
            ConfigItemID => $ConfigItemID,
        );

        if ( !$ConfigItemData ) {

            $Self->True(
                0,
                "Test $TestCount: ConfigItemGet() - get config item data."
            );
        }

        # check all config item attributes
        my $Counter = 0;
        for my $Attribute ( sort keys %{ $Test->{ReferenceData}->{ConfigItemGet} } ) {

            # set content if values are undef
            if ( !defined $ConfigItemData->{$Attribute} ) {
                $ConfigItemData->{$Attribute} = 'UNDEF-unittest';
            }
            if ( !defined $Test->{ReferenceData}->{ConfigItemGet}->{$Attribute} ) {
                $Test->{ReferenceData}->{ConfigItemGet}->{$Attribute} = 'UNDEF-unittest';
            }

            # check attributes
            $Self->Is(
                $ConfigItemData->{$Attribute},
                $Test->{ReferenceData}->{ConfigItemGet}->{$Attribute},
                "Test $TestCount: ConfigItemGet() - $Attribute",
            );

            $Counter++;
        }
    }

    # check the versions
    if (
        $Test->{ReferenceData}
        && $Test->{ReferenceData}->{VersionGet}
        && @{ $Test->{ReferenceData}->{VersionGet} }
        )
    {

        $Self->Is(
            scalar @VersionIDs,
            scalar @{ $Test->{ReferenceData}->{VersionGet} },
            "Test $TestCount: VersionAdd() - correct number of versions",
        );

        next TEST if !$ConfigItemID;
    }
    else {

        $Self->False(
            scalar @VersionIDs,
            "Test $TestCount: VersionAdd() - no versions exits",
        );
    }

    next TEST if !$Test->{ReferenceData};
    next TEST if !$Test->{ReferenceData}->{VersionGet};

    my $Counter           = 0;
    my $LastVersionIDMust = 'UNDEF-unittest';

    # Refresh ITSMConfigurationManagement cache to avoid issues with previously cached values.
    $Kernel::OM->Get('Kernel::System::Cache')->CleanUp(
        Type => 'ITSMConfigurationManagement',
    );

    VERSIONID:
    for my $VersionID (@VersionIDs) {

        # get this version
        my $VersionData = $ConfigItemObject->VersionGet(
            VersionID  => $VersionID,
            XMLDataGet => 1,
        );

        if ( !$VersionData ) {

            $Self->True(
                0,
                "Test $TestCount: VersionGet() - get version data."
            );

            next VERSIONID;
        }

        # save last version id
        $LastVersionIDMust = $VersionData->{VersionID};

        # check all version attributes
        for my $Attribute ( sort keys %{ $Test->{ReferenceData}->{VersionGet}->[$Counter] } ) {

            # extract the needed attributes
            my $VersionAttribute   = $VersionData->{$Attribute};
            my $ReferenceAttribute = $Test->{ReferenceData}->{VersionGet}->[$Counter]->{$Attribute};

            # set content if values are undef
            if ( !defined $VersionAttribute ) {
                $VersionAttribute = 'UNDEF-unittest';
            }
            if ( !defined $ReferenceAttribute ) {
                $ReferenceAttribute = 'UNDEF-unittest';
            }

            # check attributes
            $Self->IsDeeply(
                $VersionAttribute,
                $ReferenceAttribute,
                "Test $TestCount: VersionGet() - $Attribute",
            );
        }

        $Counter++;
    }

    # prepare last version id
    my $LastVersionIDActual = 'UNDEF-unittest';
    if ( $ConfigItemData->{LastVersionID} ) {
        $LastVersionIDActual = $ConfigItemData->{LastVersionID};
    }

    # check last version id
    $Self->Is(
        $ConfigItemData->{LastVersionID},
        $LastVersionIDMust,
        "Test $TestCount: last version id identical",
    );

    # check history entries
    if (
        $Test->{ReferenceData}
        && $Test->{ReferenceData}->{HistoryGet}
        && @{ $Test->{ReferenceData}->{HistoryGet} }
        )
    {
        my $CompleteHistory = $ConfigItemObject->HistoryGet(
            ConfigItemID => $ConfigItemID,
        );

        # check nr of history entries
        $Self->Is(
            scalar @{ $Test->{ReferenceData}->{HistoryGet} },
            scalar @{$CompleteHistory},
            "Test $TestCount: nr of history entries",
        );

        CHECKNR: for my $CheckNr ( 0 .. $#{$CompleteHistory} ) {
            my $Check = $Test->{ReferenceData}->{HistoryGet}->[$CheckNr];
            my $Data  = $CompleteHistory->[$CheckNr];

            next CHECKNR if !( $Check && $Data );

            for my $Key ( sort keys %{$Check} ) {

                # check history data
                $Self->Is(
                    $Check->{$Key},
                    $Data->{$Key},
                    "Test $TestCount: $Key",
                );
            }
        }
    }
}
continue {
    $TestCount++;
}

# ------------------------------------------------------------ #
# test for bugfix 4377
# ------------------------------------------------------------ #

{

    my $CI1 = $ConfigItemObject->ConfigItemLookup(
        ConfigItemNumber => $ConfigItemNumbers[72],
    );

    my $CI2 = $ConfigItemObject->ConfigItemLookup(
        ConfigItemNumber => $ConfigItemNumbers[73],
    );

    # link the CI with a CI
    my $LinkResult = $LinkObject->LinkAdd(
        SourceObject => 'ITSMConfigItem',
        SourceKey    => $CI1,
        TargetObject => 'ITSMConfigItem',
        TargetKey    => $CI2,
        Type         => 'DependsOn',
        State        => 'Valid',
        UserID       => 1,
    );

    # update incident state of CI1
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $CI1,
        Name         => 'UnitTest - Bugfix4377 - CI-A',
        DefinitionID => $ConfigItemDefinitionIDs[0],
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Incident},
        UserID       => 1,
    );

    # check if version could be added
    $Self->True(
        $VersionID,
        "Test $TestCount: VersionAdd() for $CI1 - Set to 'Incident'",
    );

    # get the latest version for CI1
    my $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI1,
    );

    # check if incident state of CI1 is 'Incident'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Incident',
        "Test $TestCount: Current incident state of CI $CI1",
    );

    # get the latest version for CI2
    $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI2,
    );

    # check if incident state of CI2 is 'Warning'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Warning',
        "Test $TestCount: Current incident state of CI $CI2",
    );

    # update incident state of CI2 to 'Incident'
    $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $CI2,
        Name         => 'UnitTest - Bugfix4377 - CI-B',
        DefinitionID => $ConfigItemDefinitionIDs[0],
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Incident},
        UserID       => 1,
    );

    # check if version could be added
    $Self->True(
        $VersionID,
        "Test $TestCount: VersionAdd() for CI $CI2 - Set to 'Incident'",
    );

    # get the latest version for CI2
    $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI2,
    );

    # check if incident state of CI2 is 'Incident'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Incident',
        "Test $TestCount: Current incident state of CI $CI2",
    );

    # update incident state of CI1 to 'Operational'
    $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $CI1,
        Name         => 'UnitTest - Bugfix4377 - CI-A',
        DefinitionID => $ConfigItemDefinitionIDs[0],
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Operational},
        UserID       => 1,
    );

    # check if version could be added
    $Self->True(
        $VersionID,
        "Test $TestCount: VersionAdd() for CI $CI1 - Set to 'Operational'",
    );

    # get the latest version for CI1
    $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI1,
    );

    # check if incident state of CI1 is 'Warning' (because of linked CI2 in state 'incident')
    $Self->Is(
        $VersionRef->{CurInciState},
        'Warning',
        "Test $TestCount: Current incident state of CI $CI1",
    );

    # update incident state of CI2 to 'Operational'
    $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $CI2,
        Name         => 'UnitTest - Bugfix4377 - CI-B',
        DefinitionID => $ConfigItemDefinitionIDs[0],
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Operational},
        UserID       => 1,
    );

    # check if version could be added
    $Self->True(
        $VersionID,
        "Test $TestCount: VersionAdd() for CI $CI2 - Set to 'Operational'",
    );

    # get the latest version for CI1
    $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI1,
    );

    # check if incident state of CI1 is 'Operational'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Operational',
        "Test $TestCount: Current incident state of CI $CI1",
    );

    # get the latest version for CI2
    $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI2,
    );

    # check if incident state of CI2 is 'Warning'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Operational',
        "Test $TestCount: Current incident state of CI $CI2",
    );

    # increase the test counter
    $TestCount++;
}

# ------------------------------------------------------------ #
# test for bugfix 10356
# ------------------------------------------------------------ #

{

    my $CI1 = $ConfigItemObject->ConfigItemLookup(
        ConfigItemNumber => $ConfigItemNumbers[72],
    );

    # add a version, set incident state to incident
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $CI1,
        Name         => 'UnitTest - Bugfix10356',
        DefinitionID => $ConfigItemDefinitionIDs[0],
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Incident},
        UserID       => 1,
    );

    # check if version could be added
    $Self->True(
        $VersionID,
        "Test $TestCount: VersionAdd() for $CI1 - Set to 'Incident'",
    );

    # get the latest version for CI1
    my $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI1,
    );

    # check if incident state of CI1 is 'Incident'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Incident',
        "Test $TestCount: Current incident state of CI $CI1",
    );

    # delete the last version
    my $VersionDeleteSuccess = $ConfigItemObject->VersionDelete(
        VersionID => $VersionRef->{VersionID},
        UserID    => 1,
    );

    # check if version could be deleted
    $Self->True(
        $VersionDeleteSuccess,
        "Test $TestCount: VersionDelete() for $CI1'",
    );

    # get the history
    my $HistoryRef = $ConfigItemObject->HistoryGet(
        ConfigItemID => $CI1,
    );

    my $LastHistoryEntry = pop @{$HistoryRef};

    # check if last history entry has the correct history type
    $Self->Is(
        $LastHistoryEntry->{HistoryType},
        'VersionDelete',
        "Test $TestCount: HistoryType of last version of CI $CI1",
    );

    # increase the test counter
    $TestCount++;
}

# ------------------------------------------------------------ #
# define general config item search tests
# ------------------------------------------------------------ #

my @SearchTests = (

    # search ALL config items in the two test classes
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the number param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            Number => $ConfigItemNumbers[50],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
        ],
    },

    # test the number param with wildcards
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            Number => '*' . $ConfigItemNumbers[50] . '*',
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
        ],
    },

    # test the number param with wildcards but with deactivated wildcard feature
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            Number         => '*' . $ConfigItemNumbers[50] . '*',
            UsingWildcards => 0,
        },
        ReferenceData => [],
    },

    # test the deployment state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [ $DeplStateListReverse{Production} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the deployment state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [ $DeplStateListReverse{Maintenance} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
        ],
    },

    # test the deployment state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [
                $DeplStateListReverse{Production},
                $DeplStateListReverse{Maintenance},
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [ $InciStateListReverse{Operational} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [ $InciStateListReverse{Incident} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test the incident state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [
                $InciStateListReverse{Incident},
                $InciStateListReverse{Operational},
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the order by param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs         => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            OrderBy          => ['CreateBy'],
            OrderByDirection => ['Up'],
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[60],
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
        ],
    },

    # test the limit param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 100,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the limit param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 3,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test the limit param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 2,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
        ],
    },

    # test the limit param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
        ],
    },

    # test the limit param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 0,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the create by param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            CreateBy => [ $UserIDs[2] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
        ],
    },

    # test the create by param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            CreateBy => [ $UserIDs[1],            $UserIDs[2] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the change by param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ChangeBy => [ $UserIDs[2] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
        ],
    },

    # test the change by param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            ChangeBy => [1],
        },
        ReferenceData => [],
    },

    # test the change by param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            ChangeBy => [ $UserIDs[1] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[60],
        ],
    },

    # test the change by param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            ChangeBy => [ $UserIDs[2] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
        ],
    },

    # search ALL config items in the two test classes using the version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the name param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name     => 'UnitTest - Class 3 ConfigItem 1 Version 1',
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
        ],
    },

    # test the name param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name     => 'UnitTest - Class 3 ConfigItem 3 Version 1',
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test the name param with an wildcard
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name     => 'UnitTest - * 1',
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the name param with an wildcard and a previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name                  => 'UnitTest - * 1',
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the name param with wildcards
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name     => 'UnitTest - Class 3*',
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test the name param with wildcards but with deactivated wildcard feature
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name           => 'UnitTest - Class 3*',
            ClassIDs       => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            UsingWildcards => 0,
        },
        ReferenceData => [],
    },

    # test the last version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name     => 'UnitTest - Class 3 ConfigItem 2 Version 1',
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [],
    },

    # test the PreviousVersionSearch param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name                  => 'UnitTest - Class 3 ConfigItem 2 Version 1',
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
        ],
    },

    # test the limit param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 100,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the limit param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 3,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test the limit param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 2,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
        ],
    },

    # test the limit param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
        ],
    },

    # test the deployment state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [
                $DeplStateListReverse{Production},
                $DeplStateListReverse{Maintenance},
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the deployment state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [ $DeplStateListReverse{Production} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the deployment state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [ $DeplStateListReverse{Maintenance} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
        ],
    },

    # test the deployment state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [
                $DeplStateListReverse{Production},
                $DeplStateListReverse{Maintenance},
            ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the deployment state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs          => [ $DeplStateListReverse{Production} ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the deployment state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs          => [ $DeplStateListReverse{Maintenance} ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
        ],
    },

    # test the incident state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [
                $InciStateListReverse{Operational},
                $InciStateListReverse{Incident},
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [ $InciStateListReverse{Operational} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [ $InciStateListReverse{Incident} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test the incident state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [
                $InciStateListReverse{Operational},
                $InciStateListReverse{Incident},
            ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs          => [ $InciStateListReverse{Operational} ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs          => [ $InciStateListReverse{Incident} ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Customer1)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            ChangeBy => [ $UserIDs[2] ],
            What     => [
                {
                    "[1]{'Version'}[1]{'Customer1'}[1]{'Content'}" => 'dummy_customer_for_unitest',
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}" => '2010-02-12',
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (DateTime1)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'DateTime1'}[1]{'Content'}" => '2010-02-12 09:14',
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Customer1, Date1, DateTime1)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Customer1'}[1]{'Content'}" => 'dummy_customer_for_unitest',
                },
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}" => '2010-02-12',
                },
                {
                    "[1]{'Version'}[1]{'DateTime1'}[1]{'Content'}" => '2010-02-12 09:14',
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, <, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '<' => '2010-02-12',
                        }
                },
            ],

        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, <, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '<' => '2010-02-13',
                        }
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, <=, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '<=' => '2010-02-11',
                        }
                },
            ],

        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, <=, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '<=' => '2010-02-12',
                        }
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, <=, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '<=' => '2010-02-13',
                        }
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, =, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '=' => '2010-02-11',
                        }
                },
            ],
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, =, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '=' => '2010-02-12',
                        }
                },
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, !=, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '!=' => '2010-02-12',
                        }
                },
            ],
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, !=, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '!=' => '2010-02-13',
                        }
                },
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, >=, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '>=' => '2010-02-13',
                        }
                },
            ],
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, >=, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '>=' => '2010-02-12',
                        }
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, >=, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '>=' => '2010-02-11',
                        }
                },
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, >, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '>' => '2010-02-12',
                        }
                },
            ],
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, >, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '>' => '2010-02-11',
                        }
                },
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, -between, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => { '-between' => [ '2010-01-01', '2010-01-31' ] }
                },
            ],
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, -between, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => { '-between' => [ '2010-02-01', '2010-02-31' ] }
                },
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with Name = 0 without any wildcards
    # should return no results
    # Bugfix# 8881
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            Name     => 0,
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with Number = 0 without any wildcards
    # should return no results
    # Bugfix# 8881
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            Number   => 0,
        },
        ReferenceData => [
        ],
    },
);

# ------------------------------------------------------------ #
# run general config item search tests
# ------------------------------------------------------------ #

# $SearchTestCount provides grouping of test cases
my $SearchTestCount = 1;

TEST:
for my $Test (@SearchTests) {

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

        $Self->True(
            0,
            "SearchTest $SearchTestCount: No SearchData found for this test.",
        );

        next TEST;
    }

    if ( !$Test->{Function} || ref $Test->{Function} ne 'ARRAY' || !@{ $Test->{Function} } ) {
        $Test->{Function} = ['ConfigItemSearchExtended'];
    }

    for my $Function ( @{ $Test->{Function} } ) {

        # start search
        my $ConfigItemList = $ConfigItemObject->$Function(
            %{ $Test->{SearchData} },
        );

        # check the config item list
        if ( $Test->{ReferenceData} ) {

            $Self->True(
                $ConfigItemList && ref $ConfigItemList eq 'ARRAY',
                "SearchTest $SearchTestCount: $Function() - List is an array reference.",
            );

            next TEST if !$ConfigItemList;
        }
        else {

            $Self->False(
                $ConfigItemList,
                "SearchTest $SearchTestCount: $Function() - Return false.",
            );

            next TEST if !$ConfigItemList;
        }

        # check number of found config items
        $Self->Is(
            scalar @{$ConfigItemList},
            scalar @{ $Test->{ReferenceData} },
            "SearchTest $SearchTestCount: $Function() - correct number of found config items",
        );

        my @ReferenceList;
        for my $Number ( @{ $Test->{ReferenceData} } ) {

            # find id of the item
            $DBObject->Prepare(
                SQL => "SELECT id FROM configitem WHERE "
                    . "configitem_number = '$Number' "
                    . "ORDER BY id DESC",
                Limit => 1,
            );

            # fetch the result
            my $ConfigItemID;
            while ( my @Row = $DBObject->FetchrowArray() ) {
                $ConfigItemID = $Row[0];
            }

            push @ReferenceList, $ConfigItemID;
        }

        # check arrays
        $Self->IsDeeply(
            $ConfigItemList,
            \@ReferenceList,
            "SearchTest $SearchTestCount: $Function() - List",
        );
    }
}
continue {
    $SearchTestCount++;
}

# ------------------------------------------------------------ #
# testing adding of invalid definitions
# ------------------------------------------------------------ #

{
    # define some invalid definitions

    my @InvalidConfigItemDefinitions;

    # Data sctructure is no array reference.
    push @InvalidConfigItemDefinitions, "
        {
            Key        => 'Customer1',
            Name       => 'Customer 1',
            Searchable => 1,
            Input      => {
                Type => 'Customer',
            },
        }
    ";

    # Data sctructure hash elements contain no data.
    push @InvalidConfigItemDefinitions, " [
        {
        },
        {
        },
    ]";

    # Missing comma between the 2 elements
    push @InvalidConfigItemDefinitions, " [
        {
            Key        => 'Customer1',
            Name       => 'Customer 1',
            Searchable => 1,
            Input      => {
                Type => 'Customer',
            },
        }
        {
            Key        => 'Date1',
            Name       => 'Date 1',
            Searchable => 1,
            Input      => {
                Type => 'Date',
            },
        },
    ]";

    # Key "Customer 1" containing a space.
    push @InvalidConfigItemDefinitions, " [
        {
            Key        => 'Customer 1',   # Key containing a space!
            Name       => 'Customer 1',
            Searchable => 1,
            Input      => {
                Type => 'Customer',
            },
        },
    ]";

    # Key "Customer1ΣՎέ" containing non-ascii characters.
    push @InvalidConfigItemDefinitions, " [
        {
            Key        => 'Customer1ΣՎέ',
            Name       => 'Customer 1',
            Searchable => 1,
            Input      => {
                Type => 'Customer',
            },
        },
    ]";

    # Invalid empty sub element.
    push @InvalidConfigItemDefinitions, " [
        {
            Key        => 'Customer1',
            Name       => 'Customer 1',
            Searchable => 1,
            Input      => {
                Type => 'Customer',
            },
            Sub => [
                {},
            ],
        },
    ]";

    # generate a random name
    my $ClassName = 'UnitTest' . $Helper->GetRandomID();

    # add an unittest config item class
    my $ClassID = $GeneralCatalogObject->ItemAdd(
        Class   => 'ITSM::ConfigItem::Class',
        Name    => $ClassName,
        ValidID => 1,
        UserID  => 1,
    );

    # check class id
    if ( !$ClassID ) {
        $Self->True(
            0,
            "Can't add new config item class.",
        );
    }

    for my $Definition (@InvalidConfigItemDefinitions) {

        # add a definition to the class
        my $DefinitionID = $ConfigItemObject->DefinitionAdd(
            ClassID    => $ClassID,
            Definition => $Definition,
            UserID     => 1,
        );

        # check definition id, must be false, because all definitions have errors
        $Self->False(
            $DefinitionID,
            "Can't add new config item definition.",
        );
    }

}

# ------------------------------------------------------------ #
# testing support for attachments
# ------------------------------------------------------------ #

my $AttachmentTestConfigItemID = $ConfigItemIDs[0];

# verify that initially no attachment exists
my @AttachmentList = $ConfigItemObject->ConfigItemAttachmentList(
    ConfigItemID => $AttachmentTestConfigItemID,
);

$Self->Is(
    scalar @AttachmentList,
    0,
    'No attachments initially',
);

my @TestFileList = (
    {
        Filename    => 'first attachment',
        Content     => 'First attachment from ITSMConfigItem.t',
        ContentType => 'text/plain',
    },
    {
        Filename    => 'second attachment',
        Content     => 'Second attachment from ITSMConfigItem.t',
        ContentType => 'text/plain',
    },
);

my $FileCount;
for my $TestFile (@TestFileList) {

    $FileCount++;

    my $AddOk = $ConfigItemObject->ConfigItemAttachmentAdd(
        %{$TestFile},
        ConfigItemID => $AttachmentTestConfigItemID,
        UserID       => 1,
    );
    $Self->True(
        $AddOk,
        "Attachment $FileCount: attachment added",
    );

    my @AttachmentList = $ConfigItemObject->ConfigItemAttachmentList(
        ConfigItemID => $AttachmentTestConfigItemID,
        UserID       => 1,
    );
    $Self->Is(
        scalar @AttachmentList,
        $FileCount,
        "Attachment $FileCount: number of attachments after adding",
    );

    # check whether the last added attachment is in the list
    my %AttachmentLookup = map { $_ => 1 } @AttachmentList;
    $Self->True(
        $AttachmentLookup{ $TestFile->{Filename} },
        "Attachment $FileCount: filename from ConfigItemAttachmentList()",
    );

    # get the attachment
    my $Attachment = $ConfigItemObject->ConfigItemAttachmentGet(
        ConfigItemID => $AttachmentTestConfigItemID,
        Filename     => $TestFile->{Filename},
    );
    $Self->True(
        $Attachment,
        "Attachment $FileCount: ConfigItemAttachmentGet() returned true",
    );

    # check attachment file attributes
    for my $Attribute (qw(Filename Content ContentType)) {
        $Self->Is(
            $Attachment->{$Attribute},
            $TestFile->{$Attribute},
            "Attachment $FileCount: $Attribute from ConfigItemAttachmentGet",
        );
    }

    # check existence of attachment
    my $AttachmentExists = $ConfigItemObject->ConfigItemAttachmentExists(
        ConfigItemID => $AttachmentTestConfigItemID,
        Filename     => $TestFile->{Filename},
        UserID       => 1,
    );
    $Self->True(
        $AttachmentExists,
        "Attachment $FileCount: attachment exists",
    );

}

# now delete the attachments
$FileCount = 0;
my $MaxTestFiles = scalar @TestFileList;
for my $TestFile (@TestFileList) {

    $FileCount++;

    my $DeleteOk = $ConfigItemObject->ConfigItemAttachmentDelete(
        ConfigItemID => $AttachmentTestConfigItemID,
        Filename     => $TestFile->{Filename},
        UserID       => 1,
    );
    $Self->True(
        $DeleteOk,
        "Attachment $FileCount: attachment deleted",
    );

    my @AttachmentList = $ConfigItemObject->ConfigItemAttachmentList(
        ConfigItemID => $AttachmentTestConfigItemID,
        UserID       => 1,
    );

    $Self->Is(
        scalar @AttachmentList,
        $MaxTestFiles - $FileCount,
        "Attachment $FileCount: number of attachments after deletion",
    );

    my $AttachmentExists = $ConfigItemObject->ConfigItemAttachmentExists(
        Filename     => $TestFile->{Filename},
        ConfigItemID => $AttachmentTestConfigItemID,
        UserID       => 1,
    );
    $Self->False(
        $AttachmentExists,
        "Attachment $FileCount: attachment is gone",
    );
}

# check config item delete
my $DeleteTestCount = 1;
for my $ConfigItemID (@ConfigItemIDs) {
    my $DeleteOk = $ConfigItemObject->ConfigItemDelete(
        ConfigItemID => $ConfigItemID,
        UserID       => 1,
    );
    $Self->True(
        $DeleteOk,
        "DeleteTest $DeleteTestCount - ConfigItemDelete() (ConfigItemID=$ConfigItemID)"
    );

    # double check if config item is really deleted
    my $ConfigItemData = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => $ConfigItemID,
        UserID       => 1,
        Cache        => 0,
    );
    $Self->False(
        $ConfigItemData->{ConfigItemID},
        "DeleteTest $DeleteTestCount - double check (ConfigItemID=$ConfigItemID)",
    );

    $DeleteTestCount++;
}

# cleanup is done by RestoreDatabase

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;

use utf8;

use vars qw($Self);

my $DBObject             = $Kernel::OM->Get('Kernel::System::DB');
my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');
my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $LinkObject           = $Kernel::OM->Get('Kernel::System::LinkObject');
my $UserObject           = $Kernel::OM->Get('Kernel::System::User');

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# define needed variable
my $RandomID = $Helper->GetRandomID();

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# perform ConfigItemCount to fill the empty fields
$ConfigItemObject->ConfigItemCount();

# create needed users
my @UserIDs;
{

    # disable email checks to create new user
    my $CheckEmailAddressesOrg = $ConfigObject->Get('CheckEmailAddresses') || 1;
    $ConfigObject->Set(
        Key   => 'CheckEmailAddresses',
        Value => 0,
    );

    for my $Counter ( 1 .. 3 ) {

        # create new users for the tests
        my $UserID = $UserObject->UserAdd(
            UserFirstname => 'ITSMConfigItem' . $Counter,
            UserLastname  => 'UnitTest',
            UserLogin     => 'UnitTest-ITSMConfigItem-' . $Counter . $RandomID,
            UserEmail     => 'UnitTest-ITSMConfigItem-' . $Counter . '@localhost',
            ValidID       => 1,
            ChangeUserID  => 1,
        );

        push @UserIDs, $UserID;
    }

    # restore original email check param
    $ConfigObject->Set(
        Key   => 'CheckEmailAddresses',
        Value => $CheckEmailAddressesOrg,
    );
}

my $GeneralCatalogClass = 'UnitTest' . $RandomID;

# add a general catalog test list
for my $Name (qw(Test1 Test2 Test3 Test4)) {

    # add a new item
    my $ItemID = $GeneralCatalogObject->ItemAdd(
        Class   => $GeneralCatalogClass,
        Name    => $Name,
        ValidID => 1,
        UserID  => 1,
    );

    # check item id
    if ( !$ItemID ) {

        $Self->True(
            0,
            "Can't add new general catalog item.",
        );
    }
}

# define the first test definition (all provided data types)
my @ConfigItemPerlDefinitions;
$ConfigItemPerlDefinitions[0] = " [
    {
        Key        => 'Customer1',
        Name       => 'Customer 1',
        Searchable => 1,
        Input      => {
            Type => 'Customer',
        },
    },
    {
        Key        => 'Date1',
        Name       => 'Date 1',
        Searchable => 1,
        Input      => {
            Type => 'Date',
        },
    },
    {
        Key        => 'DateTime1',
        Name       => 'Date Time 1',
        Searchable => 1,
        Input      => {
            Type => 'DateTime',
        },
    },
    {
        Key   => 'Dummy1',
        Name  => 'Dummy 1',
        Input => {
            Type => 'Dummy',
        },
    },
    {
        Key        => 'GeneralCatalog1',
        Name       => 'GeneralCatalog 1',
        Searchable => 1,
        Input      => {
            Type  => 'GeneralCatalog',
            Class => '$GeneralCatalogClass',
        },
    },
    {
        Key        => 'Integer1',
        Name       => 'Integer 1',
        Searchable => 1,
        Input      => {
            Type => 'Integer',
        },
    },
    {
        Key        => 'Text1',
        Name       => 'Text 1',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 50,
        },
    },
    {
        Key        => 'TextArea1',
        Name       => 'TextArea 1',
        Searchable => 1,
        Input      => {
            Type => 'TextArea',
        },
    },
] ";

# define the second test definition (sub data types)
$ConfigItemPerlDefinitions[1] = " [
    {
        Key        => 'Main1',
        Name       => 'Main 1',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 50,
        },
        CountMax => 10,
        Sub => [
            {
                Key        => 'Main1Sub1',
                Name       => 'Main 1 Sub 1',
                Searchable => 1,
                Input      => {
                    Type      => 'Text',
                    Size      => 50,
                    MaxLength => 50,
                },
                CountMax => 10,
                Sub => [
                    {
                        Key        => 'Main1Sub1SubSub1',
                        Name       => 'Main 1 Sub 1 SubSub 1',
                        Searchable => 1,
                        Input      => {
                            Type      => 'Text',
                            Size      => 50,
                            MaxLength => 50,
                        },
                        CountMax => 10,
                    },
                    {
                        Key        => 'Main1Sub1SubSub2',
                        Name       => 'Main 1 Sub 1 SubSub 2',
                        Searchable => 1,
                        Input      => {
                            Type => 'TextArea',
                        },
                        CountMax => 10,
                    },
                ],
            },
            {
                Key        => 'Main1Sub2',
                Name       => 'Main 1 Sub 2',
                Searchable => 1,
                Input      => {
                    Type => 'TextArea',
                },
                CountMax => 10,
            },
        ],
    },
    {
        Key        => 'Main2',
        Name       => 'Main 2',
        Searchable => 1,
        Input      => {
            Type => 'TextArea',
        },
        CountMax => 10,
        Sub => [
            {
                Key        => 'Main2Sub1',
                Name       => 'Main 2 Sub 1',
                Searchable => 1,
                Input      => {
                    Type      => 'Text',
                    Size      => 50,
                    MaxLength => 50,
                },
                CountMax => 10,
            },
            {
                Key        => 'Main2Sub2',
                Name       => 'Main 2 Sub 2',
                Searchable => 1,
                Input      => {
                    Type => 'TextArea',
                },
                CountMax => 10,
            },
        ],
    },
] ";

# define the third test definition (especially for search tests with XMLData)
$ConfigItemPerlDefinitions[2] = " [
    {
        Key        => 'Customer1',
        Name       => 'Customer 1',
        Searchable => 1,
        Input      => {
            Type => 'Customer',
        },
    },
    {
        Key        => 'Date1',
        Name       => 'Date 1',
        Searchable => 1,
        Input      => {
            Type => 'Date',
        },
    },
    {
        Key        => 'DateTime1',
        Name       => 'Date Time 1',
        Searchable => 1,
        Input      => {
            Type => 'DateTime',
        },
    },
] ";

# define the fourth test definition (only for search tests)
$ConfigItemPerlDefinitions[3] = " [
    {
        Key        => 'Customer1',
        Name       => 'Customer 1',
        Searchable => 1,
        Input      => {
            Type => 'Customer',
        },
    },
] ";

my $YAMLObject = $Kernel::OM->Get('Kernel::System::YAML');

my @ConfigItemDefinitions;
for my $PerlDefinition (@ConfigItemPerlDefinitions) {
    my $YAMLDefinition = $YAMLObject->Dump(
        Data => eval $PerlDefinition,    ## no critic
    );
    push @ConfigItemDefinitions, $YAMLDefinition;
}

# add the test classes
my @ConfigItemClassIDs;
my @ConfigItemClasses;
my @ConfigItemDefinitionIDs;
for my $Definition (@ConfigItemDefinitions) {

    # generate a random name
    my $ClassName = 'UnitTest' . $Helper->GetRandomID();

    # add an unittest config item class
    my $ClassID = $GeneralCatalogObject->ItemAdd(
        Class   => 'ITSM::ConfigItem::Class',
        Name    => $ClassName,
        ValidID => 1,
        UserID  => 1,
    );

    # check class id
    if ( !$ClassID ) {

        $Self->True(
            0,
            "Can't add new config item class.",
        );
    }

    push @ConfigItemClassIDs, $ClassID;
    push @ConfigItemClasses,  $ClassName;

    # add a definition to the class
    my $DefinitionID = $ConfigItemObject->DefinitionAdd(
        ClassID    => $ClassID,
        Definition => $Definition,
        UserID     => 1,
    );

    # check definition id
    if ( !$DefinitionID ) {

        $Self->True(
            0,
            "Can't add new config item definition.",
        );
    }

    push @ConfigItemDefinitionIDs, $DefinitionID;
}

# test DefinitionList for those simple cases
my $Counter = 0;
for my $ClassID (@ConfigItemClassIDs) {
    my $DefinitionListRef = $ConfigItemObject->DefinitionList(
        ClassID => $ClassID,
    );

    # expect a single definition per config item class
    $Self->Is(
        scalar @{$DefinitionListRef},
        1,
        "DefinitionList() for class id $ClassID: got a single result",
    );

    # expect the remembered definition id in the first definition
    $Self->Is(
        $DefinitionListRef->[0]->{DefinitionID},
        $ConfigItemDefinitionIDs[$Counter],
        "DefinitionList() for class id $ClassID: got expected definition id",
    );

    $Counter++;
}

# create some random numbers
my @ConfigItemNumbers;
for ( 1 .. 100 ) {
    push @ConfigItemNumbers, $Helper->GetRandomNumber();
}

# get class list
my $ClassList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::Class',
);
my %ClassListReverse = reverse %{$ClassList};

# get deployment state list
my $DeplStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %DeplStateListReverse = reverse %{$DeplStateList};

# get incident state list
my $InciStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %InciStateListReverse = reverse %{$InciStateList};

# get general catalog test list
my $GeneralCatalogList = $GeneralCatalogObject->ItemList(
    Class => $GeneralCatalogClass,
);
my %GeneralCatalogListReverse = reverse %{$GeneralCatalogList};

# ------------------------------------------------------------ #
# define general config item tests
# ------------------------------------------------------------ #

my $ConfigItemTests = [

    # ConfigItemAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                UserID => 1,
            },
        },
    },

    # ConfigItemAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                ClassID => $ConfigItemClassIDs[0],
            },
        },
    },

    # invalid class id is given (check return false)
    {
        SourceData => {
            ConfigItemAdd => {
                ClassID => $ConfigItemClassIDs[-1] + 1,
                UserID  => 1,
            },
        },
    },

    # all required config item values are given (check returned config item values)
    {
        SourceData => {
            ConfigItemAdd => {
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
        },
        ReferenceData => {
            ConfigItemGet => {
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # all required config item values are given (check number attribute)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[0],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => $UserIDs[1],
            },
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[0],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => $UserIDs[1],
                ChangeBy         => $UserIDs[1],
            },
        },
    },

    # config item with this number already exists (check return false)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[0],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
        },
    },

    # VersionAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[1],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[1],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # VersionAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[2],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name        => 'UnitTest - Class 1 ConfigItem 2 Version 1',
                    DeplStateID => $DeplStateListReverse{Production},
                    InciStateID => $InciStateListReverse{Operational},
                    UserID      => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[2],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # VersionAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[3],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 3 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[3],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # VersionAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[4],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 4 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[4],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # VersionAdd doesn't contains all data (check required attributes)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[5],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => $UserIDs[1],
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 5 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[5],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => $UserIDs[1],
                ChangeBy         => $UserIDs[1],
            },
        },
    },

    # invalid deployment state id is given (check returned values)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[6],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 6 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => 9999999,
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[6],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # invalid incident state id is given (check returned values)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[7],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 7 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => 9999999,
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[7],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                LastVersionID    => undef,
                CurDeplStateID   => undef,
                CurDeplState     => undef,
                CurDeplStateType => undef,
                CurInciStateID   => undef,
                CurInciState     => undef,
                CurInciStateType => undef,
                CreateBy         => 1,
                ChangeBy         => 1,
            },
        },
    },

    # all required values are given (check returned values)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[8],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 8 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[8],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Operational},
                CurInciState     => 'Operational',
                CurInciStateType => 'operational',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[8],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Class 1 ConfigItem 8 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
        },
    },

    # all required values are given (general check with two versions)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[9],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 9 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 9 Version 2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => $UserIDs[1],
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[9],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Operational},
                CurInciState     => 'Operational',
                CurInciStateType => 'operational',
                CreateBy         => 1,
                ChangeBy         => $UserIDs[1],
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[9],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Class 1 ConfigItem 9 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => 1,
                },
                {
                    Number           => $ConfigItemNumbers[9],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Class 1 ConfigItem 9 Version 2',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => $UserIDs[1],
                },
            ],
        },
    },

    # all required values are given (check the calculation of deployment and incident state)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[10],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 10 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Planned},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Class 1 ConfigItem 10 Version 2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[10],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Maintenance},
                CurDeplState     => 'Maintenance',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[10],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Class 1 ConfigItem 10 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Planned},
                    DeplState        => 'Planned',
                    DeplStateType    => 'preproductive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
                {
                    Number           => $ConfigItemNumbers[10],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Class 1 ConfigItem 10 Version 2',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Maintenance},
                    DeplState        => 'Maintenance',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
        },
    },

    # add config item only for later search tests
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[50],
                ClassID => $ConfigItemClassIDs[2],
                UserID  => $UserIDs[2],
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 3 ConfigItem 1 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[2],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => $UserIDs[2],
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[50],
                ClassID          => $ConfigItemClassIDs[2],
                Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => $UserIDs[2],
                ChangeBy         => $UserIDs[2],
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[50],
                    ClassID          => $ConfigItemClassIDs[2],
                    Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                    Name             => 'UnitTest - Class 3 ConfigItem 1 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[2],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => $UserIDs[2],
                },
            ],
        },
    },

    # add config item only for later search tests
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[51],
                ClassID => $ConfigItemClassIDs[2],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 3 ConfigItem 2 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[2],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Class 3 ConfigItem 2 Version 2',
                    DefinitionID => $ConfigItemDefinitionIDs[2],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => $UserIDs[1],
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[51],
                ClassID          => $ConfigItemClassIDs[2],
                Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                CurDeplStateID   => $DeplStateListReverse{Maintenance},
                CurDeplState     => 'Maintenance',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => 1,
                ChangeBy         => $UserIDs[1],
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[51],
                    ClassID          => $ConfigItemClassIDs[2],
                    Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                    Name             => 'UnitTest - Class 3 ConfigItem 2 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[2],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
                {
                    Number           => $ConfigItemNumbers[51],
                    ClassID          => $ConfigItemClassIDs[2],
                    Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                    Name             => 'UnitTest - Class 3 ConfigItem 2 Version 2',
                    DefinitionID     => $ConfigItemDefinitionIDs[2],
                    DeplStateID      => $DeplStateListReverse{Maintenance},
                    DeplState        => 'Maintenance',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => $UserIDs[1],
                },
            ],
        },
    },

    # add config item only for later search tests, including XMLData
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[52],
                ClassID => $ConfigItemClassIDs[2],
                UserID  => $UserIDs[2],
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 3 ConfigItem 3 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[2],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => $UserIDs[2],
                    XMLData      => [
                        undef,
                        {
                            Version => [
                                undef,
                                {
                                    Customer1 => [
                                        undef,
                                        {
                                            Content => 'dummy_customer_for_unitest',
                                        },
                                    ],
                                    Date1 => [
                                        undef,
                                        {
                                            Content => '2010-02-12',
                                        },
                                    ],
                                    DateTime1 => [
                                        undef,
                                        {
                                            Content => '2010-02-12 09:14',
                                        },
                                    ],
                                },
                            ],
                        },
                    ],
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[52],
                ClassID          => $ConfigItemClassIDs[2],
                Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => $UserIDs[2],
                ChangeBy         => $UserIDs[2],
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[52],
                    ClassID          => $ConfigItemClassIDs[2],
                    Class            => $ClassList->{ $ConfigItemClassIDs[2] },
                    Name             => 'UnitTest - Class 3 ConfigItem 3 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[2],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [
                        undef,
                        {
                            'TagKey'  => '[1]',
                            'Version' => [
                                undef,
                                {
                                    'Customer1' => [
                                        undef,
                                        {
                                            'Content' => 'dummy_customer_for_unitest',
                                            'TagKey'  => '[1]{\'Version\'}[1]{\'Customer1\'}[1]',
                                        },
                                    ],
                                    'Date1' => [
                                        undef,
                                        {
                                            'Content' => '2010-02-12',
                                            'TagKey'  => '[1]{\'Version\'}[1]{\'Date1\'}[1]',
                                        },
                                    ],
                                    'DateTime1' => [
                                        undef,
                                        {
                                            'Content' => '2010-02-12 09:14',
                                            'TagKey'  => '[1]{\'Version\'}[1]{\'DateTime1\'}[1]',
                                        },
                                    ],
                                    'TagKey' => '[1]{\'Version\'}[1]',
                                },
                            ],
                        },
                    ],
                    CreateBy => $UserIDs[2],
                },
            ],
        },
    },

    # add config item only for later search tests
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[60],
                ClassID => $ConfigItemClassIDs[3],
                UserID  => $UserIDs[1],
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Class 4 ConfigItem 1 Version 1',
                    DefinitionID => $ConfigItemDefinitionIDs[2],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => $UserIDs[1],
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[60],
                ClassID          => $ConfigItemClassIDs[3],
                Class            => $ClassList->{ $ConfigItemClassIDs[3] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Operational},
                CurInciState     => 'Operational',
                CurInciStateType => 'operational',
                CreateBy         => $UserIDs[1],
                ChangeBy         => $UserIDs[1],
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[60],
                    ClassID          => $ConfigItemClassIDs[3],
                    Class            => $ClassList->{ $ConfigItemClassIDs[3] },
                    Name             => 'UnitTest - Class 4 ConfigItem 1 Version 1',
                    DefinitionID     => $ConfigItemDefinitionIDs[2],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => $UserIDs[1],
                },
            ],
        },
    },

    # added to check history functions
    # all required values are given (check the calculation of deployment and incident state)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[70],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - HistoryTest',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Planned},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - HistoryTest Version 2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[70],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Maintenance},
                CurDeplState     => 'Maintenance',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[70],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - HistoryTest',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Planned},
                    DeplState        => 'Planned',
                    DeplStateType    => 'preproductive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
                {
                    Number           => $ConfigItemNumbers[70],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - HistoryTest Version 2',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Maintenance},
                    DeplState        => 'Maintenance',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
            HistoryGet => [
                {
                    HistoryType   => 'ConfigItemCreate',
                    HistoryTypeID => 1,
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'VersionCreate',
                    HistoryTypeID => 6,
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'DefinitionUpdate',
                    HistoryTypeID => 8,
                    Comment       => $ConfigItemDefinitionIDs[0],
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'NameUpdate',
                    HistoryTypeID => 5,
                    Comment       => 'UnitTest - HistoryTest%%',
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'IncidentStateUpdate',
                    HistoryTypeID => 9,
                    Comment       => $InciStateListReverse{Operational} . '%%',
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'DeploymentStateUpdate',
                    HistoryTypeID => 10,
                    Comment       => $DeplStateListReverse{Planned} . '%%',
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'VersionCreate',
                    HistoryTypeID => 6,
                    CreateBy      => 1,
                },
                {
                    HistoryType   => 'NameUpdate',
                    HistoryTypeID => 5,
                    Comment =>
                        'UnitTest - HistoryTest Version 2%%UnitTest - HistoryTest',
                    CreateBy => 1,
                },
                {
                    HistoryType   => 'IncidentStateUpdate',
                    HistoryTypeID => 9,
                    Comment       => $InciStateListReverse{Incident} . '%%'
                        . $InciStateListReverse{Operational},
                    CreateBy => 1,
                },
                {
                    HistoryType   => 'DeploymentStateUpdate',
                    HistoryTypeID => 10,
                    Comment       => $DeplStateListReverse{Maintenance} . '%%'
                        . $DeplStateListReverse{Planned},
                    CreateBy => 1,
                },
            ],
        },
    },

    # added for Bug4196
    # all required values are given (check the calculation of deployment and incident state)
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[71],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Bugfix4196',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Planned},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Bugfix4196 V2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Bugfix4196 V2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => 1,
                },
                {
                    Name         => 'UnitTest - Bugfix4196 V2',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Maintenance},
                    InciStateID  => $InciStateListReverse{Incident},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[71],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Maintenance},
                CurDeplState     => 'Maintenance',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Incident},
                CurInciState     => 'Incident',
                CurInciStateType => 'incident',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[71],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Bugfix4196',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Planned},
                    DeplState        => 'Planned',
                    DeplStateType    => 'preproductive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
                {
                    Number           => $ConfigItemNumbers[71],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Bugfix4196 V2',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Maintenance},
                    DeplState        => 'Maintenance',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Maintenance},
                    CurDeplState     => 'Maintenance',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Incident},
                    InciState        => 'Incident',
                    InciStateType    => 'incident',
                    CurInciStateID   => $InciStateListReverse{Incident},
                    CurInciState     => 'Incident',
                    CurInciStateType => 'incident',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
        },
    },

    # added for Bug 4377 - CI-A
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[72],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Bugfix4377 - CI-A',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[72],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Operational},
                CurInciState     => 'Operational',
                CurInciStateType => 'operational',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[72],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Bugfix4377 - CI-A',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
        },
    },

    # added for Bug 4377 - CI-B
    {
        SourceData => {
            ConfigItemAdd => {
                Number  => $ConfigItemNumbers[73],
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            },
            VersionAdd => [
                {
                    Name         => 'UnitTest - Bugfix4377 - CI-B',
                    DefinitionID => $ConfigItemDefinitionIDs[0],
                    DeplStateID  => $DeplStateListReverse{Production},
                    InciStateID  => $InciStateListReverse{Operational},
                    UserID       => 1,
                },
            ],
        },
        ReferenceData => {
            ConfigItemGet => {
                Number           => $ConfigItemNumbers[73],
                ClassID          => $ConfigItemClassIDs[0],
                Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                CurDeplStateID   => $DeplStateListReverse{Production},
                CurDeplState     => 'Production',
                CurDeplStateType => 'productive',
                CurInciStateID   => $InciStateListReverse{Operational},
                CurInciState     => 'Operational',
                CurInciStateType => 'operational',
                CreateBy         => 1,
                ChangeBy         => 1,
            },
            VersionGet => [
                {
                    Number           => $ConfigItemNumbers[73],
                    ClassID          => $ConfigItemClassIDs[0],
                    Class            => $ClassList->{ $ConfigItemClassIDs[0] },
                    Name             => 'UnitTest - Bugfix4377 - CI-B',
                    DefinitionID     => $ConfigItemDefinitionIDs[0],
                    DeplStateID      => $DeplStateListReverse{Production},
                    DeplState        => 'Production',
                    DeplStateType    => 'productive',
                    CurDeplStateID   => $DeplStateListReverse{Production},
                    CurDeplState     => 'Production',
                    CurDeplStateType => 'productive',
                    InciStateID      => $InciStateListReverse{Operational},
                    InciState        => 'Operational',
                    InciStateType    => 'operational',
                    CurInciStateID   => $InciStateListReverse{Operational},
                    CurInciState     => 'Operational',
                    CurInciStateType => 'operational',
                    XMLData          => [],
                    CreateBy         => 1,
                },
            ],
        },
    },

];

# ------------------------------------------------------------ #
# run general config item tests
# ------------------------------------------------------------ #

my $TestCount = 1;
my @ConfigItemIDs;

TEST:
for my $Test ( @{$ConfigItemTests} ) {

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

        $Self->True(
            0,
            "Test $TestCount: No SourceData found for this test.",
        );

        next TEST;
    }

    # extract source data
    my $SourceData = $Test->{SourceData};

    # add a new config item
    my $ConfigItemID;
    if ( $SourceData->{ConfigItemAdd} ) {

        # add the new config item
        $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
            %{ $SourceData->{ConfigItemAdd} },
        );

        if ($ConfigItemID) {
            push @ConfigItemIDs, $ConfigItemID;
        }
    }

    # check the config item
    if ( $Test->{ReferenceData} && $Test->{ReferenceData}->{ConfigItemGet} ) {

        $Self->True(
            $ConfigItemID,
            "Test $TestCount: ConfigItemAdd() - Add new config item. Insert success.",
        );

        next TEST if !$ConfigItemID;
    }
    else {

        $Self->False(
            $ConfigItemID,
            "Test $TestCount: ConfigItemAdd() - Add new config item. Return false.",
        );
    }

    # add all defined versions
    my @VersionIDs;
    my %VersionIDsSeen;
    if ( $SourceData->{VersionAdd} ) {

        for my $Version ( @{ $SourceData->{VersionAdd} } ) {

            if ($ConfigItemID) {
                $Version->{ConfigItemID} = $ConfigItemID;
            }

            # add a new version
            my $VersionID = $ConfigItemObject->VersionAdd(
                %{$Version},
            );

            if ($VersionID) {
                push @VersionIDs, $VersionID if !$VersionIDsSeen{$VersionID}++;
            }
        }
    }

    # check the config item
    my $ConfigItemData;
    if ( $Test->{ReferenceData} && $Test->{ReferenceData}->{ConfigItemGet} ) {

        # get the config item data
        $ConfigItemData = $ConfigItemObject->ConfigItemGet(
            ConfigItemID => $ConfigItemID,
        );

        if ( !$ConfigItemData ) {

            $Self->True(
                0,
                "Test $TestCount: ConfigItemGet() - get config item data."
            );
        }

        # check all config item attributes
        my $Counter = 0;
        for my $Attribute ( sort keys %{ $Test->{ReferenceData}->{ConfigItemGet} } ) {

            # set content if values are undef
            if ( !defined $ConfigItemData->{$Attribute} ) {
                $ConfigItemData->{$Attribute} = 'UNDEF-unittest';
            }
            if ( !defined $Test->{ReferenceData}->{ConfigItemGet}->{$Attribute} ) {
                $Test->{ReferenceData}->{ConfigItemGet}->{$Attribute} = 'UNDEF-unittest';
            }

            # check attributes
            $Self->Is(
                $ConfigItemData->{$Attribute},
                $Test->{ReferenceData}->{ConfigItemGet}->{$Attribute},
                "Test $TestCount: ConfigItemGet() - $Attribute",
            );

            $Counter++;
        }
    }

    # check the versions
    if (
        $Test->{ReferenceData}
        && $Test->{ReferenceData}->{VersionGet}
        && @{ $Test->{ReferenceData}->{VersionGet} }
        )
    {

        $Self->Is(
            scalar @VersionIDs,
            scalar @{ $Test->{ReferenceData}->{VersionGet} },
            "Test $TestCount: VersionAdd() - correct number of versions",
        );

        next TEST if !$ConfigItemID;
    }
    else {

        $Self->False(
            scalar @VersionIDs,
            "Test $TestCount: VersionAdd() - no versions exits",
        );
    }

    next TEST if !$Test->{ReferenceData};
    next TEST if !$Test->{ReferenceData}->{VersionGet};

    my $Counter           = 0;
    my $LastVersionIDMust = 'UNDEF-unittest';

    # Refresh ITSMConfigurationManagement cache to avoid issues with previously cached values.
    $Kernel::OM->Get('Kernel::System::Cache')->CleanUp(
        Type => 'ITSMConfigurationManagement',
    );

    VERSIONID:
    for my $VersionID (@VersionIDs) {

        # get this version
        my $VersionData = $ConfigItemObject->VersionGet(
            VersionID  => $VersionID,
            XMLDataGet => 1,
        );

        if ( !$VersionData ) {

            $Self->True(
                0,
                "Test $TestCount: VersionGet() - get version data."
            );

            next VERSIONID;
        }

        # save last version id
        $LastVersionIDMust = $VersionData->{VersionID};

        # check all version attributes
        for my $Attribute ( sort keys %{ $Test->{ReferenceData}->{VersionGet}->[$Counter] } ) {

            # extract the needed attributes
            my $VersionAttribute   = $VersionData->{$Attribute};
            my $ReferenceAttribute = $Test->{ReferenceData}->{VersionGet}->[$Counter]->{$Attribute};

            # set content if values are undef
            if ( !defined $VersionAttribute ) {
                $VersionAttribute = 'UNDEF-unittest';
            }
            if ( !defined $ReferenceAttribute ) {
                $ReferenceAttribute = 'UNDEF-unittest';
            }

            # check attributes
            $Self->IsDeeply(
                $VersionAttribute,
                $ReferenceAttribute,
                "Test $TestCount: VersionGet() - $Attribute",
            );
        }

        $Counter++;
    }

    # prepare last version id
    my $LastVersionIDActual = 'UNDEF-unittest';
    if ( $ConfigItemData->{LastVersionID} ) {
        $LastVersionIDActual = $ConfigItemData->{LastVersionID};
    }

    # check last version id
    $Self->Is(
        $ConfigItemData->{LastVersionID},
        $LastVersionIDMust,
        "Test $TestCount: last version id identical",
    );

    # check history entries
    if (
        $Test->{ReferenceData}
        && $Test->{ReferenceData}->{HistoryGet}
        && @{ $Test->{ReferenceData}->{HistoryGet} }
        )
    {
        my $CompleteHistory = $ConfigItemObject->HistoryGet(
            ConfigItemID => $ConfigItemID,
        );

        # check nr of history entries
        $Self->Is(
            scalar @{ $Test->{ReferenceData}->{HistoryGet} },
            scalar @{$CompleteHistory},
            "Test $TestCount: nr of history entries",
        );

        CHECKNR: for my $CheckNr ( 0 .. $#{$CompleteHistory} ) {
            my $Check = $Test->{ReferenceData}->{HistoryGet}->[$CheckNr];
            my $Data  = $CompleteHistory->[$CheckNr];

            next CHECKNR if !( $Check && $Data );

            for my $Key ( sort keys %{$Check} ) {

                # check history data
                $Self->Is(
                    $Check->{$Key},
                    $Data->{$Key},
                    "Test $TestCount: $Key",
                );
            }
        }
    }
}
continue {
    $TestCount++;
}

# ------------------------------------------------------------ #
# test for bugfix 4377
# ------------------------------------------------------------ #

{

    my $CI1 = $ConfigItemObject->ConfigItemLookup(
        ConfigItemNumber => $ConfigItemNumbers[72],
    );

    my $CI2 = $ConfigItemObject->ConfigItemLookup(
        ConfigItemNumber => $ConfigItemNumbers[73],
    );

    # link the CI with a CI
    my $LinkResult = $LinkObject->LinkAdd(
        SourceObject => 'ITSMConfigItem',
        SourceKey    => $CI1,
        TargetObject => 'ITSMConfigItem',
        TargetKey    => $CI2,
        Type         => 'DependsOn',
        State        => 'Valid',
        UserID       => 1,
    );

    # update incident state of CI1
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $CI1,
        Name         => 'UnitTest - Bugfix4377 - CI-A',
        DefinitionID => $ConfigItemDefinitionIDs[0],
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Incident},
        UserID       => 1,
    );

    # check if version could be added
    $Self->True(
        $VersionID,
        "Test $TestCount: VersionAdd() for $CI1 - Set to 'Incident'",
    );

    # get the latest version for CI1
    my $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI1,
    );

    # check if incident state of CI1 is 'Incident'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Incident',
        "Test $TestCount: Current incident state of CI $CI1",
    );

    # get the latest version for CI2
    $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI2,
    );

    # check if incident state of CI2 is 'Warning'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Warning',
        "Test $TestCount: Current incident state of CI $CI2",
    );

    # update incident state of CI2 to 'Incident'
    $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $CI2,
        Name         => 'UnitTest - Bugfix4377 - CI-B',
        DefinitionID => $ConfigItemDefinitionIDs[0],
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Incident},
        UserID       => 1,
    );

    # check if version could be added
    $Self->True(
        $VersionID,
        "Test $TestCount: VersionAdd() for CI $CI2 - Set to 'Incident'",
    );

    # get the latest version for CI2
    $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI2,
    );

    # check if incident state of CI2 is 'Incident'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Incident',
        "Test $TestCount: Current incident state of CI $CI2",
    );

    # update incident state of CI1 to 'Operational'
    $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $CI1,
        Name         => 'UnitTest - Bugfix4377 - CI-A',
        DefinitionID => $ConfigItemDefinitionIDs[0],
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Operational},
        UserID       => 1,
    );

    # check if version could be added
    $Self->True(
        $VersionID,
        "Test $TestCount: VersionAdd() for CI $CI1 - Set to 'Operational'",
    );

    # get the latest version for CI1
    $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI1,
    );

    # check if incident state of CI1 is 'Warning' (because of linked CI2 in state 'incident')
    $Self->Is(
        $VersionRef->{CurInciState},
        'Warning',
        "Test $TestCount: Current incident state of CI $CI1",
    );

    # update incident state of CI2 to 'Operational'
    $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $CI2,
        Name         => 'UnitTest - Bugfix4377 - CI-B',
        DefinitionID => $ConfigItemDefinitionIDs[0],
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Operational},
        UserID       => 1,
    );

    # check if version could be added
    $Self->True(
        $VersionID,
        "Test $TestCount: VersionAdd() for CI $CI2 - Set to 'Operational'",
    );

    # get the latest version for CI1
    $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI1,
    );

    # check if incident state of CI1 is 'Operational'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Operational',
        "Test $TestCount: Current incident state of CI $CI1",
    );

    # get the latest version for CI2
    $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI2,
    );

    # check if incident state of CI2 is 'Warning'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Operational',
        "Test $TestCount: Current incident state of CI $CI2",
    );

    # increase the test counter
    $TestCount++;
}

# ------------------------------------------------------------ #
# test for bugfix 10356
# ------------------------------------------------------------ #

{

    my $CI1 = $ConfigItemObject->ConfigItemLookup(
        ConfigItemNumber => $ConfigItemNumbers[72],
    );

    # add a version, set incident state to incident
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $CI1,
        Name         => 'UnitTest - Bugfix10356',
        DefinitionID => $ConfigItemDefinitionIDs[0],
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Incident},
        UserID       => 1,
    );

    # check if version could be added
    $Self->True(
        $VersionID,
        "Test $TestCount: VersionAdd() for $CI1 - Set to 'Incident'",
    );

    # get the latest version for CI1
    my $VersionRef = $ConfigItemObject->VersionGet(
        ConfigItemID => $CI1,
    );

    # check if incident state of CI1 is 'Incident'
    $Self->Is(
        $VersionRef->{CurInciState},
        'Incident',
        "Test $TestCount: Current incident state of CI $CI1",
    );

    # delete the last version
    my $VersionDeleteSuccess = $ConfigItemObject->VersionDelete(
        VersionID => $VersionRef->{VersionID},
        UserID    => 1,
    );

    # check if version could be deleted
    $Self->True(
        $VersionDeleteSuccess,
        "Test $TestCount: VersionDelete() for $CI1'",
    );

    # get the history
    my $HistoryRef = $ConfigItemObject->HistoryGet(
        ConfigItemID => $CI1,
    );

    my $LastHistoryEntry = pop @{$HistoryRef};

    # check if last history entry has the correct history type
    $Self->Is(
        $LastHistoryEntry->{HistoryType},
        'VersionDelete',
        "Test $TestCount: HistoryType of last version of CI $CI1",
    );

    # increase the test counter
    $TestCount++;
}

# ------------------------------------------------------------ #
# define general config item search tests
# ------------------------------------------------------------ #

my @SearchTests = (

    # search ALL config items in the two test classes
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the number param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            Number => $ConfigItemNumbers[50],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
        ],
    },

    # test the number param with wildcards
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            Number => '*' . $ConfigItemNumbers[50] . '*',
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
        ],
    },

    # test the number param with wildcards but with deactivated wildcard feature
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            Number         => '*' . $ConfigItemNumbers[50] . '*',
            UsingWildcards => 0,
        },
        ReferenceData => [],
    },

    # test the deployment state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [ $DeplStateListReverse{Production} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the deployment state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [ $DeplStateListReverse{Maintenance} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
        ],
    },

    # test the deployment state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [
                $DeplStateListReverse{Production},
                $DeplStateListReverse{Maintenance},
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [ $InciStateListReverse{Operational} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [ $InciStateListReverse{Incident} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test the incident state param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [
                $InciStateListReverse{Incident},
                $InciStateListReverse{Operational},
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the order by param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs         => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            OrderBy          => ['CreateBy'],
            OrderByDirection => ['Up'],
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[60],
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
        ],
    },

    # test the limit param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 100,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the limit param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 3,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test the limit param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 2,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
        ],
    },

    # test the limit param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
        ],
    },

    # test the limit param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 0,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the create by param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            CreateBy => [ $UserIDs[2] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
        ],
    },

    # test the create by param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            CreateBy => [ $UserIDs[1],            $UserIDs[2] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the change by param
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ChangeBy => [ $UserIDs[2] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
        ],
    },

    # test the change by param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            ChangeBy => [1],
        },
        ReferenceData => [],
    },

    # test the change by param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            ChangeBy => [ $UserIDs[1] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[60],
        ],
    },

    # test the change by param in combination of the class id
    {
        Function   => [ 'ConfigItemSearchExtended', 'ConfigItemSearch' ],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            ChangeBy => [ $UserIDs[2] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
        ],
    },

    # search ALL config items in the two test classes using the version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the name param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name     => 'UnitTest - Class 3 ConfigItem 1 Version 1',
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
        ],
    },

    # test the name param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name     => 'UnitTest - Class 3 ConfigItem 3 Version 1',
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test the name param with an wildcard
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name     => 'UnitTest - * 1',
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the name param with an wildcard and a previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name                  => 'UnitTest - * 1',
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the name param with wildcards
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name     => 'UnitTest - Class 3*',
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test the name param with wildcards but with deactivated wildcard feature
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name           => 'UnitTest - Class 3*',
            ClassIDs       => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            UsingWildcards => 0,
        },
        ReferenceData => [],
    },

    # test the last version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name     => 'UnitTest - Class 3 ConfigItem 2 Version 1',
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
        },
        ReferenceData => [],
    },

    # test the PreviousVersionSearch param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            Name                  => 'UnitTest - Class 3 ConfigItem 2 Version 1',
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
        ],
    },

    # test the limit param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 100,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the limit param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 3,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test the limit param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 2,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
        ],
    },

    # test the limit param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            Limit    => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
        ],
    },

    # test the deployment state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [
                $DeplStateListReverse{Production},
                $DeplStateListReverse{Maintenance},
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the deployment state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [ $DeplStateListReverse{Production} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the deployment state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [ $DeplStateListReverse{Maintenance} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
        ],
    },

    # test the deployment state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs => [
                $DeplStateListReverse{Production},
                $DeplStateListReverse{Maintenance},
            ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the deployment state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs          => [ $DeplStateListReverse{Production} ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the deployment state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            DeplStateIDs          => [ $DeplStateListReverse{Maintenance} ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
        ],
    },

    # test the incident state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [
                $InciStateListReverse{Operational},
                $InciStateListReverse{Incident},
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [ $InciStateListReverse{Operational} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [ $InciStateListReverse{Incident} ],
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test the incident state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs     => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs => [
                $InciStateListReverse{Operational},
                $InciStateListReverse{Incident},
            ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs          => [ $InciStateListReverse{Operational} ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[60],
        ],
    },

    # test the incident state param with activated previous version search
    {
        Function   => ['VersionSearch'],
        SearchData => {
            ClassIDs              => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            InciStateIDs          => [ $InciStateListReverse{Incident} ],
            PreviousVersionSearch => 1,
        },
        ReferenceData => [
            $ConfigItemNumbers[50],
            $ConfigItemNumbers[51],
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Customer1)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => [ $ConfigItemClassIDs[2], $ConfigItemClassIDs[3] ],
            ChangeBy => [ $UserIDs[2] ],
            What     => [
                {
                    "[1]{'Version'}[1]{'Customer1'}[1]{'Content'}" => 'dummy_customer_for_unitest',
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}" => '2010-02-12',
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (DateTime1)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'DateTime1'}[1]{'Content'}" => '2010-02-12 09:14',
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Customer1, Date1, DateTime1)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Customer1'}[1]{'Content'}" => 'dummy_customer_for_unitest',
                },
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}" => '2010-02-12',
                },
                {
                    "[1]{'Version'}[1]{'DateTime1'}[1]{'Content'}" => '2010-02-12 09:14',
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, <, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '<' => '2010-02-12',
                        }
                },
            ],

        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, <, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '<' => '2010-02-13',
                        }
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, <=, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '<=' => '2010-02-11',
                        }
                },
            ],

        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, <=, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '<=' => '2010-02-12',
                        }
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, <=, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '<=' => '2010-02-13',
                        }
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, =, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '=' => '2010-02-11',
                        }
                },
            ],
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, =, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '=' => '2010-02-12',
                        }
                },
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, !=, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '!=' => '2010-02-12',
                        }
                },
            ],
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, !=, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '!=' => '2010-02-13',
                        }
                },
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, >=, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '>=' => '2010-02-13',
                        }
                },
            ],
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, >=, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '>=' => '2010-02-12',
                        }
                },
            ],

        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, >=, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '>=' => '2010-02-11',
                        }
                },
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, >, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '>' => '2010-02-12',
                        }
                },
            ],
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, >, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => {
                        '>' => '2010-02-11',
                        }
                },
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, -between, false)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => { '-between' => [ '2010-01-01', '2010-01-31' ] }
                },
            ],
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with 'What' (Date1, -between, true)
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            What     => [
                {
                    "[1]{'Version'}[1]{'Date1'}[1]{'Content'}"
                        => { '-between' => [ '2010-02-01', '2010-02-31' ] }
                },
            ],
        },
        ReferenceData => [
            $ConfigItemNumbers[52],
        ],
    },

    # test ConfigItemSearchExtended() with Name = 0 without any wildcards
    # should return no results
    # Bugfix# 8881
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            Name     => 0,
        },
        ReferenceData => [
        ],
    },

    # test ConfigItemSearchExtended() with Number = 0 without any wildcards
    # should return no results
    # Bugfix# 8881
    {
        Function   => ['ConfigItemSearchExtended'],
        SearchData => {
            ClassIDs => \@ConfigItemClassIDs,
            Number   => 0,
        },
        ReferenceData => [
        ],
    },
);

# ------------------------------------------------------------ #
# run general config item search tests
# ------------------------------------------------------------ #

# $SearchTestCount provides grouping of test cases
my $SearchTestCount = 1;

TEST:
for my $Test (@SearchTests) {

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

        $Self->True(
            0,
            "SearchTest $SearchTestCount: No SearchData found for this test.",
        );

        next TEST;
    }

    if ( !$Test->{Function} || ref $Test->{Function} ne 'ARRAY' || !@{ $Test->{Function} } ) {
        $Test->{Function} = ['ConfigItemSearchExtended'];
    }

    for my $Function ( @{ $Test->{Function} } ) {

        # start search
        my $ConfigItemList = $ConfigItemObject->$Function(
            %{ $Test->{SearchData} },
        );

        # check the config item list
        if ( $Test->{ReferenceData} ) {

            $Self->True(
                $ConfigItemList && ref $ConfigItemList eq 'ARRAY',
                "SearchTest $SearchTestCount: $Function() - List is an array reference.",
            );

            next TEST if !$ConfigItemList;
        }
        else {

            $Self->False(
                $ConfigItemList,
                "SearchTest $SearchTestCount: $Function() - Return false.",
            );

            next TEST if !$ConfigItemList;
        }

        # check number of found config items
        $Self->Is(
            scalar @{$ConfigItemList},
            scalar @{ $Test->{ReferenceData} },
            "SearchTest $SearchTestCount: $Function() - correct number of found config items",
        );

        my @ReferenceList;
        for my $Number ( @{ $Test->{ReferenceData} } ) {

            # find id of the item
            $DBObject->Prepare(
                SQL => "SELECT id FROM configitem WHERE "
                    . "configitem_number = '$Number' "
                    . "ORDER BY id DESC",
                Limit => 1,
            );

            # fetch the result
            my $ConfigItemID;
            while ( my @Row = $DBObject->FetchrowArray() ) {
                $ConfigItemID = $Row[0];
            }

            push @ReferenceList, $ConfigItemID;
        }

        # check arrays
        $Self->IsDeeply(
            $ConfigItemList,
            \@ReferenceList,
            "SearchTest $SearchTestCount: $Function() - List",
        );
    }
}
continue {
    $SearchTestCount++;
}

# ------------------------------------------------------------ #
# testing adding of invalid definitions
# ------------------------------------------------------------ #

{
    # define some invalid definitions

    my @InvalidConfigItemDefinitions;

    # Data sctructure is no array reference.
    push @InvalidConfigItemDefinitions, "
        {
            Key        => 'Customer1',
            Name       => 'Customer 1',
            Searchable => 1,
            Input      => {
                Type => 'Customer',
            },
        }
    ";

    # Data sctructure hash elements contain no data.
    push @InvalidConfigItemDefinitions, " [
        {
        },
        {
        },
    ]";

    # Missing comma between the 2 elements
    push @InvalidConfigItemDefinitions, " [
        {
            Key        => 'Customer1',
            Name       => 'Customer 1',
            Searchable => 1,
            Input      => {
                Type => 'Customer',
            },
        }
        {
            Key        => 'Date1',
            Name       => 'Date 1',
            Searchable => 1,
            Input      => {
                Type => 'Date',
            },
        },
    ]";

    # Key "Customer 1" containing a space.
    push @InvalidConfigItemDefinitions, " [
        {
            Key        => 'Customer 1',   # Key containing a space!
            Name       => 'Customer 1',
            Searchable => 1,
            Input      => {
                Type => 'Customer',
            },
        },
    ]";

    # Key "Customer1ΣՎέ" containing non-ascii characters.
    push @InvalidConfigItemDefinitions, " [
        {
            Key        => 'Customer1ΣՎέ',
            Name       => 'Customer 1',
            Searchable => 1,
            Input      => {
                Type => 'Customer',
            },
        },
    ]";

    # Invalid empty sub element.
    push @InvalidConfigItemDefinitions, " [
        {
            Key        => 'Customer1',
            Name       => 'Customer 1',
            Searchable => 1,
            Input      => {
                Type => 'Customer',
            },
            Sub => [
                {},
            ],
        },
    ]";

    # generate a random name
    my $ClassName = 'UnitTest' . $Helper->GetRandomID();

    # add an unittest config item class
    my $ClassID = $GeneralCatalogObject->ItemAdd(
        Class   => 'ITSM::ConfigItem::Class',
        Name    => $ClassName,
        ValidID => 1,
        UserID  => 1,
    );

    # check class id
    if ( !$ClassID ) {
        $Self->True(
            0,
            "Can't add new config item class.",
        );
    }

    for my $Definition (@InvalidConfigItemDefinitions) {

        # add a definition to the class
        my $DefinitionID = $ConfigItemObject->DefinitionAdd(
            ClassID    => $ClassID,
            Definition => $Definition,
            UserID     => 1,
        );

        # check definition id, must be false, because all definitions have errors
        $Self->False(
            $DefinitionID,
            "Can't add new config item definition.",
        );
    }

}

# ------------------------------------------------------------ #
# testing support for attachments
# ------------------------------------------------------------ #

my $AttachmentTestConfigItemID = $ConfigItemIDs[0];

# verify that initially no attachment exists
my @AttachmentList = $ConfigItemObject->ConfigItemAttachmentList(
    ConfigItemID => $AttachmentTestConfigItemID,
);

$Self->Is(
    scalar @AttachmentList,
    0,
    'No attachments initially',
);

my @TestFileList = (
    {
        Filename    => 'first attachment',
        Content     => 'First attachment from ITSMConfigItem.t',
        ContentType => 'text/plain',
    },
    {
        Filename    => 'second attachment',
        Content     => 'Second attachment from ITSMConfigItem.t',
        ContentType => 'text/plain',
    },
);

my $FileCount;
for my $TestFile (@TestFileList) {

    $FileCount++;

    my $AddOk = $ConfigItemObject->ConfigItemAttachmentAdd(
        %{$TestFile},
        ConfigItemID => $AttachmentTestConfigItemID,
        UserID       => 1,
    );
    $Self->True(
        $AddOk,
        "Attachment $FileCount: attachment added",
    );

    my @AttachmentList = $ConfigItemObject->ConfigItemAttachmentList(
        ConfigItemID => $AttachmentTestConfigItemID,
        UserID       => 1,
    );
    $Self->Is(
        scalar @AttachmentList,
        $FileCount,
        "Attachment $FileCount: number of attachments after adding",
    );

    # check whether the last added attachment is in the list
    my %AttachmentLookup = map { $_ => 1 } @AttachmentList;
    $Self->True(
        $AttachmentLookup{ $TestFile->{Filename} },
        "Attachment $FileCount: filename from ConfigItemAttachmentList()",
    );

    # get the attachment
    my $Attachment = $ConfigItemObject->ConfigItemAttachmentGet(
        ConfigItemID => $AttachmentTestConfigItemID,
        Filename     => $TestFile->{Filename},
    );
    $Self->True(
        $Attachment,
        "Attachment $FileCount: ConfigItemAttachmentGet() returned true",
    );

    # check attachment file attributes
    for my $Attribute (qw(Filename Content ContentType)) {
        $Self->Is(
            $Attachment->{$Attribute},
            $TestFile->{$Attribute},
            "Attachment $FileCount: $Attribute from ConfigItemAttachmentGet",
        );
    }

    # check existence of attachment
    my $AttachmentExists = $ConfigItemObject->ConfigItemAttachmentExists(
        ConfigItemID => $AttachmentTestConfigItemID,
        Filename     => $TestFile->{Filename},
        UserID       => 1,
    );
    $Self->True(
        $AttachmentExists,
        "Attachment $FileCount: attachment exists",
    );

}

# now delete the attachments
$FileCount = 0;
my $MaxTestFiles = scalar @TestFileList;
for my $TestFile (@TestFileList) {

    $FileCount++;

    my $DeleteOk = $ConfigItemObject->ConfigItemAttachmentDelete(
        ConfigItemID => $AttachmentTestConfigItemID,
        Filename     => $TestFile->{Filename},
        UserID       => 1,
    );
    $Self->True(
        $DeleteOk,
        "Attachment $FileCount: attachment deleted",
    );

    my @AttachmentList = $ConfigItemObject->ConfigItemAttachmentList(
        ConfigItemID => $AttachmentTestConfigItemID,
        UserID       => 1,
    );

    $Self->Is(
        scalar @AttachmentList,
        $MaxTestFiles - $FileCount,
        "Attachment $FileCount: number of attachments after deletion",
    );

    my $AttachmentExists = $ConfigItemObject->ConfigItemAttachmentExists(
        Filename     => $TestFile->{Filename},
        ConfigItemID => $AttachmentTestConfigItemID,
        UserID       => 1,
    );
    $Self->False(
        $AttachmentExists,
        "Attachment $FileCount: attachment is gone",
    );
}

# check config item delete
my $DeleteTestCount = 1;
for my $ConfigItemID (@ConfigItemIDs) {
    my $DeleteOk = $ConfigItemObject->ConfigItemDelete(
        ConfigItemID => $ConfigItemID,
        UserID       => 1,
    );
    $Self->True(
        $DeleteOk,
        "DeleteTest $DeleteTestCount - ConfigItemDelete() (ConfigItemID=$ConfigItemID)"
    );

    # double check if config item is really deleted
    my $ConfigItemData = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => $ConfigItemID,
        UserID       => 1,
        Cache        => 0,
    );
    $Self->False(
        $ConfigItemData->{ConfigItemID},
        "DeleteTest $DeleteTestCount - double check (ConfigItemID=$ConfigItemID)",
    );

    $DeleteTestCount++;
}

# cleanup is done by RestoreDatabase

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7Cgp1c2UgdmFycyAocXcoJFNlbGYpKTsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6T2JqZWN0TWFuYWdlcjsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6VmFyaWFibGVDaGVjayBxdyg6YWxsKTsKCm15ICRIZWxwZXJPYmplY3QgICAgICAgICAgICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKbXkgJENJQXR0YWNobWVudE9iamVjdCAgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW06OkNJQXR0YWNobWVudCcpOwpteSAkVW5pdFRlc3RJVFNNQ29uZmlnSXRlbU9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SVRTTUNvbmZpZ0l0ZW0nKTsKbXkgJENvbmZpZ0l0ZW1PYmplY3QgICAgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nKTsKCm15ICRDb25maWdJdGVtID0gJFVuaXRUZXN0SVRTTUNvbmZpZ0l0ZW1PYmplY3QtPkNvbmZpZ0l0ZW1DcmVhdGUoCiAgICBOYW1lICAgICAgICAgID0+ICdNYWMnLAogICAgQ2xhc3NOYW1lICAgICA9PiAnQ29tcHV0ZXInLAogICAgRGVwbFN0YXRlTmFtZSA9PiAnUHJvZHVjdGlvbicsCiAgICBJbmNpU3RhdGVOYW1lID0+ICdPcGVyYXRpb25hbCcsCik7CgpteSAkQXR0YWNobWVudElEID0gJEhlbHBlck9iamVjdC0+R2V0UmFuZG9tSUQoKTsKCm15ICRTdWNjZXNzID0gJENJQXR0YWNobWVudE9iamVjdC0+Q0lBdHRhY2htZW50QWRkKAogICAgQXR0YWNobWVudElEICAgID0+ICRBdHRhY2htZW50SUQsCiAgICBDb25maWdJdGVtRmllbGQgPT4gJ1Rlc3RGaWVsZCcsCiAgICBDb25maWdJdGVtSUQgICAgPT4gJENvbmZpZ0l0ZW0tPntDb25maWdJdGVtSUR9LAogICAgVmVyc2lvbklEICAgICAgID0+IDEsCiAgICBGaWxlbmFtZSAgICAgICAgPT4gJ2FzZGYudHh0JywKICAgIENvbnRlbnQgICAgICAgICA9PiAnY29udGVudCBibHVicycsCiAgICBDb250ZW50VHlwZSAgICAgPT4gJ3RleHQvcGxhaW4nLAopOwoKJFNlbGYtPlRydWUoCiAgICAkU3VjY2VzcywKICAgICdDSUF0dGFjaG1lbnRBZGQgc3VjY2Vzc2Z1bGwuJywKKTsKCm15ICVBdHRhY2htZW50ID0gJENJQXR0YWNobWVudE9iamVjdC0+Q0lBdHRhY2htZW50R2V0KAogICAgQXR0YWNobWVudElEID0+ICRBdHRhY2htZW50SUQsCik7CgpteSAlRXhwZWN0ZWRBdHRhY2htZW50ID0gKAogICAgQ29udGVudCAgICAgPT4gJ2NvbnRlbnQgYmx1YnMnLAogICAgQ29udGVudFR5cGUgPT4gJ3RleHQvcGxhaW4nLAogICAgRmlsZW5hbWUgICAgPT4gJ2FzZGYudHh0JywKICAgIEZpbGVzaXplUmF3ID0+ICcxMycsCiAgICBUeXBlICAgICAgICA9PiAnYXR0YWNobWVudCcsCiAgICBGaWxlc2l6ZSAgICA9PiAnMTMgQicsCik7CgokU2VsZi0+SXNEZWVwbHkoCiAgICBcJUF0dGFjaG1lbnQsCiAgICBcJUV4cGVjdGVkQXR0YWNobWVudCwKICAgICdDSUF0dGFjaG1lbnRHZXQgd2FzIHN1Y2Nlc3NmdWxsLicsCik7CgpteSBARXhwZWN0ZWRDSUF0dGFjaG1lbnRMaXN0ID0gKCRBdHRhY2htZW50SUQpOwoKbXkgQENJQXR0YWNobWVudExpc3QgPSAkQ0lBdHRhY2htZW50T2JqZWN0LT5DSUF0dGFjaG1lbnRMaXN0KAogICAgQ29uZmlnSXRlbUZpZWxkID0+ICdUZXN0RmllbGQnLAogICAgQ29uZmlnSXRlbUlEICAgID0+ICRDb25maWdJdGVtLT57Q29uZmlnSXRlbUlEfSwKICAgIFZlcnNpb25JRCAgICAgICA9PiAxLAopOwoKJFNlbGYtPklzRGVlcGx5KAogICAgXEBDSUF0dGFjaG1lbnRMaXN0LAogICAgXEBFeHBlY3RlZENJQXR0YWNobWVudExpc3QsCiAgICAiQ0lBdHRhY2htZW50TGlzdCB3YXMgc3VjY2Vzc2Z1bGwuIChhZnRlciBDSUF0dGFjaG1lbnRBZGQpIiwKKTsKCm15IEBDb25maWdJdGVtQXR0YWNobWVudExpc3QgPSAkQ29uZmlnSXRlbU9iamVjdC0+Q29uZmlnSXRlbUF0dGFjaG1lbnRMaXN0KAogICAgQ29uZmlnSXRlbUlEID0+ICRDb25maWdJdGVtLT57Q29uZmlnSXRlbUlEfSwKKTsKCiRTZWxmLT5Jc0RlZXBseSgKICAgIFxAQ29uZmlnSXRlbUF0dGFjaG1lbnRMaXN0LAogICAgW10sCiAgICAiQ29uZmlnSXRlbUF0dGFjaG1lbnRMaXN0IHdhcyBzdWNjZXNzZnVsbC4iLAopOwoKJFN1Y2Nlc3MgPSAkQ0lBdHRhY2htZW50T2JqZWN0LT5DSUF0dGFjaG1lbnREZWxldGUoCiAgICBBdHRhY2htZW50SUQgPT4gJEF0dGFjaG1lbnRJRCwKKTsKCiRTZWxmLT5UcnVlKAogICAgJFN1Y2Nlc3MsCiAgICAnQ0lBdHRhY2htZW50RGVsZXRlIHN1Y2Nlc3NmdWxsLicsCik7CgpAQ0lBdHRhY2htZW50TGlzdCA9ICRDSUF0dGFjaG1lbnRPYmplY3QtPkNJQXR0YWNobWVudExpc3QoCiAgICBDb25maWdJdGVtRmllbGQgPT4gJ1Rlc3RGaWVsZCcsCiAgICBDb25maWdJdGVtSUQgICAgPT4gJENvbmZpZ0l0ZW0tPntDb25maWdJdGVtSUR9LAogICAgVmVyc2lvbklEICAgICAgID0+IDEsCik7CgokU2VsZi0+SXNEZWVwbHkoCiAgICBcQENJQXR0YWNobWVudExpc3QsCiAgICBbXSwKICAgICdDSUF0dGFjaG1lbnRHZXQgd2FzIHN1Y2Nlc3NmdWxsLiAoYWZ0ZXIgQ0lBdHRhY2htZW50RGVsZXRlKScsCik7CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgcXcoJFNlbGYpOwoKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCm15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAndXNlcnMnIF0sCik7Cm15ICRVc2VySUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpLT5Vc2VyTG9va3VwKCBVc2VyTG9naW4gPT4gJFRlc3RVc2VyTG9naW4gKTsKCm15ICRSYW5kb21JRCA9ICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CgpteSAkQ2xhc3NJRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtQWRkKAogICAgQ2xhc3MgICA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgICA9PiAkUmFuZG9tSUQsCiAgICBWYWxpZElEID0+IDEsCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CiRTZWxmLT5UcnVlKAogICAgJENsYXNzSUQsCiAgICAiQ2xhc3MgYWRkZWQgdG8gR2VuZXJhbENhdGFsb2ciLAopOwoKbXkgQFRlc3RzID0gKAogICAgewogICAgICAgIE5hbWUgICAgPT4gJ0VtcHR5JywKICAgICAgICBDb25maWcgID0+IHt9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdNaXNzaW5nIENsYXNzc0lEJywKICAgICAgICBDb25maWcgPT4gewogICAgICAgICAgICBEZWZpbml0aW9uID0+IDw8ICdFT0YnLAotLS0KLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgogICAgICAgICAgICBVc2VySUQgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdNaXNzaW5nIERlZmluaXRpb24nLAogICAgICAgIENvbmZpZyA9PiB7CiAgICAgICAgICAgIENsYXNzSUQgPT4gJENsYXNzSUQsCiAgICAgICAgICAgIFVzZXJJRCAgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdNaXNzaW5nIFVzZXJJRCcsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgQ2xhc3NJRCAgICA9PiAkQ2xhc3NJRCwKICAgICAgICAgICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAxCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdXcm9uZyBEZWZpbml0aW9uIChMZWdhY3kgUGVybCknLAogICAgICAgIENvbmZpZyA9PiB7CiAgICAgICAgICAgIENsYXNzSUQgICAgPT4gJENsYXNzSUQsCiAgICAgICAgICAgIERlZmluaXRpb24gPT4gPDwgJ0VPRicsClsKICAgIHsKICAgICAgICBLZXkgICAgICAgID0+ICdWZW5kb3InLAogICAgICAgIE5hbWUgICAgICAgPT4gJ1ZlbmRvcicsCiAgICAgICAgU2VhcmNoYWJsZSA9PiAxLAogICAgICAgIElucHV0ICAgICAgPT4gewogICAgICAgICAgICBUeXBlICAgICAgPT4gJ1RleHQnLAogICAgICAgICAgICBTaXplICAgICAgPT4gNTAsCiAgICAgICAgICAgIE1heExlbmd0aCA9PiA1MAogICAgICAgIH0sCiAgICB9LApdOwpFT0YKICAgICAgICAgICAgVXNlcklEID0+ICRVc2VySUQsCiAgICAgICAgfSwKICAgICAgICBTdWNjZXNzID0+IDAsCiAgICB9LAogICAgewogICAgICAgIE5hbWUgICA9PiAnV3JvbmcgRGVmaW5pdGlvbiAoSW52YWxpZCBZQU1MKScsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgQ2xhc3NJRCAgICA9PiAkQ2xhc3NJRCwKICAgICAgICAgICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tClRlc3QKIEludmFsaWQKLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgogICAgICAgICAgICBVc2VySUQgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdDb3JyZWN0IEFTQ0lJJywKICAgICAgICBDb25maWcgPT4gewogICAgICAgICAgICBDbGFzc0lEICAgID0+ICRDbGFzc0lELAogICAgICAgICAgICBEZWZpbml0aW9uID0+IDw8ICdFT0YnLAotLS0KLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgogICAgICAgICAgICBVc2VySUQgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMSwKICAgIH0sCgopOwoKbXkgJENvbmZpZ0l0ZW1PYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nKTsKClRFU1Q6CmZvciBteSAkVGVzdCAoQFRlc3RzKSB7CgogICAgbXkgJERlZmluaXRpb25JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5EZWZpbml0aW9uQWRkKCAleyAkVGVzdC0+e0NvbmZpZ30gfSApOwoKICAgIGlmICggISRUZXN0LT57U3VjY2Vzc30gKSB7CiAgICAgICAgJFNlbGYtPkZhbHNlKAogICAgICAgICAgICAkRGVmaW5pdGlvbklELAogICAgICAgICAgICAiJFRlc3QtPntOYW1lfSBEZWZpbml0aW9uQWRkKCkgLSBXaXRoIGZhbHNlIiwKICAgICAgICApOwogICAgICAgIG5leHQgVEVTVDsKICAgIH0KCiAgICAkU2VsZi0+SXNOb3QoCiAgICAgICAgJERlZmluaXRpb25JRCAvLyAwLAogICAgICAgIDAsCiAgICAgICAgIiRUZXN0LT57TmFtZX0gRGVmaW5pdGlvbkFkZCgpIC0gRGVmaW5pdGlvbklEIgogICAgKTsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgcXcoJFNlbGYpOwoKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCm15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAndXNlcnMnIF0sCik7Cm15ICRVc2VySUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpLT5Vc2VyTG9va3VwKCBVc2VyTG9naW4gPT4gJFRlc3RVc2VyTG9naW4gKTsKCm15ICRSYW5kb21JRCA9ICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CgpteSAkQ2xhc3NJRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtQWRkKAogICAgQ2xhc3MgICA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgICA9PiAkUmFuZG9tSUQsCiAgICBWYWxpZElEID0+IDEsCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CiRTZWxmLT5UcnVlKAogICAgJENsYXNzSUQsCiAgICAiQ2xhc3MgYWRkZWQgdG8gR2VuZXJhbENhdGFsb2ciLAopOwoKbXkgQFRlc3RzID0gKAogICAgewogICAgICAgIE5hbWUgICAgPT4gJ0VtcHR5JywKICAgICAgICBDb25maWcgID0+IHt9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdNaXNzaW5nIENsYXNzc0lEJywKICAgICAgICBDb25maWcgPT4gewogICAgICAgICAgICBEZWZpbml0aW9uID0+IDw8ICdFT0YnLAotLS0KLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgogICAgICAgICAgICBVc2VySUQgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdNaXNzaW5nIERlZmluaXRpb24nLAogICAgICAgIENvbmZpZyA9PiB7CiAgICAgICAgICAgIENsYXNzSUQgPT4gJENsYXNzSUQsCiAgICAgICAgICAgIFVzZXJJRCAgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdNaXNzaW5nIFVzZXJJRCcsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgQ2xhc3NJRCAgICA9PiAkQ2xhc3NJRCwKICAgICAgICAgICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAxCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdXcm9uZyBEZWZpbml0aW9uIChMZWdhY3kgUGVybCknLAogICAgICAgIENvbmZpZyA9PiB7CiAgICAgICAgICAgIENsYXNzSUQgICAgPT4gJENsYXNzSUQsCiAgICAgICAgICAgIERlZmluaXRpb24gPT4gPDwgJ0VPRicsClsKICAgIHsKICAgICAgICBLZXkgICAgICAgID0+ICdWZW5kb3InLAogICAgICAgIE5hbWUgICAgICAgPT4gJ1ZlbmRvcicsCiAgICAgICAgU2VhcmNoYWJsZSA9PiAxLAogICAgICAgIElucHV0ICAgICAgPT4gewogICAgICAgICAgICBUeXBlICAgICAgPT4gJ1RleHQnLAogICAgICAgICAgICBTaXplICAgICAgPT4gNTAsCiAgICAgICAgICAgIE1heExlbmd0aCA9PiA1MAogICAgICAgIH0sCiAgICB9LApdOwpFT0YKICAgICAgICAgICAgVXNlcklEID0+ICRVc2VySUQsCiAgICAgICAgfSwKICAgICAgICBTdWNjZXNzID0+IDAsCiAgICB9LAogICAgewogICAgICAgIE5hbWUgICA9PiAnV3JvbmcgRGVmaW5pdGlvbiAoSW52YWxpZCBZQU1MKScsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgQ2xhc3NJRCAgICA9PiAkQ2xhc3NJRCwKICAgICAgICAgICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tClRlc3QKIEludmFsaWQKLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgogICAgICAgICAgICBVc2VySUQgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdDb3JyZWN0IEFTQ0lJJywKICAgICAgICBDb25maWcgPT4gewogICAgICAgICAgICBDbGFzc0lEICAgID0+ICRDbGFzc0lELAogICAgICAgICAgICBEZWZpbml0aW9uID0+IDw8ICdFT0YnLAotLS0KLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgogICAgICAgICAgICBVc2VySUQgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMSwKICAgIH0sCgopOwoKbXkgJENvbmZpZ0l0ZW1PYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nKTsKClRFU1Q6CmZvciBteSAkVGVzdCAoQFRlc3RzKSB7CgogICAgbXkgJERlZmluaXRpb25JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5EZWZpbml0aW9uQWRkKCAleyAkVGVzdC0+e0NvbmZpZ30gfSApOwoKICAgIGlmICggISRUZXN0LT57U3VjY2Vzc30gKSB7CiAgICAgICAgJFNlbGYtPkZhbHNlKAogICAgICAgICAgICAkRGVmaW5pdGlvbklELAogICAgICAgICAgICAiJFRlc3QtPntOYW1lfSBEZWZpbml0aW9uQWRkKCkgLSBXaXRoIGZhbHNlIiwKICAgICAgICApOwogICAgICAgIG5leHQgVEVTVDsKICAgIH0KCiAgICAkU2VsZi0+SXNOb3QoCiAgICAgICAgJERlZmluaXRpb25JRCAvLyAwLAogICAgICAgIDAsCiAgICAgICAgIiRUZXN0LT57TmFtZX0gRGVmaW5pdGlvbkFkZCgpIC0gRGVmaW5pdGlvbklEIgogICAgKTsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgcXcoJFNlbGYpOwoKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCm15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAndXNlcnMnIF0sCik7Cm15ICRVc2VySUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpLT5Vc2VyTG9va3VwKCBVc2VyTG9naW4gPT4gJFRlc3RVc2VyTG9naW4gKTsKCm15ICRSYW5kb21JRCA9ICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CgpteSAkQ2xhc3NJRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtQWRkKAogICAgQ2xhc3MgICA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgICA9PiAkUmFuZG9tSUQsCiAgICBWYWxpZElEID0+IDEsCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CiRTZWxmLT5UcnVlKAogICAgJENsYXNzSUQsCiAgICAiQ2xhc3MgYWRkZWQgdG8gR2VuZXJhbENhdGFsb2ciLAopOwoKbXkgQFRlc3RzID0gKAogICAgewogICAgICAgIE5hbWUgICAgPT4gJ0VtcHR5JywKICAgICAgICBDb25maWcgID0+IHt9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdNaXNzaW5nIENsYXNzc0lEJywKICAgICAgICBDb25maWcgPT4gewogICAgICAgICAgICBEZWZpbml0aW9uID0+IDw8ICdFT0YnLAotLS0KLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgogICAgICAgICAgICBVc2VySUQgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdNaXNzaW5nIERlZmluaXRpb24nLAogICAgICAgIENvbmZpZyA9PiB7CiAgICAgICAgICAgIENsYXNzSUQgPT4gJENsYXNzSUQsCiAgICAgICAgICAgIFVzZXJJRCAgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdNaXNzaW5nIFVzZXJJRCcsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgQ2xhc3NJRCAgICA9PiAkQ2xhc3NJRCwKICAgICAgICAgICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAxCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdXcm9uZyBEZWZpbml0aW9uIChMZWdhY3kgUGVybCknLAogICAgICAgIENvbmZpZyA9PiB7CiAgICAgICAgICAgIENsYXNzSUQgICAgPT4gJENsYXNzSUQsCiAgICAgICAgICAgIERlZmluaXRpb24gPT4gPDwgJ0VPRicsClsKICAgIHsKICAgICAgICBLZXkgICAgICAgID0+ICdWZW5kb3InLAogICAgICAgIE5hbWUgICAgICAgPT4gJ1ZlbmRvcicsCiAgICAgICAgU2VhcmNoYWJsZSA9PiAxLAogICAgICAgIElucHV0ICAgICAgPT4gewogICAgICAgICAgICBUeXBlICAgICAgPT4gJ1RleHQnLAogICAgICAgICAgICBTaXplICAgICAgPT4gNTAsCiAgICAgICAgICAgIE1heExlbmd0aCA9PiA1MAogICAgICAgIH0sCiAgICB9LApdOwpFT0YKICAgICAgICAgICAgVXNlcklEID0+ICRVc2VySUQsCiAgICAgICAgfSwKICAgICAgICBTdWNjZXNzID0+IDAsCiAgICB9LAogICAgewogICAgICAgIE5hbWUgICA9PiAnV3JvbmcgRGVmaW5pdGlvbiAoSW52YWxpZCBZQU1MKScsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgQ2xhc3NJRCAgICA9PiAkQ2xhc3NJRCwKICAgICAgICAgICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tClRlc3QKIEludmFsaWQKLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgogICAgICAgICAgICBVc2VySUQgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdDb3JyZWN0IEFTQ0lJJywKICAgICAgICBDb25maWcgPT4gewogICAgICAgICAgICBDbGFzc0lEICAgID0+ICRDbGFzc0lELAogICAgICAgICAgICBEZWZpbml0aW9uID0+IDw8ICdFT0YnLAotLS0KLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgogICAgICAgICAgICBVc2VySUQgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMSwKICAgIH0sCgopOwoKbXkgJENvbmZpZ0l0ZW1PYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nKTsKClRFU1Q6CmZvciBteSAkVGVzdCAoQFRlc3RzKSB7CgogICAgbXkgJERlZmluaXRpb25JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5EZWZpbml0aW9uQWRkKCAleyAkVGVzdC0+e0NvbmZpZ30gfSApOwoKICAgIGlmICggISRUZXN0LT57U3VjY2Vzc30gKSB7CiAgICAgICAgJFNlbGYtPkZhbHNlKAogICAgICAgICAgICAkRGVmaW5pdGlvbklELAogICAgICAgICAgICAiJFRlc3QtPntOYW1lfSBEZWZpbml0aW9uQWRkKCkgLSBXaXRoIGZhbHNlIiwKICAgICAgICApOwogICAgICAgIG5leHQgVEVTVDsKICAgIH0KCiAgICAkU2VsZi0+SXNOb3QoCiAgICAgICAgJERlZmluaXRpb25JRCAvLyAwLAogICAgICAgIDAsCiAgICAgICAgIiRUZXN0LT57TmFtZX0gRGVmaW5pdGlvbkFkZCgpIC0gRGVmaW5pdGlvbklEIgogICAgKTsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgcXcoJFNlbGYpOwoKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCm15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAndXNlcnMnIF0sCik7Cm15ICRVc2VySUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpLT5Vc2VyTG9va3VwKCBVc2VyTG9naW4gPT4gJFRlc3RVc2VyTG9naW4gKTsKCm15ICRSYW5kb21JRCA9ICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CgpteSAkQ2xhc3NJRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtQWRkKAogICAgQ2xhc3MgICA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgICA9PiAkUmFuZG9tSUQsCiAgICBWYWxpZElEID0+IDEsCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CiRTZWxmLT5UcnVlKAogICAgJENsYXNzSUQsCiAgICAiQ2xhc3MgYWRkZWQgdG8gR2VuZXJhbENhdGFsb2ciLAopOwoKbXkgQFRlc3RzID0gKAogICAgewogICAgICAgIE5hbWUgICAgPT4gJ0VtcHR5JywKICAgICAgICBDb25maWcgID0+IHt9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdNaXNzaW5nIENsYXNzc0lEJywKICAgICAgICBDb25maWcgPT4gewogICAgICAgICAgICBEZWZpbml0aW9uID0+IDw8ICdFT0YnLAotLS0KLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgogICAgICAgICAgICBVc2VySUQgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdNaXNzaW5nIERlZmluaXRpb24nLAogICAgICAgIENvbmZpZyA9PiB7CiAgICAgICAgICAgIENsYXNzSUQgPT4gJENsYXNzSUQsCiAgICAgICAgICAgIFVzZXJJRCAgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdNaXNzaW5nIFVzZXJJRCcsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgQ2xhc3NJRCAgICA9PiAkQ2xhc3NJRCwKICAgICAgICAgICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAxCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdXcm9uZyBEZWZpbml0aW9uIChMZWdhY3kgUGVybCknLAogICAgICAgIENvbmZpZyA9PiB7CiAgICAgICAgICAgIENsYXNzSUQgICAgPT4gJENsYXNzSUQsCiAgICAgICAgICAgIERlZmluaXRpb24gPT4gPDwgJ0VPRicsClsKICAgIHsKICAgICAgICBLZXkgICAgICAgID0+ICdWZW5kb3InLAogICAgICAgIE5hbWUgICAgICAgPT4gJ1ZlbmRvcicsCiAgICAgICAgU2VhcmNoYWJsZSA9PiAxLAogICAgICAgIElucHV0ICAgICAgPT4gewogICAgICAgICAgICBUeXBlICAgICAgPT4gJ1RleHQnLAogICAgICAgICAgICBTaXplICAgICAgPT4gNTAsCiAgICAgICAgICAgIE1heExlbmd0aCA9PiA1MAogICAgICAgIH0sCiAgICB9LApdOwpFT0YKICAgICAgICAgICAgVXNlcklEID0+ICRVc2VySUQsCiAgICAgICAgfSwKICAgICAgICBTdWNjZXNzID0+IDAsCiAgICB9LAogICAgewogICAgICAgIE5hbWUgICA9PiAnV3JvbmcgRGVmaW5pdGlvbiAoSW52YWxpZCBZQU1MKScsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgQ2xhc3NJRCAgICA9PiAkQ2xhc3NJRCwKICAgICAgICAgICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tClRlc3QKIEludmFsaWQKLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgogICAgICAgICAgICBVc2VySUQgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMCwKICAgIH0sCiAgICB7CiAgICAgICAgTmFtZSAgID0+ICdDb3JyZWN0IEFTQ0lJJywKICAgICAgICBDb25maWcgPT4gewogICAgICAgICAgICBDbGFzc0lEICAgID0+ICRDbGFzc0lELAogICAgICAgICAgICBEZWZpbml0aW9uID0+IDw8ICdFT0YnLAotLS0KLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgogICAgICAgICAgICBVc2VySUQgPT4gJFVzZXJJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgPT4gMSwKICAgIH0sCgopOwoKbXkgJENvbmZpZ0l0ZW1PYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nKTsKClRFU1Q6CmZvciBteSAkVGVzdCAoQFRlc3RzKSB7CgogICAgbXkgJERlZmluaXRpb25JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5EZWZpbml0aW9uQWRkKCAleyAkVGVzdC0+e0NvbmZpZ30gfSApOwoKICAgIGlmICggISRUZXN0LT57U3VjY2Vzc30gKSB7CiAgICAgICAgJFNlbGYtPkZhbHNlKAogICAgICAgICAgICAkRGVmaW5pdGlvbklELAogICAgICAgICAgICAiJFRlc3QtPntOYW1lfSBEZWZpbml0aW9uQWRkKCkgLSBXaXRoIGZhbHNlIiwKICAgICAgICApOwogICAgICAgIG5leHQgVEVTVDsKICAgIH0KCiAgICAkU2VsZi0+SXNOb3QoCiAgICAgICAgJERlZmluaXRpb25JRCAvLyAwLAogICAgICAgIDAsCiAgICAgICAgIiRUZXN0LT57TmFtZX0gRGVmaW5pdGlvbkFkZCgpIC0gRGVmaW5pdGlvbklEIgogICAgKTsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgcXcoJFNlbGYpOwoKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCm15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAndXNlcnMnIF0sCik7Cm15ICRVc2VySUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpLT5Vc2VyTG9va3VwKCBVc2VyTG9naW4gPT4gJFRlc3RVc2VyTG9naW4gKTsKCm15ICRSYW5kb21JRCA9ICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CgpteSAkQ2xhc3NJRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtQWRkKAogICAgQ2xhc3MgICA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgICA9PiAkUmFuZG9tSUQsCiAgICBWYWxpZElEID0+IDEsCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CiRTZWxmLT5UcnVlKAogICAgJENsYXNzSUQsCiAgICAiQ2xhc3MgYWRkZWQgdG8gR2VuZXJhbENhdGFsb2ciLAopOwoKbXkgJENvbmZpZ0l0ZW1PYmplY3QgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbScpOwpteSAlRGVmaW5pdGlvblRlbXBsYXRlID0gKAogICAgQ2xhc3NJRCAgICA9PiAkQ2xhc3NJRCwKICAgIENsYXNzICAgICAgPT4gJFJhbmRvbUlELAogICAgQ3JlYXRlQnkgICA9PiAkVXNlcklELAogICAgVmVyc2lvbiAgICA9PiAxLAogICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAxCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKICAgIERlZmluaXRpb25SZWYgPT4gWwogICAgICAgIHsKICAgICAgICAgICAgS2V5ICAgICAgICA9PiAnVmVuZG9yJywKICAgICAgICAgICAgTmFtZSAgICAgICA9PiAnVmVuZG9yJywKICAgICAgICAgICAgU2VhcmNoYWJsZSA9PiAxLAogICAgICAgICAgICBJbnB1dCAgICAgID0+IHsKICAgICAgICAgICAgICAgIFR5cGUgICAgICA9PiAnVGV4dCcsCiAgICAgICAgICAgICAgICBTaXplICAgICAgPT4gNTAsCiAgICAgICAgICAgICAgICBNYXhMZW5ndGggPT4gNTAsCiAgICAgICAgICAgIH0sCiAgICAgICAgICAgIENvdW50TWluICAgICA9PiAxLAogICAgICAgICAgICBDb3VudE1heCAgICAgPT4gMSwKICAgICAgICAgICAgQ291bnREZWZhdWx0ID0+IDEsCiAgICAgICAgfSwKICAgIF0sCik7Cm15ICREZWZpbml0aW9uSUQgPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkFkZCgKICAgICVEZWZpbml0aW9uVGVtcGxhdGUsCiAgICBVc2VySUQgPT4gJFVzZXJJRCwKKTsKJFNlbGYtPlRydWUoCiAgICAkRGVmaW5pdGlvbklELAogICAgIkRlZmluaXRpb25BZGQoKSIsCik7CgpteSBAVGVzdHMgPSAoCiAgICB7CiAgICAgICAgTmFtZSAgICA9PiAnRW1wdHknLAogICAgICAgIENvbmZpZyAgPT4ge30sCiAgICAgICAgU3VjY2VzcyA9PiAwLAogICAgfSwKICAgIHsKICAgICAgICBOYW1lICAgPT4gJ0J5IERlZmluaXRpb25JRCcsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgRGVmaW5pdGlvbklEID0+ICREZWZpbml0aW9uSUQsCiAgICAgICAgfSwKICAgICAgICBTdWNjZXNzICAgICAgICAgPT4gMSwKICAgICAgICBFeHBlY3RlZFJlc3VsdHMgPT4gewogICAgICAgICAgICAlRGVmaW5pdGlvblRlbXBsYXRlLAogICAgICAgICAgICBEZWZpbml0aW9uSUQgPT4gJERlZmluaXRpb25JRCwKICAgICAgICB9LAogICAgfSwKICAgIHsKICAgICAgICBOYW1lICAgPT4gJ0J5IENsYXNzSUQnLAogICAgICAgIENvbmZpZyA9PiB7CiAgICAgICAgICAgIENsYXNzSUQgPT4gJENsYXNzSUQsCiAgICAgICAgfSwKICAgICAgICBTdWNjZXNzICAgICAgICAgPT4gMSwKICAgICAgICBFeHBlY3RlZFJlc3VsdHMgPT4gewogICAgICAgICAgICAlRGVmaW5pdGlvblRlbXBsYXRlLAogICAgICAgICAgICBEZWZpbml0aW9uSUQgPT4gJERlZmluaXRpb25JRCwKICAgICAgICB9LAogICAgfSwKKTsKClRFU1Q6CmZvciBteSAkVGVzdCAoQFRlc3RzKSB7CgogICAgbXkgJERlZmluaXRpb24gPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkdldCggJXsgJFRlc3QtPntDb25maWd9IH0gKTsKCiAgICBpZiAoICEkVGVzdC0+e1N1Y2Nlc3N9ICkgewogICAgICAgICRTZWxmLT5GYWxzZSgKICAgICAgICAgICAgJERlZmluaXRpb24sCiAgICAgICAgICAgICIkVGVzdC0+e05hbWV9IERlZmluaXRpb25HZXQoKSAtIFdpdGggZmFsc2UiLAogICAgICAgICk7CiAgICAgICAgbmV4dCBURVNUOwogICAgfQoKICAgIGRlbGV0ZSAkRGVmaW5pdGlvbi0+e0NyZWF0ZVRpbWV9OwoKICAgICRTZWxmLT5Jc0RlZXBseSgKICAgICAgICAkRGVmaW5pdGlvbiwKICAgICAgICAkVGVzdC0+e0V4cGVjdGVkUmVzdWx0c30sCiAgICAgICAgIiRUZXN0LT57TmFtZX0gRGVmaW5pdGlvbkdldCgpIC0gRGVmaW5pdGlvbiIKICAgICk7Cn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgcXcoJFNlbGYpOwoKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCm15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAndXNlcnMnIF0sCik7Cm15ICRVc2VySUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpLT5Vc2VyTG9va3VwKCBVc2VyTG9naW4gPT4gJFRlc3RVc2VyTG9naW4gKTsKCm15ICRSYW5kb21JRCA9ICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CgpteSAkQ2xhc3NJRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtQWRkKAogICAgQ2xhc3MgICA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgICA9PiAkUmFuZG9tSUQsCiAgICBWYWxpZElEID0+IDEsCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CiRTZWxmLT5UcnVlKAogICAgJENsYXNzSUQsCiAgICAiQ2xhc3MgYWRkZWQgdG8gR2VuZXJhbENhdGFsb2ciLAopOwoKbXkgJENvbmZpZ0l0ZW1PYmplY3QgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbScpOwpteSAlRGVmaW5pdGlvblRlbXBsYXRlID0gKAogICAgQ2xhc3NJRCAgICA9PiAkQ2xhc3NJRCwKICAgIENsYXNzICAgICAgPT4gJFJhbmRvbUlELAogICAgQ3JlYXRlQnkgICA9PiAkVXNlcklELAogICAgVmVyc2lvbiAgICA9PiAxLAogICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAxCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKICAgIERlZmluaXRpb25SZWYgPT4gWwogICAgICAgIHsKICAgICAgICAgICAgS2V5ICAgICAgICA9PiAnVmVuZG9yJywKICAgICAgICAgICAgTmFtZSAgICAgICA9PiAnVmVuZG9yJywKICAgICAgICAgICAgU2VhcmNoYWJsZSA9PiAxLAogICAgICAgICAgICBJbnB1dCAgICAgID0+IHsKICAgICAgICAgICAgICAgIFR5cGUgICAgICA9PiAnVGV4dCcsCiAgICAgICAgICAgICAgICBTaXplICAgICAgPT4gNTAsCiAgICAgICAgICAgICAgICBNYXhMZW5ndGggPT4gNTAsCiAgICAgICAgICAgIH0sCiAgICAgICAgICAgIENvdW50TWluICAgICA9PiAxLAogICAgICAgICAgICBDb3VudE1heCAgICAgPT4gMSwKICAgICAgICAgICAgQ291bnREZWZhdWx0ID0+IDEsCiAgICAgICAgfSwKICAgIF0sCik7Cm15ICREZWZpbml0aW9uSUQgPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkFkZCgKICAgICVEZWZpbml0aW9uVGVtcGxhdGUsCiAgICBVc2VySUQgPT4gJFVzZXJJRCwKKTsKJFNlbGYtPlRydWUoCiAgICAkRGVmaW5pdGlvbklELAogICAgIkRlZmluaXRpb25BZGQoKSIsCik7CgpteSBAVGVzdHMgPSAoCiAgICB7CiAgICAgICAgTmFtZSAgICA9PiAnRW1wdHknLAogICAgICAgIENvbmZpZyAgPT4ge30sCiAgICAgICAgU3VjY2VzcyA9PiAwLAogICAgfSwKICAgIHsKICAgICAgICBOYW1lICAgPT4gJ0J5IERlZmluaXRpb25JRCcsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgRGVmaW5pdGlvbklEID0+ICREZWZpbml0aW9uSUQsCiAgICAgICAgfSwKICAgICAgICBTdWNjZXNzICAgICAgICAgPT4gMSwKICAgICAgICBFeHBlY3RlZFJlc3VsdHMgPT4gewogICAgICAgICAgICAlRGVmaW5pdGlvblRlbXBsYXRlLAogICAgICAgICAgICBEZWZpbml0aW9uSUQgPT4gJERlZmluaXRpb25JRCwKICAgICAgICB9LAogICAgfSwKICAgIHsKICAgICAgICBOYW1lICAgPT4gJ0J5IENsYXNzSUQnLAogICAgICAgIENvbmZpZyA9PiB7CiAgICAgICAgICAgIENsYXNzSUQgPT4gJENsYXNzSUQsCiAgICAgICAgfSwKICAgICAgICBTdWNjZXNzICAgICAgICAgPT4gMSwKICAgICAgICBFeHBlY3RlZFJlc3VsdHMgPT4gewogICAgICAgICAgICAlRGVmaW5pdGlvblRlbXBsYXRlLAogICAgICAgICAgICBEZWZpbml0aW9uSUQgPT4gJERlZmluaXRpb25JRCwKICAgICAgICB9LAogICAgfSwKKTsKClRFU1Q6CmZvciBteSAkVGVzdCAoQFRlc3RzKSB7CgogICAgbXkgJERlZmluaXRpb24gPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkdldCggJXsgJFRlc3QtPntDb25maWd9IH0gKTsKCiAgICBpZiAoICEkVGVzdC0+e1N1Y2Nlc3N9ICkgewogICAgICAgICRTZWxmLT5GYWxzZSgKICAgICAgICAgICAgJERlZmluaXRpb24sCiAgICAgICAgICAgICIkVGVzdC0+e05hbWV9IERlZmluaXRpb25HZXQoKSAtIFdpdGggZmFsc2UiLAogICAgICAgICk7CiAgICAgICAgbmV4dCBURVNUOwogICAgfQoKICAgIGRlbGV0ZSAkRGVmaW5pdGlvbi0+e0NyZWF0ZVRpbWV9OwoKICAgICRTZWxmLT5Jc0RlZXBseSgKICAgICAgICAkRGVmaW5pdGlvbiwKICAgICAgICAkVGVzdC0+e0V4cGVjdGVkUmVzdWx0c30sCiAgICAgICAgIiRUZXN0LT57TmFtZX0gRGVmaW5pdGlvbkdldCgpIC0gRGVmaW5pdGlvbiIKICAgICk7Cn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgcXcoJFNlbGYpOwoKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCm15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAndXNlcnMnIF0sCik7Cm15ICRVc2VySUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpLT5Vc2VyTG9va3VwKCBVc2VyTG9naW4gPT4gJFRlc3RVc2VyTG9naW4gKTsKCm15ICRSYW5kb21JRCA9ICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CgpteSAkQ2xhc3NJRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtQWRkKAogICAgQ2xhc3MgICA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgICA9PiAkUmFuZG9tSUQsCiAgICBWYWxpZElEID0+IDEsCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CiRTZWxmLT5UcnVlKAogICAgJENsYXNzSUQsCiAgICAiQ2xhc3MgYWRkZWQgdG8gR2VuZXJhbENhdGFsb2ciLAopOwoKbXkgJENvbmZpZ0l0ZW1PYmplY3QgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbScpOwpteSAlRGVmaW5pdGlvblRlbXBsYXRlID0gKAogICAgQ2xhc3NJRCAgICA9PiAkQ2xhc3NJRCwKICAgIENsYXNzICAgICAgPT4gJFJhbmRvbUlELAogICAgQ3JlYXRlQnkgICA9PiAkVXNlcklELAogICAgVmVyc2lvbiAgICA9PiAxLAogICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAxCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKICAgIERlZmluaXRpb25SZWYgPT4gWwogICAgICAgIHsKICAgICAgICAgICAgS2V5ICAgICAgICA9PiAnVmVuZG9yJywKICAgICAgICAgICAgTmFtZSAgICAgICA9PiAnVmVuZG9yJywKICAgICAgICAgICAgU2VhcmNoYWJsZSA9PiAxLAogICAgICAgICAgICBJbnB1dCAgICAgID0+IHsKICAgICAgICAgICAgICAgIFR5cGUgICAgICA9PiAnVGV4dCcsCiAgICAgICAgICAgICAgICBTaXplICAgICAgPT4gNTAsCiAgICAgICAgICAgICAgICBNYXhMZW5ndGggPT4gNTAsCiAgICAgICAgICAgIH0sCiAgICAgICAgICAgIENvdW50TWluICAgICA9PiAxLAogICAgICAgICAgICBDb3VudE1heCAgICAgPT4gMSwKICAgICAgICAgICAgQ291bnREZWZhdWx0ID0+IDEsCiAgICAgICAgfSwKICAgIF0sCik7Cm15ICREZWZpbml0aW9uSUQgPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkFkZCgKICAgICVEZWZpbml0aW9uVGVtcGxhdGUsCiAgICBVc2VySUQgPT4gJFVzZXJJRCwKKTsKJFNlbGYtPlRydWUoCiAgICAkRGVmaW5pdGlvbklELAogICAgIkRlZmluaXRpb25BZGQoKSIsCik7CgpteSBAVGVzdHMgPSAoCiAgICB7CiAgICAgICAgTmFtZSAgICA9PiAnRW1wdHknLAogICAgICAgIENvbmZpZyAgPT4ge30sCiAgICAgICAgU3VjY2VzcyA9PiAwLAogICAgfSwKICAgIHsKICAgICAgICBOYW1lICAgPT4gJ0J5IERlZmluaXRpb25JRCcsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgRGVmaW5pdGlvbklEID0+ICREZWZpbml0aW9uSUQsCiAgICAgICAgfSwKICAgICAgICBTdWNjZXNzICAgICAgICAgPT4gMSwKICAgICAgICBFeHBlY3RlZFJlc3VsdHMgPT4gewogICAgICAgICAgICAlRGVmaW5pdGlvblRlbXBsYXRlLAogICAgICAgICAgICBEZWZpbml0aW9uSUQgPT4gJERlZmluaXRpb25JRCwKICAgICAgICB9LAogICAgfSwKICAgIHsKICAgICAgICBOYW1lICAgPT4gJ0J5IENsYXNzSUQnLAogICAgICAgIENvbmZpZyA9PiB7CiAgICAgICAgICAgIENsYXNzSUQgPT4gJENsYXNzSUQsCiAgICAgICAgfSwKICAgICAgICBTdWNjZXNzICAgICAgICAgPT4gMSwKICAgICAgICBFeHBlY3RlZFJlc3VsdHMgPT4gewogICAgICAgICAgICAlRGVmaW5pdGlvblRlbXBsYXRlLAogICAgICAgICAgICBEZWZpbml0aW9uSUQgPT4gJERlZmluaXRpb25JRCwKICAgICAgICB9LAogICAgfSwKKTsKClRFU1Q6CmZvciBteSAkVGVzdCAoQFRlc3RzKSB7CgogICAgbXkgJERlZmluaXRpb24gPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkdldCggJXsgJFRlc3QtPntDb25maWd9IH0gKTsKCiAgICBpZiAoICEkVGVzdC0+e1N1Y2Nlc3N9ICkgewogICAgICAgICRTZWxmLT5GYWxzZSgKICAgICAgICAgICAgJERlZmluaXRpb24sCiAgICAgICAgICAgICIkVGVzdC0+e05hbWV9IERlZmluaXRpb25HZXQoKSAtIFdpdGggZmFsc2UiLAogICAgICAgICk7CiAgICAgICAgbmV4dCBURVNUOwogICAgfQoKICAgIGRlbGV0ZSAkRGVmaW5pdGlvbi0+e0NyZWF0ZVRpbWV9OwoKICAgICRTZWxmLT5Jc0RlZXBseSgKICAgICAgICAkRGVmaW5pdGlvbiwKICAgICAgICAkVGVzdC0+e0V4cGVjdGVkUmVzdWx0c30sCiAgICAgICAgIiRUZXN0LT57TmFtZX0gRGVmaW5pdGlvbkdldCgpIC0gRGVmaW5pdGlvbiIKICAgICk7Cn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgcXcoJFNlbGYpOwoKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCm15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAndXNlcnMnIF0sCik7Cm15ICRVc2VySUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpLT5Vc2VyTG9va3VwKCBVc2VyTG9naW4gPT4gJFRlc3RVc2VyTG9naW4gKTsKCm15ICRSYW5kb21JRCA9ICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CgpteSAkQ2xhc3NJRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtQWRkKAogICAgQ2xhc3MgICA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgICA9PiAkUmFuZG9tSUQsCiAgICBWYWxpZElEID0+IDEsCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CiRTZWxmLT5UcnVlKAogICAgJENsYXNzSUQsCiAgICAiQ2xhc3MgYWRkZWQgdG8gR2VuZXJhbENhdGFsb2ciLAopOwoKbXkgJENvbmZpZ0l0ZW1PYmplY3QgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbScpOwpteSAlRGVmaW5pdGlvblRlbXBsYXRlID0gKAogICAgQ2xhc3NJRCAgICA9PiAkQ2xhc3NJRCwKICAgIENsYXNzICAgICAgPT4gJFJhbmRvbUlELAogICAgQ3JlYXRlQnkgICA9PiAkVXNlcklELAogICAgVmVyc2lvbiAgICA9PiAxLAogICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAxCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKICAgIERlZmluaXRpb25SZWYgPT4gWwogICAgICAgIHsKICAgICAgICAgICAgS2V5ICAgICAgICA9PiAnVmVuZG9yJywKICAgICAgICAgICAgTmFtZSAgICAgICA9PiAnVmVuZG9yJywKICAgICAgICAgICAgU2VhcmNoYWJsZSA9PiAxLAogICAgICAgICAgICBJbnB1dCAgICAgID0+IHsKICAgICAgICAgICAgICAgIFR5cGUgICAgICA9PiAnVGV4dCcsCiAgICAgICAgICAgICAgICBTaXplICAgICAgPT4gNTAsCiAgICAgICAgICAgICAgICBNYXhMZW5ndGggPT4gNTAsCiAgICAgICAgICAgIH0sCiAgICAgICAgICAgIENvdW50TWluICAgICA9PiAxLAogICAgICAgICAgICBDb3VudE1heCAgICAgPT4gMSwKICAgICAgICAgICAgQ291bnREZWZhdWx0ID0+IDEsCiAgICAgICAgfSwKICAgIF0sCik7Cm15ICREZWZpbml0aW9uSUQgPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkFkZCgKICAgICVEZWZpbml0aW9uVGVtcGxhdGUsCiAgICBVc2VySUQgPT4gJFVzZXJJRCwKKTsKJFNlbGYtPlRydWUoCiAgICAkRGVmaW5pdGlvbklELAogICAgIkRlZmluaXRpb25BZGQoKSIsCik7CgpteSBAVGVzdHMgPSAoCiAgICB7CiAgICAgICAgTmFtZSAgICA9PiAnRW1wdHknLAogICAgICAgIENvbmZpZyAgPT4ge30sCiAgICAgICAgU3VjY2VzcyA9PiAwLAogICAgfSwKICAgIHsKICAgICAgICBOYW1lICAgPT4gJ0J5IERlZmluaXRpb25JRCcsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgRGVmaW5pdGlvbklEID0+ICREZWZpbml0aW9uSUQsCiAgICAgICAgfSwKICAgICAgICBTdWNjZXNzICAgICAgICAgPT4gMSwKICAgICAgICBFeHBlY3RlZFJlc3VsdHMgPT4gewogICAgICAgICAgICAlRGVmaW5pdGlvblRlbXBsYXRlLAogICAgICAgICAgICBEZWZpbml0aW9uSUQgPT4gJERlZmluaXRpb25JRCwKICAgICAgICB9LAogICAgfSwKICAgIHsKICAgICAgICBOYW1lICAgPT4gJ0J5IENsYXNzSUQnLAogICAgICAgIENvbmZpZyA9PiB7CiAgICAgICAgICAgIENsYXNzSUQgPT4gJENsYXNzSUQsCiAgICAgICAgfSwKICAgICAgICBTdWNjZXNzICAgICAgICAgPT4gMSwKICAgICAgICBFeHBlY3RlZFJlc3VsdHMgPT4gewogICAgICAgICAgICAlRGVmaW5pdGlvblRlbXBsYXRlLAogICAgICAgICAgICBEZWZpbml0aW9uSUQgPT4gJERlZmluaXRpb25JRCwKICAgICAgICB9LAogICAgfSwKKTsKClRFU1Q6CmZvciBteSAkVGVzdCAoQFRlc3RzKSB7CgogICAgbXkgJERlZmluaXRpb24gPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkdldCggJXsgJFRlc3QtPntDb25maWd9IH0gKTsKCiAgICBpZiAoICEkVGVzdC0+e1N1Y2Nlc3N9ICkgewogICAgICAgICRTZWxmLT5GYWxzZSgKICAgICAgICAgICAgJERlZmluaXRpb24sCiAgICAgICAgICAgICIkVGVzdC0+e05hbWV9IERlZmluaXRpb25HZXQoKSAtIFdpdGggZmFsc2UiLAogICAgICAgICk7CiAgICAgICAgbmV4dCBURVNUOwogICAgfQoKICAgIGRlbGV0ZSAkRGVmaW5pdGlvbi0+e0NyZWF0ZVRpbWV9OwoKICAgICRTZWxmLT5Jc0RlZXBseSgKICAgICAgICAkRGVmaW5pdGlvbiwKICAgICAgICAkVGVzdC0+e0V4cGVjdGVkUmVzdWx0c30sCiAgICAgICAgIiRUZXN0LT57TmFtZX0gRGVmaW5pdGlvbkdldCgpIC0gRGVmaW5pdGlvbiIKICAgICk7Cn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgcXcoJFNlbGYpOwoKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCm15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAndXNlcnMnIF0sCik7Cm15ICRVc2VySUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpLT5Vc2VyTG9va3VwKCBVc2VyTG9naW4gPT4gJFRlc3RVc2VyTG9naW4gKTsKCm15ICRSYW5kb21JRCA9ICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CgpteSAkQ2xhc3NJRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtQWRkKAogICAgQ2xhc3MgICA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgICA9PiAkUmFuZG9tSUQsCiAgICBWYWxpZElEID0+IDEsCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CiRTZWxmLT5UcnVlKAogICAgJENsYXNzSUQsCiAgICAiQ2xhc3MgYWRkZWQgdG8gR2VuZXJhbENhdGFsb2ciLAopOwoKbXkgJENvbmZpZ0l0ZW1PYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nKTsKbXkgJERlZmluaXRpb25JRDEgICAgPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkFkZCgKICAgIENsYXNzSUQgICAgPT4gJENsYXNzSUQsCiAgICBVc2VySUQgICAgID0+ICRVc2VySUQsCiAgICBDcmVhdGVCeSAgID0+ICRVc2VySUQsCiAgICBEZWZpbml0aW9uID0+IDw8ICdFT0YnLAotLS0KLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgopOwokU2VsZi0+VHJ1ZSgKICAgICREZWZpbml0aW9uSUQxLAogICAgIkRlZmluaXRpb25BZGQoKSIsCik7Cm15ICREZWZpbml0aW9uSUQyID0gJENvbmZpZ0l0ZW1PYmplY3QtPkRlZmluaXRpb25BZGQoCiAgICBDbGFzc0lEICAgID0+ICRDbGFzc0lELAogICAgVXNlcklEICAgICA9PiAkVXNlcklELAogICAgQ3JlYXRlQnkgICA9PiAkVXNlcklELAogICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAwCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKKTsKCiRTZWxmLT5UcnVlKAogICAgJERlZmluaXRpb25JRDIsCiAgICAiRGVmaW5pdGlvbkFkZCgpIiwKKTsKCm15IEBUZXN0cyA9ICgKICAgIHsKICAgICAgICBOYW1lICAgID0+ICdFbXB0eScsCiAgICAgICAgQ29uZmlnICA9PiB7fSwKICAgICAgICBTdWNjZXNzID0+IDAsCiAgICB9LAogICAgewogICAgICAgIE5hbWUgICA9PiAnQnkgQ2xhc3NJRCcsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgQ2xhc3NJRCA9PiAkQ2xhc3NJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgICAgICAgICA9PiAxLAogICAgICAgIEV4cGVjdGVkUmVzdWx0cyA9PiBbCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIENyZWF0ZUJ5ICAgPT4gJFVzZXJJRCwKICAgICAgICAgICAgICAgIERlZmluaXRpb24gPT4gPDwgJ0VPRicsCi0tLQotIEtleTogVmVuZG9yCiAgTmFtZTogVmVuZG9yCiAgU2VhcmNoYWJsZTogMQogIElucHV0OgogICAgVHlwZTogVGV4dAogICAgU2l6ZTogNTAKICAgIE1heExlbmd0aDogNTAKRU9GCiAgICAgICAgICAgICAgICBWZXJzaW9uICAgICAgPT4gMSwKICAgICAgICAgICAgICAgIERlZmluaXRpb25JRCA9PiAkRGVmaW5pdGlvbklEMSwKICAgICAgICAgICAgfSwKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgQ3JlYXRlQnkgICA9PiAkVXNlcklELAogICAgICAgICAgICAgICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAwCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKICAgICAgICAgICAgICAgIFZlcnNpb24gICAgICA9PiAyLAogICAgICAgICAgICAgICAgRGVmaW5pdGlvbklEID0+ICREZWZpbml0aW9uSUQyLAogICAgICAgICAgICB9LAogICAgICAgIF0sCiAgICB9LAopOwoKVEVTVDoKZm9yIG15ICRUZXN0IChAVGVzdHMpIHsKCiAgICBteSAkRGVmaW5pdGlvbkxpc3QgPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkxpc3QoICV7ICRUZXN0LT57Q29uZmlnfSB9ICk7CgogICAgaWYgKCAhJFRlc3QtPntTdWNjZXNzfSApIHsKICAgICAgICAkU2VsZi0+RmFsc2UoCiAgICAgICAgICAgICREZWZpbml0aW9uTGlzdCwKICAgICAgICAgICAgIiRUZXN0LT57TmFtZX0gRGVmaW5pdGlvbkxpc3QoKSAtIFdpdGggZmFsc2UiLAogICAgICAgICk7CiAgICAgICAgbmV4dCBURVNUOwogICAgfQoKICAgIGZvciBteSAkRGVmaW5pdGlvbiAoIEB7JERlZmluaXRpb25MaXN0fSApIHsKICAgICAgICBkZWxldGUgJERlZmluaXRpb24tPntDcmVhdGVUaW1lfTsKICAgIH0KCiAgICAkU2VsZi0+SXNEZWVwbHkoCiAgICAgICAgJERlZmluaXRpb25MaXN0LAogICAgICAgICRUZXN0LT57RXhwZWN0ZWRSZXN1bHRzfSwKICAgICAgICAiJFRlc3QtPntOYW1lfSBEZWZpbml0aW9uTGlzdCgpIC0gRGVmaW5pdGlvbiIKICAgICk7Cn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgcXcoJFNlbGYpOwoKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCm15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAndXNlcnMnIF0sCik7Cm15ICRVc2VySUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpLT5Vc2VyTG9va3VwKCBVc2VyTG9naW4gPT4gJFRlc3RVc2VyTG9naW4gKTsKCm15ICRSYW5kb21JRCA9ICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CgpteSAkQ2xhc3NJRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtQWRkKAogICAgQ2xhc3MgICA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgICA9PiAkUmFuZG9tSUQsCiAgICBWYWxpZElEID0+IDEsCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CiRTZWxmLT5UcnVlKAogICAgJENsYXNzSUQsCiAgICAiQ2xhc3MgYWRkZWQgdG8gR2VuZXJhbENhdGFsb2ciLAopOwoKbXkgJENvbmZpZ0l0ZW1PYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nKTsKbXkgJERlZmluaXRpb25JRDEgICAgPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkFkZCgKICAgIENsYXNzSUQgICAgPT4gJENsYXNzSUQsCiAgICBVc2VySUQgICAgID0+ICRVc2VySUQsCiAgICBDcmVhdGVCeSAgID0+ICRVc2VySUQsCiAgICBEZWZpbml0aW9uID0+IDw8ICdFT0YnLAotLS0KLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgopOwokU2VsZi0+VHJ1ZSgKICAgICREZWZpbml0aW9uSUQxLAogICAgIkRlZmluaXRpb25BZGQoKSIsCik7Cm15ICREZWZpbml0aW9uSUQyID0gJENvbmZpZ0l0ZW1PYmplY3QtPkRlZmluaXRpb25BZGQoCiAgICBDbGFzc0lEICAgID0+ICRDbGFzc0lELAogICAgVXNlcklEICAgICA9PiAkVXNlcklELAogICAgQ3JlYXRlQnkgICA9PiAkVXNlcklELAogICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAwCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKKTsKCiRTZWxmLT5UcnVlKAogICAgJERlZmluaXRpb25JRDIsCiAgICAiRGVmaW5pdGlvbkFkZCgpIiwKKTsKCm15IEBUZXN0cyA9ICgKICAgIHsKICAgICAgICBOYW1lICAgID0+ICdFbXB0eScsCiAgICAgICAgQ29uZmlnICA9PiB7fSwKICAgICAgICBTdWNjZXNzID0+IDAsCiAgICB9LAogICAgewogICAgICAgIE5hbWUgICA9PiAnQnkgQ2xhc3NJRCcsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgQ2xhc3NJRCA9PiAkQ2xhc3NJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgICAgICAgICA9PiAxLAogICAgICAgIEV4cGVjdGVkUmVzdWx0cyA9PiBbCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIENyZWF0ZUJ5ICAgPT4gJFVzZXJJRCwKICAgICAgICAgICAgICAgIERlZmluaXRpb24gPT4gPDwgJ0VPRicsCi0tLQotIEtleTogVmVuZG9yCiAgTmFtZTogVmVuZG9yCiAgU2VhcmNoYWJsZTogMQogIElucHV0OgogICAgVHlwZTogVGV4dAogICAgU2l6ZTogNTAKICAgIE1heExlbmd0aDogNTAKRU9GCiAgICAgICAgICAgICAgICBWZXJzaW9uICAgICAgPT4gMSwKICAgICAgICAgICAgICAgIERlZmluaXRpb25JRCA9PiAkRGVmaW5pdGlvbklEMSwKICAgICAgICAgICAgfSwKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgQ3JlYXRlQnkgICA9PiAkVXNlcklELAogICAgICAgICAgICAgICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAwCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKICAgICAgICAgICAgICAgIFZlcnNpb24gICAgICA9PiAyLAogICAgICAgICAgICAgICAgRGVmaW5pdGlvbklEID0+ICREZWZpbml0aW9uSUQyLAogICAgICAgICAgICB9LAogICAgICAgIF0sCiAgICB9LAopOwoKVEVTVDoKZm9yIG15ICRUZXN0IChAVGVzdHMpIHsKCiAgICBteSAkRGVmaW5pdGlvbkxpc3QgPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkxpc3QoICV7ICRUZXN0LT57Q29uZmlnfSB9ICk7CgogICAgaWYgKCAhJFRlc3QtPntTdWNjZXNzfSApIHsKICAgICAgICAkU2VsZi0+RmFsc2UoCiAgICAgICAgICAgICREZWZpbml0aW9uTGlzdCwKICAgICAgICAgICAgIiRUZXN0LT57TmFtZX0gRGVmaW5pdGlvbkxpc3QoKSAtIFdpdGggZmFsc2UiLAogICAgICAgICk7CiAgICAgICAgbmV4dCBURVNUOwogICAgfQoKICAgIGZvciBteSAkRGVmaW5pdGlvbiAoIEB7JERlZmluaXRpb25MaXN0fSApIHsKICAgICAgICBkZWxldGUgJERlZmluaXRpb24tPntDcmVhdGVUaW1lfTsKICAgIH0KCiAgICAkU2VsZi0+SXNEZWVwbHkoCiAgICAgICAgJERlZmluaXRpb25MaXN0LAogICAgICAgICRUZXN0LT57RXhwZWN0ZWRSZXN1bHRzfSwKICAgICAgICAiJFRlc3QtPntOYW1lfSBEZWZpbml0aW9uTGlzdCgpIC0gRGVmaW5pdGlvbiIKICAgICk7Cn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgcXcoJFNlbGYpOwoKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCm15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAndXNlcnMnIF0sCik7Cm15ICRVc2VySUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpLT5Vc2VyTG9va3VwKCBVc2VyTG9naW4gPT4gJFRlc3RVc2VyTG9naW4gKTsKCm15ICRSYW5kb21JRCA9ICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CgpteSAkQ2xhc3NJRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtQWRkKAogICAgQ2xhc3MgICA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgICA9PiAkUmFuZG9tSUQsCiAgICBWYWxpZElEID0+IDEsCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CiRTZWxmLT5UcnVlKAogICAgJENsYXNzSUQsCiAgICAiQ2xhc3MgYWRkZWQgdG8gR2VuZXJhbENhdGFsb2ciLAopOwoKbXkgJENvbmZpZ0l0ZW1PYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nKTsKbXkgJERlZmluaXRpb25JRDEgICAgPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkFkZCgKICAgIENsYXNzSUQgICAgPT4gJENsYXNzSUQsCiAgICBVc2VySUQgICAgID0+ICRVc2VySUQsCiAgICBDcmVhdGVCeSAgID0+ICRVc2VySUQsCiAgICBEZWZpbml0aW9uID0+IDw8ICdFT0YnLAotLS0KLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgopOwokU2VsZi0+VHJ1ZSgKICAgICREZWZpbml0aW9uSUQxLAogICAgIkRlZmluaXRpb25BZGQoKSIsCik7Cm15ICREZWZpbml0aW9uSUQyID0gJENvbmZpZ0l0ZW1PYmplY3QtPkRlZmluaXRpb25BZGQoCiAgICBDbGFzc0lEICAgID0+ICRDbGFzc0lELAogICAgVXNlcklEICAgICA9PiAkVXNlcklELAogICAgQ3JlYXRlQnkgICA9PiAkVXNlcklELAogICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAwCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKKTsKCiRTZWxmLT5UcnVlKAogICAgJERlZmluaXRpb25JRDIsCiAgICAiRGVmaW5pdGlvbkFkZCgpIiwKKTsKCm15IEBUZXN0cyA9ICgKICAgIHsKICAgICAgICBOYW1lICAgID0+ICdFbXB0eScsCiAgICAgICAgQ29uZmlnICA9PiB7fSwKICAgICAgICBTdWNjZXNzID0+IDAsCiAgICB9LAogICAgewogICAgICAgIE5hbWUgICA9PiAnQnkgQ2xhc3NJRCcsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgQ2xhc3NJRCA9PiAkQ2xhc3NJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgICAgICAgICA9PiAxLAogICAgICAgIEV4cGVjdGVkUmVzdWx0cyA9PiBbCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIENyZWF0ZUJ5ICAgPT4gJFVzZXJJRCwKICAgICAgICAgICAgICAgIERlZmluaXRpb24gPT4gPDwgJ0VPRicsCi0tLQotIEtleTogVmVuZG9yCiAgTmFtZTogVmVuZG9yCiAgU2VhcmNoYWJsZTogMQogIElucHV0OgogICAgVHlwZTogVGV4dAogICAgU2l6ZTogNTAKICAgIE1heExlbmd0aDogNTAKRU9GCiAgICAgICAgICAgICAgICBWZXJzaW9uICAgICAgPT4gMSwKICAgICAgICAgICAgICAgIERlZmluaXRpb25JRCA9PiAkRGVmaW5pdGlvbklEMSwKICAgICAgICAgICAgfSwKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgQ3JlYXRlQnkgICA9PiAkVXNlcklELAogICAgICAgICAgICAgICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAwCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKICAgICAgICAgICAgICAgIFZlcnNpb24gICAgICA9PiAyLAogICAgICAgICAgICAgICAgRGVmaW5pdGlvbklEID0+ICREZWZpbml0aW9uSUQyLAogICAgICAgICAgICB9LAogICAgICAgIF0sCiAgICB9LAopOwoKVEVTVDoKZm9yIG15ICRUZXN0IChAVGVzdHMpIHsKCiAgICBteSAkRGVmaW5pdGlvbkxpc3QgPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkxpc3QoICV7ICRUZXN0LT57Q29uZmlnfSB9ICk7CgogICAgaWYgKCAhJFRlc3QtPntTdWNjZXNzfSApIHsKICAgICAgICAkU2VsZi0+RmFsc2UoCiAgICAgICAgICAgICREZWZpbml0aW9uTGlzdCwKICAgICAgICAgICAgIiRUZXN0LT57TmFtZX0gRGVmaW5pdGlvbkxpc3QoKSAtIFdpdGggZmFsc2UiLAogICAgICAgICk7CiAgICAgICAgbmV4dCBURVNUOwogICAgfQoKICAgIGZvciBteSAkRGVmaW5pdGlvbiAoIEB7JERlZmluaXRpb25MaXN0fSApIHsKICAgICAgICBkZWxldGUgJERlZmluaXRpb24tPntDcmVhdGVUaW1lfTsKICAgIH0KCiAgICAkU2VsZi0+SXNEZWVwbHkoCiAgICAgICAgJERlZmluaXRpb25MaXN0LAogICAgICAgICRUZXN0LT57RXhwZWN0ZWRSZXN1bHRzfSwKICAgICAgICAiJFRlc3QtPntOYW1lfSBEZWZpbml0aW9uTGlzdCgpIC0gRGVmaW5pdGlvbiIKICAgICk7Cn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKIyMgbm8gY3JpdGljIChNb2R1bGVzOjpSZXF1aXJlRXhwbGljaXRQYWNrYWdlKQp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSB1dGY4OwoKdXNlIHZhcnMgcXcoJFNlbGYpOwoKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCm15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAndXNlcnMnIF0sCik7Cm15ICRVc2VySUQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpLT5Vc2VyTG9va3VwKCBVc2VyTG9naW4gPT4gJFRlc3RVc2VyTG9naW4gKTsKCm15ICRSYW5kb21JRCA9ICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CgpteSAkQ2xhc3NJRCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtQWRkKAogICAgQ2xhc3MgICA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgICA9PiAkUmFuZG9tSUQsCiAgICBWYWxpZElEID0+IDEsCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CiRTZWxmLT5UcnVlKAogICAgJENsYXNzSUQsCiAgICAiQ2xhc3MgYWRkZWQgdG8gR2VuZXJhbENhdGFsb2ciLAopOwoKbXkgJENvbmZpZ0l0ZW1PYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nKTsKbXkgJERlZmluaXRpb25JRDEgICAgPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkFkZCgKICAgIENsYXNzSUQgICAgPT4gJENsYXNzSUQsCiAgICBVc2VySUQgICAgID0+ICRVc2VySUQsCiAgICBDcmVhdGVCeSAgID0+ICRVc2VySUQsCiAgICBEZWZpbml0aW9uID0+IDw8ICdFT0YnLAotLS0KLSBLZXk6IFZlbmRvcgogIE5hbWU6IFZlbmRvcgogIFNlYXJjaGFibGU6IDEKICBJbnB1dDoKICAgIFR5cGU6IFRleHQKICAgIFNpemU6IDUwCiAgICBNYXhMZW5ndGg6IDUwCkVPRgopOwokU2VsZi0+VHJ1ZSgKICAgICREZWZpbml0aW9uSUQxLAogICAgIkRlZmluaXRpb25BZGQoKSIsCik7Cm15ICREZWZpbml0aW9uSUQyID0gJENvbmZpZ0l0ZW1PYmplY3QtPkRlZmluaXRpb25BZGQoCiAgICBDbGFzc0lEICAgID0+ICRDbGFzc0lELAogICAgVXNlcklEICAgICA9PiAkVXNlcklELAogICAgQ3JlYXRlQnkgICA9PiAkVXNlcklELAogICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAwCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKKTsKCiRTZWxmLT5UcnVlKAogICAgJERlZmluaXRpb25JRDIsCiAgICAiRGVmaW5pdGlvbkFkZCgpIiwKKTsKCm15IEBUZXN0cyA9ICgKICAgIHsKICAgICAgICBOYW1lICAgID0+ICdFbXB0eScsCiAgICAgICAgQ29uZmlnICA9PiB7fSwKICAgICAgICBTdWNjZXNzID0+IDAsCiAgICB9LAogICAgewogICAgICAgIE5hbWUgICA9PiAnQnkgQ2xhc3NJRCcsCiAgICAgICAgQ29uZmlnID0+IHsKICAgICAgICAgICAgQ2xhc3NJRCA9PiAkQ2xhc3NJRCwKICAgICAgICB9LAogICAgICAgIFN1Y2Nlc3MgICAgICAgICA9PiAxLAogICAgICAgIEV4cGVjdGVkUmVzdWx0cyA9PiBbCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIENyZWF0ZUJ5ICAgPT4gJFVzZXJJRCwKICAgICAgICAgICAgICAgIERlZmluaXRpb24gPT4gPDwgJ0VPRicsCi0tLQotIEtleTogVmVuZG9yCiAgTmFtZTogVmVuZG9yCiAgU2VhcmNoYWJsZTogMQogIElucHV0OgogICAgVHlwZTogVGV4dAogICAgU2l6ZTogNTAKICAgIE1heExlbmd0aDogNTAKRU9GCiAgICAgICAgICAgICAgICBWZXJzaW9uICAgICAgPT4gMSwKICAgICAgICAgICAgICAgIERlZmluaXRpb25JRCA9PiAkRGVmaW5pdGlvbklEMSwKICAgICAgICAgICAgfSwKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgQ3JlYXRlQnkgICA9PiAkVXNlcklELAogICAgICAgICAgICAgICAgRGVmaW5pdGlvbiA9PiA8PCAnRU9GJywKLS0tCi0gS2V5OiBWZW5kb3IKICBOYW1lOiBWZW5kb3IKICBTZWFyY2hhYmxlOiAwCiAgSW5wdXQ6CiAgICBUeXBlOiBUZXh0CiAgICBTaXplOiA1MAogICAgTWF4TGVuZ3RoOiA1MApFT0YKICAgICAgICAgICAgICAgIFZlcnNpb24gICAgICA9PiAyLAogICAgICAgICAgICAgICAgRGVmaW5pdGlvbklEID0+ICREZWZpbml0aW9uSUQyLAogICAgICAgICAgICB9LAogICAgICAgIF0sCiAgICB9LAopOwoKVEVTVDoKZm9yIG15ICRUZXN0IChAVGVzdHMpIHsKCiAgICBteSAkRGVmaW5pdGlvbkxpc3QgPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkxpc3QoICV7ICRUZXN0LT57Q29uZmlnfSB9ICk7CgogICAgaWYgKCAhJFRlc3QtPntTdWNjZXNzfSApIHsKICAgICAgICAkU2VsZi0+RmFsc2UoCiAgICAgICAgICAgICREZWZpbml0aW9uTGlzdCwKICAgICAgICAgICAgIiRUZXN0LT57TmFtZX0gRGVmaW5pdGlvbkxpc3QoKSAtIFdpdGggZmFsc2UiLAogICAgICAgICk7CiAgICAgICAgbmV4dCBURVNUOwogICAgfQoKICAgIGZvciBteSAkRGVmaW5pdGlvbiAoIEB7JERlZmluaXRpb25MaXN0fSApIHsKICAgICAgICBkZWxldGUgJERlZmluaXRpb24tPntDcmVhdGVUaW1lfTsKICAgIH0KCiAgICAkU2VsZi0+SXNEZWVwbHkoCiAgICAgICAgJERlZmluaXRpb25MaXN0LAogICAgICAgICRUZXN0LT57RXhwZWN0ZWRSZXN1bHRzfSwKICAgICAgICAiJFRlc3QtPntOYW1lfSBEZWZpbml0aW9uTGlzdCgpIC0gRGVmaW5pdGlvbiIKICAgICk7Cn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7Cgp1c2UgdmFycyAocXcoJFNlbGYpKTsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6T2JqZWN0TWFuYWdlcjsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6VmFyaWFibGVDaGVjayBxdyg6YWxsKTsKCiRLZXJuZWw6Ok9NLT5PYmplY3RQYXJhbUFkZCgKICAgICdLZXJuZWw6OlN5c3RlbTo6VW5pdFRlc3Q6OkhlbHBlcicgPT4gewogICAgICAgIFJlc3RvcmVEYXRhYmFzZSA9PiAxLAogICAgfSwKKTsKCm15ICRDSUF0dGFjaG1lbnRPYmplY3QgICAgICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtOjpDSUF0dGFjaG1lbnQnKTsKbXkgJENvbmZpZ0l0ZW1PYmplY3QgICAgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nKTsKbXkgJEhlbHBlck9iamVjdCAgICAgICAgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VW5pdFRlc3Q6OkhlbHBlcicpOwpteSAkTGF5b3V0Q0lBdHRhY2htZW50T2JqZWN0ICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpJVFNNQ29uZmlnSXRlbTo6TGF5b3V0Q0lBdHRhY2htZW50Jyk7Cm15ICRVbml0VGVzdElUU01Db25maWdJdGVtT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpJVFNNQ29uZmlnSXRlbScpOwpteSAkVW5pdFRlc3RQYXJhbU9iamVjdCAgICAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6UGFyYW0nKTsKbXkgJFVwbG9hZENhY2hlT2JqZWN0ICAgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6V2ViOjpVcGxvYWRDYWNoZScpOwoKbXkgJEF0dGFjaG1lbnRJRCA9ICRIZWxwZXJPYmplY3QtPkdldFJhbmRvbUlEKCk7Cm15ICRGb3JtSUQgICAgICAgPSAkVXBsb2FkQ2FjaGVPYmplY3QtPkZvcm1JRENyZWF0ZSgpOwoKbXkgJVRlc3RBdHRhY2htZW50ID0gKAogICAgRmlsZW5hbWUgICAgPT4gJ3Rlc3QudHh0JywKICAgIENvbnRlbnRUeXBlID0+ICd0ZXh0L3BsYWluJywKICAgIENvbnRlbnQgICAgID0+ICdibHViIGJsdWInLAopOwoKJFVuaXRUZXN0UGFyYW1PYmplY3QtPlBhcmFtU2V0KAogICAgTmFtZSAgPT4gJ1Rlc3RGaWVsZDo6MScsCiAgICBWYWx1ZSA9PiAkQXR0YWNobWVudElELAopOwokVW5pdFRlc3RQYXJhbU9iamVjdC0+UGFyYW1TZXQoCiAgICBOYW1lICA9PiAnVGVzdEZpZWxkOjoxOjpDSUF0dGFjaG1lbnQnLAogICAgVmFsdWUgPT4gMSwKKTsKJFVuaXRUZXN0UGFyYW1PYmplY3QtPlBhcmFtU2V0KAogICAgTmFtZSAgPT4gJ0Zvcm1JRCcsCiAgICBWYWx1ZSA9PiAkRm9ybUlELAopOwoKbXkgJEZvcm1JREF0dGFjaG1lbnQgPSAkTGF5b3V0Q0lBdHRhY2htZW50T2JqZWN0LT5Gb3JtSURHZXQoCiAgICBBdHRhY2htZW50S2V5ID0+ICdUZXN0RmllbGQnLAopOwoKJFVwbG9hZENhY2hlT2JqZWN0LT5Gb3JtSURBZGRGaWxlKAogICAgJVRlc3RBdHRhY2htZW50LAogICAgRm9ybUlEID0+ICRGb3JtSURBdHRhY2htZW50LAopOwoKbXkgJENvbmZpZ0l0ZW0gPSAkVW5pdFRlc3RJVFNNQ29uZmlnSXRlbU9iamVjdC0+Q29uZmlnSXRlbUNyZWF0ZSgKICAgIE5hbWUgICAgICAgICAgPT4gJ01hYycsCiAgICBDbGFzc05hbWUgICAgID0+ICdDb21wdXRlcicsCiAgICBEZXBsU3RhdGVOYW1lID0+ICdQcm9kdWN0aW9uJywKICAgIEluY2lTdGF0ZU5hbWUgPT4gJ09wZXJhdGlvbmFsJywKKTsKCm15IEBBdHRhY2htZW50cyA9ICRDSUF0dGFjaG1lbnRPYmplY3QtPkNJQXR0YWNobWVudExpc3QoCiAgICBDb25maWdJdGVtRmllbGQgPT4gJ1Rlc3RGaWVsZCcsCiAgICBDb25maWdJdGVtSUQgICAgPT4gJENvbmZpZ0l0ZW0tPntDb25maWdJdGVtSUR9LAogICAgVmVyc2lvbklEICAgICAgID0+ICRDb25maWdJdGVtLT57TGFzdFZlcnNpb25JRH0sCik7CgokU2VsZi0+SXMoCiAgICAkQXR0YWNobWVudHNbMF0sCiAgICAkQXR0YWNobWVudElELAogICAgJ0ZvdW5kIGF0dGFjaG1lbnQgaW4gQ0lBdHRhY2htZW50TGlzdCcsCik7CgpteSAlQXR0YWNobWVudCA9ICRDSUF0dGFjaG1lbnRPYmplY3QtPkNJQXR0YWNobWVudEdldCgKICAgIEF0dGFjaG1lbnRJRCA9PiAkQXR0YWNobWVudElELAopOwoKJFNlbGYtPlRydWUoCiAgICAoICVBdHRhY2htZW50ID8gMSA6IDAgKSwKICAgICdGb3VuZCBhdHRhY2htZW50IGluIENJQXR0YWNobWVudEdldCcsCik7CgokQ29uZmlnSXRlbU9iamVjdC0+VmVyc2lvbkRlbGV0ZSgKICAgIFZlcnNpb25JRCA9PiAkQ29uZmlnSXRlbS0+e0xhc3RWZXJzaW9uSUR9LAogICAgVXNlcklEICAgID0+IDEsCik7CgpAQXR0YWNobWVudHMgPSAkQ0lBdHRhY2htZW50T2JqZWN0LT5DSUF0dGFjaG1lbnRMaXN0KAogICAgQ29uZmlnSXRlbUZpZWxkID0+ICdUZXN0RmllbGQnLAogICAgQ29uZmlnSXRlbUlEICAgID0+ICRDb25maWdJdGVtLT57Q29uZmlnSXRlbUlEfSwKICAgIFZlcnNpb25JRCAgICAgICA9PiAkQ29uZmlnSXRlbS0+e0xhc3RWZXJzaW9uSUR9LAopOwoKJFNlbGYtPkZhbHNlKAogICAgQEF0dGFjaG1lbnRzID8gMSA6IDAsCiAgICAkQXR0YWNobWVudElELAogICAgJ0ZvdW5kIG5vIGF0dGFjaG1lbnQgaW4gQ0lBdHRhY2htZW50TGlzdCcsCik7CgolQXR0YWNobWVudCA9ICRDSUF0dGFjaG1lbnRPYmplY3QtPkNJQXR0YWNobWVudEdldCgKICAgIEF0dGFjaG1lbnRJRCA9PiAkQXR0YWNobWVudElELAopOwoKJFNlbGYtPkZhbHNlKAogICAgJUF0dGFjaG1lbnQgPyAxIDogMCwKICAgICdGb3VuZCBubyBhdHRhY2htZW50IGluIENJQXR0YWNobWVudEdldCcsCik7CgoxOwo=
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;

use utf8;

use vars qw($Self);

my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');
my $ServiceObject        = $Kernel::OM->Get('Kernel::System::Service');
my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
my $LinkObject           = $Kernel::OM->Get('Kernel::System::LinkObject');

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# define needed variable
my $RandomID = $Helper->GetRandomID();

my $ConfigItemName = 'UnitTestConfigItemTest' . $RandomID;
my $ServiceName    = 'UnitTestServiceTest' . $RandomID;

my @ConfigItemIDs;
my @ServiceIDs;

my $CheckExpectedResults;
$CheckExpectedResults = sub {

    # get parameters
    my (%Param) = @_;

    my %ExpectedIncidentStates = %{ $Param{ExpectedIncidentStates} };
    my %ObjectNameSuffix2ID    = %{ $Param{ObjectNameSuffix2ID} };

    # check the results
    for my $Object ( sort keys %ExpectedIncidentStates ) {

        if ( $Object eq 'ITSMConfigItem' ) {

            for my $NameSuffix ( sort keys %{ $ExpectedIncidentStates{$Object} } ) {

                # get config item data
                my $ConfigItem = $ConfigItemObject->ConfigItemGet(
                    ConfigItemID => $ObjectNameSuffix2ID{$Object}->{$NameSuffix},
                );

                # check the result
                $Self->Is(
                    $ConfigItem->{CurInciState},
                    $ExpectedIncidentStates{$Object}->{$NameSuffix},
                    "Check incident state of config item $NameSuffix.",
                );
            }
        }
        elsif ( $Object eq 'Service' ) {

            for my $NameSuffix ( sort keys %{ $ExpectedIncidentStates{$Object} } ) {

                # get service data (including the current incident state)
                my %ServiceData = $ServiceObject->ServiceGet(
                    ServiceID     => $ObjectNameSuffix2ID{$Object}->{$NameSuffix},
                    IncidentState => 1,
                    UserID        => 1,
                );

                # check the result
                $Self->Is(
                    $ServiceData{CurInciState},
                    $ExpectedIncidentStates{$Object}->{$NameSuffix},
                    "Check incident state of service $NameSuffix.",
                );
            }
        }
    }

    # Done for second run (after recalculation).
    return 1 if $Param{Recalculate};

    # Trigger recalculation of incident states for each config item.
    my $ExitCode
        = $Kernel::OM->Get('Kernel::System::Console::Command::Admin::ITSM::IncidentState::Recalculate')->Execute();
    $Self->Is(
        $ExitCode,
        0,
        "Admin::ITSM::IncidentState::Recalculate exit code",
    );

    # Check results again after recalculation (call myself recursively).
    $CheckExpectedResults->(
        %Param,
        Recalculate => 1,
    );

    # Done for first run (before recalculation).
    return 1;
};

# get class list
my $ClassList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::Class',
);
my %ClassListReverse = reverse %{$ClassList};

# get deployment state list
my $DeplStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %DeplStateListReverse = reverse %{$DeplStateList};

# get incident state list
my $InciStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %InciStateListReverse = reverse %{$InciStateList};

# get definition for 'Hardware' class
my $DefinitionRef = $ConfigItemObject->DefinitionGet(
    ClassID => $ClassListReverse{Hardware},
);

my %ObjectNameSuffix2ID;

# create config items
for my $NameSuffix ( 1 .. 7, qw(A B C D E F G) ) {

    # add a configitem
    my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        ClassID => $ClassListReverse{Hardware},
        UserID  => 1,
    );

    $Self->True(
        $ConfigItemID,
        "Added configitem id $ConfigItemID.",
    );

    # remember the config item id
    $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix} = $ConfigItemID;

    push @ConfigItemIDs, $ConfigItemID;

    # set a name for each configitem
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ConfigItemID,
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Operational},
        XMLData      => [
            undef,
            {
                Version => [
                    undef,
                    {
                        Vendor => [
                            undef,
                            {
                                Content => 'TestVendor',
                            },
                        ],
                    },
                ],
            },
        ],
        UserID => 1,
    );

    $Self->True(
        $VersionID,
        "Added a version for the configitem id $ConfigItemID",
    );
}

# create services
for my $NameSuffix ( 1 .. 2 ) {

    my $ServiceID = $ServiceObject->ServiceAdd(
        Name        => $ServiceName . '_' . $NameSuffix,
        ValidID     => 1,
        UserID      => 1,
        TypeID      => 1,
        Criticality => '3 normal',
    );

    $Self->True(
        $ServiceID,
        "Added service id $ServiceID.",
    );

    # remember the service id
    $ObjectNameSuffix2ID{Service}->{$NameSuffix} = $ServiceID;

    push @ServiceIDs, $ServiceID;
}

# read the original setting for IncidentLinkTypeDirection
my $OrigIncidentLinkTypeDirectionSetting = $ConfigObject->Get('ITSM::Core::IncidentLinkTypeDirection');

# set new config for IncidentLinkTypeDirection
$ConfigObject->Set(
    Key   => 'ITSM::Core::IncidentLinkTypeDirection',
    Value => {
        DependsOn => 'Source',
        Includes  => 'Source',
    },
);

# ################################################################################################################
#                                            Link Diagram
# ################################################################################################################
#
#
#
#                     6                         Service 1
#                     |                             ^
#                     |                             |
#                     |                             |
#                     v                             |
#         C <******** 5 ------> 4 ------> 3 ------> 1 *********> A *******> F **********> G
#         *           ^                   |                      ^          ^
#         *           |                   |                      *          *
#         *           |                   |                      *          *
#         *           |                   |                      *          *
#         *           |                   |         B            *          *
#         *           |                   |         *            *          *
#         *           |                   |         *            *          *
#         *           |                   |         *            *          *
#         v           |                   |         v            *          *
#         D ********> 7                   +-------> 2 ************          E
#                                                   ^
#                                                   |
#                                                   |
#                                                   |
#                                                Service 2
#
#
#
#
#  Explanation:
#               1 .. 7 and A .. G are ITSMConfigItems
#
#               DependsOn Links are shown as ----->
#               Includes  Links are shown as *****>
#
# ################################################################################################################

# define the links between CIs and Services
my %Links = (
    DependsOn => {
        ITSMConfigItem => {
            '7' => {
                ITSMConfigItem => ['5'],
            },
            '6' => {
                ITSMConfigItem => ['5'],
            },
            '5' => {
                ITSMConfigItem => ['4'],
            },
            '4' => {
                ITSMConfigItem => ['3'],
            },
            '3' => {
                ITSMConfigItem => [ '1', '2' ],
            },
            '1' => {
                Service => ['1'],
            },
        },
        Service => {
            '2' => {
                ITSMConfigItem => ['2'],
            }
        },
    },
    Includes => {
        ITSMConfigItem => {
            '5' => {
                ITSMConfigItem => ['C'],
            },
            'C' => {
                ITSMConfigItem => ['D'],
            },
            'D' => {
                ITSMConfigItem => ['7'],
            },
            'B' => {
                ITSMConfigItem => ['2'],
            },
            '2' => {
                ITSMConfigItem => ['A'],
            },
            '1' => {
                ITSMConfigItem => ['A'],
            },
            'A' => {
                ITSMConfigItem => ['F'],
            },
            'F' => {
                ITSMConfigItem => ['G'],
            },
            'E' => {
                ITSMConfigItem => ['F'],
            },
        },
    },
);

# link the config items and services as shown in the diagram
for my $LinkType ( sort keys %Links ) {
    for my $TargetObject ( sort keys %{ $Links{$LinkType} } ) {
        for my $TargetKey ( sort keys %{ $Links{$LinkType}->{$TargetObject} } ) {
            for my $SourceObject ( sort keys %{ $Links{$LinkType}->{$TargetObject}->{$TargetKey} } )
            {
                for my $SourceKey (
                    @{ $Links{$LinkType}->{$TargetObject}->{$TargetKey}->{$SourceObject} }
                    )
                {

                    # add the links
                    my $Success = $LinkObject->LinkAdd(
                        SourceObject => $SourceObject,
                        SourceKey    => $ObjectNameSuffix2ID{$SourceObject}->{$SourceKey},
                        TargetObject => $TargetObject,
                        TargetKey    => $ObjectNameSuffix2ID{$TargetObject}->{$TargetKey},
                        Type         => $LinkType,
                        State        => 'Valid',
                        UserID       => 1,
                    );

                    $Self->True(
                        $Success,
                        "LinkAdd() - $SourceObject:$SourceKey linked with $TargetObject:$TargetKey with LinkType '$LinkType'.",
                    );
                }
            }
        }
    }
}

# ------------------------------------------------------------ #
# set CI6 to "Incident" and check the results
# ------------------------------------------------------------ #

{

    my $NameSuffix    = 6;
    my $IncidentState = 'Incident';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Warning',
                '2' => 'Warning',
                '3' => 'Warning',
                '4' => 'Warning',
                '5' => 'Warning',
                '6' => 'Incident',
                '7' => 'Operational',
                'A' => 'Operational',
                'B' => 'Operational',
                'C' => 'Operational',
                'D' => 'Operational',
                'E' => 'Operational',
                'F' => 'Operational',
                'G' => 'Operational',
            },
            Service => {
                '1' => 'Warning',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# ------------------------------------------------------------ #
# set CI6 back to "Operational" and check the results
# ------------------------------------------------------------ #

{

    my $NameSuffix    = 6;
    my $IncidentState = 'Operational';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Operational',
                '2' => 'Operational',
                '3' => 'Operational',
                '4' => 'Operational',
                '5' => 'Operational',
                '6' => 'Operational',
                '7' => 'Operational',
                'A' => 'Operational',
                'B' => 'Operational',
                'C' => 'Operational',
                'D' => 'Operational',
                'E' => 'Operational',
                'F' => 'Operational',
                'G' => 'Operational',
            },
            Service => {
                '1' => 'Operational',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# ------------------------------------------------------------ #
# set CI1 to "Incident" and check the results
# ------------------------------------------------------------ #

{

    my $NameSuffix    = 1;
    my $IncidentState = 'Incident';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Incident',
                '2' => 'Operational',
                '3' => 'Operational',
                '4' => 'Operational',
                '5' => 'Operational',
                '6' => 'Operational',
                '7' => 'Operational',
                'A' => 'Warning',
                'B' => 'Operational',
                'C' => 'Operational',
                'D' => 'Operational',
                'E' => 'Operational',
                'F' => 'Warning',
                'G' => 'Warning',
            },
            Service => {
                '1' => 'Incident',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# ------------------------------------------------------------------------ #
# set CI5 to "Incident" and check the results (CI1 is still in "Incident")
# ------------------------------------------------------------------------ #

{

    my $NameSuffix    = 5;
    my $IncidentState = 'Incident';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Incident',
                '2' => 'Warning',
                '3' => 'Warning',
                '4' => 'Warning',
                '5' => 'Incident',
                '6' => 'Operational',
                '7' => 'Warning',
                'A' => 'Warning',
                'B' => 'Operational',
                'C' => 'Warning',
                'D' => 'Warning',
                'E' => 'Operational',
                'F' => 'Warning',
                'G' => 'Warning',
            },
            Service => {
                '1' => 'Incident',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# -------------------------------------------------------------------------- #
# set CI1 to "Operational" and check the results (CI5 is still in "Incident")
# -------------------------------------------------------------------------- #

{

    my $NameSuffix    = 1;
    my $IncidentState = 'Operational';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Warning',
                '2' => 'Warning',
                '3' => 'Warning',
                '4' => 'Warning',
                '5' => 'Incident',
                '6' => 'Operational',
                '7' => 'Warning',
                'A' => 'Operational',
                'B' => 'Operational',
                'C' => 'Warning',
                'D' => 'Warning',
                'E' => 'Operational',
                'F' => 'Operational',
                'G' => 'Operational',
            },
            Service => {
                '1' => 'Warning',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# -------------------------------------------------------------------------- #
# set CI5 to "Operational" and check the results
# -------------------------------------------------------------------------- #

{

    my $NameSuffix    = 5;
    my $IncidentState = 'Operational';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Operational',
                '2' => 'Operational',
                '3' => 'Operational',
                '4' => 'Operational',
                '5' => 'Operational',
                '6' => 'Operational',
                '7' => 'Operational',
                'A' => 'Operational',
                'B' => 'Operational',
                'C' => 'Operational',
                'D' => 'Operational',
                'E' => 'Operational',
                'F' => 'Operational',
                'G' => 'Operational',
            },
            Service => {
                '1' => 'Operational',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# reset the enabled setting for IncidentLinkTypeDirection to its original value
$ConfigObject->Set(
    Key   => 'ITSM::Core::IncidentLinkTypeDirection',
    Value => $OrigIncidentLinkTypeDirectionSetting,
);

# Check CI's incident state update after 'LinkAdd' action (see bug#14382).
# Set some configs for link status.
$ConfigObject->Set(
    Valid => 1,
    Key   => 'ITSMConfigItem::SetIncidentStateOnLink',
    Value => 1,
);
$ConfigObject->Set(
    Valid => 1,
    Key   => 'ITSMConfigItem::LinkStatus::DeploymentStates',
    Value => ['Production'],
);
$ConfigObject->Set(
    Valid => 1,
    Key   => 'ITSMConfigItem::LinkStatus::IncidentStates',
    Value => [ 'Incident', 'Warning', 'Operational' ],
);
$ConfigObject->Set(
    Valid => 1,
    Key   => 'ITSMConfigItem::LinkStatus::LinkTypes',
    Value => {
        RelevantTo => 'Incident',
    },
);
$ConfigObject->Set(
    Valid => 1,
    Key   => 'ITSMConfigItem::LinkStatus::TicketTypes',
    Value => ['Incident'],
);

$RandomID = $Helper->GetRandomID();

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

my %Types = reverse $TicketObject->TicketTypeList(
    UserID => 1,
);

if ( !$Types{Incident} ) {
    my $TypeObject = $Kernel::OM->Get('Kernel::System::Type');
    my $ID         = $TypeObject->TypeAdd(
        Name    => 'Incident',
        ValidID => 1,
        UserID  => 1,
    );
}

# Create test ticket.
my $TicketID = $TicketObject->TicketCreate(
    Title        => "TN-$RandomID",
    Queue        => 'Raw',
    Lock         => 'unlock',
    Priority     => '3 normal',
    State        => 'open',
    Type         => 'Incident',
    CustomerNo   => '123465',
    CustomerUser => 'customer@example.com',
    OwnerID      => 1,
    UserID       => 1,
);
$Self->True(
    $TicketID,
    "TicketID $TicketID is created",
);

my $XMLData = [
    undef,
    {
        Version => [
            undef,
            {
                Vendor => [
                    undef,
                    {
                        Content => 'TestVendor',
                    },
                ],
            },
        ],
    },
];

my @Tests = (
    {
        Name              => "CI-01-$RandomID",
        DeplStateID       => $DeplStateListReverse{Production},
        InciStateID       => $InciStateListReverse{Operational},
        LinkType          => 'RelevantTo',
        ExpectedInciState => 'Incident',
    },
    {
        Name              => "CI-02-$RandomID",
        DeplStateID       => $DeplStateListReverse{Production},
        InciStateID       => $InciStateListReverse{Operational},
        LinkType          => 'DependsOn',
        ExpectedInciState => 'Operational',
    },
    {
        Name              => "CI-03-$RandomID",
        DeplStateID       => $DeplStateListReverse{Production},
        InciStateID       => $InciStateListReverse{Incident},
        LinkType          => 'RelevantTo',
        ExpectedInciState => 'Incident',
    },
    {
        Name              => "CI-04-$RandomID",
        DeplStateID       => $DeplStateListReverse{Maintenance},
        InciStateID       => $InciStateListReverse{Operational},
        LinkType          => 'RelevantTo',
        ExpectedInciState => 'Operational',
    },
);

my @CIIDs;

for my $Test (@Tests) {

    # Create CI.
    my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        ClassID => $ClassListReverse{Hardware},
        UserID  => 1,
    );
    $Self->True(
        $ConfigItemID,
        "ConfigItemID $ConfigItemID is created",
    );

    push @CIIDs, $ConfigItemID;

    # Add a version.
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ConfigItemID,
        Name         => $Test->{Name},
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $Test->{DeplStateID},
        InciStateID  => $Test->{InciStateID},
        XMLData      => $XMLData,
        UserID       => 1,
    );
    $Self->True(
        $VersionID,
        "VersionID $VersionID is created",
    );

    # Add a link.
    my $Success = $LinkObject->LinkAdd(
        SourceObject => 'Ticket',
        SourceKey    => $TicketID,
        TargetObject => 'ITSMConfigItem',
        TargetKey    => $ConfigItemID,
        Type         => $Test->{LinkType},
        State        => 'Valid',
        UserID       => 1,
    );
    $Self->True(
        $Success,
        "TicketID $TicketID linked with ConfigItemID $ConfigItemID with LinkType '$Test->{LinkType}'",
    );

    my $ConfigItem = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => $ConfigItemID,
    );

    # Check CI's incident state.
    $Self->Is(
        $ConfigItem->{CurInciState},
        $Test->{ExpectedInciState},
        "$Test->{Name} - CurInciState: $Test->{ExpectedInciState}",
    );
}

# Set ticket type to something different than 'Incident' to verify CI's incident state update.
my $Success = $TicketObject->TicketTypeSet(
    Type     => 'Unclassified',
    TicketID => $TicketID,
    UserID   => 1,
);
$Self->True(
    $Success,
    "Type for TicketID $TicketID is changed to 'Unclassified'",
);

for my $CIID (@CIIDs) {
    my $CI = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => $CIID,
    );
    $Self->Is(
        $CI->{CurInciState},
        'Operational',
        "CIID $CIID is in 'Operational' state after ticket type update",
    );
}

# cleanup is done by RestoreDatabase

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;

use utf8;

use vars qw($Self);

my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');
my $ServiceObject        = $Kernel::OM->Get('Kernel::System::Service');
my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
my $LinkObject           = $Kernel::OM->Get('Kernel::System::LinkObject');

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# define needed variable
my $RandomID = $Helper->GetRandomID();

my $ConfigItemName = 'UnitTestConfigItemTest' . $RandomID;
my $ServiceName    = 'UnitTestServiceTest' . $RandomID;

my @ConfigItemIDs;
my @ServiceIDs;

my $CheckExpectedResults;
$CheckExpectedResults = sub {

    # get parameters
    my (%Param) = @_;

    my %ExpectedIncidentStates = %{ $Param{ExpectedIncidentStates} };
    my %ObjectNameSuffix2ID    = %{ $Param{ObjectNameSuffix2ID} };

    # check the results
    for my $Object ( sort keys %ExpectedIncidentStates ) {

        if ( $Object eq 'ITSMConfigItem' ) {

            for my $NameSuffix ( sort keys %{ $ExpectedIncidentStates{$Object} } ) {

                # get config item data
                my $ConfigItem = $ConfigItemObject->ConfigItemGet(
                    ConfigItemID => $ObjectNameSuffix2ID{$Object}->{$NameSuffix},
                );

                # check the result
                $Self->Is(
                    $ConfigItem->{CurInciState},
                    $ExpectedIncidentStates{$Object}->{$NameSuffix},
                    "Check incident state of config item $NameSuffix.",
                );
            }
        }
        elsif ( $Object eq 'Service' ) {

            for my $NameSuffix ( sort keys %{ $ExpectedIncidentStates{$Object} } ) {

                # get service data (including the current incident state)
                my %ServiceData = $ServiceObject->ServiceGet(
                    ServiceID     => $ObjectNameSuffix2ID{$Object}->{$NameSuffix},
                    IncidentState => 1,
                    UserID        => 1,
                );

                # check the result
                $Self->Is(
                    $ServiceData{CurInciState},
                    $ExpectedIncidentStates{$Object}->{$NameSuffix},
                    "Check incident state of service $NameSuffix.",
                );
            }
        }
    }

    # Done for second run (after recalculation).
    return 1 if $Param{Recalculate};

    # Trigger recalculation of incident states for each config item.
    my $ExitCode
        = $Kernel::OM->Get('Kernel::System::Console::Command::Admin::ITSM::IncidentState::Recalculate')->Execute();
    $Self->Is(
        $ExitCode,
        0,
        "Admin::ITSM::IncidentState::Recalculate exit code",
    );

    # Check results again after recalculation (call myself recursively).
    $CheckExpectedResults->(
        %Param,
        Recalculate => 1,
    );

    # Done for first run (before recalculation).
    return 1;
};

# get class list
my $ClassList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::Class',
);
my %ClassListReverse = reverse %{$ClassList};

# get deployment state list
my $DeplStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %DeplStateListReverse = reverse %{$DeplStateList};

# get incident state list
my $InciStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %InciStateListReverse = reverse %{$InciStateList};

# get definition for 'Hardware' class
my $DefinitionRef = $ConfigItemObject->DefinitionGet(
    ClassID => $ClassListReverse{Hardware},
);

my %ObjectNameSuffix2ID;

# create config items
for my $NameSuffix ( 1 .. 7, qw(A B C D E F G) ) {

    # add a configitem
    my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        ClassID => $ClassListReverse{Hardware},
        UserID  => 1,
    );

    $Self->True(
        $ConfigItemID,
        "Added configitem id $ConfigItemID.",
    );

    # remember the config item id
    $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix} = $ConfigItemID;

    push @ConfigItemIDs, $ConfigItemID;

    # set a name for each configitem
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ConfigItemID,
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Operational},
        XMLData      => [
            undef,
            {
                Version => [
                    undef,
                    {
                        Vendor => [
                            undef,
                            {
                                Content => 'TestVendor',
                            },
                        ],
                    },
                ],
            },
        ],
        UserID => 1,
    );

    $Self->True(
        $VersionID,
        "Added a version for the configitem id $ConfigItemID",
    );
}

# create services
for my $NameSuffix ( 1 .. 2 ) {

    my $ServiceID = $ServiceObject->ServiceAdd(
        Name        => $ServiceName . '_' . $NameSuffix,
        ValidID     => 1,
        UserID      => 1,
        TypeID      => 1,
        Criticality => '3 normal',
    );

    $Self->True(
        $ServiceID,
        "Added service id $ServiceID.",
    );

    # remember the service id
    $ObjectNameSuffix2ID{Service}->{$NameSuffix} = $ServiceID;

    push @ServiceIDs, $ServiceID;
}

# read the original setting for IncidentLinkTypeDirection
my $OrigIncidentLinkTypeDirectionSetting = $ConfigObject->Get('ITSM::Core::IncidentLinkTypeDirection');

# set new config for IncidentLinkTypeDirection
$ConfigObject->Set(
    Key   => 'ITSM::Core::IncidentLinkTypeDirection',
    Value => {
        DependsOn => 'Source',
        Includes  => 'Source',
    },
);

# ################################################################################################################
#                                            Link Diagram
# ################################################################################################################
#
#
#
#                     6                         Service 1
#                     |                             ^
#                     |                             |
#                     |                             |
#                     v                             |
#         C <******** 5 ------> 4 ------> 3 ------> 1 *********> A *******> F **********> G
#         *           ^                   |                      ^          ^
#         *           |                   |                      *          *
#         *           |                   |                      *          *
#         *           |                   |                      *          *
#         *           |                   |         B            *          *
#         *           |                   |         *            *          *
#         *           |                   |         *            *          *
#         *           |                   |         *            *          *
#         v           |                   |         v            *          *
#         D ********> 7                   +-------> 2 ************          E
#                                                   ^
#                                                   |
#                                                   |
#                                                   |
#                                                Service 2
#
#
#
#
#  Explanation:
#               1 .. 7 and A .. G are ITSMConfigItems
#
#               DependsOn Links are shown as ----->
#               Includes  Links are shown as *****>
#
# ################################################################################################################

# define the links between CIs and Services
my %Links = (
    DependsOn => {
        ITSMConfigItem => {
            '7' => {
                ITSMConfigItem => ['5'],
            },
            '6' => {
                ITSMConfigItem => ['5'],
            },
            '5' => {
                ITSMConfigItem => ['4'],
            },
            '4' => {
                ITSMConfigItem => ['3'],
            },
            '3' => {
                ITSMConfigItem => [ '1', '2' ],
            },
            '1' => {
                Service => ['1'],
            },
        },
        Service => {
            '2' => {
                ITSMConfigItem => ['2'],
            }
        },
    },
    Includes => {
        ITSMConfigItem => {
            '5' => {
                ITSMConfigItem => ['C'],
            },
            'C' => {
                ITSMConfigItem => ['D'],
            },
            'D' => {
                ITSMConfigItem => ['7'],
            },
            'B' => {
                ITSMConfigItem => ['2'],
            },
            '2' => {
                ITSMConfigItem => ['A'],
            },
            '1' => {
                ITSMConfigItem => ['A'],
            },
            'A' => {
                ITSMConfigItem => ['F'],
            },
            'F' => {
                ITSMConfigItem => ['G'],
            },
            'E' => {
                ITSMConfigItem => ['F'],
            },
        },
    },
);

# link the config items and services as shown in the diagram
for my $LinkType ( sort keys %Links ) {
    for my $TargetObject ( sort keys %{ $Links{$LinkType} } ) {
        for my $TargetKey ( sort keys %{ $Links{$LinkType}->{$TargetObject} } ) {
            for my $SourceObject ( sort keys %{ $Links{$LinkType}->{$TargetObject}->{$TargetKey} } )
            {
                for my $SourceKey (
                    @{ $Links{$LinkType}->{$TargetObject}->{$TargetKey}->{$SourceObject} }
                    )
                {

                    # add the links
                    my $Success = $LinkObject->LinkAdd(
                        SourceObject => $SourceObject,
                        SourceKey    => $ObjectNameSuffix2ID{$SourceObject}->{$SourceKey},
                        TargetObject => $TargetObject,
                        TargetKey    => $ObjectNameSuffix2ID{$TargetObject}->{$TargetKey},
                        Type         => $LinkType,
                        State        => 'Valid',
                        UserID       => 1,
                    );

                    $Self->True(
                        $Success,
                        "LinkAdd() - $SourceObject:$SourceKey linked with $TargetObject:$TargetKey with LinkType '$LinkType'.",
                    );
                }
            }
        }
    }
}

# ------------------------------------------------------------ #
# set CI6 to "Incident" and check the results
# ------------------------------------------------------------ #

{

    my $NameSuffix    = 6;
    my $IncidentState = 'Incident';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Warning',
                '2' => 'Warning',
                '3' => 'Warning',
                '4' => 'Warning',
                '5' => 'Warning',
                '6' => 'Incident',
                '7' => 'Operational',
                'A' => 'Operational',
                'B' => 'Operational',
                'C' => 'Operational',
                'D' => 'Operational',
                'E' => 'Operational',
                'F' => 'Operational',
                'G' => 'Operational',
            },
            Service => {
                '1' => 'Warning',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# ------------------------------------------------------------ #
# set CI6 back to "Operational" and check the results
# ------------------------------------------------------------ #

{

    my $NameSuffix    = 6;
    my $IncidentState = 'Operational';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Operational',
                '2' => 'Operational',
                '3' => 'Operational',
                '4' => 'Operational',
                '5' => 'Operational',
                '6' => 'Operational',
                '7' => 'Operational',
                'A' => 'Operational',
                'B' => 'Operational',
                'C' => 'Operational',
                'D' => 'Operational',
                'E' => 'Operational',
                'F' => 'Operational',
                'G' => 'Operational',
            },
            Service => {
                '1' => 'Operational',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# ------------------------------------------------------------ #
# set CI1 to "Incident" and check the results
# ------------------------------------------------------------ #

{

    my $NameSuffix    = 1;
    my $IncidentState = 'Incident';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Incident',
                '2' => 'Operational',
                '3' => 'Operational',
                '4' => 'Operational',
                '5' => 'Operational',
                '6' => 'Operational',
                '7' => 'Operational',
                'A' => 'Warning',
                'B' => 'Operational',
                'C' => 'Operational',
                'D' => 'Operational',
                'E' => 'Operational',
                'F' => 'Warning',
                'G' => 'Warning',
            },
            Service => {
                '1' => 'Incident',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# ------------------------------------------------------------------------ #
# set CI5 to "Incident" and check the results (CI1 is still in "Incident")
# ------------------------------------------------------------------------ #

{

    my $NameSuffix    = 5;
    my $IncidentState = 'Incident';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Incident',
                '2' => 'Warning',
                '3' => 'Warning',
                '4' => 'Warning',
                '5' => 'Incident',
                '6' => 'Operational',
                '7' => 'Warning',
                'A' => 'Warning',
                'B' => 'Operational',
                'C' => 'Warning',
                'D' => 'Warning',
                'E' => 'Operational',
                'F' => 'Warning',
                'G' => 'Warning',
            },
            Service => {
                '1' => 'Incident',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# -------------------------------------------------------------------------- #
# set CI1 to "Operational" and check the results (CI5 is still in "Incident")
# -------------------------------------------------------------------------- #

{

    my $NameSuffix    = 1;
    my $IncidentState = 'Operational';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Warning',
                '2' => 'Warning',
                '3' => 'Warning',
                '4' => 'Warning',
                '5' => 'Incident',
                '6' => 'Operational',
                '7' => 'Warning',
                'A' => 'Operational',
                'B' => 'Operational',
                'C' => 'Warning',
                'D' => 'Warning',
                'E' => 'Operational',
                'F' => 'Operational',
                'G' => 'Operational',
            },
            Service => {
                '1' => 'Warning',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# -------------------------------------------------------------------------- #
# set CI5 to "Operational" and check the results
# -------------------------------------------------------------------------- #

{

    my $NameSuffix    = 5;
    my $IncidentState = 'Operational';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Operational',
                '2' => 'Operational',
                '3' => 'Operational',
                '4' => 'Operational',
                '5' => 'Operational',
                '6' => 'Operational',
                '7' => 'Operational',
                'A' => 'Operational',
                'B' => 'Operational',
                'C' => 'Operational',
                'D' => 'Operational',
                'E' => 'Operational',
                'F' => 'Operational',
                'G' => 'Operational',
            },
            Service => {
                '1' => 'Operational',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# reset the enabled setting for IncidentLinkTypeDirection to its original value
$ConfigObject->Set(
    Key   => 'ITSM::Core::IncidentLinkTypeDirection',
    Value => $OrigIncidentLinkTypeDirectionSetting,
);

# Check CI's incident state update after 'LinkAdd' action (see bug#14382).
# Set some configs for link status.
$ConfigObject->Set(
    Valid => 1,
    Key   => 'ITSMConfigItem::SetIncidentStateOnLink',
    Value => 1,
);
$ConfigObject->Set(
    Valid => 1,
    Key   => 'ITSMConfigItem::LinkStatus::DeploymentStates',
    Value => ['Production'],
);
$ConfigObject->Set(
    Valid => 1,
    Key   => 'ITSMConfigItem::LinkStatus::IncidentStates',
    Value => [ 'Incident', 'Warning', 'Operational' ],
);
$ConfigObject->Set(
    Valid => 1,
    Key   => 'ITSMConfigItem::LinkStatus::LinkTypes',
    Value => {
        RelevantTo => 'Incident',
    },
);
$ConfigObject->Set(
    Valid => 1,
    Key   => 'ITSMConfigItem::LinkStatus::TicketTypes',
    Value => ['Incident'],
);

$RandomID = $Helper->GetRandomID();

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

my %Types = reverse $TicketObject->TicketTypeList(
    UserID => 1,
);

if ( !$Types{Incident} ) {
    my $TypeObject = $Kernel::OM->Get('Kernel::System::Type');
    my $ID         = $TypeObject->TypeAdd(
        Name    => 'Incident',
        ValidID => 1,
        UserID  => 1,
    );
}

# Create test ticket.
my $TicketID = $TicketObject->TicketCreate(
    Title        => "TN-$RandomID",
    Queue        => 'Raw',
    Lock         => 'unlock',
    Priority     => '3 normal',
    State        => 'open',
    Type         => 'Incident',
    CustomerNo   => '123465',
    CustomerUser => 'customer@example.com',
    OwnerID      => 1,
    UserID       => 1,
);
$Self->True(
    $TicketID,
    "TicketID $TicketID is created",
);

my $XMLData = [
    undef,
    {
        Version => [
            undef,
            {
                Vendor => [
                    undef,
                    {
                        Content => 'TestVendor',
                    },
                ],
            },
        ],
    },
];

my @Tests = (
    {
        Name              => "CI-01-$RandomID",
        DeplStateID       => $DeplStateListReverse{Production},
        InciStateID       => $InciStateListReverse{Operational},
        LinkType          => 'RelevantTo',
        ExpectedInciState => 'Incident',
    },
    {
        Name              => "CI-02-$RandomID",
        DeplStateID       => $DeplStateListReverse{Production},
        InciStateID       => $InciStateListReverse{Operational},
        LinkType          => 'DependsOn',
        ExpectedInciState => 'Operational',
    },
    {
        Name              => "CI-03-$RandomID",
        DeplStateID       => $DeplStateListReverse{Production},
        InciStateID       => $InciStateListReverse{Incident},
        LinkType          => 'RelevantTo',
        ExpectedInciState => 'Incident',
    },
    {
        Name              => "CI-04-$RandomID",
        DeplStateID       => $DeplStateListReverse{Maintenance},
        InciStateID       => $InciStateListReverse{Operational},
        LinkType          => 'RelevantTo',
        ExpectedInciState => 'Operational',
    },
);

my @CIIDs;

for my $Test (@Tests) {

    # Create CI.
    my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        ClassID => $ClassListReverse{Hardware},
        UserID  => 1,
    );
    $Self->True(
        $ConfigItemID,
        "ConfigItemID $ConfigItemID is created",
    );

    push @CIIDs, $ConfigItemID;

    # Add a version.
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ConfigItemID,
        Name         => $Test->{Name},
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $Test->{DeplStateID},
        InciStateID  => $Test->{InciStateID},
        XMLData      => $XMLData,
        UserID       => 1,
    );
    $Self->True(
        $VersionID,
        "VersionID $VersionID is created",
    );

    # Add a link.
    my $Success = $LinkObject->LinkAdd(
        SourceObject => 'Ticket',
        SourceKey    => $TicketID,
        TargetObject => 'ITSMConfigItem',
        TargetKey    => $ConfigItemID,
        Type         => $Test->{LinkType},
        State        => 'Valid',
        UserID       => 1,
    );
    $Self->True(
        $Success,
        "TicketID $TicketID linked with ConfigItemID $ConfigItemID with LinkType '$Test->{LinkType}'",
    );

    my $ConfigItem = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => $ConfigItemID,
    );

    # Check CI's incident state.
    $Self->Is(
        $ConfigItem->{CurInciState},
        $Test->{ExpectedInciState},
        "$Test->{Name} - CurInciState: $Test->{ExpectedInciState}",
    );
}

# Set ticket type to something different than 'Incident' to verify CI's incident state update.
my $Success = $TicketObject->TicketTypeSet(
    Type     => 'Unclassified',
    TicketID => $TicketID,
    UserID   => 1,
);
$Self->True(
    $Success,
    "Type for TicketID $TicketID is changed to 'Unclassified'",
);

for my $CIID (@CIIDs) {
    my $CI = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => $CIID,
    );
    $Self->Is(
        $CI->{CurInciState},
        'Operational',
        "CIID $CIID is in 'Operational' state after ticket type update",
    );
}

# cleanup is done by RestoreDatabase

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;

use utf8;

use vars qw($Self);

my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');
my $ServiceObject        = $Kernel::OM->Get('Kernel::System::Service');
my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
my $LinkObject           = $Kernel::OM->Get('Kernel::System::LinkObject');

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# define needed variable
my $RandomID = $Helper->GetRandomID();

my $ConfigItemName = 'UnitTestConfigItemTest' . $RandomID;
my $ServiceName    = 'UnitTestServiceTest' . $RandomID;

my @ConfigItemIDs;
my @ServiceIDs;

my $CheckExpectedResults;
$CheckExpectedResults = sub {

    # get parameters
    my (%Param) = @_;

    my %ExpectedIncidentStates = %{ $Param{ExpectedIncidentStates} };
    my %ObjectNameSuffix2ID    = %{ $Param{ObjectNameSuffix2ID} };

    # check the results
    for my $Object ( sort keys %ExpectedIncidentStates ) {

        if ( $Object eq 'ITSMConfigItem' ) {

            for my $NameSuffix ( sort keys %{ $ExpectedIncidentStates{$Object} } ) {

                # get config item data
                my $ConfigItem = $ConfigItemObject->ConfigItemGet(
                    ConfigItemID => $ObjectNameSuffix2ID{$Object}->{$NameSuffix},
                );

                # check the result
                $Self->Is(
                    $ConfigItem->{CurInciState},
                    $ExpectedIncidentStates{$Object}->{$NameSuffix},
                    "Check incident state of config item $NameSuffix.",
                );
            }
        }
        elsif ( $Object eq 'Service' ) {

            for my $NameSuffix ( sort keys %{ $ExpectedIncidentStates{$Object} } ) {

                # get service data (including the current incident state)
                my %ServiceData = $ServiceObject->ServiceGet(
                    ServiceID     => $ObjectNameSuffix2ID{$Object}->{$NameSuffix},
                    IncidentState => 1,
                    UserID        => 1,
                );

                # check the result
                $Self->Is(
                    $ServiceData{CurInciState},
                    $ExpectedIncidentStates{$Object}->{$NameSuffix},
                    "Check incident state of service $NameSuffix.",
                );
            }
        }
    }

    # Done for second run (after recalculation).
    return 1 if $Param{Recalculate};

    # Trigger recalculation of incident states for each config item.
    my $ExitCode
        = $Kernel::OM->Get('Kernel::System::Console::Command::Admin::ITSM::IncidentState::Recalculate')->Execute();
    $Self->Is(
        $ExitCode,
        0,
        "Admin::ITSM::IncidentState::Recalculate exit code",
    );

    # Check results again after recalculation (call myself recursively).
    $CheckExpectedResults->(
        %Param,
        Recalculate => 1,
    );

    # Done for first run (before recalculation).
    return 1;
};

# get class list
my $ClassList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::Class',
);
my %ClassListReverse = reverse %{$ClassList};

# get deployment state list
my $DeplStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %DeplStateListReverse = reverse %{$DeplStateList};

# get incident state list
my $InciStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %InciStateListReverse = reverse %{$InciStateList};

# get definition for 'Hardware' class
my $DefinitionRef = $ConfigItemObject->DefinitionGet(
    ClassID => $ClassListReverse{Hardware},
);

my %ObjectNameSuffix2ID;

# create config items
for my $NameSuffix ( 1 .. 7, qw(A B C D E F G) ) {

    # add a configitem
    my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        ClassID => $ClassListReverse{Hardware},
        UserID  => 1,
    );

    $Self->True(
        $ConfigItemID,
        "Added configitem id $ConfigItemID.",
    );

    # remember the config item id
    $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix} = $ConfigItemID;

    push @ConfigItemIDs, $ConfigItemID;

    # set a name for each configitem
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ConfigItemID,
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{Operational},
        XMLData      => [
            undef,
            {
                Version => [
                    undef,
                    {
                        Vendor => [
                            undef,
                            {
                                Content => 'TestVendor',
                            },
                        ],
                    },
                ],
            },
        ],
        UserID => 1,
    );

    $Self->True(
        $VersionID,
        "Added a version for the configitem id $ConfigItemID",
    );
}

# create services
for my $NameSuffix ( 1 .. 2 ) {

    my $ServiceID = $ServiceObject->ServiceAdd(
        Name        => $ServiceName . '_' . $NameSuffix,
        ValidID     => 1,
        UserID      => 1,
        TypeID      => 1,
        Criticality => '3 normal',
    );

    $Self->True(
        $ServiceID,
        "Added service id $ServiceID.",
    );

    # remember the service id
    $ObjectNameSuffix2ID{Service}->{$NameSuffix} = $ServiceID;

    push @ServiceIDs, $ServiceID;
}

# read the original setting for IncidentLinkTypeDirection
my $OrigIncidentLinkTypeDirectionSetting = $ConfigObject->Get('ITSM::Core::IncidentLinkTypeDirection');

# set new config for IncidentLinkTypeDirection
$ConfigObject->Set(
    Key   => 'ITSM::Core::IncidentLinkTypeDirection',
    Value => {
        DependsOn => 'Source',
        Includes  => 'Source',
    },
);

# ################################################################################################################
#                                            Link Diagram
# ################################################################################################################
#
#
#
#                     6                         Service 1
#                     |                             ^
#                     |                             |
#                     |                             |
#                     v                             |
#         C <******** 5 ------> 4 ------> 3 ------> 1 *********> A *******> F **********> G
#         *           ^                   |                      ^          ^
#         *           |                   |                      *          *
#         *           |                   |                      *          *
#         *           |                   |                      *          *
#         *           |                   |         B            *          *
#         *           |                   |         *            *          *
#         *           |                   |         *            *          *
#         *           |                   |         *            *          *
#         v           |                   |         v            *          *
#         D ********> 7                   +-------> 2 ************          E
#                                                   ^
#                                                   |
#                                                   |
#                                                   |
#                                                Service 2
#
#
#
#
#  Explanation:
#               1 .. 7 and A .. G are ITSMConfigItems
#
#               DependsOn Links are shown as ----->
#               Includes  Links are shown as *****>
#
# ################################################################################################################

# define the links between CIs and Services
my %Links = (
    DependsOn => {
        ITSMConfigItem => {
            '7' => {
                ITSMConfigItem => ['5'],
            },
            '6' => {
                ITSMConfigItem => ['5'],
            },
            '5' => {
                ITSMConfigItem => ['4'],
            },
            '4' => {
                ITSMConfigItem => ['3'],
            },
            '3' => {
                ITSMConfigItem => [ '1', '2' ],
            },
            '1' => {
                Service => ['1'],
            },
        },
        Service => {
            '2' => {
                ITSMConfigItem => ['2'],
            }
        },
    },
    Includes => {
        ITSMConfigItem => {
            '5' => {
                ITSMConfigItem => ['C'],
            },
            'C' => {
                ITSMConfigItem => ['D'],
            },
            'D' => {
                ITSMConfigItem => ['7'],
            },
            'B' => {
                ITSMConfigItem => ['2'],
            },
            '2' => {
                ITSMConfigItem => ['A'],
            },
            '1' => {
                ITSMConfigItem => ['A'],
            },
            'A' => {
                ITSMConfigItem => ['F'],
            },
            'F' => {
                ITSMConfigItem => ['G'],
            },
            'E' => {
                ITSMConfigItem => ['F'],
            },
        },
    },
);

# link the config items and services as shown in the diagram
for my $LinkType ( sort keys %Links ) {
    for my $TargetObject ( sort keys %{ $Links{$LinkType} } ) {
        for my $TargetKey ( sort keys %{ $Links{$LinkType}->{$TargetObject} } ) {
            for my $SourceObject ( sort keys %{ $Links{$LinkType}->{$TargetObject}->{$TargetKey} } )
            {
                for my $SourceKey (
                    @{ $Links{$LinkType}->{$TargetObject}->{$TargetKey}->{$SourceObject} }
                    )
                {

                    # add the links
                    my $Success = $LinkObject->LinkAdd(
                        SourceObject => $SourceObject,
                        SourceKey    => $ObjectNameSuffix2ID{$SourceObject}->{$SourceKey},
                        TargetObject => $TargetObject,
                        TargetKey    => $ObjectNameSuffix2ID{$TargetObject}->{$TargetKey},
                        Type         => $LinkType,
                        State        => 'Valid',
                        UserID       => 1,
                    );

                    $Self->True(
                        $Success,
                        "LinkAdd() - $SourceObject:$SourceKey linked with $TargetObject:$TargetKey with LinkType '$LinkType'.",
                    );
                }
            }
        }
    }
}

# ------------------------------------------------------------ #
# set CI6 to "Incident" and check the results
# ------------------------------------------------------------ #

{

    my $NameSuffix    = 6;
    my $IncidentState = 'Incident';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Warning',
                '2' => 'Warning',
                '3' => 'Warning',
                '4' => 'Warning',
                '5' => 'Warning',
                '6' => 'Incident',
                '7' => 'Operational',
                'A' => 'Operational',
                'B' => 'Operational',
                'C' => 'Operational',
                'D' => 'Operational',
                'E' => 'Operational',
                'F' => 'Operational',
                'G' => 'Operational',
            },
            Service => {
                '1' => 'Warning',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# ------------------------------------------------------------ #
# set CI6 back to "Operational" and check the results
# ------------------------------------------------------------ #

{

    my $NameSuffix    = 6;
    my $IncidentState = 'Operational';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Operational',
                '2' => 'Operational',
                '3' => 'Operational',
                '4' => 'Operational',
                '5' => 'Operational',
                '6' => 'Operational',
                '7' => 'Operational',
                'A' => 'Operational',
                'B' => 'Operational',
                'C' => 'Operational',
                'D' => 'Operational',
                'E' => 'Operational',
                'F' => 'Operational',
                'G' => 'Operational',
            },
            Service => {
                '1' => 'Operational',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# ------------------------------------------------------------ #
# set CI1 to "Incident" and check the results
# ------------------------------------------------------------ #

{

    my $NameSuffix    = 1;
    my $IncidentState = 'Incident';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Incident',
                '2' => 'Operational',
                '3' => 'Operational',
                '4' => 'Operational',
                '5' => 'Operational',
                '6' => 'Operational',
                '7' => 'Operational',
                'A' => 'Warning',
                'B' => 'Operational',
                'C' => 'Operational',
                'D' => 'Operational',
                'E' => 'Operational',
                'F' => 'Warning',
                'G' => 'Warning',
            },
            Service => {
                '1' => 'Incident',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# ------------------------------------------------------------------------ #
# set CI5 to "Incident" and check the results (CI1 is still in "Incident")
# ------------------------------------------------------------------------ #

{

    my $NameSuffix    = 5;
    my $IncidentState = 'Incident';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Incident',
                '2' => 'Warning',
                '3' => 'Warning',
                '4' => 'Warning',
                '5' => 'Incident',
                '6' => 'Operational',
                '7' => 'Warning',
                'A' => 'Warning',
                'B' => 'Operational',
                'C' => 'Warning',
                'D' => 'Warning',
                'E' => 'Operational',
                'F' => 'Warning',
                'G' => 'Warning',
            },
            Service => {
                '1' => 'Incident',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# -------------------------------------------------------------------------- #
# set CI1 to "Operational" and check the results (CI5 is still in "Incident")
# -------------------------------------------------------------------------- #

{

    my $NameSuffix    = 1;
    my $IncidentState = 'Operational';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Warning',
                '2' => 'Warning',
                '3' => 'Warning',
                '4' => 'Warning',
                '5' => 'Incident',
                '6' => 'Operational',
                '7' => 'Warning',
                'A' => 'Operational',
                'B' => 'Operational',
                'C' => 'Warning',
                'D' => 'Warning',
                'E' => 'Operational',
                'F' => 'Operational',
                'G' => 'Operational',
            },
            Service => {
                '1' => 'Warning',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# -------------------------------------------------------------------------- #
# set CI5 to "Operational" and check the results
# -------------------------------------------------------------------------- #

{

    my $NameSuffix    = 5;
    my $IncidentState = 'Operational';

    # change incident state
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ObjectNameSuffix2ID{ITSMConfigItem}->{$NameSuffix},
        Name         => $ConfigItemName . '_Hardware_' . $NameSuffix,
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $DeplStateListReverse{Production},
        InciStateID  => $InciStateListReverse{$IncidentState},
        UserID       => 1,
    );

    $Self->True(
        $VersionID,
        "Set config item id $NameSuffix to state '$IncidentState'.",
    );

    $CheckExpectedResults->(
        ExpectedIncidentStates => {
            ITSMConfigItem => {
                '1' => 'Operational',
                '2' => 'Operational',
                '3' => 'Operational',
                '4' => 'Operational',
                '5' => 'Operational',
                '6' => 'Operational',
                '7' => 'Operational',
                'A' => 'Operational',
                'B' => 'Operational',
                'C' => 'Operational',
                'D' => 'Operational',
                'E' => 'Operational',
                'F' => 'Operational',
                'G' => 'Operational',
            },
            Service => {
                '1' => 'Operational',
                '2' => 'Operational',
            },
        },
        ObjectNameSuffix2ID => \%ObjectNameSuffix2ID,
    );

}

# reset the enabled setting for IncidentLinkTypeDirection to its original value
$ConfigObject->Set(
    Key   => 'ITSM::Core::IncidentLinkTypeDirection',
    Value => $OrigIncidentLinkTypeDirectionSetting,
);

# Check CI's incident state update after 'LinkAdd' action (see bug#14382).
# Set some configs for link status.
$ConfigObject->Set(
    Valid => 1,
    Key   => 'ITSMConfigItem::SetIncidentStateOnLink',
    Value => 1,
);
$ConfigObject->Set(
    Valid => 1,
    Key   => 'ITSMConfigItem::LinkStatus::DeploymentStates',
    Value => ['Production'],
);
$ConfigObject->Set(
    Valid => 1,
    Key   => 'ITSMConfigItem::LinkStatus::IncidentStates',
    Value => [ 'Incident', 'Warning', 'Operational' ],
);
$ConfigObject->Set(
    Valid => 1,
    Key   => 'ITSMConfigItem::LinkStatus::LinkTypes',
    Value => {
        RelevantTo => 'Incident',
    },
);
$ConfigObject->Set(
    Valid => 1,
    Key   => 'ITSMConfigItem::LinkStatus::TicketTypes',
    Value => ['Incident'],
);

$RandomID = $Helper->GetRandomID();

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

my %Types = reverse $TicketObject->TicketTypeList(
    UserID => 1,
);

if ( !$Types{Incident} ) {
    my $TypeObject = $Kernel::OM->Get('Kernel::System::Type');
    my $ID         = $TypeObject->TypeAdd(
        Name    => 'Incident',
        ValidID => 1,
        UserID  => 1,
    );
}

# Create test ticket.
my $TicketID = $TicketObject->TicketCreate(
    Title        => "TN-$RandomID",
    Queue        => 'Raw',
    Lock         => 'unlock',
    Priority     => '3 normal',
    State        => 'open',
    Type         => 'Incident',
    CustomerNo   => '123465',
    CustomerUser => 'customer@example.com',
    OwnerID      => 1,
    UserID       => 1,
);
$Self->True(
    $TicketID,
    "TicketID $TicketID is created",
);

my $XMLData = [
    undef,
    {
        Version => [
            undef,
            {
                Vendor => [
                    undef,
                    {
                        Content => 'TestVendor',
                    },
                ],
            },
        ],
    },
];

my @Tests = (
    {
        Name              => "CI-01-$RandomID",
        DeplStateID       => $DeplStateListReverse{Production},
        InciStateID       => $InciStateListReverse{Operational},
        LinkType          => 'RelevantTo',
        ExpectedInciState => 'Incident',
    },
    {
        Name              => "CI-02-$RandomID",
        DeplStateID       => $DeplStateListReverse{Production},
        InciStateID       => $InciStateListReverse{Operational},
        LinkType          => 'DependsOn',
        ExpectedInciState => 'Operational',
    },
    {
        Name              => "CI-03-$RandomID",
        DeplStateID       => $DeplStateListReverse{Production},
        InciStateID       => $InciStateListReverse{Incident},
        LinkType          => 'RelevantTo',
        ExpectedInciState => 'Incident',
    },
    {
        Name              => "CI-04-$RandomID",
        DeplStateID       => $DeplStateListReverse{Maintenance},
        InciStateID       => $InciStateListReverse{Operational},
        LinkType          => 'RelevantTo',
        ExpectedInciState => 'Operational',
    },
);

my @CIIDs;

for my $Test (@Tests) {

    # Create CI.
    my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        ClassID => $ClassListReverse{Hardware},
        UserID  => 1,
    );
    $Self->True(
        $ConfigItemID,
        "ConfigItemID $ConfigItemID is created",
    );

    push @CIIDs, $ConfigItemID;

    # Add a version.
    my $VersionID = $ConfigItemObject->VersionAdd(
        ConfigItemID => $ConfigItemID,
        Name         => $Test->{Name},
        DefinitionID => $DefinitionRef->{DefinitionID},
        DeplStateID  => $Test->{DeplStateID},
        InciStateID  => $Test->{InciStateID},
        XMLData      => $XMLData,
        UserID       => 1,
    );
    $Self->True(
        $VersionID,
        "VersionID $VersionID is created",
    );

    # Add a link.
    my $Success = $LinkObject->LinkAdd(
        SourceObject => 'Ticket',
        SourceKey    => $TicketID,
        TargetObject => 'ITSMConfigItem',
        TargetKey    => $ConfigItemID,
        Type         => $Test->{LinkType},
        State        => 'Valid',
        UserID       => 1,
    );
    $Self->True(
        $Success,
        "TicketID $TicketID linked with ConfigItemID $ConfigItemID with LinkType '$Test->{LinkType}'",
    );

    my $ConfigItem = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => $ConfigItemID,
    );

    # Check CI's incident state.
    $Self->Is(
        $ConfigItem->{CurInciState},
        $Test->{ExpectedInciState},
        "$Test->{Name} - CurInciState: $Test->{ExpectedInciState}",
    );
}

# Set ticket type to something different than 'Incident' to verify CI's incident state update.
my $Success = $TicketObject->TicketTypeSet(
    Type     => 'Unclassified',
    TicketID => $TicketID,
    UserID   => 1,
);
$Self->True(
    $Success,
    "Type for TicketID $TicketID is changed to 'Unclassified'",
);

for my $CIID (@CIIDs) {
    my $CI = $ConfigItemObject->ConfigItemGet(
        ConfigItemID => $CIID,
    );
    $Self->Is(
        $CI->{CurInciState},
        'Operational',
        "CIID $CIID is in 'Operational' state after ticket type update",
    );
}

# cleanup is done by RestoreDatabase

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');
my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# define needed variable
my $RandomID = $Helper->GetRandomID();

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# add the test classes
my @ConfigItemClassIDs;
my @ConfigItemClasses;
my @ConfigItemDefinitionIDs;

# generate a random name
my $FirstClassName  = 'UnitTestClass1' . $RandomID;
my $SecondClassName = 'UnitTestClass2' . $RandomID;

# set a name prefix
my $NamePrefix = 'UnitTestName' . $RandomID;

# add both unittest config item classes
my $FirstClassID = $GeneralCatalogObject->ItemAdd(
    Class   => 'ITSM::ConfigItem::Class',
    Name    => $FirstClassName,
    ValidID => 1,
    UserID  => 1,
);

# check first class id
if ( !$FirstClassID ) {

    $Self->True(
        0,
        "Can't add first config item class.",
    );
}

push @ConfigItemClassIDs, $FirstClassID;
push @ConfigItemClasses,  $FirstClassName;

my $SecondClassID = $GeneralCatalogObject->ItemAdd(
    Class   => 'ITSM::ConfigItem::Class',
    Name    => $SecondClassName,
    ValidID => 1,
    UserID  => 1,
);

# check second class id
if ( !$SecondClassID ) {

    $Self->True(
        0,
        "Can't add second config item class.",
    );
}

push @ConfigItemClassIDs, $SecondClassID;
push @ConfigItemClasses,  $SecondClassName;

# add an empty definition to the class. the definition doesn't need any elements, as we're only
# testing the name which isn't part of the definition, but of the config item itself
my $FirstDefinitionID = $ConfigItemObject->DefinitionAdd(
    ClassID    => $FirstClassID,
    Definition => "--- []",
    UserID     => 1,
);

# check definition id
if ( !$FirstDefinitionID ) {

    $Self->True(
        0,
        "Can't add first config item definition.",
    );
}

push @ConfigItemDefinitionIDs, $FirstDefinitionID;

my $SecondDefinitionID = $ConfigItemObject->DefinitionAdd(
    ClassID    => $SecondClassID,
    Definition => "--- []",
    UserID     => 1,
);

# check definition id
if ( !$SecondDefinitionID ) {

    $Self->True(
        0,
        "Can't add second config item definition.",
    );
}

push @ConfigItemDefinitionIDs, $SecondDefinitionID;

my @ConfigItemIDs;

# add a configitem to each class
my $FirstConfigItemID = $ConfigItemObject->ConfigItemAdd(
    ClassID => $FirstClassID,
    UserID  => 1,
);

if ( !$FirstConfigItemID ) {
    $Self->True(
        0,
        "Failed to add the first configitem",
    );
}

push @ConfigItemIDs, $FirstConfigItemID;

my $SecondConfigItemID = $ConfigItemObject->ConfigItemAdd(
    ClassID => $SecondClassID,
    UserID  => 1,
);

if ( !$SecondConfigItemID ) {
    $Self->True(
        0,
        "Failed to add the second configitem",
    );
}

push @ConfigItemIDs, $SecondConfigItemID;

# create a 3rd configitem in the 2nd class
my $ThirdConfigItemID = $ConfigItemObject->ConfigItemAdd(
    ClassID => $SecondClassID,
    UserID  => 1,
);

if ( !$ThirdConfigItemID ) {
    $Self->True(
        0,
        "Failed to add the third configitem",
    );
}

push @ConfigItemIDs, $ThirdConfigItemID;

# get deployment state list
my $DeplStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %DeplStateListReverse = reverse %{$DeplStateList};

# get incident state list
my $InciStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %InciStateListReverse = reverse %{$InciStateList};

# set a name for each configitem
my $FirstInitialVersionID = $ConfigItemObject->VersionAdd(
    ConfigItemID => $FirstConfigItemID,
    Name         => $NamePrefix . 'First#001',
    DefinitionID => $FirstDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

if ( !$FirstInitialVersionID ) {
    $Self->True(
        0,
        "Failed to add the initial version for the first configitem",
    );
}

my $SecondInitialVersionID = $ConfigItemObject->VersionAdd(
    ConfigItemID => $SecondConfigItemID,
    Name         => $NamePrefix . 'Second#001',
    DefinitionID => $SecondDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

if ( !$SecondInitialVersionID ) {
    $Self->True(
        0,
        "Failed to add the initial version for the second configitem",
    );
}

my $ThirdInitialVersionID = $ConfigItemObject->VersionAdd(
    ConfigItemID => $ThirdConfigItemID,
    Name         => $NamePrefix . 'Second#002',
    DefinitionID => $SecondDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

if ( !$ThirdInitialVersionID ) {
    $Self->True(
        0,
        "Failed to add the initial version for the third configitem",
    );
}

# ------------------------------------------------------------ #
# run the actual tests
# ------------------------------------------------------------ #

# read the original setting for the setting EnableUniquenessCheck
my $OrigEnableSetting = $ConfigObject->Get('UniqueCIName::EnableUniquenessCheck');

# enable the uniqueness check
$ConfigObject->Set(
    Key   => 'UniqueCIName::EnableUniquenessCheck',
    Value => 1,
);

# read the original setting for the scope of the uniqueness check
my $OrigScope = $ConfigObject->Get('UniqueCIName::UniquenessCheckScope');

# make sure, the scope for the uniqueness check is set to 'global'
$ConfigObject->Set(
    Key   => 'UniqueCIName::UniquenessCheckScope',
    Value => 'global',
);

my $RenameSuccess;

# try to give the 1st configitem the same name as the 2nd one
$RenameSuccess = $ConfigItemObject->VersionAdd(
    ConfigItemID => $FirstConfigItemID,
    Name         => $NamePrefix . 'Second#001',
    DefinitionID => $FirstDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

$Self->False(
    $RenameSuccess,
    "Scope => global: Renaming First#001 to already existing Second#001 successfully prevented"
);

# try to give the 2nd configitem the same name as the 3rd one
$RenameSuccess = $ConfigItemObject->VersionAdd(
    ConfigItemID => $SecondConfigItemID,
    Name         => $NamePrefix . 'Second#002',
    DefinitionID => $FirstDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

$Self->False(
    $RenameSuccess,
    "Scope => global: Renaming Second#001 to already existing Second#002 successfully prevented"
);

# set the scope for the uniqueness check to 'class'
$ConfigObject->Set(
    Key   => 'UniqueCIName::UniquenessCheckScope',
    Value => 'class',
);

# try to rename First#001 again to Second#001 which should work now, due to the different class
$RenameSuccess = $ConfigItemObject->VersionAdd(
    ConfigItemID => $FirstConfigItemID,
    Name         => $NamePrefix . 'Second#001',
    DefinitionID => $FirstDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

$Self->True(
    $RenameSuccess,
    "Scope => class: Renaming First#001 to already existing Second#001 succeeded"
);

# trying now to create a duplicate name within a class
$RenameSuccess = $ConfigItemObject->VersionAdd(
    ConfigItemID => $SecondConfigItemID,
    Name         => $NamePrefix . 'Second#002',
    DefinitionID => $SecondDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

$Self->False(
    $RenameSuccess,
    "Scope => class: Renaming Second#001 to already existing Second#002 successfully prevented"
);

# reset the enabled setting for the uniqueness check to its original value
$ConfigObject->Set(
    Key   => 'UniqueCIName::EnableUniquenessCheck',
    Value => $OrigEnableSetting,
);

# reset the scope for the uniqueness check to its original value
$ConfigObject->Set(
    Key   => 'UniqueCIName::UniquenessCheckScope',
    Value => $OrigScope,
);

# cleanup is done by RestoreDatabase

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');
my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# define needed variable
my $RandomID = $Helper->GetRandomID();

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# add the test classes
my @ConfigItemClassIDs;
my @ConfigItemClasses;
my @ConfigItemDefinitionIDs;

# generate a random name
my $FirstClassName  = 'UnitTestClass1' . $RandomID;
my $SecondClassName = 'UnitTestClass2' . $RandomID;

# set a name prefix
my $NamePrefix = 'UnitTestName' . $RandomID;

# add both unittest config item classes
my $FirstClassID = $GeneralCatalogObject->ItemAdd(
    Class   => 'ITSM::ConfigItem::Class',
    Name    => $FirstClassName,
    ValidID => 1,
    UserID  => 1,
);

# check first class id
if ( !$FirstClassID ) {

    $Self->True(
        0,
        "Can't add first config item class.",
    );
}

push @ConfigItemClassIDs, $FirstClassID;
push @ConfigItemClasses,  $FirstClassName;

my $SecondClassID = $GeneralCatalogObject->ItemAdd(
    Class   => 'ITSM::ConfigItem::Class',
    Name    => $SecondClassName,
    ValidID => 1,
    UserID  => 1,
);

# check second class id
if ( !$SecondClassID ) {

    $Self->True(
        0,
        "Can't add second config item class.",
    );
}

push @ConfigItemClassIDs, $SecondClassID;
push @ConfigItemClasses,  $SecondClassName;

# add an empty definition to the class. the definition doesn't need any elements, as we're only
# testing the name which isn't part of the definition, but of the config item itself
my $FirstDefinitionID = $ConfigItemObject->DefinitionAdd(
    ClassID    => $FirstClassID,
    Definition => "--- []",
    UserID     => 1,
);

# check definition id
if ( !$FirstDefinitionID ) {

    $Self->True(
        0,
        "Can't add first config item definition.",
    );
}

push @ConfigItemDefinitionIDs, $FirstDefinitionID;

my $SecondDefinitionID = $ConfigItemObject->DefinitionAdd(
    ClassID    => $SecondClassID,
    Definition => "--- []",
    UserID     => 1,
);

# check definition id
if ( !$SecondDefinitionID ) {

    $Self->True(
        0,
        "Can't add second config item definition.",
    );
}

push @ConfigItemDefinitionIDs, $SecondDefinitionID;

my @ConfigItemIDs;

# add a configitem to each class
my $FirstConfigItemID = $ConfigItemObject->ConfigItemAdd(
    ClassID => $FirstClassID,
    UserID  => 1,
);

if ( !$FirstConfigItemID ) {
    $Self->True(
        0,
        "Failed to add the first configitem",
    );
}

push @ConfigItemIDs, $FirstConfigItemID;

my $SecondConfigItemID = $ConfigItemObject->ConfigItemAdd(
    ClassID => $SecondClassID,
    UserID  => 1,
);

if ( !$SecondConfigItemID ) {
    $Self->True(
        0,
        "Failed to add the second configitem",
    );
}

push @ConfigItemIDs, $SecondConfigItemID;

# create a 3rd configitem in the 2nd class
my $ThirdConfigItemID = $ConfigItemObject->ConfigItemAdd(
    ClassID => $SecondClassID,
    UserID  => 1,
);

if ( !$ThirdConfigItemID ) {
    $Self->True(
        0,
        "Failed to add the third configitem",
    );
}

push @ConfigItemIDs, $ThirdConfigItemID;

# get deployment state list
my $DeplStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %DeplStateListReverse = reverse %{$DeplStateList};

# get incident state list
my $InciStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %InciStateListReverse = reverse %{$InciStateList};

# set a name for each configitem
my $FirstInitialVersionID = $ConfigItemObject->VersionAdd(
    ConfigItemID => $FirstConfigItemID,
    Name         => $NamePrefix . 'First#001',
    DefinitionID => $FirstDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

if ( !$FirstInitialVersionID ) {
    $Self->True(
        0,
        "Failed to add the initial version for the first configitem",
    );
}

my $SecondInitialVersionID = $ConfigItemObject->VersionAdd(
    ConfigItemID => $SecondConfigItemID,
    Name         => $NamePrefix . 'Second#001',
    DefinitionID => $SecondDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

if ( !$SecondInitialVersionID ) {
    $Self->True(
        0,
        "Failed to add the initial version for the second configitem",
    );
}

my $ThirdInitialVersionID = $ConfigItemObject->VersionAdd(
    ConfigItemID => $ThirdConfigItemID,
    Name         => $NamePrefix . 'Second#002',
    DefinitionID => $SecondDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

if ( !$ThirdInitialVersionID ) {
    $Self->True(
        0,
        "Failed to add the initial version for the third configitem",
    );
}

# ------------------------------------------------------------ #
# run the actual tests
# ------------------------------------------------------------ #

# read the original setting for the setting EnableUniquenessCheck
my $OrigEnableSetting = $ConfigObject->Get('UniqueCIName::EnableUniquenessCheck');

# enable the uniqueness check
$ConfigObject->Set(
    Key   => 'UniqueCIName::EnableUniquenessCheck',
    Value => 1,
);

# read the original setting for the scope of the uniqueness check
my $OrigScope = $ConfigObject->Get('UniqueCIName::UniquenessCheckScope');

# make sure, the scope for the uniqueness check is set to 'global'
$ConfigObject->Set(
    Key   => 'UniqueCIName::UniquenessCheckScope',
    Value => 'global',
);

my $RenameSuccess;

# try to give the 1st configitem the same name as the 2nd one
$RenameSuccess = $ConfigItemObject->VersionAdd(
    ConfigItemID => $FirstConfigItemID,
    Name         => $NamePrefix . 'Second#001',
    DefinitionID => $FirstDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

$Self->False(
    $RenameSuccess,
    "Scope => global: Renaming First#001 to already existing Second#001 successfully prevented"
);

# try to give the 2nd configitem the same name as the 3rd one
$RenameSuccess = $ConfigItemObject->VersionAdd(
    ConfigItemID => $SecondConfigItemID,
    Name         => $NamePrefix . 'Second#002',
    DefinitionID => $FirstDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

$Self->False(
    $RenameSuccess,
    "Scope => global: Renaming Second#001 to already existing Second#002 successfully prevented"
);

# set the scope for the uniqueness check to 'class'
$ConfigObject->Set(
    Key   => 'UniqueCIName::UniquenessCheckScope',
    Value => 'class',
);

# try to rename First#001 again to Second#001 which should work now, due to the different class
$RenameSuccess = $ConfigItemObject->VersionAdd(
    ConfigItemID => $FirstConfigItemID,
    Name         => $NamePrefix . 'Second#001',
    DefinitionID => $FirstDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

$Self->True(
    $RenameSuccess,
    "Scope => class: Renaming First#001 to already existing Second#001 succeeded"
);

# trying now to create a duplicate name within a class
$RenameSuccess = $ConfigItemObject->VersionAdd(
    ConfigItemID => $SecondConfigItemID,
    Name         => $NamePrefix . 'Second#002',
    DefinitionID => $SecondDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

$Self->False(
    $RenameSuccess,
    "Scope => class: Renaming Second#001 to already existing Second#002 successfully prevented"
);

# reset the enabled setting for the uniqueness check to its original value
$ConfigObject->Set(
    Key   => 'UniqueCIName::EnableUniquenessCheck',
    Value => $OrigEnableSetting,
);

# reset the scope for the uniqueness check to its original value
$ConfigObject->Set(
    Key   => 'UniqueCIName::UniquenessCheckScope',
    Value => $OrigScope,
);

# cleanup is done by RestoreDatabase

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');
my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# define needed variable
my $RandomID = $Helper->GetRandomID();

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# add the test classes
my @ConfigItemClassIDs;
my @ConfigItemClasses;
my @ConfigItemDefinitionIDs;

# generate a random name
my $FirstClassName  = 'UnitTestClass1' . $RandomID;
my $SecondClassName = 'UnitTestClass2' . $RandomID;

# set a name prefix
my $NamePrefix = 'UnitTestName' . $RandomID;

# add both unittest config item classes
my $FirstClassID = $GeneralCatalogObject->ItemAdd(
    Class   => 'ITSM::ConfigItem::Class',
    Name    => $FirstClassName,
    ValidID => 1,
    UserID  => 1,
);

# check first class id
if ( !$FirstClassID ) {

    $Self->True(
        0,
        "Can't add first config item class.",
    );
}

push @ConfigItemClassIDs, $FirstClassID;
push @ConfigItemClasses,  $FirstClassName;

my $SecondClassID = $GeneralCatalogObject->ItemAdd(
    Class   => 'ITSM::ConfigItem::Class',
    Name    => $SecondClassName,
    ValidID => 1,
    UserID  => 1,
);

# check second class id
if ( !$SecondClassID ) {

    $Self->True(
        0,
        "Can't add second config item class.",
    );
}

push @ConfigItemClassIDs, $SecondClassID;
push @ConfigItemClasses,  $SecondClassName;

# add an empty definition to the class. the definition doesn't need any elements, as we're only
# testing the name which isn't part of the definition, but of the config item itself
my $FirstDefinitionID = $ConfigItemObject->DefinitionAdd(
    ClassID    => $FirstClassID,
    Definition => "--- []",
    UserID     => 1,
);

# check definition id
if ( !$FirstDefinitionID ) {

    $Self->True(
        0,
        "Can't add first config item definition.",
    );
}

push @ConfigItemDefinitionIDs, $FirstDefinitionID;

my $SecondDefinitionID = $ConfigItemObject->DefinitionAdd(
    ClassID    => $SecondClassID,
    Definition => "--- []",
    UserID     => 1,
);

# check definition id
if ( !$SecondDefinitionID ) {

    $Self->True(
        0,
        "Can't add second config item definition.",
    );
}

push @ConfigItemDefinitionIDs, $SecondDefinitionID;

my @ConfigItemIDs;

# add a configitem to each class
my $FirstConfigItemID = $ConfigItemObject->ConfigItemAdd(
    ClassID => $FirstClassID,
    UserID  => 1,
);

if ( !$FirstConfigItemID ) {
    $Self->True(
        0,
        "Failed to add the first configitem",
    );
}

push @ConfigItemIDs, $FirstConfigItemID;

my $SecondConfigItemID = $ConfigItemObject->ConfigItemAdd(
    ClassID => $SecondClassID,
    UserID  => 1,
);

if ( !$SecondConfigItemID ) {
    $Self->True(
        0,
        "Failed to add the second configitem",
    );
}

push @ConfigItemIDs, $SecondConfigItemID;

# create a 3rd configitem in the 2nd class
my $ThirdConfigItemID = $ConfigItemObject->ConfigItemAdd(
    ClassID => $SecondClassID,
    UserID  => 1,
);

if ( !$ThirdConfigItemID ) {
    $Self->True(
        0,
        "Failed to add the third configitem",
    );
}

push @ConfigItemIDs, $ThirdConfigItemID;

# get deployment state list
my $DeplStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %DeplStateListReverse = reverse %{$DeplStateList};

# get incident state list
my $InciStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %InciStateListReverse = reverse %{$InciStateList};

# set a name for each configitem
my $FirstInitialVersionID = $ConfigItemObject->VersionAdd(
    ConfigItemID => $FirstConfigItemID,
    Name         => $NamePrefix . 'First#001',
    DefinitionID => $FirstDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

if ( !$FirstInitialVersionID ) {
    $Self->True(
        0,
        "Failed to add the initial version for the first configitem",
    );
}

my $SecondInitialVersionID = $ConfigItemObject->VersionAdd(
    ConfigItemID => $SecondConfigItemID,
    Name         => $NamePrefix . 'Second#001',
    DefinitionID => $SecondDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

if ( !$SecondInitialVersionID ) {
    $Self->True(
        0,
        "Failed to add the initial version for the second configitem",
    );
}

my $ThirdInitialVersionID = $ConfigItemObject->VersionAdd(
    ConfigItemID => $ThirdConfigItemID,
    Name         => $NamePrefix . 'Second#002',
    DefinitionID => $SecondDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

if ( !$ThirdInitialVersionID ) {
    $Self->True(
        0,
        "Failed to add the initial version for the third configitem",
    );
}

# ------------------------------------------------------------ #
# run the actual tests
# ------------------------------------------------------------ #

# read the original setting for the setting EnableUniquenessCheck
my $OrigEnableSetting = $ConfigObject->Get('UniqueCIName::EnableUniquenessCheck');

# enable the uniqueness check
$ConfigObject->Set(
    Key   => 'UniqueCIName::EnableUniquenessCheck',
    Value => 1,
);

# read the original setting for the scope of the uniqueness check
my $OrigScope = $ConfigObject->Get('UniqueCIName::UniquenessCheckScope');

# make sure, the scope for the uniqueness check is set to 'global'
$ConfigObject->Set(
    Key   => 'UniqueCIName::UniquenessCheckScope',
    Value => 'global',
);

my $RenameSuccess;

# try to give the 1st configitem the same name as the 2nd one
$RenameSuccess = $ConfigItemObject->VersionAdd(
    ConfigItemID => $FirstConfigItemID,
    Name         => $NamePrefix . 'Second#001',
    DefinitionID => $FirstDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

$Self->False(
    $RenameSuccess,
    "Scope => global: Renaming First#001 to already existing Second#001 successfully prevented"
);

# try to give the 2nd configitem the same name as the 3rd one
$RenameSuccess = $ConfigItemObject->VersionAdd(
    ConfigItemID => $SecondConfigItemID,
    Name         => $NamePrefix . 'Second#002',
    DefinitionID => $FirstDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

$Self->False(
    $RenameSuccess,
    "Scope => global: Renaming Second#001 to already existing Second#002 successfully prevented"
);

# set the scope for the uniqueness check to 'class'
$ConfigObject->Set(
    Key   => 'UniqueCIName::UniquenessCheckScope',
    Value => 'class',
);

# try to rename First#001 again to Second#001 which should work now, due to the different class
$RenameSuccess = $ConfigItemObject->VersionAdd(
    ConfigItemID => $FirstConfigItemID,
    Name         => $NamePrefix . 'Second#001',
    DefinitionID => $FirstDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

$Self->True(
    $RenameSuccess,
    "Scope => class: Renaming First#001 to already existing Second#001 succeeded"
);

# trying now to create a duplicate name within a class
$RenameSuccess = $ConfigItemObject->VersionAdd(
    ConfigItemID => $SecondConfigItemID,
    Name         => $NamePrefix . 'Second#002',
    DefinitionID => $SecondDefinitionID,
    DeplStateID  => $DeplStateListReverse{Production},
    InciStateID  => $InciStateListReverse{Operational},
    UserID       => 1,
);

$Self->False(
    $RenameSuccess,
    "Scope => class: Renaming Second#001 to already existing Second#002 successfully prevented"
);

# reset the enabled setting for the uniqueness check to its original value
$ConfigObject->Set(
    Key   => 'UniqueCIName::EnableUniquenessCheck',
    Value => $OrigEnableSetting,
);

# reset the scope for the uniqueness check to its original value
$ConfigObject->Set(
    Key   => 'UniqueCIName::UniquenessCheckScope',
    Value => $OrigScope,
);

# cleanup is done by RestoreDatabase

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

# run this test only if the ImportExport package is installed
{
    # get ImportExport module directory
    my $ImportExportModule = $Kernel::OM->Get('Kernel::Config')->Get('Home') . '/Kernel/System/ImportExport.pm';

    # Return early if ImportExport package is not installed.
    if ( !-e $ImportExportModule ) {
        $Self->False(
            0,
            'ImportExport package not installed'
        );

        return 1;
    }
}

my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
my $ImportExportObject   = $Kernel::OM->Get('Kernel::System::ImportExport');
my $ObjectBackendObject  = $Kernel::OM->Get('Kernel::System::ImportExport::ObjectBackend::ITSMConfigItem');
my $XMLObject            = $Kernel::OM->Get('Kernel::System::XML');

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# define needed variable
my $RandomID = $Helper->GetRandomID();

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# add some test templates for later checks
my @TemplateIDs;
my $Counter = 0;
for ( 1 .. 30 ) {

    # add a test template for later checks
    my $TemplateID = $ImportExportObject->TemplateAdd(
        Object  => 'ITSMConfigItem',
        Format  => 'UnitTest' . $Counter . $RandomID,
        Name    => 'UnitTest' . $Counter . $RandomID,
        ValidID => 1,
        UserID  => 1,
    );

    push @TemplateIDs, $TemplateID;

    $Counter++;
}

# ------------------------------------------------------------ #
# ObjectList test 1 (check CSV item)
# ------------------------------------------------------------ #

# get object list
my $ObjectList1 = $ImportExportObject->ObjectList();

# check object list
$Self->True(
    $ObjectList1 && ref $ObjectList1 eq 'HASH' && $ObjectList1->{ITSMConfigItem},
    "ObjectList() - ITSMConfigItem exists",
);

# ------------------------------------------------------------ #
# ObjectAttributesGet test 1 (check attribute hash)
# ------------------------------------------------------------ #

{

    # get object attributes
    my $ObjectAttributesGet1 = $ImportExportObject->ObjectAttributesGet(
        TemplateID => $TemplateIDs[0],
        UserID     => 1,
    );

    # check object attribute reference
    $Self->True(
        $ObjectAttributesGet1 && ref $ObjectAttributesGet1 eq 'ARRAY',
        "ObjectAttributesGet() - check array reference",
    );

    # get class list
    my $ClassList = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # define the reference hash
    my $ObjectAttributesGet1Reference = [
        {
            Key   => 'ClassID',
            Name  => 'Class',
            Input => {
                Type         => 'Selection',
                Data         => $ClassList,
                Required     => 1,
                Translation  => 0,
                PossibleNone => 1,
            },
        },
        {
            Key   => 'CountMax',
            Name  => 'Maximum number of one element',
            Input => {
                Type         => 'Text',
                ValueDefault => '10',
                Required     => 1,
                Regex        => qr{ \A \d+ \z }xms,
                Translation  => 0,
                Size         => 5,
                MaxLength    => 5,
                DataType     => 'IntegerBiggerThanZero',
            },
        },
        {
            'Input' => {
                'Type' => 'Checkbox'
            },
            'Name' => 'Empty fields indicate that the current values are kept',
            'Key'  => 'EmptyFieldsLeaveTheOldValues',
        }
    ];

    $Self->IsDeeply(
        $ObjectAttributesGet1,
        $ObjectAttributesGet1Reference,
        "ObjectAttributesGet() - attributes of the row are identical",
    );
}

# ------------------------------------------------------------ #
# ObjectAttributesGet test 2 (check with non existing template)
# ------------------------------------------------------------ #

# get object attributes
my $ObjectAttributesGet2 = $ImportExportObject->ObjectAttributesGet(
    TemplateID => $TemplateIDs[-1] + 1,
    UserID     => 1,
);

# check false return
$Self->False(
    $ObjectAttributesGet2,
    "ObjectAttributesGet() - check false return",
);

# ------------------------------------------------------------ #
# MappingObjectAttributesGet test 1 (check attribute hash)
# ------------------------------------------------------------ #

# get mapping object attributes
my $MappingObjectAttributesGet1 = $ImportExportObject->MappingObjectAttributesGet(
    TemplateID => $TemplateIDs[0],
    UserID     => 1,
);

# check mapping object attribute reference
$Self->True(
    $MappingObjectAttributesGet1 && ref $MappingObjectAttributesGet1 eq 'ARRAY',
    "MappingObjectAttributesGet() - check array reference",
);

# ------------------------------------------------------------ #
# MappingObjectAttributesGet test 2 (check with non existing template)
# ------------------------------------------------------------ #

# get mapping object attributes
my $MappingObjectAttributesGet2 = $ImportExportObject->MappingObjectAttributesGet(
    TemplateID => $TemplateIDs[-1] + 1,
    UserID     => 1,
);

# check false return
$Self->False(
    $MappingObjectAttributesGet2,
    "MappingObjectAttributesGet() - check false return",
);

# ------------------------------------------------------------ #
# make preparations to test ExportDataGet() and ImportDataSave()
# ------------------------------------------------------------ #

my $GeneralCatalogClass = 'UnitTest' . $RandomID;

# add a general catalog test list
for my $Name (qw(Test1 Test2 Test3 Test4)) {

    # add a new item
    my $ItemID = $GeneralCatalogObject->ItemAdd(
        Class   => $GeneralCatalogClass,
        Name    => $Name,
        ValidID => 1,
        UserID  => 1,
    );

    # check item id
    if ( !$ItemID ) {

        $Self->True(
            0,
            "Can't add new general catalog item.",
        );
    }
}

# define the first test definition (all provided data types)
my @ConfigItemPerlDefinitions;
$ConfigItemPerlDefinitions[0] = " [
    {
        Key        => 'Customer1',
        Name       => 'Customer 1',
        Searchable => 1,
        Input      => {
            Type => 'Customer',
        },
    },
    {
        Key        => 'Date1',
        Name       => 'Date 1',
        Searchable => 1,
        Input      => {
            Type => 'Date',
        },
    },
    {
        Key        => 'DateTime1',
        Name       => 'Date Time 1',
        Searchable => 1,
        Input      => {
            Type => 'DateTime',
        },
    },
    {
        Key   => 'Dummy1',
        Name  => 'Dummy 1',
        Input => {
            Type => 'Dummy',
        },
    },
    {
        Key        => 'GeneralCatalog1',
        Name       => 'GeneralCatalog 1',
        Searchable => 1,
        Input      => {
            Type  => 'GeneralCatalog',
            Class => '$GeneralCatalogClass',
        },
    },
    {
        Key        => 'Integer1',
        Name       => 'Integer 1',
        Searchable => 1,
        Input      => {
            Type => 'Integer',
        },
    },
    {
        Key        => 'Text1',
        Name       => 'Text 1',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 50,
        },
    },
    {
        Key        => 'TextArea1',
        Name       => 'TextArea 1',
        Searchable => 1,
        Input      => {
            Type => 'TextArea',
        },
    },
] ";

# define the second test definition (sub data types)
$ConfigItemPerlDefinitions[1] = " [
    {
        Key        => 'Main1',
        Name       => 'Main 1',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 50,
            Required  => 1,
        },
        CountMax => 10,
        Sub => [
            {
                Key        => 'Main1Sub1',
                Name       => 'Main 1 Sub 1',
                Searchable => 1,
                Input      => {
                    Type      => 'Text',
                    Size      => 50,
                    MaxLength => 50,
                },
                CountMax => 10,
                Sub => [
                    {
                        Key        => 'Main1Sub1SubSub1',
                        Name       => 'Main 1 Sub 1 SubSub 1',
                        Searchable => 1,
                        Input      => {
                            Type      => 'Text',
                            Size      => 50,
                            MaxLength => 50,
                        },
                        CountMax => 10,
                    },
                    {
                        Key        => 'Main1Sub1SubSub2',
                        Name       => 'Main 1 Sub 1 SubSub 2',
                        Searchable => 1,
                        Input      => {
                            Type => 'TextArea',
                        },
                        CountMax => 10,
                    },
                ],
            },
            {
                Key        => 'Main1Sub2',
                Name       => 'Main 1 Sub 2',
                Searchable => 1,
                Input      => {
                    Type => 'TextArea',
                },
                CountMax => 10,
            },
        ],
    },
    {
        Key        => 'Main2',
        Name       => 'Main 2',
        Searchable => 1,
        Input      => {
            Type => 'TextArea',
        },
        CountMax => 10,
        Sub => [
            {
                Key        => 'Main2Sub1',
                Name       => 'Main 2 Sub 1',
                Searchable => 1,
                Input      => {
                    Type      => 'Text',
                    Size      => 50,
                    MaxLength => 50,
                },
                CountMax => 10,
            },
            {
                Key        => 'Main2Sub2',
                Name       => 'Main 2 Sub 2',
                Searchable => 1,
                Input      => {
                    Type => 'TextArea',
                },
                CountMax => 10,
            },
        ],
    },
] ";

# Define the third test definition with required attribute 'Type'.
$ConfigItemPerlDefinitions[2] = " [
    {
        'Key' => 'Type',
        'Input' => {
            'Class' => '$GeneralCatalogClass',
            'Required' => 1,
            'Translation' => 1,
            'Type' => 'GeneralCatalog'
        },
        'Searchable' => 1,
        'CountMin' => 1,
        'CountDefault' => 1,
        'CountMax' => 3,
        'Name' => 'Type',
    },
] ";

my $YAMLObject = $Kernel::OM->Get('Kernel::System::YAML');

my @ConfigItemDefinitions;
for my $PerlDefinition (@ConfigItemPerlDefinitions) {
    my $YAMLDefinition = $YAMLObject->Dump(
        Data => eval $PerlDefinition,    ## no critic
    );
    push @ConfigItemDefinitions, $YAMLDefinition;
}

# add the test classes
my @ConfigItemClassIDs;
my @ConfigItemDefinitionIDs;
for my $Definition (@ConfigItemDefinitions) {

    # generate a random name
    my $ClassName = 'UnitTest' . $Helper->GetRandomID();

    # add an unittest config item class
    my $ClassID = $GeneralCatalogObject->ItemAdd(
        Class   => 'ITSM::ConfigItem::Class',
        Name    => $ClassName,
        ValidID => 1,
        UserID  => 1,
    );

    # check class id
    if ( !$ClassID ) {

        $Self->True(
            0,
            "Can't add new config item class.",
        );
    }

    push @ConfigItemClassIDs, $ClassID;

    # add a definition to the class
    my $DefinitionID = $ConfigItemObject->DefinitionAdd(
        ClassID    => $ClassID,
        Definition => $Definition,
        UserID     => 1,
    );

    # check definition id
    if ( !$DefinitionID ) {

        $Self->True(
            0,
            "Can't add new config item definition.",
        );
    }

    push @ConfigItemDefinitionIDs, $DefinitionID;
}

# create some random numbers
my @ConfigItemNumbers;
for ( 1 .. 10 ) {
    push @ConfigItemNumbers, $Helper->GetRandomNumber();
}

# get deployment state list
my $DeplStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %DeplStateListReverse = reverse %{$DeplStateList};

# get incident state list
my $InciStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %InciStateListReverse = reverse %{$InciStateList};

# get general catalog test list
my $GeneralCatalogList = $GeneralCatalogObject->ItemList(
    Class => $GeneralCatalogClass,
);
my %GeneralCatalogListReverse = reverse %{$GeneralCatalogList};

# define the test config items
my @ConfigItems = (

    # config item for all provided data types
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[0],
            ClassID => $ConfigItemClassIDs[0],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 1 Version 1',
                DefinitionID => $ConfigItemDefinitionIDs[0],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Customer1 => [
                                    undef,
                                    {
                                        Content => 'UnitTest',
                                    },
                                ],
                                Date1 => [
                                    undef,
                                    {
                                        Content => '2008-02-01',
                                    },
                                ],
                                DateTime1 => [
                                    undef,
                                    {
                                        Content => '2008-02-01 03:59',
                                    },
                                ],
                                GeneralCatalog1 => [
                                    undef,
                                    {
                                        Content => $GeneralCatalogListReverse{Test1},
                                    },
                                ],
                                Integer1 => [
                                    undef,
                                    {
                                        Content => '1',
                                    },
                                ],
                                Text1 => [
                                    undef,
                                    {
                                        Content => 'Test Text Test',
                                    },
                                ],
                                TextArea1 => [
                                    undef,
                                    {
                                        Content => "Test\nText Array\nTest",
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },

    # a second config item for all provided data types
    # (duplicate name of first version for search checks)
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[1],
            ClassID => $ConfigItemClassIDs[0],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 1 Version 1',    # duplicate name for tests
                DefinitionID => $ConfigItemDefinitionIDs[0],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Customer1 => [
                                    undef,
                                    {
                                        Content => 'UnitTest',
                                    },
                                ],
                                Date1 => [
                                    undef,
                                    {
                                        Content => '2008-02-01',
                                    },
                                ],
                                DateTime1 => [
                                    undef,
                                    {
                                        Content => '2008-02-01 03:59',
                                    },
                                ],
                                GeneralCatalog1 => [
                                    undef,
                                    {
                                        Content => $GeneralCatalogListReverse{Test1},
                                    },
                                ],
                                Integer1 => [
                                    undef,
                                    {
                                        Content => '1',
                                    },
                                ],
                                Text1 => [
                                    undef,
                                    {
                                        Content => 'Test Text Test',
                                    },
                                ],
                                TextArea1 => [
                                    undef,
                                    {
                                        Content => "Test\nText Array\nTest",
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
            {
                Name         => 'UnitTest - ConfigItem 2 Version 2',
                DefinitionID => $ConfigItemDefinitionIDs[0],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Customer1 => [
                                    undef,
                                    {
                                        Content => 'UnitTest2',
                                    },
                                ],
                                Date1 => [
                                    undef,
                                    {
                                        Content => '2008-02-02',
                                    },
                                ],
                                DateTime1 => [
                                    undef,
                                    {
                                        Content => '2008-02-02 03:59',
                                    },
                                ],
                                GeneralCatalog1 => [
                                    undef,
                                    {
                                        Content => $GeneralCatalogListReverse{Test2},
                                    },
                                ],
                                Integer1 => [
                                    undef,
                                    {
                                        Content => '2',
                                    },
                                ],
                                Text1 => [
                                    undef,
                                    {
                                        Content => 'Test Text Test2',
                                    },
                                ],
                                TextArea1 => [
                                    undef,
                                    {
                                        Content => "Test2\nText Array\nTest 2",
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },

    # config item for sub element tests
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[2],
            ClassID => $ConfigItemClassIDs[1],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 3 Version 1',
                DefinitionID => $ConfigItemDefinitionIDs[1],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Main1 => [
                                    undef,
                                    {
                                        Content   => 'Main1 (1)',
                                        Main1Sub1 => [
                                            undef,
                                            {
                                                Content          => 'Main1 (1) Sub1 (1)',
                                                Main1Sub1SubSub1 => [
                                                    undef,
                                                    {
                                                        Content => 'Main1 (1) Sub1 (1) SubSub1 (1)',
                                                    },
                                                    {
                                                        Content => 'Main1 (1) Sub1 (1) SubSub1 (2)',
                                                    },
                                                    {
                                                        Content => 'Main1 (1) Sub1 (1) SubSub1 (3)',
                                                    },
                                                ],
                                                Main1Sub1SubSub2 => [
                                                    undef,
                                                    {
                                                        Content => 'Main1 (1) Sub1 (1) SubSub2 (1)',
                                                    },
                                                ],
                                            },
                                            {
                                                Content          => 'Main1 (1) Sub1 (2)',
                                                Main1Sub1SubSub1 => [
                                                    undef,
                                                    {
                                                        Content => 'Main1 (1) Sub1 (2) SubSub1 (1)',
                                                    },
                                                ],
                                                Main1Sub1SubSub2 => [
                                                    undef,
                                                    {
                                                        Content => 'Main1 (1) Sub1 (2) SubSub2 (1)',
                                                    },
                                                    {
                                                        Content => 'Main1 (1) Sub1 (2) SubSub2 (2)',
                                                    },
                                                ],
                                            },
                                        ],
                                        Main1Sub2 => [
                                            undef,
                                            {
                                                Content => 'Main1 (1) Sub2 (1)',
                                            },
                                            {
                                                Content => 'Main1 (1) Sub2 (2)',
                                            },
                                        ],
                                    },
                                ],
                                Main2 => [
                                    undef,
                                    {
                                        Content   => 'Main2 (1)',
                                        Main2Sub1 => [
                                            undef,
                                            {
                                                Content => 'Main2 (1) Sub1 (1)',
                                            },
                                        ],
                                        Main2Sub2 => [
                                            undef,
                                            {
                                                'Content' => 'Main2 (1) Sub2 (1)',
                                            },
                                            {
                                                'Content' => 'Main2 (1) Sub2 (2)',
                                            },
                                        ],
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },

    # config item for sub element tests
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[3],
            ClassID => $ConfigItemClassIDs[1],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 4 Version 1',
                DefinitionID => $ConfigItemDefinitionIDs[1],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Main1 => [
                                    undef,
                                    {
                                        Content   => '',
                                        Main1Sub1 => [
                                            undef,
                                            {
                                                Content          => '',
                                                Main1Sub1SubSub1 => [
                                                    undef,
                                                    {
                                                        Content => '',
                                                    },
                                                ],
                                                Main1Sub1SubSub2 => [
                                                    undef,
                                                    {
                                                        Content => '',
                                                    },
                                                ],
                                            },
                                        ],
                                        Main1Sub2 => [
                                            undef,
                                            {
                                                Content => '',
                                            },
                                        ],
                                    },
                                ],
                                Main2 => [
                                    undef,
                                    {
                                        Content   => '',
                                        Main2Sub1 => [
                                            undef,
                                            {
                                                Content => '',
                                            },
                                        ],
                                        Main2Sub2 => [
                                            undef,
                                            {
                                                Content => '',
                                            },
                                        ],
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },

    # config item for special character tests
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[4],
            ClassID => $ConfigItemClassIDs[1],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 5 Version 1',
                DefinitionID => $ConfigItemDefinitionIDs[1],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Main1 => [
                                    undef,
                                    {
                                        Content   => '"";;::..--__##',
                                        Main1Sub1 => [
                                            undef,
                                            {
                                                Content          => 'Test;:_°^!"§$%&/()=?´`*+Test',
                                                Main1Sub1SubSub1 => [
                                                    undef,
                                                    {
                                                        Content => '><@~\'}{[]\\',
                                                    },
                                                ],
                                                Main1Sub1SubSub2 => [
                                                    undef,
                                                    {
                                                        Content => '><@~\'}{[]\\',
                                                    },
                                                ],
                                            },
                                        ],
                                        Main1Sub2 => [
                                            undef,
                                            {
                                                Content => 'Test;:_°^!"§$%&/()=?´`*+Test',
                                            },
                                        ],
                                    },
                                ],
                                Main2 => [
                                    undef,
                                    {
                                        Content   => '"";;::..--__##',
                                        Main2Sub1 => [
                                            undef,
                                            {
                                                Content => 'Test Test',
                                            },
                                        ],
                                        Main2Sub2 => [
                                            undef,
                                            {
                                                Content => "Test\nTest\tTest",
                                            },
                                        ],
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },

    # config item for UTF-8 tests
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[5],
            ClassID => $ConfigItemClassIDs[1],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 6 Version 1',
                DefinitionID => $ConfigItemDefinitionIDs[1],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Main1 => [
                                    undef,
                                    {
                                        Content   => 'ↂ ⅻ ⅛',
                                        Main1Sub1 => [
                                            undef,
                                            {
                                                Content          => '☄ ↮ ↹ →',
                                                Main1Sub1SubSub1 => [
                                                    undef,
                                                    {
                                                        Content => '₤ ₡ ₩ ₯ ₵',
                                                    },
                                                ],
                                                Main1Sub1SubSub2 => [
                                                    undef,
                                                    {
                                                        Content => '♊ ♈ ♉ ♊ ♋ ♍ ♑',
                                                    },
                                                ],
                                            },
                                        ],
                                        Main1Sub2 => [
                                            undef,
                                            {
                                                Content => '✈ ❤ ☮',
                                            },
                                        ],
                                    },
                                ],
                                Main2 => [
                                    undef,
                                    {
                                        Content   => 'Պ Մ Հ',
                                        Main2Sub1 => [
                                            undef,
                                            {
                                                Content => '® ©',
                                            },
                                        ],
                                        Main2Sub2 => [
                                            undef,
                                            {
                                                Content => 'か げ を',
                                            },
                                        ],
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },
);

# add the test config items
my @ConfigItemIDs;
for my $ConfigItem (@ConfigItems) {

    # add a config item
    my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        %{ $ConfigItem->{ConfigItem} },
    );

    # check config item id
    if ( !$ConfigItemID ) {

        $Self->True(
            0,
            "Can't add new config item.",
        );
    }

    push @ConfigItemIDs, $ConfigItemID;

    # add the versions
    for my $Version ( @{ $ConfigItem->{Versions} } ) {

        # add a version
        my $VersionID = $ConfigItemObject->VersionAdd(
            %{$Version},
            ConfigItemID => $ConfigItemID,
        );

        # check version id
        if ( !$VersionID ) {

            $Self->True(
                0,
                "Can't add new version.",
            );
        }
    }
}

# ------------------------------------------------------------ #
# define general ExportDataGet tests
# ------------------------------------------------------------ #

my @ExportDataTests = (

    # ImportDataGet doesn't contains all data (check required attributes)
    {
        SourceExportData => {
            ExportDataGet => {
                UserID => 1,
            },
        },
    },

    # ImportDataGet doesn't contains all data (check required attributes)
    {
        SourceExportData => {
            ExportDataGet => {
                TemplateID => $TemplateIDs[1],
            },
        },
    },

    # no existing template id is given (check return false)
    {
        SourceExportData => {
            ExportDataGet => {
                TemplateID => $TemplateIDs[-1] + 1,
                UserID     => 1,
            },
        },
    },

    # no class id is given (check return false)
    {
        SourceExportData => {
            ExportDataGet => {
                TemplateID => $TemplateIDs[2],
                UserID     => 1,
            },
        },
    },

    # invalid class id is given (check return false)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[-1] + 1,
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[2],
                UserID     => 1,
            },
        },
    },

    # mapping list is empty (check return false)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[3],
                UserID     => 1,
            },
        },
    },

    # all required values are given (number search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[0],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (name search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Name => 'UnitTest - ConfigItem 1 Version 1',
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (case insensitive name search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Name => 'unittest - configitem 1 version 1',
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (name and number search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[0],
                Name   => 'UnitTest - ConfigItem 1 Version 1',
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (deployment state search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                DeplStateIDs => $DeplStateListReverse{Production},
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
            [ $ConfigItemNumbers[1] ],
        ],
    },

    # all required values are given (incident state search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                InciStateIDs => $InciStateListReverse{Operational},
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
            [ $ConfigItemNumbers[1] ],
        ],
    },

    # all required values are given (combined search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Number       => $ConfigItemNumbers[0],
                Name         => 'UnitTest - ConfigItem 1 Version 1',
                DeplStateIDs => $DeplStateListReverse{Production},
                InciStateIDs => $InciStateListReverse{Operational},
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (XML data search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Date1           => '2008-02-01',
                TextArea1       => "Test\nText Array\nTest",
                Customer1       => 'UnitTest',
                Text1           => 'Test Text Test',
                DateTime1       => '2008-02-01 03:59',
                Integer1        => '1',
                GeneralCatalog1 => $GeneralCatalogListReverse{Test1},
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (combined all search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Number          => $ConfigItemNumbers[0],
                Name            => 'UnitTest - ConfigItem 1 Version 1',
                DeplStateIDs    => $DeplStateListReverse{Production},
                InciStateIDs    => $InciStateListReverse{Operational},
                Date1           => '2008-02-01',
                TextArea1       => "Test\nText Array\nTest",
                Customer1       => 'UnitTest',
                Text1           => 'Test Text Test',
                DateTime1       => '2008-02-01 03:59',
                Integer1        => '1',
                GeneralCatalog1 => $GeneralCatalogListReverse{Test1},
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (check the returned array)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'Dummy1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[0],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[6],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[0],
                'UnitTest - ConfigItem 1 Version 1',
                'Production',
                'Operational',
                'UnitTest',
                '2008-02-01',
                '2008-02-01 03:59',
                undef,
                'Test1',
                '1',
                'Test Text Test',
                "Test\nText Array\nTest",
            ],
        ],
    },

    # all required values are given (double element checks)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'Dummy1::1',
                },
                {
                    Key => 'Dummy1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[0],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[6],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[0],
                $ConfigItemNumbers[0],
                'UnitTest - ConfigItem 1 Version 1',
                'UnitTest - ConfigItem 1 Version 1',
                'Production',
                'Production',
                'Operational',
                'Operational',
                'UnitTest',
                'UnitTest',
                '2008-02-01',
                '2008-02-01',
                '2008-02-01 03:59',
                '2008-02-01 03:59',
                undef,
                undef,
                'Test1',
                'Test1',
                '1',
                '1',
                'Test Text Test',
                'Test Text Test',
                "Test\nText Array\nTest",
                "Test\nText Array\nTest",
            ],
        ],
    },

    # all required values are given (sub element checks)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::2',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::2',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[2],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[7],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[2],
                'UnitTest - ConfigItem 3 Version 1',
                'Production',
                'Operational',
                'Main1 (1)',
                'Main1 (1) Sub1 (1)',
                'Main1 (1) Sub1 (1) SubSub1 (1)',
                'Main1 (1) Sub1 (1) SubSub1 (2)',
                'Main1 (1) Sub1 (1) SubSub1 (3)',
                'Main1 (1) Sub1 (1) SubSub2 (1)',
                'Main1 (1) Sub1 (2)',
                'Main1 (1) Sub1 (2) SubSub1 (1)',
                'Main1 (1) Sub1 (2) SubSub2 (1)',
                'Main1 (1) Sub1 (2) SubSub2 (2)',
                'Main1 (1) Sub2 (1)',
                'Main1 (1) Sub2 (2)',
                'Main2 (1)',
                'Main2 (1) Sub1 (1)',
                'Main2 (1) Sub2 (1)',
                'Main2 (1) Sub2 (2)',
            ],
        ],
    },

    # all required values are given (sub element checks with undef values)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::4',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::3',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub2::3',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::2',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::2',
                },
                {
                    Key => 'Main2::1::Main2Sub2::3',
                },
                {
                    Key => 'Main2::2',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[2],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[7],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[2],
                'UnitTest - ConfigItem 3 Version 1',
                'Production',
                'Operational',
                'Main1 (1)',
                'Main1 (1) Sub1 (1)',
                'Main1 (1) Sub1 (1) SubSub1 (1)',
                'Main1 (1) Sub1 (1) SubSub1 (2)',
                'Main1 (1) Sub1 (1) SubSub1 (3)',
                undef,
                'Main1 (1) Sub1 (1) SubSub2 (1)',
                undef,
                'Main1 (1) Sub1 (2)',
                'Main1 (1) Sub1 (2) SubSub1 (1)',
                undef,
                'Main1 (1) Sub1 (2) SubSub2 (1)',
                'Main1 (1) Sub1 (2) SubSub2 (2)',
                undef,
                'Main1 (1) Sub2 (1)',
                'Main1 (1) Sub2 (2)',
                undef,
                'Main2 (1)',
                'Main2 (1) Sub1 (1)',
                undef,
                'Main2 (1) Sub2 (1)',
                'Main2 (1) Sub2 (2)',
                undef,
                undef,
            ],
        ],
    },

    # all required values are given (sub element checks with undef values and empty strings)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::4',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::3',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub2::3',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::2',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::2',
                },
                {
                    Key => 'Main2::1::Main2Sub2::3',
                },
                {
                    Key => 'Main2::2',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[3],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[7],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[3],
                'UnitTest - ConfigItem 4 Version 1',
                'Production',
                'Operational',
                '',
                '',
                '',
                undef,
                undef,
                undef,
                '',
                undef,
                undef,
                undef,
                undef,
                undef,
                undef,
                undef,
                '',
                undef,
                undef,
                '',
                '',
                undef,
                '',
                undef,
                undef,
                undef,
            ],
        ],
    },

    # all required values are given (special character checks)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[4],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[8],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[4],
                'UnitTest - ConfigItem 5 Version 1',
                'Production',
                'Operational',
                '"";;::..--__##',
                'Test;:_°^!"§$%&/()=?´`*+Test',
                '><@~\'}{[]\\',
                '><@~\'}{[]\\',
                'Test;:_°^!"§$%&/()=?´`*+Test',
                '"";;::..--__##',
                'Test Test',
                "Test\nTest\tTest",
            ],
        ],
    },

    # all required values are given (UTF-8 checks)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[5],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[9],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[5],
                'UnitTest - ConfigItem 6 Version 1',
                'Production',
                'Operational',
                'ↂ ⅻ ⅛',
                '☄ ↮ ↹ →',
                '₤ ₡ ₩ ₯ ₵',
                '♊ ♈ ♉ ♊ ♋ ♍ ♑',
                '✈ ❤ ☮',
                'Պ Մ Հ',
                '® ©',
                'か げ を',
            ],
        ],
    },
);

# ------------------------------------------------------------ #
# run general ExportDataGet tests
# ------------------------------------------------------------ #

my $ExportTestCount = 1;
TEST:
for my $Test (@ExportDataTests) {

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

        $Self->True(
            0,
            "ExportTest $ExportTestCount: No SourceExportData found for this test."
        );

        next TEST;
    }

    # set the object data
    if (
        $Test->{SourceExportData}->{ObjectData}
        && ref $Test->{SourceExportData}->{ObjectData} eq 'HASH'
        && $Test->{SourceExportData}->{ExportDataGet}->{TemplateID}
        )
    {

        # save object data
        $ImportExportObject->ObjectDataSave(
            TemplateID => $Test->{SourceExportData}->{ExportDataGet}->{TemplateID},
            ObjectData => $Test->{SourceExportData}->{ObjectData},
            UserID     => 1,
        );
    }

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

        # delete all existing mapping data
        $ImportExportObject->MappingDelete(
            TemplateID => $Test->{SourceExportData}->{ExportDataGet}->{TemplateID},
            UserID     => 1,
        );

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

            # add a new mapping row
            my $MappingID = $ImportExportObject->MappingAdd(
                TemplateID => $Test->{SourceExportData}->{ExportDataGet}->{TemplateID},
                UserID     => 1,
            );

            # add the mapping object data
            $ImportExportObject->MappingObjectDataSave(
                MappingID         => $MappingID,
                MappingObjectData => $MappingObjectData,
                UserID            => 1,
            );
        }
    }

    # add the search data
    if (
        $Test->{SourceExportData}->{SearchData}
        && ref $Test->{SourceExportData}->{SearchData} eq 'HASH'
        && $Test->{SourceExportData}->{ExportDataGet}->{TemplateID}
        )
    {

        # save search data
        $ImportExportObject->SearchDataSave(
            TemplateID => $Test->{SourceExportData}->{ExportDataGet}->{TemplateID},
            SearchData => $Test->{SourceExportData}->{SearchData},
            UserID     => 1,
        );
    }

    # get export data
    my $ExportData = $ObjectBackendObject->ExportDataGet(
        %{ $Test->{SourceExportData}->{ExportDataGet} },
    );

    if ( !$Test->{ReferenceExportData} ) {

        $Self->False(
            $ExportData,
            "ExportTest $ExportTestCount: ExportDataGet() - return false",
        );

        next TEST;
    }

    if ( ref $ExportData ne 'ARRAY' ) {

        # check array reference
        $Self->True(
            0,
            "ExportTest $ExportTestCount: ExportDataGet() - return value is an array reference",
        );

        next TEST;
    }

    # check number of rows
    $Self->Is(
        scalar @{$ExportData},
        scalar @{ $Test->{ReferenceExportData} },
        "ExportTest $ExportTestCount: ExportDataGet() - correct number of rows",
    );

    # check content of export data
    my $CounterRow = 0;
    ROW:
    for my $ExportRow ( @{$ExportData} ) {

        # extract reference row
        my $ReferenceRow = $Test->{ReferenceExportData}->[$CounterRow];

        if ( ref $ExportRow ne 'ARRAY' || ref $ReferenceRow ne 'ARRAY' ) {

            # check array reference
            $Self->True(
                0,
                "ExportTest $ExportTestCount: ExportDataGet() - export row and reference row matched",
            );

            next TEST;
        }

        # check number of columns
        $Self->Is(
            scalar @{$ExportRow},
            scalar @{$ReferenceRow},
            "ExportTest $ExportTestCount: ExportDataGet() - correct number of columns",
        );

        my $CounterColumn = 0;
        for my $Cell ( @{$ExportRow} ) {

            # set content if values are undef
            if ( !defined $Cell ) {
                $Cell = 'UNDEF-unittest';
            }
            if ( !defined $ReferenceRow->[$CounterColumn] ) {
                $ReferenceRow->[$CounterColumn] = 'UNDEF-unittest';
            }

            # check cell data
            $Self->Is(
                $Cell,
                $ReferenceRow->[$CounterColumn],
                "ExportTest $ExportTestCount: ExportDataGet() ",
            );

            $CounterColumn++;
        }

        $CounterRow++;
    }
}
continue {
    $ExportTestCount++;
}

# ------------------------------------------------------------ #
# define general ImportDataSave tests
# ------------------------------------------------------------ #

my @ImportDataTests = (

    # ImportDataSave doesn't contains all data (check required attributes)
    {
        SourceImportData => {
            ImportDataSave => {
                ImportDataRow => [],
                UserID        => 1,
            },
        },
    },

    # ImportDataSave doesn't contains all data (check required attributes)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID => $TemplateIDs[20],
                UserID     => 1,
            },
        },
    },

    # ImportDataSave doesn't contains all data (check required attributes)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID    => $TemplateIDs[20],
                ImportDataRow => [],
            },
        },
    },

    # import data row must be an array reference (check return false)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID    => $TemplateIDs[20],
                ImportDataRow => '',
                UserID        => 1,
            },
        },
    },

    # import data row must be an array reference (check return false)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID    => $TemplateIDs[20],
                ImportDataRow => {},
                UserID        => 1,
            },
        },
    },

    # no existing template id is given (check return false)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID    => $TemplateIDs[-1] + 1,
                ImportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # no class id is given (check return false)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID    => $TemplateIDs[21],
                ImportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # invalid class id is given (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[-1] + 1,
            },
            ImportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ImportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # mapping list is empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # more than one identifier with the same name (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '123', '321' ],
                UserID        => 1,
            },
        },
    },

    # identifier is empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [''],
                UserID        => 1,
            },
        },
    },

    # identifier is undef (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [undef],
                UserID        => 1,
            },
        },
    },

    # both identifiers are empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '', '' ],
                UserID        => 1,
            },
        },
    },

    # both identifiers are undef (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ undef, undef ],
                UserID        => 1,
            },
        },
    },

    # one identifiers is empty, one is undef (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '', undef ],
                UserID        => 1,
            },
        },
    },

    # one of the identifiers is empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '123', '' ],
                UserID        => 1,
            },
        },
    },

    # one of the identifiers is undef (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '123', undef ],
                UserID        => 1,
            },
        },
    },

    # one of the identifiers is empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '', '123' ],
                UserID        => 1,
            },
        },
    },

    # one of the identifiers is empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ undef, '123' ],
                UserID        => 1,
            },
        },
    },

    # TODO Add some identifier tests

    # empty name is given (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ImportDataRow => [ '', 'Production', 'Operational' ],
                UserID        => 1,
            },
        },
    },

    # invalid deployment state is given (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ImportDataRow => [ 'UnitTest - Importtest 1', 'Dummy', 'Operational' ],
                UserID        => 1,
            },
        },
    },

    # invalid incident state is given (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ImportDataRow => [ 'UnitTest - Importtest 2', 'Production', 'Dummy' ],
                UserID        => 1,
            },
        },
    },

    # all required values are given (a NEW config item must be created)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 3',
                    'Production',
                    'Operational',
                    'UnitTest',
                    '2008-06-05',
                    '2008-08-05 04:50',
                    'Test3',
                    '3',
                    'Test3 Text3 Test3',
                    "Test3\nTextArray3\nTest3",
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 1,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 3',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Customer1::1'       => 'UnitTest',
                'Date1::1'           => '2008-06-05',
                'DateTime1::1'       => '2008-08-05 04:50',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test3},
                'Integer1::1'        => '3',
                'Text1::1'           => 'Test3 Text3 Test3',
                'TextArea1::1'       => "Test3\nTextArray3\nTest3",
            },
        },
    },

    # all required values are given (a second NEW config item must be created)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 4',
                    'Production',
                    'Operational',
                    'UnitTest',
                    '2008-09-05',
                    '2008-12-05 04:50',
                    'Test4',
                    '4',
                    'Test4 Text4 Test4',
                    "Test4\nTextArray4\nTest4",
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 1,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 4',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Customer1::1'       => 'UnitTest',
                'Date1::1'           => '2008-09-05',
                'DateTime1::1'       => '2008-12-05 04:50',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test4},
                'Integer1::1'        => '4',
                'Text1::1'           => 'Test4 Text4 Test4',
                'TextArea1::1'       => "Test4\nTextArray4\nTest4",
            },
        },
    },

    # all required values are given (a new version must be added to first test config item)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[26],
                ImportDataRow => [
                    $ConfigItemNumbers[0],
                    'UnitTest - ConfigItem 1 Version 2',
                    'Pilot',
                    'Incident',
                    'UnitTest',
                    '2008-02-02',
                    '2008-02-02 03:59',
                    'Test2',
                    '2',
                    'Test Text UPDATE1 Test',
                    "Test\nText Array UPDATE1\nTest",
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 2,
            LastVersion   => {
                Name                 => 'UnitTest - ConfigItem 1 Version 2',
                DeplState            => 'Pilot',
                InciState            => 'Incident',
                'Customer1::1'       => 'UnitTest',
                'Date1::1'           => '2008-02-02',
                'DateTime1::1'       => '2008-02-02 03:59',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test2},
                'Integer1::1'        => '2',
                'Text1::1'           => 'Test Text UPDATE1 Test',
                'TextArea1::1'       => "Test\nText Array UPDATE1\nTest",
            },
        },
    },

    # all required values are given (a new version must be added to first test config item again)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[27],
                ImportDataRow => [
                    $ConfigItemNumbers[0],
                    'UnitTest - ConfigItem 1 Version 3',
                    'Repair',
                    'Operational',
                    'UnitTest',
                    '2008-02-03',
                    '2008-02-03 03:59',
                    'Test3',
                    '3',
                    'Test Text UPDATE2 Test',
                    "Test\nText Array UPDATE2\nTest",
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 3,
            LastVersion   => {
                Name                 => 'UnitTest - ConfigItem 1 Version 3',
                DeplState            => 'Repair',
                InciState            => 'Operational',
                'Customer1::1'       => 'UnitTest',
                'Date1::1'           => '2008-02-03',
                'DateTime1::1'       => '2008-02-03 03:59',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test3},
                'Integer1::1'        => '3',
                'Text1::1'           => 'Test Text UPDATE2 Test',
                'TextArea1::1'       => "Test\nText Array UPDATE2\nTest",
            },
        },
    },

    # all required values are given (a new version must be added to third test config item)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::2',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::2',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[27],
                ImportDataRow => [
                    $ConfigItemNumbers[2],
                    'UnitTest - ConfigItem 3 Version 2',
                    'Production',
                    'Operational',
                    'Main1 (1)',
                    'Main1 (1) Main1Sub1 (1)',
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (1)',
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (2)',
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (3)',
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub2 (1)',
                    'Main1 (1) Main1Sub1 (2)',
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub1 (1)',
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (1)',
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (2)',
                    'Main1 (1) Main1Sub2 (1)',
                    'Main1 (1) Main1Sub2 (2)',
                    'Main2 (1)',
                    'Main2 (1) Main2Sub1 (1)',
                    'Main2 (1) Main2Sub2 (1)',
                    'Main2 (1) Main2Sub2 (2)',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 2,
            LastVersion   => {
                Name                     => 'UnitTest - ConfigItem 3 Version 2',
                DeplState                => 'Production',
                InciState                => 'Operational',
                'Main1::1'               => 'Main1 (1)',
                'Main1::1::Main1Sub1::1' => 'Main1 (1) Main1Sub1 (1)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (1)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (2)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (3)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub2 (1)',
                'Main1::1::Main1Sub1::2' => 'Main1 (1) Main1Sub1 (2)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub1 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (2)',
                'Main1::1::Main1Sub2::1' => 'Main1 (1) Main1Sub2 (1)',
                'Main1::1::Main1Sub2::2' => 'Main1 (1) Main1Sub2 (2)',
                'Main2::1'               => 'Main2 (1)',
                'Main2::1::Main2Sub1::1' => 'Main2 (1) Main2Sub1 (1)',
                'Main2::1::Main2Sub2::1' => 'Main2 (1) Main2Sub2 (1)',
                'Main2::1::Main2Sub2::2' => 'Main2 (1) Main2Sub2 (2)',
            },
        },
    },

    # all required values are given (special character checks)
    # In 'UnitTest - ConfigItem 3 Version 2' 16 Attributes were imported,
    # so there will be 8 lingering attributes.
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[27],
                ImportDataRow => [
                    $ConfigItemNumbers[2],
                    'UnitTest - ConfigItem 3 Version 3',
                    'Production',
                    'Operational',
                    '"";;::..--__##',
                    'Test;:_°^!"§$%&/()=?´`*+Test',
                    '><@~\'}{[]\\',
                    '><@~\'}{[]\\',
                    'Test;:_°^!"§$%&/()=?´`*+Test',
                    '"";;::..--__##',
                    'Test Test',
                    "Test\nTest\tTest",
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 3,
            LastVersion   => {
                Name                                          => 'UnitTest - ConfigItem 3 Version 3',
                DeplState                                     => 'Production',
                InciState                                     => 'Operational',
                'Main1::1'                                    => '"";;::..--__##',
                'Main1::1::Main1Sub1::1'                      => 'Test;:_°^!"§$%&/()=?´`*+Test',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1' => '><@~\'}{[]\\',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1' => '><@~\'}{[]\\',
                'Main1::1::Main1Sub2::1'                      => 'Test;:_°^!"§$%&/()=?´`*+Test',
                'Main2::1'                                    => '"";;::..--__##',
                'Main2::1::Main2Sub1::1'                      => 'Test Test',
                'Main2::1::Main2Sub2::1'                      => "Test\nTest\tTest",

                # lingering from 'UnitTest - ConfigItem 3 Version 2',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (2)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (3)',
                'Main1::1::Main1Sub1::2' => 'Main1 (1) Main1Sub1 (2)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub1 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (2)',
                'Main1::1::Main1Sub2::2' => 'Main1 (1) Main1Sub2 (2)',
                'Main2::1::Main2Sub2::2' => 'Main2 (1) Main2Sub2 (2)',
            },
        },
    },

    # all required values are given (UTF-8 checks)
    # In 'UnitTest - ConfigItem 3 Version 2' 16 Attributes were imported,
    # so there will be 8 lingering attributes.
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[27],
                ImportDataRow => [
                    $ConfigItemNumbers[2],
                    'UnitTest - ConfigItem 3 Version 4',
                    'Production',
                    'Operational',
                    'Ϋ δ λ',
                    'π χ Ϙ',
                    'Ϻ ϱ Ϯ',
                    'ɯ ʓ ʠ',
                    'ʬ ʯ',
                    'й ф щ',
                    'њ ё',
                    'Ѭ Ѧ',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 4,
            LastVersion   => {
                Name                                          => 'UnitTest - ConfigItem 3 Version 4',
                DeplState                                     => 'Production',
                InciState                                     => 'Operational',
                'Main1::1'                                    => 'Ϋ δ λ',
                'Main1::1::Main1Sub1::1'                      => 'π χ Ϙ',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1' => 'Ϻ ϱ Ϯ',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1' => 'ɯ ʓ ʠ',
                'Main1::1::Main1Sub2::1'                      => 'ʬ ʯ',
                'Main2::1'                                    => 'й ф щ',
                'Main2::1::Main2Sub1::1'                      => 'њ ё',
                'Main2::1::Main2Sub2::1'                      => 'Ѭ Ѧ',

                # lingering from 'UnitTest - ConfigItem 3 Version 2',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (2)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (3)',
                'Main1::1::Main1Sub1::2' => 'Main1 (1) Main1Sub1 (2)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub1 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (2)',
                'Main1::1::Main1Sub2::2' => 'Main1 (1) Main1Sub2 (2)',
                'Main2::1::Main2Sub2::2' => 'Main2 (1) Main2Sub2 (2)',
            },
        },
    },

    # a simple import for testing the overriding behavior of empty values
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    'Importtest 5 for behavior of empty values',
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 1,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => 'Importtest 5 for behavior of empty values',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an empty value for Text1, with EmptyFieldsLeaveTheOldValues turned on
    # no new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '',
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 1,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => 'Importtest 5 for behavior of empty values',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import undef for Text1, with EmptyFieldsLeaveTheOldValues turned on
    # no new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    undef,
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 1,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => 'Importtest 5 for behavior of empty values',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an empty value for Text1, with EmptyFieldsLeaveTheOldValues turned off
    # a new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => '',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '',
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 2,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => '',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import a single space value for Text1, with EmptyFieldsLeaveTheOldValues turned on
    # a new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => '',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    ' ',
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 3,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => ' ',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import the string '0' value for Text1, with EmptyFieldsLeaveTheOldValues turned on
    # a new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => '',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '0',
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 4,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => '0',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an empty value for GeneralCatalog1, with EmptyFieldsLeaveTheOldValues turned on
    # no new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '',
                    '',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 4,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => '0',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an invalid value for GeneralCatalog1, with EmptyFieldsLeaveTheOldValues turned on
    # the import should fail
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '',
                    'non-existent general catalog entry',
                ],
                UserID => 1,
            },
        },
    },

    # import an invalid value for GeneralCatalog1, with EmptyFieldsLeaveTheOldValues turned off
    # the import should fail
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => '',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '',
                    'non-existent general catalog entry',
                ],
                UserID => 1,
            },
        },
    },

    # import an empty value for DeplState, with EmptyFieldsLeaveTheOldValues turned on
    # no new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    '',
                    'Operational',
                    '',
                    '',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 4,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => '0',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an invalid value for DeplState, with EmptyFieldsLeaveTheOldValues turned on
    # an error should be generated
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'invalid deployment state',
                    'Operational',
                    '',
                    '',
                ],
                UserID => 1,
            },
        },
    },

    # import an empty value for InciState, with EmptyFieldsLeaveTheOldValues turned on
    # no new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    '',
                    '',
                    '',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 4,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => '0',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an invalid value for InciState, with EmptyFieldsLeaveTheOldValues turned on
    # an error should be generated
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'invalid incident state',
                    '',
                    '',
                ],
                UserID => 1,
            },
        },
    },

    # Import without required attribute 'Type', an error should be generated (see bug#14098).
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[2],
                EmptyFieldsLeaveTheOldValues => '',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                ],
                UserID => 1,
            },
        },
    },
);

# ------------------------------------------------------------ #
# run general ExportDataGet tests
# ------------------------------------------------------------ #

my $ImportTestCount = 1;
TEST:
for my $Test (@ImportDataTests) {

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

        $Self->True(
            0,
            "ImportTest $ImportTestCount: No SourceImportData found for this test."
        );

        next TEST;
    }

    # set the object data
    if (
        $Test->{SourceImportData}->{ObjectData}
        && ref $Test->{SourceImportData}->{ObjectData} eq 'HASH'
        && $Test->{SourceImportData}->{ImportDataSave}->{TemplateID}
        )
    {

        # save object data
        $ImportExportObject->ObjectDataSave(
            TemplateID => $Test->{SourceImportData}->{ImportDataSave}->{TemplateID},
            ObjectData => $Test->{SourceImportData}->{ObjectData},
            UserID     => 1,
        );
    }

    # set the mapping object data
    if (
        $Test->{SourceImportData}->{MappingObjectData}
        && ref $Test->{SourceImportData}->{MappingObjectData} eq 'ARRAY'
        && $Test->{SourceImportData}->{ImportDataSave}->{TemplateID}
        )
    {

        # delete all existing mapping data
        $ImportExportObject->MappingDelete(
            TemplateID => $Test->{SourceImportData}->{ImportDataSave}->{TemplateID},
            UserID     => 1,
        );

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

            # add a new mapping row
            my $MappingID = $ImportExportObject->MappingAdd(
                TemplateID => $Test->{SourceImportData}->{ImportDataSave}->{TemplateID},
                UserID     => 1,
            );

            # add the mapping object data
            $ImportExportObject->MappingObjectDataSave(
                MappingID         => $MappingID,
                MappingObjectData => $MappingObjectData,
                UserID            => 1,
            );
        }
    }

    # import data save
    my ( $ConfigItemID, $RetCode ) = $ObjectBackendObject->ImportDataSave(
        %{ $Test->{SourceImportData}->{ImportDataSave} },
        Counter => $ImportTestCount,
    );

    if ( !$Test->{ReferenceImportData} ) {

        $Self->False(
            $ConfigItemID,
            "ImportTest $ImportTestCount: ImportDataSave() - return no ConfigItemID"
        );
        $Self->False(
            $RetCode,
            "ImportTest $ImportTestCount: ImportDataSave() - return no RetCode"
        );

        next TEST;
    }

    $Self->True(
        $ConfigItemID,
        "ImportTest $ImportTestCount: ImportDataSave() - return ConfigItemID"
    );
    $Self->True(
        $RetCode,
        "ImportTest $ImportTestCount: ImportDataSave() - return RetCode"
    );

    # get the version list
    my $VersionList = $ConfigItemObject->VersionList(
        ConfigItemID => $ConfigItemID,
    ) // [];

    # check number of versions
    $Self->Is(
        scalar @{$VersionList},
        $Test->{ReferenceImportData}->{VersionNumber} || 0,
        "ImportTest $ImportTestCount: ImportDataSave() - correct number of versions",
    );

    # get the last version
    my $VersionData = $ConfigItemObject->VersionGet(
        ConfigItemID => $ConfigItemID,
        XMLDataGet   => 1,
    );

    # translate xmldata in a 2d hash
    my %XMLHash = $XMLObject->XMLHash2D(
        XMLHash => $VersionData->{XMLData},
    );

    # clean the xml hash
    KEY:
    for my $Key ( sort keys %XMLHash ) {

        next KEY if $Key =~ m{ \{'Content'\} \z }xms;

        delete $XMLHash{$Key};
    }

    # check general elements
    ELEMENT:
    for my $Element (qw(Number Name DeplState InciState)) {

        next ELEMENT if !exists $Test->{ReferenceImportData}->{LastVersion}->{$Element};

        # set content if values are undef
        if ( !defined $Test->{ReferenceImportData}->{LastVersion}->{$Element} ) {
            $Test->{ReferenceImportData}->{LastVersion}->{$Element} = 'UNDEF-unittest';
        }
        if ( !defined $Test->{ReferenceImportData}->{LastVersion}->{$Element} ) {
            $Test->{ReferenceImportData}->{LastVersion}->{$Element} = 'UNDEF-unittest';
        }

        # check element
        $Self->Is(
            $VersionData->{$Element},
            $Test->{ReferenceImportData}->{LastVersion}->{$Element},
            "ImportTest $ImportTestCount: ImportDataSave() $Element is identical",
        );

        delete $Test->{ReferenceImportData}->{LastVersion}->{$Element};
    }

    # check number of XML elements
    $Self->Is(
        scalar keys %XMLHash,
        scalar keys %{ $Test->{ReferenceImportData}->{LastVersion} },
        "ImportTest $ImportTestCount: ImportDataSave() - correct number of XML elements",
    );

    # check XML elements
    ELEMENT:
    for my $Key ( sort keys %{ $Test->{ReferenceImportData}->{LastVersion} } ) {

        # duplicate key
        my $XMLKey = $Key;

        # prepare key
        my $Counter = 0;
        while ( $XMLKey =~ m{ :: }xms ) {

            if ( $Counter % 2 ) {
                $XMLKey =~ s{ :: }{]\{'}xms;
            }
            else {
                $XMLKey =~ s{ :: }{'\}[}xms;
            }

            $Counter++;
        }

        next ELEMENT if !exists $XMLHash{ '[1]{\'Version\'}[1]{\'' . $XMLKey . ']{\'Content\'}' };

        # set content if values are undef
        if ( !defined $XMLHash{ '[1]{\'Version\'}[1]{\'' . $XMLKey . ']{\'Content\'}' } ) {
            $XMLHash{ '[1]{\'Version\'}[1]{\'' . $XMLKey . ']{\'Content\'}' } = 'UNDEF-unittest';
        }
        if ( !defined $Test->{ReferenceImportData}->{LastVersion}->{$Key} ) {
            $Test->{ReferenceImportData}->{LastVersion}->{$Key} = 'UNDEF-unittest';
        }

        # check XML element
        $Self->Is(
            $XMLHash{ '[1]{\'Version\'}[1]{\'' . $XMLKey . ']{\'Content\'}' },
            $Test->{ReferenceImportData}->{LastVersion}->{$Key},
            "ImportTest $ImportTestCount: ImportDataSave() $Key is identical",
        );
    }
}
continue {
    $ImportTestCount++;
}

# cleanup is done by RestoreDatabase

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

# run this test only if the ImportExport package is installed
{
    # get ImportExport module directory
    my $ImportExportModule = $Kernel::OM->Get('Kernel::Config')->Get('Home') . '/Kernel/System/ImportExport.pm';

    # Return early if ImportExport package is not installed.
    if ( !-e $ImportExportModule ) {
        $Self->False(
            0,
            'ImportExport package not installed'
        );

        return 1;
    }
}

my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
my $ImportExportObject   = $Kernel::OM->Get('Kernel::System::ImportExport');
my $ObjectBackendObject  = $Kernel::OM->Get('Kernel::System::ImportExport::ObjectBackend::ITSMConfigItem');
my $XMLObject            = $Kernel::OM->Get('Kernel::System::XML');

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# define needed variable
my $RandomID = $Helper->GetRandomID();

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# add some test templates for later checks
my @TemplateIDs;
my $Counter = 0;
for ( 1 .. 30 ) {

    # add a test template for later checks
    my $TemplateID = $ImportExportObject->TemplateAdd(
        Object  => 'ITSMConfigItem',
        Format  => 'UnitTest' . $Counter . $RandomID,
        Name    => 'UnitTest' . $Counter . $RandomID,
        ValidID => 1,
        UserID  => 1,
    );

    push @TemplateIDs, $TemplateID;

    $Counter++;
}

# ------------------------------------------------------------ #
# ObjectList test 1 (check CSV item)
# ------------------------------------------------------------ #

# get object list
my $ObjectList1 = $ImportExportObject->ObjectList();

# check object list
$Self->True(
    $ObjectList1 && ref $ObjectList1 eq 'HASH' && $ObjectList1->{ITSMConfigItem},
    "ObjectList() - ITSMConfigItem exists",
);

# ------------------------------------------------------------ #
# ObjectAttributesGet test 1 (check attribute hash)
# ------------------------------------------------------------ #

{

    # get object attributes
    my $ObjectAttributesGet1 = $ImportExportObject->ObjectAttributesGet(
        TemplateID => $TemplateIDs[0],
        UserID     => 1,
    );

    # check object attribute reference
    $Self->True(
        $ObjectAttributesGet1 && ref $ObjectAttributesGet1 eq 'ARRAY',
        "ObjectAttributesGet() - check array reference",
    );

    # get class list
    my $ClassList = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # define the reference hash
    my $ObjectAttributesGet1Reference = [
        {
            Key   => 'ClassID',
            Name  => 'Class',
            Input => {
                Type         => 'Selection',
                Data         => $ClassList,
                Required     => 1,
                Translation  => 0,
                PossibleNone => 1,
            },
        },
        {
            Key   => 'CountMax',
            Name  => 'Maximum number of one element',
            Input => {
                Type         => 'Text',
                ValueDefault => '10',
                Required     => 1,
                Regex        => qr{ \A \d+ \z }xms,
                Translation  => 0,
                Size         => 5,
                MaxLength    => 5,
                DataType     => 'IntegerBiggerThanZero',
            },
        },
        {
            'Input' => {
                'Type' => 'Checkbox'
            },
            'Name' => 'Empty fields indicate that the current values are kept',
            'Key'  => 'EmptyFieldsLeaveTheOldValues',
        }
    ];

    $Self->IsDeeply(
        $ObjectAttributesGet1,
        $ObjectAttributesGet1Reference,
        "ObjectAttributesGet() - attributes of the row are identical",
    );
}

# ------------------------------------------------------------ #
# ObjectAttributesGet test 2 (check with non existing template)
# ------------------------------------------------------------ #

# get object attributes
my $ObjectAttributesGet2 = $ImportExportObject->ObjectAttributesGet(
    TemplateID => $TemplateIDs[-1] + 1,
    UserID     => 1,
);

# check false return
$Self->False(
    $ObjectAttributesGet2,
    "ObjectAttributesGet() - check false return",
);

# ------------------------------------------------------------ #
# MappingObjectAttributesGet test 1 (check attribute hash)
# ------------------------------------------------------------ #

# get mapping object attributes
my $MappingObjectAttributesGet1 = $ImportExportObject->MappingObjectAttributesGet(
    TemplateID => $TemplateIDs[0],
    UserID     => 1,
);

# check mapping object attribute reference
$Self->True(
    $MappingObjectAttributesGet1 && ref $MappingObjectAttributesGet1 eq 'ARRAY',
    "MappingObjectAttributesGet() - check array reference",
);

# ------------------------------------------------------------ #
# MappingObjectAttributesGet test 2 (check with non existing template)
# ------------------------------------------------------------ #

# get mapping object attributes
my $MappingObjectAttributesGet2 = $ImportExportObject->MappingObjectAttributesGet(
    TemplateID => $TemplateIDs[-1] + 1,
    UserID     => 1,
);

# check false return
$Self->False(
    $MappingObjectAttributesGet2,
    "MappingObjectAttributesGet() - check false return",
);

# ------------------------------------------------------------ #
# make preparations to test ExportDataGet() and ImportDataSave()
# ------------------------------------------------------------ #

my $GeneralCatalogClass = 'UnitTest' . $RandomID;

# add a general catalog test list
for my $Name (qw(Test1 Test2 Test3 Test4)) {

    # add a new item
    my $ItemID = $GeneralCatalogObject->ItemAdd(
        Class   => $GeneralCatalogClass,
        Name    => $Name,
        ValidID => 1,
        UserID  => 1,
    );

    # check item id
    if ( !$ItemID ) {

        $Self->True(
            0,
            "Can't add new general catalog item.",
        );
    }
}

# define the first test definition (all provided data types)
my @ConfigItemPerlDefinitions;
$ConfigItemPerlDefinitions[0] = " [
    {
        Key        => 'Customer1',
        Name       => 'Customer 1',
        Searchable => 1,
        Input      => {
            Type => 'Customer',
        },
    },
    {
        Key        => 'Date1',
        Name       => 'Date 1',
        Searchable => 1,
        Input      => {
            Type => 'Date',
        },
    },
    {
        Key        => 'DateTime1',
        Name       => 'Date Time 1',
        Searchable => 1,
        Input      => {
            Type => 'DateTime',
        },
    },
    {
        Key   => 'Dummy1',
        Name  => 'Dummy 1',
        Input => {
            Type => 'Dummy',
        },
    },
    {
        Key        => 'GeneralCatalog1',
        Name       => 'GeneralCatalog 1',
        Searchable => 1,
        Input      => {
            Type  => 'GeneralCatalog',
            Class => '$GeneralCatalogClass',
        },
    },
    {
        Key        => 'Integer1',
        Name       => 'Integer 1',
        Searchable => 1,
        Input      => {
            Type => 'Integer',
        },
    },
    {
        Key        => 'Text1',
        Name       => 'Text 1',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 50,
        },
    },
    {
        Key        => 'TextArea1',
        Name       => 'TextArea 1',
        Searchable => 1,
        Input      => {
            Type => 'TextArea',
        },
    },
] ";

# define the second test definition (sub data types)
$ConfigItemPerlDefinitions[1] = " [
    {
        Key        => 'Main1',
        Name       => 'Main 1',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 50,
            Required  => 1,
        },
        CountMax => 10,
        Sub => [
            {
                Key        => 'Main1Sub1',
                Name       => 'Main 1 Sub 1',
                Searchable => 1,
                Input      => {
                    Type      => 'Text',
                    Size      => 50,
                    MaxLength => 50,
                },
                CountMax => 10,
                Sub => [
                    {
                        Key        => 'Main1Sub1SubSub1',
                        Name       => 'Main 1 Sub 1 SubSub 1',
                        Searchable => 1,
                        Input      => {
                            Type      => 'Text',
                            Size      => 50,
                            MaxLength => 50,
                        },
                        CountMax => 10,
                    },
                    {
                        Key        => 'Main1Sub1SubSub2',
                        Name       => 'Main 1 Sub 1 SubSub 2',
                        Searchable => 1,
                        Input      => {
                            Type => 'TextArea',
                        },
                        CountMax => 10,
                    },
                ],
            },
            {
                Key        => 'Main1Sub2',
                Name       => 'Main 1 Sub 2',
                Searchable => 1,
                Input      => {
                    Type => 'TextArea',
                },
                CountMax => 10,
            },
        ],
    },
    {
        Key        => 'Main2',
        Name       => 'Main 2',
        Searchable => 1,
        Input      => {
            Type => 'TextArea',
        },
        CountMax => 10,
        Sub => [
            {
                Key        => 'Main2Sub1',
                Name       => 'Main 2 Sub 1',
                Searchable => 1,
                Input      => {
                    Type      => 'Text',
                    Size      => 50,
                    MaxLength => 50,
                },
                CountMax => 10,
            },
            {
                Key        => 'Main2Sub2',
                Name       => 'Main 2 Sub 2',
                Searchable => 1,
                Input      => {
                    Type => 'TextArea',
                },
                CountMax => 10,
            },
        ],
    },
] ";

# Define the third test definition with required attribute 'Type'.
$ConfigItemPerlDefinitions[2] = " [
    {
        'Key' => 'Type',
        'Input' => {
            'Class' => '$GeneralCatalogClass',
            'Required' => 1,
            'Translation' => 1,
            'Type' => 'GeneralCatalog'
        },
        'Searchable' => 1,
        'CountMin' => 1,
        'CountDefault' => 1,
        'CountMax' => 3,
        'Name' => 'Type',
    },
] ";

my $YAMLObject = $Kernel::OM->Get('Kernel::System::YAML');

my @ConfigItemDefinitions;
for my $PerlDefinition (@ConfigItemPerlDefinitions) {
    my $YAMLDefinition = $YAMLObject->Dump(
        Data => eval $PerlDefinition,    ## no critic
    );
    push @ConfigItemDefinitions, $YAMLDefinition;
}

# add the test classes
my @ConfigItemClassIDs;
my @ConfigItemDefinitionIDs;
for my $Definition (@ConfigItemDefinitions) {

    # generate a random name
    my $ClassName = 'UnitTest' . $Helper->GetRandomID();

    # add an unittest config item class
    my $ClassID = $GeneralCatalogObject->ItemAdd(
        Class   => 'ITSM::ConfigItem::Class',
        Name    => $ClassName,
        ValidID => 1,
        UserID  => 1,
    );

    # check class id
    if ( !$ClassID ) {

        $Self->True(
            0,
            "Can't add new config item class.",
        );
    }

    push @ConfigItemClassIDs, $ClassID;

    # add a definition to the class
    my $DefinitionID = $ConfigItemObject->DefinitionAdd(
        ClassID    => $ClassID,
        Definition => $Definition,
        UserID     => 1,
    );

    # check definition id
    if ( !$DefinitionID ) {

        $Self->True(
            0,
            "Can't add new config item definition.",
        );
    }

    push @ConfigItemDefinitionIDs, $DefinitionID;
}

# create some random numbers
my @ConfigItemNumbers;
for ( 1 .. 10 ) {
    push @ConfigItemNumbers, $Helper->GetRandomNumber();
}

# get deployment state list
my $DeplStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %DeplStateListReverse = reverse %{$DeplStateList};

# get incident state list
my $InciStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %InciStateListReverse = reverse %{$InciStateList};

# get general catalog test list
my $GeneralCatalogList = $GeneralCatalogObject->ItemList(
    Class => $GeneralCatalogClass,
);
my %GeneralCatalogListReverse = reverse %{$GeneralCatalogList};

# define the test config items
my @ConfigItems = (

    # config item for all provided data types
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[0],
            ClassID => $ConfigItemClassIDs[0],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 1 Version 1',
                DefinitionID => $ConfigItemDefinitionIDs[0],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Customer1 => [
                                    undef,
                                    {
                                        Content => 'UnitTest',
                                    },
                                ],
                                Date1 => [
                                    undef,
                                    {
                                        Content => '2008-02-01',
                                    },
                                ],
                                DateTime1 => [
                                    undef,
                                    {
                                        Content => '2008-02-01 03:59',
                                    },
                                ],
                                GeneralCatalog1 => [
                                    undef,
                                    {
                                        Content => $GeneralCatalogListReverse{Test1},
                                    },
                                ],
                                Integer1 => [
                                    undef,
                                    {
                                        Content => '1',
                                    },
                                ],
                                Text1 => [
                                    undef,
                                    {
                                        Content => 'Test Text Test',
                                    },
                                ],
                                TextArea1 => [
                                    undef,
                                    {
                                        Content => "Test\nText Array\nTest",
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },

    # a second config item for all provided data types
    # (duplicate name of first version for search checks)
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[1],
            ClassID => $ConfigItemClassIDs[0],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 1 Version 1',    # duplicate name for tests
                DefinitionID => $ConfigItemDefinitionIDs[0],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Customer1 => [
                                    undef,
                                    {
                                        Content => 'UnitTest',
                                    },
                                ],
                                Date1 => [
                                    undef,
                                    {
                                        Content => '2008-02-01',
                                    },
                                ],
                                DateTime1 => [
                                    undef,
                                    {
                                        Content => '2008-02-01 03:59',
                                    },
                                ],
                                GeneralCatalog1 => [
                                    undef,
                                    {
                                        Content => $GeneralCatalogListReverse{Test1},
                                    },
                                ],
                                Integer1 => [
                                    undef,
                                    {
                                        Content => '1',
                                    },
                                ],
                                Text1 => [
                                    undef,
                                    {
                                        Content => 'Test Text Test',
                                    },
                                ],
                                TextArea1 => [
                                    undef,
                                    {
                                        Content => "Test\nText Array\nTest",
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
            {
                Name         => 'UnitTest - ConfigItem 2 Version 2',
                DefinitionID => $ConfigItemDefinitionIDs[0],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Customer1 => [
                                    undef,
                                    {
                                        Content => 'UnitTest2',
                                    },
                                ],
                                Date1 => [
                                    undef,
                                    {
                                        Content => '2008-02-02',
                                    },
                                ],
                                DateTime1 => [
                                    undef,
                                    {
                                        Content => '2008-02-02 03:59',
                                    },
                                ],
                                GeneralCatalog1 => [
                                    undef,
                                    {
                                        Content => $GeneralCatalogListReverse{Test2},
                                    },
                                ],
                                Integer1 => [
                                    undef,
                                    {
                                        Content => '2',
                                    },
                                ],
                                Text1 => [
                                    undef,
                                    {
                                        Content => 'Test Text Test2',
                                    },
                                ],
                                TextArea1 => [
                                    undef,
                                    {
                                        Content => "Test2\nText Array\nTest 2",
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },

    # config item for sub element tests
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[2],
            ClassID => $ConfigItemClassIDs[1],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 3 Version 1',
                DefinitionID => $ConfigItemDefinitionIDs[1],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Main1 => [
                                    undef,
                                    {
                                        Content   => 'Main1 (1)',
                                        Main1Sub1 => [
                                            undef,
                                            {
                                                Content          => 'Main1 (1) Sub1 (1)',
                                                Main1Sub1SubSub1 => [
                                                    undef,
                                                    {
                                                        Content => 'Main1 (1) Sub1 (1) SubSub1 (1)',
                                                    },
                                                    {
                                                        Content => 'Main1 (1) Sub1 (1) SubSub1 (2)',
                                                    },
                                                    {
                                                        Content => 'Main1 (1) Sub1 (1) SubSub1 (3)',
                                                    },
                                                ],
                                                Main1Sub1SubSub2 => [
                                                    undef,
                                                    {
                                                        Content => 'Main1 (1) Sub1 (1) SubSub2 (1)',
                                                    },
                                                ],
                                            },
                                            {
                                                Content          => 'Main1 (1) Sub1 (2)',
                                                Main1Sub1SubSub1 => [
                                                    undef,
                                                    {
                                                        Content => 'Main1 (1) Sub1 (2) SubSub1 (1)',
                                                    },
                                                ],
                                                Main1Sub1SubSub2 => [
                                                    undef,
                                                    {
                                                        Content => 'Main1 (1) Sub1 (2) SubSub2 (1)',
                                                    },
                                                    {
                                                        Content => 'Main1 (1) Sub1 (2) SubSub2 (2)',
                                                    },
                                                ],
                                            },
                                        ],
                                        Main1Sub2 => [
                                            undef,
                                            {
                                                Content => 'Main1 (1) Sub2 (1)',
                                            },
                                            {
                                                Content => 'Main1 (1) Sub2 (2)',
                                            },
                                        ],
                                    },
                                ],
                                Main2 => [
                                    undef,
                                    {
                                        Content   => 'Main2 (1)',
                                        Main2Sub1 => [
                                            undef,
                                            {
                                                Content => 'Main2 (1) Sub1 (1)',
                                            },
                                        ],
                                        Main2Sub2 => [
                                            undef,
                                            {
                                                'Content' => 'Main2 (1) Sub2 (1)',
                                            },
                                            {
                                                'Content' => 'Main2 (1) Sub2 (2)',
                                            },
                                        ],
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },

    # config item for sub element tests
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[3],
            ClassID => $ConfigItemClassIDs[1],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 4 Version 1',
                DefinitionID => $ConfigItemDefinitionIDs[1],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Main1 => [
                                    undef,
                                    {
                                        Content   => '',
                                        Main1Sub1 => [
                                            undef,
                                            {
                                                Content          => '',
                                                Main1Sub1SubSub1 => [
                                                    undef,
                                                    {
                                                        Content => '',
                                                    },
                                                ],
                                                Main1Sub1SubSub2 => [
                                                    undef,
                                                    {
                                                        Content => '',
                                                    },
                                                ],
                                            },
                                        ],
                                        Main1Sub2 => [
                                            undef,
                                            {
                                                Content => '',
                                            },
                                        ],
                                    },
                                ],
                                Main2 => [
                                    undef,
                                    {
                                        Content   => '',
                                        Main2Sub1 => [
                                            undef,
                                            {
                                                Content => '',
                                            },
                                        ],
                                        Main2Sub2 => [
                                            undef,
                                            {
                                                Content => '',
                                            },
                                        ],
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },

    # config item for special character tests
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[4],
            ClassID => $ConfigItemClassIDs[1],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 5 Version 1',
                DefinitionID => $ConfigItemDefinitionIDs[1],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Main1 => [
                                    undef,
                                    {
                                        Content   => '"";;::..--__##',
                                        Main1Sub1 => [
                                            undef,
                                            {
                                                Content          => 'Test;:_°^!"§$%&/()=?´`*+Test',
                                                Main1Sub1SubSub1 => [
                                                    undef,
                                                    {
                                                        Content => '><@~\'}{[]\\',
                                                    },
                                                ],
                                                Main1Sub1SubSub2 => [
                                                    undef,
                                                    {
                                                        Content => '><@~\'}{[]\\',
                                                    },
                                                ],
                                            },
                                        ],
                                        Main1Sub2 => [
                                            undef,
                                            {
                                                Content => 'Test;:_°^!"§$%&/()=?´`*+Test',
                                            },
                                        ],
                                    },
                                ],
                                Main2 => [
                                    undef,
                                    {
                                        Content   => '"";;::..--__##',
                                        Main2Sub1 => [
                                            undef,
                                            {
                                                Content => 'Test Test',
                                            },
                                        ],
                                        Main2Sub2 => [
                                            undef,
                                            {
                                                Content => "Test\nTest\tTest",
                                            },
                                        ],
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },

    # config item for UTF-8 tests
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[5],
            ClassID => $ConfigItemClassIDs[1],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 6 Version 1',
                DefinitionID => $ConfigItemDefinitionIDs[1],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Main1 => [
                                    undef,
                                    {
                                        Content   => 'ↂ ⅻ ⅛',
                                        Main1Sub1 => [
                                            undef,
                                            {
                                                Content          => '☄ ↮ ↹ →',
                                                Main1Sub1SubSub1 => [
                                                    undef,
                                                    {
                                                        Content => '₤ ₡ ₩ ₯ ₵',
                                                    },
                                                ],
                                                Main1Sub1SubSub2 => [
                                                    undef,
                                                    {
                                                        Content => '♊ ♈ ♉ ♊ ♋ ♍ ♑',
                                                    },
                                                ],
                                            },
                                        ],
                                        Main1Sub2 => [
                                            undef,
                                            {
                                                Content => '✈ ❤ ☮',
                                            },
                                        ],
                                    },
                                ],
                                Main2 => [
                                    undef,
                                    {
                                        Content   => 'Պ Մ Հ',
                                        Main2Sub1 => [
                                            undef,
                                            {
                                                Content => '® ©',
                                            },
                                        ],
                                        Main2Sub2 => [
                                            undef,
                                            {
                                                Content => 'か げ を',
                                            },
                                        ],
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },
);

# add the test config items
my @ConfigItemIDs;
for my $ConfigItem (@ConfigItems) {

    # add a config item
    my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        %{ $ConfigItem->{ConfigItem} },
    );

    # check config item id
    if ( !$ConfigItemID ) {

        $Self->True(
            0,
            "Can't add new config item.",
        );
    }

    push @ConfigItemIDs, $ConfigItemID;

    # add the versions
    for my $Version ( @{ $ConfigItem->{Versions} } ) {

        # add a version
        my $VersionID = $ConfigItemObject->VersionAdd(
            %{$Version},
            ConfigItemID => $ConfigItemID,
        );

        # check version id
        if ( !$VersionID ) {

            $Self->True(
                0,
                "Can't add new version.",
            );
        }
    }
}

# ------------------------------------------------------------ #
# define general ExportDataGet tests
# ------------------------------------------------------------ #

my @ExportDataTests = (

    # ImportDataGet doesn't contains all data (check required attributes)
    {
        SourceExportData => {
            ExportDataGet => {
                UserID => 1,
            },
        },
    },

    # ImportDataGet doesn't contains all data (check required attributes)
    {
        SourceExportData => {
            ExportDataGet => {
                TemplateID => $TemplateIDs[1],
            },
        },
    },

    # no existing template id is given (check return false)
    {
        SourceExportData => {
            ExportDataGet => {
                TemplateID => $TemplateIDs[-1] + 1,
                UserID     => 1,
            },
        },
    },

    # no class id is given (check return false)
    {
        SourceExportData => {
            ExportDataGet => {
                TemplateID => $TemplateIDs[2],
                UserID     => 1,
            },
        },
    },

    # invalid class id is given (check return false)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[-1] + 1,
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[2],
                UserID     => 1,
            },
        },
    },

    # mapping list is empty (check return false)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[3],
                UserID     => 1,
            },
        },
    },

    # all required values are given (number search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[0],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (name search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Name => 'UnitTest - ConfigItem 1 Version 1',
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (case insensitive name search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Name => 'unittest - configitem 1 version 1',
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (name and number search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[0],
                Name   => 'UnitTest - ConfigItem 1 Version 1',
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (deployment state search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                DeplStateIDs => $DeplStateListReverse{Production},
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
            [ $ConfigItemNumbers[1] ],
        ],
    },

    # all required values are given (incident state search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                InciStateIDs => $InciStateListReverse{Operational},
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
            [ $ConfigItemNumbers[1] ],
        ],
    },

    # all required values are given (combined search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Number       => $ConfigItemNumbers[0],
                Name         => 'UnitTest - ConfigItem 1 Version 1',
                DeplStateIDs => $DeplStateListReverse{Production},
                InciStateIDs => $InciStateListReverse{Operational},
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (XML data search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Date1           => '2008-02-01',
                TextArea1       => "Test\nText Array\nTest",
                Customer1       => 'UnitTest',
                Text1           => 'Test Text Test',
                DateTime1       => '2008-02-01 03:59',
                Integer1        => '1',
                GeneralCatalog1 => $GeneralCatalogListReverse{Test1},
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (combined all search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Number          => $ConfigItemNumbers[0],
                Name            => 'UnitTest - ConfigItem 1 Version 1',
                DeplStateIDs    => $DeplStateListReverse{Production},
                InciStateIDs    => $InciStateListReverse{Operational},
                Date1           => '2008-02-01',
                TextArea1       => "Test\nText Array\nTest",
                Customer1       => 'UnitTest',
                Text1           => 'Test Text Test',
                DateTime1       => '2008-02-01 03:59',
                Integer1        => '1',
                GeneralCatalog1 => $GeneralCatalogListReverse{Test1},
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (check the returned array)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'Dummy1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[0],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[6],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[0],
                'UnitTest - ConfigItem 1 Version 1',
                'Production',
                'Operational',
                'UnitTest',
                '2008-02-01',
                '2008-02-01 03:59',
                undef,
                'Test1',
                '1',
                'Test Text Test',
                "Test\nText Array\nTest",
            ],
        ],
    },

    # all required values are given (double element checks)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'Dummy1::1',
                },
                {
                    Key => 'Dummy1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[0],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[6],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[0],
                $ConfigItemNumbers[0],
                'UnitTest - ConfigItem 1 Version 1',
                'UnitTest - ConfigItem 1 Version 1',
                'Production',
                'Production',
                'Operational',
                'Operational',
                'UnitTest',
                'UnitTest',
                '2008-02-01',
                '2008-02-01',
                '2008-02-01 03:59',
                '2008-02-01 03:59',
                undef,
                undef,
                'Test1',
                'Test1',
                '1',
                '1',
                'Test Text Test',
                'Test Text Test',
                "Test\nText Array\nTest",
                "Test\nText Array\nTest",
            ],
        ],
    },

    # all required values are given (sub element checks)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::2',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::2',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[2],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[7],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[2],
                'UnitTest - ConfigItem 3 Version 1',
                'Production',
                'Operational',
                'Main1 (1)',
                'Main1 (1) Sub1 (1)',
                'Main1 (1) Sub1 (1) SubSub1 (1)',
                'Main1 (1) Sub1 (1) SubSub1 (2)',
                'Main1 (1) Sub1 (1) SubSub1 (3)',
                'Main1 (1) Sub1 (1) SubSub2 (1)',
                'Main1 (1) Sub1 (2)',
                'Main1 (1) Sub1 (2) SubSub1 (1)',
                'Main1 (1) Sub1 (2) SubSub2 (1)',
                'Main1 (1) Sub1 (2) SubSub2 (2)',
                'Main1 (1) Sub2 (1)',
                'Main1 (1) Sub2 (2)',
                'Main2 (1)',
                'Main2 (1) Sub1 (1)',
                'Main2 (1) Sub2 (1)',
                'Main2 (1) Sub2 (2)',
            ],
        ],
    },

    # all required values are given (sub element checks with undef values)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::4',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::3',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub2::3',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::2',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::2',
                },
                {
                    Key => 'Main2::1::Main2Sub2::3',
                },
                {
                    Key => 'Main2::2',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[2],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[7],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[2],
                'UnitTest - ConfigItem 3 Version 1',
                'Production',
                'Operational',
                'Main1 (1)',
                'Main1 (1) Sub1 (1)',
                'Main1 (1) Sub1 (1) SubSub1 (1)',
                'Main1 (1) Sub1 (1) SubSub1 (2)',
                'Main1 (1) Sub1 (1) SubSub1 (3)',
                undef,
                'Main1 (1) Sub1 (1) SubSub2 (1)',
                undef,
                'Main1 (1) Sub1 (2)',
                'Main1 (1) Sub1 (2) SubSub1 (1)',
                undef,
                'Main1 (1) Sub1 (2) SubSub2 (1)',
                'Main1 (1) Sub1 (2) SubSub2 (2)',
                undef,
                'Main1 (1) Sub2 (1)',
                'Main1 (1) Sub2 (2)',
                undef,
                'Main2 (1)',
                'Main2 (1) Sub1 (1)',
                undef,
                'Main2 (1) Sub2 (1)',
                'Main2 (1) Sub2 (2)',
                undef,
                undef,
            ],
        ],
    },

    # all required values are given (sub element checks with undef values and empty strings)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::4',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::3',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub2::3',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::2',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::2',
                },
                {
                    Key => 'Main2::1::Main2Sub2::3',
                },
                {
                    Key => 'Main2::2',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[3],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[7],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[3],
                'UnitTest - ConfigItem 4 Version 1',
                'Production',
                'Operational',
                '',
                '',
                '',
                undef,
                undef,
                undef,
                '',
                undef,
                undef,
                undef,
                undef,
                undef,
                undef,
                undef,
                '',
                undef,
                undef,
                '',
                '',
                undef,
                '',
                undef,
                undef,
                undef,
            ],
        ],
    },

    # all required values are given (special character checks)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[4],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[8],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[4],
                'UnitTest - ConfigItem 5 Version 1',
                'Production',
                'Operational',
                '"";;::..--__##',
                'Test;:_°^!"§$%&/()=?´`*+Test',
                '><@~\'}{[]\\',
                '><@~\'}{[]\\',
                'Test;:_°^!"§$%&/()=?´`*+Test',
                '"";;::..--__##',
                'Test Test',
                "Test\nTest\tTest",
            ],
        ],
    },

    # all required values are given (UTF-8 checks)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[5],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[9],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[5],
                'UnitTest - ConfigItem 6 Version 1',
                'Production',
                'Operational',
                'ↂ ⅻ ⅛',
                '☄ ↮ ↹ →',
                '₤ ₡ ₩ ₯ ₵',
                '♊ ♈ ♉ ♊ ♋ ♍ ♑',
                '✈ ❤ ☮',
                'Պ Մ Հ',
                '® ©',
                'か げ を',
            ],
        ],
    },
);

# ------------------------------------------------------------ #
# run general ExportDataGet tests
# ------------------------------------------------------------ #

my $ExportTestCount = 1;
TEST:
for my $Test (@ExportDataTests) {

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

        $Self->True(
            0,
            "ExportTest $ExportTestCount: No SourceExportData found for this test."
        );

        next TEST;
    }

    # set the object data
    if (
        $Test->{SourceExportData}->{ObjectData}
        && ref $Test->{SourceExportData}->{ObjectData} eq 'HASH'
        && $Test->{SourceExportData}->{ExportDataGet}->{TemplateID}
        )
    {

        # save object data
        $ImportExportObject->ObjectDataSave(
            TemplateID => $Test->{SourceExportData}->{ExportDataGet}->{TemplateID},
            ObjectData => $Test->{SourceExportData}->{ObjectData},
            UserID     => 1,
        );
    }

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

        # delete all existing mapping data
        $ImportExportObject->MappingDelete(
            TemplateID => $Test->{SourceExportData}->{ExportDataGet}->{TemplateID},
            UserID     => 1,
        );

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

            # add a new mapping row
            my $MappingID = $ImportExportObject->MappingAdd(
                TemplateID => $Test->{SourceExportData}->{ExportDataGet}->{TemplateID},
                UserID     => 1,
            );

            # add the mapping object data
            $ImportExportObject->MappingObjectDataSave(
                MappingID         => $MappingID,
                MappingObjectData => $MappingObjectData,
                UserID            => 1,
            );
        }
    }

    # add the search data
    if (
        $Test->{SourceExportData}->{SearchData}
        && ref $Test->{SourceExportData}->{SearchData} eq 'HASH'
        && $Test->{SourceExportData}->{ExportDataGet}->{TemplateID}
        )
    {

        # save search data
        $ImportExportObject->SearchDataSave(
            TemplateID => $Test->{SourceExportData}->{ExportDataGet}->{TemplateID},
            SearchData => $Test->{SourceExportData}->{SearchData},
            UserID     => 1,
        );
    }

    # get export data
    my $ExportData = $ObjectBackendObject->ExportDataGet(
        %{ $Test->{SourceExportData}->{ExportDataGet} },
    );

    if ( !$Test->{ReferenceExportData} ) {

        $Self->False(
            $ExportData,
            "ExportTest $ExportTestCount: ExportDataGet() - return false",
        );

        next TEST;
    }

    if ( ref $ExportData ne 'ARRAY' ) {

        # check array reference
        $Self->True(
            0,
            "ExportTest $ExportTestCount: ExportDataGet() - return value is an array reference",
        );

        next TEST;
    }

    # check number of rows
    $Self->Is(
        scalar @{$ExportData},
        scalar @{ $Test->{ReferenceExportData} },
        "ExportTest $ExportTestCount: ExportDataGet() - correct number of rows",
    );

    # check content of export data
    my $CounterRow = 0;
    ROW:
    for my $ExportRow ( @{$ExportData} ) {

        # extract reference row
        my $ReferenceRow = $Test->{ReferenceExportData}->[$CounterRow];

        if ( ref $ExportRow ne 'ARRAY' || ref $ReferenceRow ne 'ARRAY' ) {

            # check array reference
            $Self->True(
                0,
                "ExportTest $ExportTestCount: ExportDataGet() - export row and reference row matched",
            );

            next TEST;
        }

        # check number of columns
        $Self->Is(
            scalar @{$ExportRow},
            scalar @{$ReferenceRow},
            "ExportTest $ExportTestCount: ExportDataGet() - correct number of columns",
        );

        my $CounterColumn = 0;
        for my $Cell ( @{$ExportRow} ) {

            # set content if values are undef
            if ( !defined $Cell ) {
                $Cell = 'UNDEF-unittest';
            }
            if ( !defined $ReferenceRow->[$CounterColumn] ) {
                $ReferenceRow->[$CounterColumn] = 'UNDEF-unittest';
            }

            # check cell data
            $Self->Is(
                $Cell,
                $ReferenceRow->[$CounterColumn],
                "ExportTest $ExportTestCount: ExportDataGet() ",
            );

            $CounterColumn++;
        }

        $CounterRow++;
    }
}
continue {
    $ExportTestCount++;
}

# ------------------------------------------------------------ #
# define general ImportDataSave tests
# ------------------------------------------------------------ #

my @ImportDataTests = (

    # ImportDataSave doesn't contains all data (check required attributes)
    {
        SourceImportData => {
            ImportDataSave => {
                ImportDataRow => [],
                UserID        => 1,
            },
        },
    },

    # ImportDataSave doesn't contains all data (check required attributes)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID => $TemplateIDs[20],
                UserID     => 1,
            },
        },
    },

    # ImportDataSave doesn't contains all data (check required attributes)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID    => $TemplateIDs[20],
                ImportDataRow => [],
            },
        },
    },

    # import data row must be an array reference (check return false)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID    => $TemplateIDs[20],
                ImportDataRow => '',
                UserID        => 1,
            },
        },
    },

    # import data row must be an array reference (check return false)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID    => $TemplateIDs[20],
                ImportDataRow => {},
                UserID        => 1,
            },
        },
    },

    # no existing template id is given (check return false)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID    => $TemplateIDs[-1] + 1,
                ImportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # no class id is given (check return false)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID    => $TemplateIDs[21],
                ImportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # invalid class id is given (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[-1] + 1,
            },
            ImportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ImportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # mapping list is empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # more than one identifier with the same name (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '123', '321' ],
                UserID        => 1,
            },
        },
    },

    # identifier is empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [''],
                UserID        => 1,
            },
        },
    },

    # identifier is undef (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [undef],
                UserID        => 1,
            },
        },
    },

    # both identifiers are empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '', '' ],
                UserID        => 1,
            },
        },
    },

    # both identifiers are undef (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ undef, undef ],
                UserID        => 1,
            },
        },
    },

    # one identifiers is empty, one is undef (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '', undef ],
                UserID        => 1,
            },
        },
    },

    # one of the identifiers is empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '123', '' ],
                UserID        => 1,
            },
        },
    },

    # one of the identifiers is undef (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '123', undef ],
                UserID        => 1,
            },
        },
    },

    # one of the identifiers is empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '', '123' ],
                UserID        => 1,
            },
        },
    },

    # one of the identifiers is empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ undef, '123' ],
                UserID        => 1,
            },
        },
    },

    # TODO Add some identifier tests

    # empty name is given (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ImportDataRow => [ '', 'Production', 'Operational' ],
                UserID        => 1,
            },
        },
    },

    # invalid deployment state is given (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ImportDataRow => [ 'UnitTest - Importtest 1', 'Dummy', 'Operational' ],
                UserID        => 1,
            },
        },
    },

    # invalid incident state is given (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ImportDataRow => [ 'UnitTest - Importtest 2', 'Production', 'Dummy' ],
                UserID        => 1,
            },
        },
    },

    # all required values are given (a NEW config item must be created)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 3',
                    'Production',
                    'Operational',
                    'UnitTest',
                    '2008-06-05',
                    '2008-08-05 04:50',
                    'Test3',
                    '3',
                    'Test3 Text3 Test3',
                    "Test3\nTextArray3\nTest3",
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 1,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 3',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Customer1::1'       => 'UnitTest',
                'Date1::1'           => '2008-06-05',
                'DateTime1::1'       => '2008-08-05 04:50',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test3},
                'Integer1::1'        => '3',
                'Text1::1'           => 'Test3 Text3 Test3',
                'TextArea1::1'       => "Test3\nTextArray3\nTest3",
            },
        },
    },

    # all required values are given (a second NEW config item must be created)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 4',
                    'Production',
                    'Operational',
                    'UnitTest',
                    '2008-09-05',
                    '2008-12-05 04:50',
                    'Test4',
                    '4',
                    'Test4 Text4 Test4',
                    "Test4\nTextArray4\nTest4",
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 1,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 4',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Customer1::1'       => 'UnitTest',
                'Date1::1'           => '2008-09-05',
                'DateTime1::1'       => '2008-12-05 04:50',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test4},
                'Integer1::1'        => '4',
                'Text1::1'           => 'Test4 Text4 Test4',
                'TextArea1::1'       => "Test4\nTextArray4\nTest4",
            },
        },
    },

    # all required values are given (a new version must be added to first test config item)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[26],
                ImportDataRow => [
                    $ConfigItemNumbers[0],
                    'UnitTest - ConfigItem 1 Version 2',
                    'Pilot',
                    'Incident',
                    'UnitTest',
                    '2008-02-02',
                    '2008-02-02 03:59',
                    'Test2',
                    '2',
                    'Test Text UPDATE1 Test',
                    "Test\nText Array UPDATE1\nTest",
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 2,
            LastVersion   => {
                Name                 => 'UnitTest - ConfigItem 1 Version 2',
                DeplState            => 'Pilot',
                InciState            => 'Incident',
                'Customer1::1'       => 'UnitTest',
                'Date1::1'           => '2008-02-02',
                'DateTime1::1'       => '2008-02-02 03:59',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test2},
                'Integer1::1'        => '2',
                'Text1::1'           => 'Test Text UPDATE1 Test',
                'TextArea1::1'       => "Test\nText Array UPDATE1\nTest",
            },
        },
    },

    # all required values are given (a new version must be added to first test config item again)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[27],
                ImportDataRow => [
                    $ConfigItemNumbers[0],
                    'UnitTest - ConfigItem 1 Version 3',
                    'Repair',
                    'Operational',
                    'UnitTest',
                    '2008-02-03',
                    '2008-02-03 03:59',
                    'Test3',
                    '3',
                    'Test Text UPDATE2 Test',
                    "Test\nText Array UPDATE2\nTest",
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 3,
            LastVersion   => {
                Name                 => 'UnitTest - ConfigItem 1 Version 3',
                DeplState            => 'Repair',
                InciState            => 'Operational',
                'Customer1::1'       => 'UnitTest',
                'Date1::1'           => '2008-02-03',
                'DateTime1::1'       => '2008-02-03 03:59',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test3},
                'Integer1::1'        => '3',
                'Text1::1'           => 'Test Text UPDATE2 Test',
                'TextArea1::1'       => "Test\nText Array UPDATE2\nTest",
            },
        },
    },

    # all required values are given (a new version must be added to third test config item)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::2',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::2',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[27],
                ImportDataRow => [
                    $ConfigItemNumbers[2],
                    'UnitTest - ConfigItem 3 Version 2',
                    'Production',
                    'Operational',
                    'Main1 (1)',
                    'Main1 (1) Main1Sub1 (1)',
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (1)',
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (2)',
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (3)',
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub2 (1)',
                    'Main1 (1) Main1Sub1 (2)',
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub1 (1)',
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (1)',
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (2)',
                    'Main1 (1) Main1Sub2 (1)',
                    'Main1 (1) Main1Sub2 (2)',
                    'Main2 (1)',
                    'Main2 (1) Main2Sub1 (1)',
                    'Main2 (1) Main2Sub2 (1)',
                    'Main2 (1) Main2Sub2 (2)',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 2,
            LastVersion   => {
                Name                     => 'UnitTest - ConfigItem 3 Version 2',
                DeplState                => 'Production',
                InciState                => 'Operational',
                'Main1::1'               => 'Main1 (1)',
                'Main1::1::Main1Sub1::1' => 'Main1 (1) Main1Sub1 (1)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (1)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (2)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (3)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub2 (1)',
                'Main1::1::Main1Sub1::2' => 'Main1 (1) Main1Sub1 (2)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub1 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (2)',
                'Main1::1::Main1Sub2::1' => 'Main1 (1) Main1Sub2 (1)',
                'Main1::1::Main1Sub2::2' => 'Main1 (1) Main1Sub2 (2)',
                'Main2::1'               => 'Main2 (1)',
                'Main2::1::Main2Sub1::1' => 'Main2 (1) Main2Sub1 (1)',
                'Main2::1::Main2Sub2::1' => 'Main2 (1) Main2Sub2 (1)',
                'Main2::1::Main2Sub2::2' => 'Main2 (1) Main2Sub2 (2)',
            },
        },
    },

    # all required values are given (special character checks)
    # In 'UnitTest - ConfigItem 3 Version 2' 16 Attributes were imported,
    # so there will be 8 lingering attributes.
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[27],
                ImportDataRow => [
                    $ConfigItemNumbers[2],
                    'UnitTest - ConfigItem 3 Version 3',
                    'Production',
                    'Operational',
                    '"";;::..--__##',
                    'Test;:_°^!"§$%&/()=?´`*+Test',
                    '><@~\'}{[]\\',
                    '><@~\'}{[]\\',
                    'Test;:_°^!"§$%&/()=?´`*+Test',
                    '"";;::..--__##',
                    'Test Test',
                    "Test\nTest\tTest",
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 3,
            LastVersion   => {
                Name                                          => 'UnitTest - ConfigItem 3 Version 3',
                DeplState                                     => 'Production',
                InciState                                     => 'Operational',
                'Main1::1'                                    => '"";;::..--__##',
                'Main1::1::Main1Sub1::1'                      => 'Test;:_°^!"§$%&/()=?´`*+Test',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1' => '><@~\'}{[]\\',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1' => '><@~\'}{[]\\',
                'Main1::1::Main1Sub2::1'                      => 'Test;:_°^!"§$%&/()=?´`*+Test',
                'Main2::1'                                    => '"";;::..--__##',
                'Main2::1::Main2Sub1::1'                      => 'Test Test',
                'Main2::1::Main2Sub2::1'                      => "Test\nTest\tTest",

                # lingering from 'UnitTest - ConfigItem 3 Version 2',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (2)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (3)',
                'Main1::1::Main1Sub1::2' => 'Main1 (1) Main1Sub1 (2)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub1 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (2)',
                'Main1::1::Main1Sub2::2' => 'Main1 (1) Main1Sub2 (2)',
                'Main2::1::Main2Sub2::2' => 'Main2 (1) Main2Sub2 (2)',
            },
        },
    },

    # all required values are given (UTF-8 checks)
    # In 'UnitTest - ConfigItem 3 Version 2' 16 Attributes were imported,
    # so there will be 8 lingering attributes.
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[27],
                ImportDataRow => [
                    $ConfigItemNumbers[2],
                    'UnitTest - ConfigItem 3 Version 4',
                    'Production',
                    'Operational',
                    'Ϋ δ λ',
                    'π χ Ϙ',
                    'Ϻ ϱ Ϯ',
                    'ɯ ʓ ʠ',
                    'ʬ ʯ',
                    'й ф щ',
                    'њ ё',
                    'Ѭ Ѧ',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 4,
            LastVersion   => {
                Name                                          => 'UnitTest - ConfigItem 3 Version 4',
                DeplState                                     => 'Production',
                InciState                                     => 'Operational',
                'Main1::1'                                    => 'Ϋ δ λ',
                'Main1::1::Main1Sub1::1'                      => 'π χ Ϙ',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1' => 'Ϻ ϱ Ϯ',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1' => 'ɯ ʓ ʠ',
                'Main1::1::Main1Sub2::1'                      => 'ʬ ʯ',
                'Main2::1'                                    => 'й ф щ',
                'Main2::1::Main2Sub1::1'                      => 'њ ё',
                'Main2::1::Main2Sub2::1'                      => 'Ѭ Ѧ',

                # lingering from 'UnitTest - ConfigItem 3 Version 2',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (2)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (3)',
                'Main1::1::Main1Sub1::2' => 'Main1 (1) Main1Sub1 (2)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub1 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (2)',
                'Main1::1::Main1Sub2::2' => 'Main1 (1) Main1Sub2 (2)',
                'Main2::1::Main2Sub2::2' => 'Main2 (1) Main2Sub2 (2)',
            },
        },
    },

    # a simple import for testing the overriding behavior of empty values
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    'Importtest 5 for behavior of empty values',
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 1,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => 'Importtest 5 for behavior of empty values',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an empty value for Text1, with EmptyFieldsLeaveTheOldValues turned on
    # no new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '',
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 1,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => 'Importtest 5 for behavior of empty values',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import undef for Text1, with EmptyFieldsLeaveTheOldValues turned on
    # no new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    undef,
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 1,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => 'Importtest 5 for behavior of empty values',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an empty value for Text1, with EmptyFieldsLeaveTheOldValues turned off
    # a new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => '',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '',
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 2,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => '',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import a single space value for Text1, with EmptyFieldsLeaveTheOldValues turned on
    # a new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => '',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    ' ',
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 3,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => ' ',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import the string '0' value for Text1, with EmptyFieldsLeaveTheOldValues turned on
    # a new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => '',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '0',
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 4,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => '0',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an empty value for GeneralCatalog1, with EmptyFieldsLeaveTheOldValues turned on
    # no new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '',
                    '',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 4,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => '0',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an invalid value for GeneralCatalog1, with EmptyFieldsLeaveTheOldValues turned on
    # the import should fail
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '',
                    'non-existent general catalog entry',
                ],
                UserID => 1,
            },
        },
    },

    # import an invalid value for GeneralCatalog1, with EmptyFieldsLeaveTheOldValues turned off
    # the import should fail
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => '',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '',
                    'non-existent general catalog entry',
                ],
                UserID => 1,
            },
        },
    },

    # import an empty value for DeplState, with EmptyFieldsLeaveTheOldValues turned on
    # no new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    '',
                    'Operational',
                    '',
                    '',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 4,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => '0',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an invalid value for DeplState, with EmptyFieldsLeaveTheOldValues turned on
    # an error should be generated
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'invalid deployment state',
                    'Operational',
                    '',
                    '',
                ],
                UserID => 1,
            },
        },
    },

    # import an empty value for InciState, with EmptyFieldsLeaveTheOldValues turned on
    # no new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    '',
                    '',
                    '',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 4,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => '0',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an invalid value for InciState, with EmptyFieldsLeaveTheOldValues turned on
    # an error should be generated
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'invalid incident state',
                    '',
                    '',
                ],
                UserID => 1,
            },
        },
    },

    # Import without required attribute 'Type', an error should be generated (see bug#14098).
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[2],
                EmptyFieldsLeaveTheOldValues => '',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                ],
                UserID => 1,
            },
        },
    },
);

# ------------------------------------------------------------ #
# run general ExportDataGet tests
# ------------------------------------------------------------ #

my $ImportTestCount = 1;
TEST:
for my $Test (@ImportDataTests) {

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

        $Self->True(
            0,
            "ImportTest $ImportTestCount: No SourceImportData found for this test."
        );

        next TEST;
    }

    # set the object data
    if (
        $Test->{SourceImportData}->{ObjectData}
        && ref $Test->{SourceImportData}->{ObjectData} eq 'HASH'
        && $Test->{SourceImportData}->{ImportDataSave}->{TemplateID}
        )
    {

        # save object data
        $ImportExportObject->ObjectDataSave(
            TemplateID => $Test->{SourceImportData}->{ImportDataSave}->{TemplateID},
            ObjectData => $Test->{SourceImportData}->{ObjectData},
            UserID     => 1,
        );
    }

    # set the mapping object data
    if (
        $Test->{SourceImportData}->{MappingObjectData}
        && ref $Test->{SourceImportData}->{MappingObjectData} eq 'ARRAY'
        && $Test->{SourceImportData}->{ImportDataSave}->{TemplateID}
        )
    {

        # delete all existing mapping data
        $ImportExportObject->MappingDelete(
            TemplateID => $Test->{SourceImportData}->{ImportDataSave}->{TemplateID},
            UserID     => 1,
        );

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

            # add a new mapping row
            my $MappingID = $ImportExportObject->MappingAdd(
                TemplateID => $Test->{SourceImportData}->{ImportDataSave}->{TemplateID},
                UserID     => 1,
            );

            # add the mapping object data
            $ImportExportObject->MappingObjectDataSave(
                MappingID         => $MappingID,
                MappingObjectData => $MappingObjectData,
                UserID            => 1,
            );
        }
    }

    # import data save
    my ( $ConfigItemID, $RetCode ) = $ObjectBackendObject->ImportDataSave(
        %{ $Test->{SourceImportData}->{ImportDataSave} },
        Counter => $ImportTestCount,
    );

    if ( !$Test->{ReferenceImportData} ) {

        $Self->False(
            $ConfigItemID,
            "ImportTest $ImportTestCount: ImportDataSave() - return no ConfigItemID"
        );
        $Self->False(
            $RetCode,
            "ImportTest $ImportTestCount: ImportDataSave() - return no RetCode"
        );

        next TEST;
    }

    $Self->True(
        $ConfigItemID,
        "ImportTest $ImportTestCount: ImportDataSave() - return ConfigItemID"
    );
    $Self->True(
        $RetCode,
        "ImportTest $ImportTestCount: ImportDataSave() - return RetCode"
    );

    # get the version list
    my $VersionList = $ConfigItemObject->VersionList(
        ConfigItemID => $ConfigItemID,
    ) // [];

    # check number of versions
    $Self->Is(
        scalar @{$VersionList},
        $Test->{ReferenceImportData}->{VersionNumber} || 0,
        "ImportTest $ImportTestCount: ImportDataSave() - correct number of versions",
    );

    # get the last version
    my $VersionData = $ConfigItemObject->VersionGet(
        ConfigItemID => $ConfigItemID,
        XMLDataGet   => 1,
    );

    # translate xmldata in a 2d hash
    my %XMLHash = $XMLObject->XMLHash2D(
        XMLHash => $VersionData->{XMLData},
    );

    # clean the xml hash
    KEY:
    for my $Key ( sort keys %XMLHash ) {

        next KEY if $Key =~ m{ \{'Content'\} \z }xms;

        delete $XMLHash{$Key};
    }

    # check general elements
    ELEMENT:
    for my $Element (qw(Number Name DeplState InciState)) {

        next ELEMENT if !exists $Test->{ReferenceImportData}->{LastVersion}->{$Element};

        # set content if values are undef
        if ( !defined $Test->{ReferenceImportData}->{LastVersion}->{$Element} ) {
            $Test->{ReferenceImportData}->{LastVersion}->{$Element} = 'UNDEF-unittest';
        }
        if ( !defined $Test->{ReferenceImportData}->{LastVersion}->{$Element} ) {
            $Test->{ReferenceImportData}->{LastVersion}->{$Element} = 'UNDEF-unittest';
        }

        # check element
        $Self->Is(
            $VersionData->{$Element},
            $Test->{ReferenceImportData}->{LastVersion}->{$Element},
            "ImportTest $ImportTestCount: ImportDataSave() $Element is identical",
        );

        delete $Test->{ReferenceImportData}->{LastVersion}->{$Element};
    }

    # check number of XML elements
    $Self->Is(
        scalar keys %XMLHash,
        scalar keys %{ $Test->{ReferenceImportData}->{LastVersion} },
        "ImportTest $ImportTestCount: ImportDataSave() - correct number of XML elements",
    );

    # check XML elements
    ELEMENT:
    for my $Key ( sort keys %{ $Test->{ReferenceImportData}->{LastVersion} } ) {

        # duplicate key
        my $XMLKey = $Key;

        # prepare key
        my $Counter = 0;
        while ( $XMLKey =~ m{ :: }xms ) {

            if ( $Counter % 2 ) {
                $XMLKey =~ s{ :: }{]\{'}xms;
            }
            else {
                $XMLKey =~ s{ :: }{'\}[}xms;
            }

            $Counter++;
        }

        next ELEMENT if !exists $XMLHash{ '[1]{\'Version\'}[1]{\'' . $XMLKey . ']{\'Content\'}' };

        # set content if values are undef
        if ( !defined $XMLHash{ '[1]{\'Version\'}[1]{\'' . $XMLKey . ']{\'Content\'}' } ) {
            $XMLHash{ '[1]{\'Version\'}[1]{\'' . $XMLKey . ']{\'Content\'}' } = 'UNDEF-unittest';
        }
        if ( !defined $Test->{ReferenceImportData}->{LastVersion}->{$Key} ) {
            $Test->{ReferenceImportData}->{LastVersion}->{$Key} = 'UNDEF-unittest';
        }

        # check XML element
        $Self->Is(
            $XMLHash{ '[1]{\'Version\'}[1]{\'' . $XMLKey . ']{\'Content\'}' },
            $Test->{ReferenceImportData}->{LastVersion}->{$Key},
            "ImportTest $ImportTestCount: ImportDataSave() $Key is identical",
        );
    }
}
continue {
    $ImportTestCount++;
}

# cleanup is done by RestoreDatabase

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

# run this test only if the ImportExport package is installed
{
    # get ImportExport module directory
    my $ImportExportModule = $Kernel::OM->Get('Kernel::Config')->Get('Home') . '/Kernel/System/ImportExport.pm';

    # Return early if ImportExport package is not installed.
    if ( !-e $ImportExportModule ) {
        $Self->False(
            0,
            'ImportExport package not installed'
        );

        return 1;
    }
}

my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
my $ImportExportObject   = $Kernel::OM->Get('Kernel::System::ImportExport');
my $ObjectBackendObject  = $Kernel::OM->Get('Kernel::System::ImportExport::ObjectBackend::ITSMConfigItem');
my $XMLObject            = $Kernel::OM->Get('Kernel::System::XML');

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# define needed variable
my $RandomID = $Helper->GetRandomID();

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# add some test templates for later checks
my @TemplateIDs;
my $Counter = 0;
for ( 1 .. 30 ) {

    # add a test template for later checks
    my $TemplateID = $ImportExportObject->TemplateAdd(
        Object  => 'ITSMConfigItem',
        Format  => 'UnitTest' . $Counter . $RandomID,
        Name    => 'UnitTest' . $Counter . $RandomID,
        ValidID => 1,
        UserID  => 1,
    );

    push @TemplateIDs, $TemplateID;

    $Counter++;
}

# ------------------------------------------------------------ #
# ObjectList test 1 (check CSV item)
# ------------------------------------------------------------ #

# get object list
my $ObjectList1 = $ImportExportObject->ObjectList();

# check object list
$Self->True(
    $ObjectList1 && ref $ObjectList1 eq 'HASH' && $ObjectList1->{ITSMConfigItem},
    "ObjectList() - ITSMConfigItem exists",
);

# ------------------------------------------------------------ #
# ObjectAttributesGet test 1 (check attribute hash)
# ------------------------------------------------------------ #

{

    # get object attributes
    my $ObjectAttributesGet1 = $ImportExportObject->ObjectAttributesGet(
        TemplateID => $TemplateIDs[0],
        UserID     => 1,
    );

    # check object attribute reference
    $Self->True(
        $ObjectAttributesGet1 && ref $ObjectAttributesGet1 eq 'ARRAY',
        "ObjectAttributesGet() - check array reference",
    );

    # get class list
    my $ClassList = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # define the reference hash
    my $ObjectAttributesGet1Reference = [
        {
            Key   => 'ClassID',
            Name  => 'Class',
            Input => {
                Type         => 'Selection',
                Data         => $ClassList,
                Required     => 1,
                Translation  => 0,
                PossibleNone => 1,
            },
        },
        {
            Key   => 'CountMax',
            Name  => 'Maximum number of one element',
            Input => {
                Type         => 'Text',
                ValueDefault => '10',
                Required     => 1,
                Regex        => qr{ \A \d+ \z }xms,
                Translation  => 0,
                Size         => 5,
                MaxLength    => 5,
                DataType     => 'IntegerBiggerThanZero',
            },
        },
        {
            'Input' => {
                'Type' => 'Checkbox'
            },
            'Name' => 'Empty fields indicate that the current values are kept',
            'Key'  => 'EmptyFieldsLeaveTheOldValues',
        }
    ];

    $Self->IsDeeply(
        $ObjectAttributesGet1,
        $ObjectAttributesGet1Reference,
        "ObjectAttributesGet() - attributes of the row are identical",
    );
}

# ------------------------------------------------------------ #
# ObjectAttributesGet test 2 (check with non existing template)
# ------------------------------------------------------------ #

# get object attributes
my $ObjectAttributesGet2 = $ImportExportObject->ObjectAttributesGet(
    TemplateID => $TemplateIDs[-1] + 1,
    UserID     => 1,
);

# check false return
$Self->False(
    $ObjectAttributesGet2,
    "ObjectAttributesGet() - check false return",
);

# ------------------------------------------------------------ #
# MappingObjectAttributesGet test 1 (check attribute hash)
# ------------------------------------------------------------ #

# get mapping object attributes
my $MappingObjectAttributesGet1 = $ImportExportObject->MappingObjectAttributesGet(
    TemplateID => $TemplateIDs[0],
    UserID     => 1,
);

# check mapping object attribute reference
$Self->True(
    $MappingObjectAttributesGet1 && ref $MappingObjectAttributesGet1 eq 'ARRAY',
    "MappingObjectAttributesGet() - check array reference",
);

# ------------------------------------------------------------ #
# MappingObjectAttributesGet test 2 (check with non existing template)
# ------------------------------------------------------------ #

# get mapping object attributes
my $MappingObjectAttributesGet2 = $ImportExportObject->MappingObjectAttributesGet(
    TemplateID => $TemplateIDs[-1] + 1,
    UserID     => 1,
);

# check false return
$Self->False(
    $MappingObjectAttributesGet2,
    "MappingObjectAttributesGet() - check false return",
);

# ------------------------------------------------------------ #
# make preparations to test ExportDataGet() and ImportDataSave()
# ------------------------------------------------------------ #

my $GeneralCatalogClass = 'UnitTest' . $RandomID;

# add a general catalog test list
for my $Name (qw(Test1 Test2 Test3 Test4)) {

    # add a new item
    my $ItemID = $GeneralCatalogObject->ItemAdd(
        Class   => $GeneralCatalogClass,
        Name    => $Name,
        ValidID => 1,
        UserID  => 1,
    );

    # check item id
    if ( !$ItemID ) {

        $Self->True(
            0,
            "Can't add new general catalog item.",
        );
    }
}

# define the first test definition (all provided data types)
my @ConfigItemPerlDefinitions;
$ConfigItemPerlDefinitions[0] = " [
    {
        Key        => 'Customer1',
        Name       => 'Customer 1',
        Searchable => 1,
        Input      => {
            Type => 'Customer',
        },
    },
    {
        Key        => 'Date1',
        Name       => 'Date 1',
        Searchable => 1,
        Input      => {
            Type => 'Date',
        },
    },
    {
        Key        => 'DateTime1',
        Name       => 'Date Time 1',
        Searchable => 1,
        Input      => {
            Type => 'DateTime',
        },
    },
    {
        Key   => 'Dummy1',
        Name  => 'Dummy 1',
        Input => {
            Type => 'Dummy',
        },
    },
    {
        Key        => 'GeneralCatalog1',
        Name       => 'GeneralCatalog 1',
        Searchable => 1,
        Input      => {
            Type  => 'GeneralCatalog',
            Class => '$GeneralCatalogClass',
        },
    },
    {
        Key        => 'Integer1',
        Name       => 'Integer 1',
        Searchable => 1,
        Input      => {
            Type => 'Integer',
        },
    },
    {
        Key        => 'Text1',
        Name       => 'Text 1',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 50,
        },
    },
    {
        Key        => 'TextArea1',
        Name       => 'TextArea 1',
        Searchable => 1,
        Input      => {
            Type => 'TextArea',
        },
    },
] ";

# define the second test definition (sub data types)
$ConfigItemPerlDefinitions[1] = " [
    {
        Key        => 'Main1',
        Name       => 'Main 1',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 50,
            Required  => 1,
        },
        CountMax => 10,
        Sub => [
            {
                Key        => 'Main1Sub1',
                Name       => 'Main 1 Sub 1',
                Searchable => 1,
                Input      => {
                    Type      => 'Text',
                    Size      => 50,
                    MaxLength => 50,
                },
                CountMax => 10,
                Sub => [
                    {
                        Key        => 'Main1Sub1SubSub1',
                        Name       => 'Main 1 Sub 1 SubSub 1',
                        Searchable => 1,
                        Input      => {
                            Type      => 'Text',
                            Size      => 50,
                            MaxLength => 50,
                        },
                        CountMax => 10,
                    },
                    {
                        Key        => 'Main1Sub1SubSub2',
                        Name       => 'Main 1 Sub 1 SubSub 2',
                        Searchable => 1,
                        Input      => {
                            Type => 'TextArea',
                        },
                        CountMax => 10,
                    },
                ],
            },
            {
                Key        => 'Main1Sub2',
                Name       => 'Main 1 Sub 2',
                Searchable => 1,
                Input      => {
                    Type => 'TextArea',
                },
                CountMax => 10,
            },
        ],
    },
    {
        Key        => 'Main2',
        Name       => 'Main 2',
        Searchable => 1,
        Input      => {
            Type => 'TextArea',
        },
        CountMax => 10,
        Sub => [
            {
                Key        => 'Main2Sub1',
                Name       => 'Main 2 Sub 1',
                Searchable => 1,
                Input      => {
                    Type      => 'Text',
                    Size      => 50,
                    MaxLength => 50,
                },
                CountMax => 10,
            },
            {
                Key        => 'Main2Sub2',
                Name       => 'Main 2 Sub 2',
                Searchable => 1,
                Input      => {
                    Type => 'TextArea',
                },
                CountMax => 10,
            },
        ],
    },
] ";

# Define the third test definition with required attribute 'Type'.
$ConfigItemPerlDefinitions[2] = " [
    {
        'Key' => 'Type',
        'Input' => {
            'Class' => '$GeneralCatalogClass',
            'Required' => 1,
            'Translation' => 1,
            'Type' => 'GeneralCatalog'
        },
        'Searchable' => 1,
        'CountMin' => 1,
        'CountDefault' => 1,
        'CountMax' => 3,
        'Name' => 'Type',
    },
] ";

my $YAMLObject = $Kernel::OM->Get('Kernel::System::YAML');

my @ConfigItemDefinitions;
for my $PerlDefinition (@ConfigItemPerlDefinitions) {
    my $YAMLDefinition = $YAMLObject->Dump(
        Data => eval $PerlDefinition,    ## no critic
    );
    push @ConfigItemDefinitions, $YAMLDefinition;
}

# add the test classes
my @ConfigItemClassIDs;
my @ConfigItemDefinitionIDs;
for my $Definition (@ConfigItemDefinitions) {

    # generate a random name
    my $ClassName = 'UnitTest' . $Helper->GetRandomID();

    # add an unittest config item class
    my $ClassID = $GeneralCatalogObject->ItemAdd(
        Class   => 'ITSM::ConfigItem::Class',
        Name    => $ClassName,
        ValidID => 1,
        UserID  => 1,
    );

    # check class id
    if ( !$ClassID ) {

        $Self->True(
            0,
            "Can't add new config item class.",
        );
    }

    push @ConfigItemClassIDs, $ClassID;

    # add a definition to the class
    my $DefinitionID = $ConfigItemObject->DefinitionAdd(
        ClassID    => $ClassID,
        Definition => $Definition,
        UserID     => 1,
    );

    # check definition id
    if ( !$DefinitionID ) {

        $Self->True(
            0,
            "Can't add new config item definition.",
        );
    }

    push @ConfigItemDefinitionIDs, $DefinitionID;
}

# create some random numbers
my @ConfigItemNumbers;
for ( 1 .. 10 ) {
    push @ConfigItemNumbers, $Helper->GetRandomNumber();
}

# get deployment state list
my $DeplStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::DeploymentState',
);
my %DeplStateListReverse = reverse %{$DeplStateList};

# get incident state list
my $InciStateList = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::Core::IncidentState',
);
my %InciStateListReverse = reverse %{$InciStateList};

# get general catalog test list
my $GeneralCatalogList = $GeneralCatalogObject->ItemList(
    Class => $GeneralCatalogClass,
);
my %GeneralCatalogListReverse = reverse %{$GeneralCatalogList};

# define the test config items
my @ConfigItems = (

    # config item for all provided data types
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[0],
            ClassID => $ConfigItemClassIDs[0],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 1 Version 1',
                DefinitionID => $ConfigItemDefinitionIDs[0],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Customer1 => [
                                    undef,
                                    {
                                        Content => 'UnitTest',
                                    },
                                ],
                                Date1 => [
                                    undef,
                                    {
                                        Content => '2008-02-01',
                                    },
                                ],
                                DateTime1 => [
                                    undef,
                                    {
                                        Content => '2008-02-01 03:59',
                                    },
                                ],
                                GeneralCatalog1 => [
                                    undef,
                                    {
                                        Content => $GeneralCatalogListReverse{Test1},
                                    },
                                ],
                                Integer1 => [
                                    undef,
                                    {
                                        Content => '1',
                                    },
                                ],
                                Text1 => [
                                    undef,
                                    {
                                        Content => 'Test Text Test',
                                    },
                                ],
                                TextArea1 => [
                                    undef,
                                    {
                                        Content => "Test\nText Array\nTest",
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },

    # a second config item for all provided data types
    # (duplicate name of first version for search checks)
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[1],
            ClassID => $ConfigItemClassIDs[0],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 1 Version 1',    # duplicate name for tests
                DefinitionID => $ConfigItemDefinitionIDs[0],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Customer1 => [
                                    undef,
                                    {
                                        Content => 'UnitTest',
                                    },
                                ],
                                Date1 => [
                                    undef,
                                    {
                                        Content => '2008-02-01',
                                    },
                                ],
                                DateTime1 => [
                                    undef,
                                    {
                                        Content => '2008-02-01 03:59',
                                    },
                                ],
                                GeneralCatalog1 => [
                                    undef,
                                    {
                                        Content => $GeneralCatalogListReverse{Test1},
                                    },
                                ],
                                Integer1 => [
                                    undef,
                                    {
                                        Content => '1',
                                    },
                                ],
                                Text1 => [
                                    undef,
                                    {
                                        Content => 'Test Text Test',
                                    },
                                ],
                                TextArea1 => [
                                    undef,
                                    {
                                        Content => "Test\nText Array\nTest",
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
            {
                Name         => 'UnitTest - ConfigItem 2 Version 2',
                DefinitionID => $ConfigItemDefinitionIDs[0],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Customer1 => [
                                    undef,
                                    {
                                        Content => 'UnitTest2',
                                    },
                                ],
                                Date1 => [
                                    undef,
                                    {
                                        Content => '2008-02-02',
                                    },
                                ],
                                DateTime1 => [
                                    undef,
                                    {
                                        Content => '2008-02-02 03:59',
                                    },
                                ],
                                GeneralCatalog1 => [
                                    undef,
                                    {
                                        Content => $GeneralCatalogListReverse{Test2},
                                    },
                                ],
                                Integer1 => [
                                    undef,
                                    {
                                        Content => '2',
                                    },
                                ],
                                Text1 => [
                                    undef,
                                    {
                                        Content => 'Test Text Test2',
                                    },
                                ],
                                TextArea1 => [
                                    undef,
                                    {
                                        Content => "Test2\nText Array\nTest 2",
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },

    # config item for sub element tests
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[2],
            ClassID => $ConfigItemClassIDs[1],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 3 Version 1',
                DefinitionID => $ConfigItemDefinitionIDs[1],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Main1 => [
                                    undef,
                                    {
                                        Content   => 'Main1 (1)',
                                        Main1Sub1 => [
                                            undef,
                                            {
                                                Content          => 'Main1 (1) Sub1 (1)',
                                                Main1Sub1SubSub1 => [
                                                    undef,
                                                    {
                                                        Content => 'Main1 (1) Sub1 (1) SubSub1 (1)',
                                                    },
                                                    {
                                                        Content => 'Main1 (1) Sub1 (1) SubSub1 (2)',
                                                    },
                                                    {
                                                        Content => 'Main1 (1) Sub1 (1) SubSub1 (3)',
                                                    },
                                                ],
                                                Main1Sub1SubSub2 => [
                                                    undef,
                                                    {
                                                        Content => 'Main1 (1) Sub1 (1) SubSub2 (1)',
                                                    },
                                                ],
                                            },
                                            {
                                                Content          => 'Main1 (1) Sub1 (2)',
                                                Main1Sub1SubSub1 => [
                                                    undef,
                                                    {
                                                        Content => 'Main1 (1) Sub1 (2) SubSub1 (1)',
                                                    },
                                                ],
                                                Main1Sub1SubSub2 => [
                                                    undef,
                                                    {
                                                        Content => 'Main1 (1) Sub1 (2) SubSub2 (1)',
                                                    },
                                                    {
                                                        Content => 'Main1 (1) Sub1 (2) SubSub2 (2)',
                                                    },
                                                ],
                                            },
                                        ],
                                        Main1Sub2 => [
                                            undef,
                                            {
                                                Content => 'Main1 (1) Sub2 (1)',
                                            },
                                            {
                                                Content => 'Main1 (1) Sub2 (2)',
                                            },
                                        ],
                                    },
                                ],
                                Main2 => [
                                    undef,
                                    {
                                        Content   => 'Main2 (1)',
                                        Main2Sub1 => [
                                            undef,
                                            {
                                                Content => 'Main2 (1) Sub1 (1)',
                                            },
                                        ],
                                        Main2Sub2 => [
                                            undef,
                                            {
                                                'Content' => 'Main2 (1) Sub2 (1)',
                                            },
                                            {
                                                'Content' => 'Main2 (1) Sub2 (2)',
                                            },
                                        ],
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },

    # config item for sub element tests
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[3],
            ClassID => $ConfigItemClassIDs[1],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 4 Version 1',
                DefinitionID => $ConfigItemDefinitionIDs[1],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Main1 => [
                                    undef,
                                    {
                                        Content   => '',
                                        Main1Sub1 => [
                                            undef,
                                            {
                                                Content          => '',
                                                Main1Sub1SubSub1 => [
                                                    undef,
                                                    {
                                                        Content => '',
                                                    },
                                                ],
                                                Main1Sub1SubSub2 => [
                                                    undef,
                                                    {
                                                        Content => '',
                                                    },
                                                ],
                                            },
                                        ],
                                        Main1Sub2 => [
                                            undef,
                                            {
                                                Content => '',
                                            },
                                        ],
                                    },
                                ],
                                Main2 => [
                                    undef,
                                    {
                                        Content   => '',
                                        Main2Sub1 => [
                                            undef,
                                            {
                                                Content => '',
                                            },
                                        ],
                                        Main2Sub2 => [
                                            undef,
                                            {
                                                Content => '',
                                            },
                                        ],
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },

    # config item for special character tests
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[4],
            ClassID => $ConfigItemClassIDs[1],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 5 Version 1',
                DefinitionID => $ConfigItemDefinitionIDs[1],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Main1 => [
                                    undef,
                                    {
                                        Content   => '"";;::..--__##',
                                        Main1Sub1 => [
                                            undef,
                                            {
                                                Content          => 'Test;:_°^!"§$%&/()=?´`*+Test',
                                                Main1Sub1SubSub1 => [
                                                    undef,
                                                    {
                                                        Content => '><@~\'}{[]\\',
                                                    },
                                                ],
                                                Main1Sub1SubSub2 => [
                                                    undef,
                                                    {
                                                        Content => '><@~\'}{[]\\',
                                                    },
                                                ],
                                            },
                                        ],
                                        Main1Sub2 => [
                                            undef,
                                            {
                                                Content => 'Test;:_°^!"§$%&/()=?´`*+Test',
                                            },
                                        ],
                                    },
                                ],
                                Main2 => [
                                    undef,
                                    {
                                        Content   => '"";;::..--__##',
                                        Main2Sub1 => [
                                            undef,
                                            {
                                                Content => 'Test Test',
                                            },
                                        ],
                                        Main2Sub2 => [
                                            undef,
                                            {
                                                Content => "Test\nTest\tTest",
                                            },
                                        ],
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },

    # config item for UTF-8 tests
    {
        ConfigItem => {
            Number  => $ConfigItemNumbers[5],
            ClassID => $ConfigItemClassIDs[1],
            UserID  => 1,
        },
        Versions => [
            {
                Name         => 'UnitTest - ConfigItem 6 Version 1',
                DefinitionID => $ConfigItemDefinitionIDs[1],
                DeplStateID  => $DeplStateListReverse{Production},
                InciStateID  => $InciStateListReverse{Operational},
                XMLData      => [
                    undef,
                    {
                        Version => [
                            undef,
                            {
                                Main1 => [
                                    undef,
                                    {
                                        Content   => 'ↂ ⅻ ⅛',
                                        Main1Sub1 => [
                                            undef,
                                            {
                                                Content          => '☄ ↮ ↹ →',
                                                Main1Sub1SubSub1 => [
                                                    undef,
                                                    {
                                                        Content => '₤ ₡ ₩ ₯ ₵',
                                                    },
                                                ],
                                                Main1Sub1SubSub2 => [
                                                    undef,
                                                    {
                                                        Content => '♊ ♈ ♉ ♊ ♋ ♍ ♑',
                                                    },
                                                ],
                                            },
                                        ],
                                        Main1Sub2 => [
                                            undef,
                                            {
                                                Content => '✈ ❤ ☮',
                                            },
                                        ],
                                    },
                                ],
                                Main2 => [
                                    undef,
                                    {
                                        Content   => 'Պ Մ Հ',
                                        Main2Sub1 => [
                                            undef,
                                            {
                                                Content => '® ©',
                                            },
                                        ],
                                        Main2Sub2 => [
                                            undef,
                                            {
                                                Content => 'か げ を',
                                            },
                                        ],
                                    },
                                ],
                            },
                        ],
                    },
                ],
                UserID => 1,
            },
        ],
    },
);

# add the test config items
my @ConfigItemIDs;
for my $ConfigItem (@ConfigItems) {

    # add a config item
    my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
        %{ $ConfigItem->{ConfigItem} },
    );

    # check config item id
    if ( !$ConfigItemID ) {

        $Self->True(
            0,
            "Can't add new config item.",
        );
    }

    push @ConfigItemIDs, $ConfigItemID;

    # add the versions
    for my $Version ( @{ $ConfigItem->{Versions} } ) {

        # add a version
        my $VersionID = $ConfigItemObject->VersionAdd(
            %{$Version},
            ConfigItemID => $ConfigItemID,
        );

        # check version id
        if ( !$VersionID ) {

            $Self->True(
                0,
                "Can't add new version.",
            );
        }
    }
}

# ------------------------------------------------------------ #
# define general ExportDataGet tests
# ------------------------------------------------------------ #

my @ExportDataTests = (

    # ImportDataGet doesn't contains all data (check required attributes)
    {
        SourceExportData => {
            ExportDataGet => {
                UserID => 1,
            },
        },
    },

    # ImportDataGet doesn't contains all data (check required attributes)
    {
        SourceExportData => {
            ExportDataGet => {
                TemplateID => $TemplateIDs[1],
            },
        },
    },

    # no existing template id is given (check return false)
    {
        SourceExportData => {
            ExportDataGet => {
                TemplateID => $TemplateIDs[-1] + 1,
                UserID     => 1,
            },
        },
    },

    # no class id is given (check return false)
    {
        SourceExportData => {
            ExportDataGet => {
                TemplateID => $TemplateIDs[2],
                UserID     => 1,
            },
        },
    },

    # invalid class id is given (check return false)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[-1] + 1,
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[2],
                UserID     => 1,
            },
        },
    },

    # mapping list is empty (check return false)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[3],
                UserID     => 1,
            },
        },
    },

    # all required values are given (number search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[0],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (name search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Name => 'UnitTest - ConfigItem 1 Version 1',
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (case insensitive name search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Name => 'unittest - configitem 1 version 1',
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (name and number search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[0],
                Name   => 'UnitTest - ConfigItem 1 Version 1',
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (deployment state search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                DeplStateIDs => $DeplStateListReverse{Production},
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
            [ $ConfigItemNumbers[1] ],
        ],
    },

    # all required values are given (incident state search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                InciStateIDs => $InciStateListReverse{Operational},
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
            [ $ConfigItemNumbers[1] ],
        ],
    },

    # all required values are given (combined search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Number       => $ConfigItemNumbers[0],
                Name         => 'UnitTest - ConfigItem 1 Version 1',
                DeplStateIDs => $DeplStateListReverse{Production},
                InciStateIDs => $InciStateListReverse{Operational},
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (XML data search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Date1           => '2008-02-01',
                TextArea1       => "Test\nText Array\nTest",
                Customer1       => 'UnitTest',
                Text1           => 'Test Text Test',
                DateTime1       => '2008-02-01 03:59',
                Integer1        => '1',
                GeneralCatalog1 => $GeneralCatalogListReverse{Test1},
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (combined all search check)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
            ],
            SearchData => {
                Number          => $ConfigItemNumbers[0],
                Name            => 'UnitTest - ConfigItem 1 Version 1',
                DeplStateIDs    => $DeplStateListReverse{Production},
                InciStateIDs    => $InciStateListReverse{Operational},
                Date1           => '2008-02-01',
                TextArea1       => "Test\nText Array\nTest",
                Customer1       => 'UnitTest',
                Text1           => 'Test Text Test',
                DateTime1       => '2008-02-01 03:59',
                Integer1        => '1',
                GeneralCatalog1 => $GeneralCatalogListReverse{Test1},
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[5],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [ $ConfigItemNumbers[0] ],
        ],
    },

    # all required values are given (check the returned array)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'Dummy1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[0],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[6],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[0],
                'UnitTest - ConfigItem 1 Version 1',
                'Production',
                'Operational',
                'UnitTest',
                '2008-02-01',
                '2008-02-01 03:59',
                undef,
                'Test1',
                '1',
                'Test Text Test',
                "Test\nText Array\nTest",
            ],
        ],
    },

    # all required values are given (double element checks)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'Dummy1::1',
                },
                {
                    Key => 'Dummy1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[0],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[6],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[0],
                $ConfigItemNumbers[0],
                'UnitTest - ConfigItem 1 Version 1',
                'UnitTest - ConfigItem 1 Version 1',
                'Production',
                'Production',
                'Operational',
                'Operational',
                'UnitTest',
                'UnitTest',
                '2008-02-01',
                '2008-02-01',
                '2008-02-01 03:59',
                '2008-02-01 03:59',
                undef,
                undef,
                'Test1',
                'Test1',
                '1',
                '1',
                'Test Text Test',
                'Test Text Test',
                "Test\nText Array\nTest",
                "Test\nText Array\nTest",
            ],
        ],
    },

    # all required values are given (sub element checks)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::2',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::2',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[2],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[7],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[2],
                'UnitTest - ConfigItem 3 Version 1',
                'Production',
                'Operational',
                'Main1 (1)',
                'Main1 (1) Sub1 (1)',
                'Main1 (1) Sub1 (1) SubSub1 (1)',
                'Main1 (1) Sub1 (1) SubSub1 (2)',
                'Main1 (1) Sub1 (1) SubSub1 (3)',
                'Main1 (1) Sub1 (1) SubSub2 (1)',
                'Main1 (1) Sub1 (2)',
                'Main1 (1) Sub1 (2) SubSub1 (1)',
                'Main1 (1) Sub1 (2) SubSub2 (1)',
                'Main1 (1) Sub1 (2) SubSub2 (2)',
                'Main1 (1) Sub2 (1)',
                'Main1 (1) Sub2 (2)',
                'Main2 (1)',
                'Main2 (1) Sub1 (1)',
                'Main2 (1) Sub2 (1)',
                'Main2 (1) Sub2 (2)',
            ],
        ],
    },

    # all required values are given (sub element checks with undef values)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::4',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::3',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub2::3',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::2',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::2',
                },
                {
                    Key => 'Main2::1::Main2Sub2::3',
                },
                {
                    Key => 'Main2::2',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[2],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[7],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[2],
                'UnitTest - ConfigItem 3 Version 1',
                'Production',
                'Operational',
                'Main1 (1)',
                'Main1 (1) Sub1 (1)',
                'Main1 (1) Sub1 (1) SubSub1 (1)',
                'Main1 (1) Sub1 (1) SubSub1 (2)',
                'Main1 (1) Sub1 (1) SubSub1 (3)',
                undef,
                'Main1 (1) Sub1 (1) SubSub2 (1)',
                undef,
                'Main1 (1) Sub1 (2)',
                'Main1 (1) Sub1 (2) SubSub1 (1)',
                undef,
                'Main1 (1) Sub1 (2) SubSub2 (1)',
                'Main1 (1) Sub1 (2) SubSub2 (2)',
                undef,
                'Main1 (1) Sub2 (1)',
                'Main1 (1) Sub2 (2)',
                undef,
                'Main2 (1)',
                'Main2 (1) Sub1 (1)',
                undef,
                'Main2 (1) Sub2 (1)',
                'Main2 (1) Sub2 (2)',
                undef,
                undef,
            ],
        ],
    },

    # all required values are given (sub element checks with undef values and empty strings)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::4',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::3',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub2::3',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::2',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::2',
                },
                {
                    Key => 'Main2::1::Main2Sub2::3',
                },
                {
                    Key => 'Main2::2',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[3],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[7],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[3],
                'UnitTest - ConfigItem 4 Version 1',
                'Production',
                'Operational',
                '',
                '',
                '',
                undef,
                undef,
                undef,
                '',
                undef,
                undef,
                undef,
                undef,
                undef,
                undef,
                undef,
                '',
                undef,
                undef,
                '',
                '',
                undef,
                '',
                undef,
                undef,
                undef,
            ],
        ],
    },

    # all required values are given (special character checks)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[4],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[8],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[4],
                'UnitTest - ConfigItem 5 Version 1',
                'Production',
                'Operational',
                '"";;::..--__##',
                'Test;:_°^!"§$%&/()=?´`*+Test',
                '><@~\'}{[]\\',
                '><@~\'}{[]\\',
                'Test;:_°^!"§$%&/()=?´`*+Test',
                '"";;::..--__##',
                'Test Test',
                "Test\nTest\tTest",
            ],
        ],
    },

    # all required values are given (UTF-8 checks)
    {
        SourceExportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key => 'Number',
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
            ],
            SearchData => {
                Number => $ConfigItemNumbers[5],
            },
            ExportDataGet => {
                TemplateID => $TemplateIDs[9],
                UserID     => 1,
            },
        },
        ReferenceExportData => [
            [
                $ConfigItemNumbers[5],
                'UnitTest - ConfigItem 6 Version 1',
                'Production',
                'Operational',
                'ↂ ⅻ ⅛',
                '☄ ↮ ↹ →',
                '₤ ₡ ₩ ₯ ₵',
                '♊ ♈ ♉ ♊ ♋ ♍ ♑',
                '✈ ❤ ☮',
                'Պ Մ Հ',
                '® ©',
                'か げ を',
            ],
        ],
    },
);

# ------------------------------------------------------------ #
# run general ExportDataGet tests
# ------------------------------------------------------------ #

my $ExportTestCount = 1;
TEST:
for my $Test (@ExportDataTests) {

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

        $Self->True(
            0,
            "ExportTest $ExportTestCount: No SourceExportData found for this test."
        );

        next TEST;
    }

    # set the object data
    if (
        $Test->{SourceExportData}->{ObjectData}
        && ref $Test->{SourceExportData}->{ObjectData} eq 'HASH'
        && $Test->{SourceExportData}->{ExportDataGet}->{TemplateID}
        )
    {

        # save object data
        $ImportExportObject->ObjectDataSave(
            TemplateID => $Test->{SourceExportData}->{ExportDataGet}->{TemplateID},
            ObjectData => $Test->{SourceExportData}->{ObjectData},
            UserID     => 1,
        );
    }

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

        # delete all existing mapping data
        $ImportExportObject->MappingDelete(
            TemplateID => $Test->{SourceExportData}->{ExportDataGet}->{TemplateID},
            UserID     => 1,
        );

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

            # add a new mapping row
            my $MappingID = $ImportExportObject->MappingAdd(
                TemplateID => $Test->{SourceExportData}->{ExportDataGet}->{TemplateID},
                UserID     => 1,
            );

            # add the mapping object data
            $ImportExportObject->MappingObjectDataSave(
                MappingID         => $MappingID,
                MappingObjectData => $MappingObjectData,
                UserID            => 1,
            );
        }
    }

    # add the search data
    if (
        $Test->{SourceExportData}->{SearchData}
        && ref $Test->{SourceExportData}->{SearchData} eq 'HASH'
        && $Test->{SourceExportData}->{ExportDataGet}->{TemplateID}
        )
    {

        # save search data
        $ImportExportObject->SearchDataSave(
            TemplateID => $Test->{SourceExportData}->{ExportDataGet}->{TemplateID},
            SearchData => $Test->{SourceExportData}->{SearchData},
            UserID     => 1,
        );
    }

    # get export data
    my $ExportData = $ObjectBackendObject->ExportDataGet(
        %{ $Test->{SourceExportData}->{ExportDataGet} },
    );

    if ( !$Test->{ReferenceExportData} ) {

        $Self->False(
            $ExportData,
            "ExportTest $ExportTestCount: ExportDataGet() - return false",
        );

        next TEST;
    }

    if ( ref $ExportData ne 'ARRAY' ) {

        # check array reference
        $Self->True(
            0,
            "ExportTest $ExportTestCount: ExportDataGet() - return value is an array reference",
        );

        next TEST;
    }

    # check number of rows
    $Self->Is(
        scalar @{$ExportData},
        scalar @{ $Test->{ReferenceExportData} },
        "ExportTest $ExportTestCount: ExportDataGet() - correct number of rows",
    );

    # check content of export data
    my $CounterRow = 0;
    ROW:
    for my $ExportRow ( @{$ExportData} ) {

        # extract reference row
        my $ReferenceRow = $Test->{ReferenceExportData}->[$CounterRow];

        if ( ref $ExportRow ne 'ARRAY' || ref $ReferenceRow ne 'ARRAY' ) {

            # check array reference
            $Self->True(
                0,
                "ExportTest $ExportTestCount: ExportDataGet() - export row and reference row matched",
            );

            next TEST;
        }

        # check number of columns
        $Self->Is(
            scalar @{$ExportRow},
            scalar @{$ReferenceRow},
            "ExportTest $ExportTestCount: ExportDataGet() - correct number of columns",
        );

        my $CounterColumn = 0;
        for my $Cell ( @{$ExportRow} ) {

            # set content if values are undef
            if ( !defined $Cell ) {
                $Cell = 'UNDEF-unittest';
            }
            if ( !defined $ReferenceRow->[$CounterColumn] ) {
                $ReferenceRow->[$CounterColumn] = 'UNDEF-unittest';
            }

            # check cell data
            $Self->Is(
                $Cell,
                $ReferenceRow->[$CounterColumn],
                "ExportTest $ExportTestCount: ExportDataGet() ",
            );

            $CounterColumn++;
        }

        $CounterRow++;
    }
}
continue {
    $ExportTestCount++;
}

# ------------------------------------------------------------ #
# define general ImportDataSave tests
# ------------------------------------------------------------ #

my @ImportDataTests = (

    # ImportDataSave doesn't contains all data (check required attributes)
    {
        SourceImportData => {
            ImportDataSave => {
                ImportDataRow => [],
                UserID        => 1,
            },
        },
    },

    # ImportDataSave doesn't contains all data (check required attributes)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID => $TemplateIDs[20],
                UserID     => 1,
            },
        },
    },

    # ImportDataSave doesn't contains all data (check required attributes)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID    => $TemplateIDs[20],
                ImportDataRow => [],
            },
        },
    },

    # import data row must be an array reference (check return false)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID    => $TemplateIDs[20],
                ImportDataRow => '',
                UserID        => 1,
            },
        },
    },

    # import data row must be an array reference (check return false)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID    => $TemplateIDs[20],
                ImportDataRow => {},
                UserID        => 1,
            },
        },
    },

    # no existing template id is given (check return false)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID    => $TemplateIDs[-1] + 1,
                ImportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # no class id is given (check return false)
    {
        SourceImportData => {
            ImportDataSave => {
                TemplateID    => $TemplateIDs[21],
                ImportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # invalid class id is given (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[-1] + 1,
            },
            ImportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ImportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # mapping list is empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # more than one identifier with the same name (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '123', '321' ],
                UserID        => 1,
            },
        },
    },

    # identifier is empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [''],
                UserID        => 1,
            },
        },
    },

    # identifier is undef (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [undef],
                UserID        => 1,
            },
        },
    },

    # both identifiers are empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '', '' ],
                UserID        => 1,
            },
        },
    },

    # both identifiers are undef (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ undef, undef ],
                UserID        => 1,
            },
        },
    },

    # one identifiers is empty, one is undef (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '', undef ],
                UserID        => 1,
            },
        },
    },

    # one of the identifiers is empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '123', '' ],
                UserID        => 1,
            },
        },
    },

    # one of the identifiers is undef (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '123', undef ],
                UserID        => 1,
            },
        },
    },

    # one of the identifiers is empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ '', '123' ],
                UserID        => 1,
            },
        },
    },

    # one of the identifiers is empty (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ImportDataRow => [ undef, '123' ],
                UserID        => 1,
            },
        },
    },

    # TODO Add some identifier tests

    # empty name is given (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ImportDataRow => [ '', 'Production', 'Operational' ],
                UserID        => 1,
            },
        },
    },

    # invalid deployment state is given (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ImportDataRow => [ 'UnitTest - Importtest 1', 'Dummy', 'Operational' ],
                UserID        => 1,
            },
        },
    },

    # invalid incident state is given (check return false)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ImportDataRow => [ 'UnitTest - Importtest 2', 'Production', 'Dummy' ],
                UserID        => 1,
            },
        },
    },

    # all required values are given (a NEW config item must be created)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 3',
                    'Production',
                    'Operational',
                    'UnitTest',
                    '2008-06-05',
                    '2008-08-05 04:50',
                    'Test3',
                    '3',
                    'Test3 Text3 Test3',
                    "Test3\nTextArray3\nTest3",
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 1,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 3',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Customer1::1'       => 'UnitTest',
                'Date1::1'           => '2008-06-05',
                'DateTime1::1'       => '2008-08-05 04:50',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test3},
                'Integer1::1'        => '3',
                'Text1::1'           => 'Test3 Text3 Test3',
                'TextArea1::1'       => "Test3\nTextArray3\nTest3",
            },
        },
    },

    # all required values are given (a second NEW config item must be created)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 4',
                    'Production',
                    'Operational',
                    'UnitTest',
                    '2008-09-05',
                    '2008-12-05 04:50',
                    'Test4',
                    '4',
                    'Test4 Text4 Test4',
                    "Test4\nTextArray4\nTest4",
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 1,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 4',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Customer1::1'       => 'UnitTest',
                'Date1::1'           => '2008-09-05',
                'DateTime1::1'       => '2008-12-05 04:50',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test4},
                'Integer1::1'        => '4',
                'Text1::1'           => 'Test4 Text4 Test4',
                'TextArea1::1'       => "Test4\nTextArray4\nTest4",
            },
        },
    },

    # all required values are given (a new version must be added to first test config item)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[26],
                ImportDataRow => [
                    $ConfigItemNumbers[0],
                    'UnitTest - ConfigItem 1 Version 2',
                    'Pilot',
                    'Incident',
                    'UnitTest',
                    '2008-02-02',
                    '2008-02-02 03:59',
                    'Test2',
                    '2',
                    'Test Text UPDATE1 Test',
                    "Test\nText Array UPDATE1\nTest",
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 2,
            LastVersion   => {
                Name                 => 'UnitTest - ConfigItem 1 Version 2',
                DeplState            => 'Pilot',
                InciState            => 'Incident',
                'Customer1::1'       => 'UnitTest',
                'Date1::1'           => '2008-02-02',
                'DateTime1::1'       => '2008-02-02 03:59',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test2},
                'Integer1::1'        => '2',
                'Text1::1'           => 'Test Text UPDATE1 Test',
                'TextArea1::1'       => "Test\nText Array UPDATE1\nTest",
            },
        },
    },

    # all required values are given (a new version must be added to first test config item again)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Customer1::1',
                },
                {
                    Key => 'Date1::1',
                },
                {
                    Key => 'DateTime1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
                {
                    Key => 'Integer1::1',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'TextArea1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[27],
                ImportDataRow => [
                    $ConfigItemNumbers[0],
                    'UnitTest - ConfigItem 1 Version 3',
                    'Repair',
                    'Operational',
                    'UnitTest',
                    '2008-02-03',
                    '2008-02-03 03:59',
                    'Test3',
                    '3',
                    'Test Text UPDATE2 Test',
                    "Test\nText Array UPDATE2\nTest",
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 3,
            LastVersion   => {
                Name                 => 'UnitTest - ConfigItem 1 Version 3',
                DeplState            => 'Repair',
                InciState            => 'Operational',
                'Customer1::1'       => 'UnitTest',
                'Date1::1'           => '2008-02-03',
                'DateTime1::1'       => '2008-02-03 03:59',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test3},
                'Integer1::1'        => '3',
                'Text1::1'           => 'Test Text UPDATE2 Test',
                'TextArea1::1'       => "Test\nText Array UPDATE2\nTest",
            },
        },
    },

    # all required values are given (a new version must be added to third test config item)
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::2',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::2',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[27],
                ImportDataRow => [
                    $ConfigItemNumbers[2],
                    'UnitTest - ConfigItem 3 Version 2',
                    'Production',
                    'Operational',
                    'Main1 (1)',
                    'Main1 (1) Main1Sub1 (1)',
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (1)',
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (2)',
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (3)',
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub2 (1)',
                    'Main1 (1) Main1Sub1 (2)',
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub1 (1)',
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (1)',
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (2)',
                    'Main1 (1) Main1Sub2 (1)',
                    'Main1 (1) Main1Sub2 (2)',
                    'Main2 (1)',
                    'Main2 (1) Main2Sub1 (1)',
                    'Main2 (1) Main2Sub2 (1)',
                    'Main2 (1) Main2Sub2 (2)',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 2,
            LastVersion   => {
                Name                     => 'UnitTest - ConfigItem 3 Version 2',
                DeplState                => 'Production',
                InciState                => 'Operational',
                'Main1::1'               => 'Main1 (1)',
                'Main1::1::Main1Sub1::1' => 'Main1 (1) Main1Sub1 (1)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (1)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (2)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (3)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub2 (1)',
                'Main1::1::Main1Sub1::2' => 'Main1 (1) Main1Sub1 (2)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub1 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (2)',
                'Main1::1::Main1Sub2::1' => 'Main1 (1) Main1Sub2 (1)',
                'Main1::1::Main1Sub2::2' => 'Main1 (1) Main1Sub2 (2)',
                'Main2::1'               => 'Main2 (1)',
                'Main2::1::Main2Sub1::1' => 'Main2 (1) Main2Sub1 (1)',
                'Main2::1::Main2Sub2::1' => 'Main2 (1) Main2Sub2 (1)',
                'Main2::1::Main2Sub2::2' => 'Main2 (1) Main2Sub2 (2)',
            },
        },
    },

    # all required values are given (special character checks)
    # In 'UnitTest - ConfigItem 3 Version 2' 16 Attributes were imported,
    # so there will be 8 lingering attributes.
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[27],
                ImportDataRow => [
                    $ConfigItemNumbers[2],
                    'UnitTest - ConfigItem 3 Version 3',
                    'Production',
                    'Operational',
                    '"";;::..--__##',
                    'Test;:_°^!"§$%&/()=?´`*+Test',
                    '><@~\'}{[]\\',
                    '><@~\'}{[]\\',
                    'Test;:_°^!"§$%&/()=?´`*+Test',
                    '"";;::..--__##',
                    'Test Test',
                    "Test\nTest\tTest",
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 3,
            LastVersion   => {
                Name                                          => 'UnitTest - ConfigItem 3 Version 3',
                DeplState                                     => 'Production',
                InciState                                     => 'Operational',
                'Main1::1'                                    => '"";;::..--__##',
                'Main1::1::Main1Sub1::1'                      => 'Test;:_°^!"§$%&/()=?´`*+Test',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1' => '><@~\'}{[]\\',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1' => '><@~\'}{[]\\',
                'Main1::1::Main1Sub2::1'                      => 'Test;:_°^!"§$%&/()=?´`*+Test',
                'Main2::1'                                    => '"";;::..--__##',
                'Main2::1::Main2Sub1::1'                      => 'Test Test',
                'Main2::1::Main2Sub2::1'                      => "Test\nTest\tTest",

                # lingering from 'UnitTest - ConfigItem 3 Version 2',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (2)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (3)',
                'Main1::1::Main1Sub1::2' => 'Main1 (1) Main1Sub1 (2)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub1 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (2)',
                'Main1::1::Main1Sub2::2' => 'Main1 (1) Main1Sub2 (2)',
                'Main2::1::Main2Sub2::2' => 'Main2 (1) Main2Sub2 (2)',
            },
        },
    },

    # all required values are given (UTF-8 checks)
    # In 'UnitTest - ConfigItem 3 Version 2' 16 Attributes were imported,
    # so there will be 8 lingering attributes.
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[1],
            },
            MappingObjectData => [
                {
                    Key        => 'Number',
                    Identifier => 1,
                },
                {
                    Key => 'Name',
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Main1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1',
                },
                {
                    Key => 'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1',
                },
                {
                    Key => 'Main1::1::Main1Sub2::1',
                },
                {
                    Key => 'Main2::1',
                },
                {
                    Key => 'Main2::1::Main2Sub1::1',
                },
                {
                    Key => 'Main2::1::Main2Sub2::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[27],
                ImportDataRow => [
                    $ConfigItemNumbers[2],
                    'UnitTest - ConfigItem 3 Version 4',
                    'Production',
                    'Operational',
                    'Ϋ δ λ',
                    'π χ Ϙ',
                    'Ϻ ϱ Ϯ',
                    'ɯ ʓ ʠ',
                    'ʬ ʯ',
                    'й ф щ',
                    'њ ё',
                    'Ѭ Ѧ',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 4,
            LastVersion   => {
                Name                                          => 'UnitTest - ConfigItem 3 Version 4',
                DeplState                                     => 'Production',
                InciState                                     => 'Operational',
                'Main1::1'                                    => 'Ϋ δ λ',
                'Main1::1::Main1Sub1::1'                      => 'π χ Ϙ',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::1' => 'Ϻ ϱ Ϯ',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub2::1' => 'ɯ ʓ ʠ',
                'Main1::1::Main1Sub2::1'                      => 'ʬ ʯ',
                'Main2::1'                                    => 'й ф щ',
                'Main2::1::Main2Sub1::1'                      => 'њ ё',
                'Main2::1::Main2Sub2::1'                      => 'Ѭ Ѧ',

                # lingering from 'UnitTest - ConfigItem 3 Version 2',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::2' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (2)',
                'Main1::1::Main1Sub1::1::Main1Sub1SubSub1::3' =>
                    'Main1 (1) Main1Sub1 (1) Main1Sub1SubSub1 (3)',
                'Main1::1::Main1Sub1::2' => 'Main1 (1) Main1Sub1 (2)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub1::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub1 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::1' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (1)',
                'Main1::1::Main1Sub1::2::Main1Sub1SubSub2::2' =>
                    'Main1 (1) Main1Sub1 (2) Main1Sub1SubSub2 (2)',
                'Main1::1::Main1Sub2::2' => 'Main1 (1) Main1Sub2 (2)',
                'Main2::1::Main2Sub2::2' => 'Main2 (1) Main2Sub2 (2)',
            },
        },
    },

    # a simple import for testing the overriding behavior of empty values
    {
        SourceImportData => {
            ObjectData => {
                ClassID => $ConfigItemClassIDs[0],
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    'Importtest 5 for behavior of empty values',
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 1,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => 'Importtest 5 for behavior of empty values',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an empty value for Text1, with EmptyFieldsLeaveTheOldValues turned on
    # no new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '',
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 1,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => 'Importtest 5 for behavior of empty values',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import undef for Text1, with EmptyFieldsLeaveTheOldValues turned on
    # no new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    undef,
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 1,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => 'Importtest 5 for behavior of empty values',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an empty value for Text1, with EmptyFieldsLeaveTheOldValues turned off
    # a new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => '',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '',
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 2,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => '',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import a single space value for Text1, with EmptyFieldsLeaveTheOldValues turned on
    # a new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => '',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    ' ',
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 3,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => ' ',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import the string '0' value for Text1, with EmptyFieldsLeaveTheOldValues turned on
    # a new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => '',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '0',
                    'Test1',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 4,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => '0',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an empty value for GeneralCatalog1, with EmptyFieldsLeaveTheOldValues turned on
    # no new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '',
                    '',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 4,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => '0',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an invalid value for GeneralCatalog1, with EmptyFieldsLeaveTheOldValues turned on
    # the import should fail
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '',
                    'non-existent general catalog entry',
                ],
                UserID => 1,
            },
        },
    },

    # import an invalid value for GeneralCatalog1, with EmptyFieldsLeaveTheOldValues turned off
    # the import should fail
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => '',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                    '',
                    'non-existent general catalog entry',
                ],
                UserID => 1,
            },
        },
    },

    # import an empty value for DeplState, with EmptyFieldsLeaveTheOldValues turned on
    # no new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    '',
                    'Operational',
                    '',
                    '',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 4,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => '0',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an invalid value for DeplState, with EmptyFieldsLeaveTheOldValues turned on
    # an error should be generated
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'invalid deployment state',
                    'Operational',
                    '',
                    '',
                ],
                UserID => 1,
            },
        },
    },

    # import an empty value for InciState, with EmptyFieldsLeaveTheOldValues turned on
    # no new version should be created
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    '',
                    '',
                    '',
                ],
                UserID => 1,
            },
        },
        ReferenceImportData => {
            VersionNumber => 4,
            LastVersion   => {
                Name                 => 'UnitTest - Importtest 5',
                DeplState            => 'Production',
                InciState            => 'Operational',
                'Text1::1'           => '0',
                'GeneralCatalog1::1' => $GeneralCatalogListReverse{Test1},
            },
        },
    },

    # import an invalid value for InciState, with EmptyFieldsLeaveTheOldValues turned on
    # an error should be generated
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[0],
                EmptyFieldsLeaveTheOldValues => 'on',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
                {
                    Key => 'Text1::1',
                },
                {
                    Key => 'GeneralCatalog1::1',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'invalid incident state',
                    '',
                    '',
                ],
                UserID => 1,
            },
        },
    },

    # Import without required attribute 'Type', an error should be generated (see bug#14098).
    {
        SourceImportData => {
            ObjectData => {
                ClassID                      => $ConfigItemClassIDs[2],
                EmptyFieldsLeaveTheOldValues => '',
            },
            MappingObjectData => [
                {
                    Key        => 'Name',
                    Identifier => 1,
                },
                {
                    Key => 'DeplState',
                },
                {
                    Key => 'InciState',
                },
            ],
            ImportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ImportDataRow => [
                    'UnitTest - Importtest 5',
                    'Production',
                    'Operational',
                ],
                UserID => 1,
            },
        },
    },
);

# ------------------------------------------------------------ #
# run general ExportDataGet tests
# ------------------------------------------------------------ #

my $ImportTestCount = 1;
TEST:
for my $Test (@ImportDataTests) {

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

        $Self->True(
            0,
            "ImportTest $ImportTestCount: No SourceImportData found for this test."
        );

        next TEST;
    }

    # set the object data
    if (
        $Test->{SourceImportData}->{ObjectData}
        && ref $Test->{SourceImportData}->{ObjectData} eq 'HASH'
        && $Test->{SourceImportData}->{ImportDataSave}->{TemplateID}
        )
    {

        # save object data
        $ImportExportObject->ObjectDataSave(
            TemplateID => $Test->{SourceImportData}->{ImportDataSave}->{TemplateID},
            ObjectData => $Test->{SourceImportData}->{ObjectData},
            UserID     => 1,
        );
    }

    # set the mapping object data
    if (
        $Test->{SourceImportData}->{MappingObjectData}
        && ref $Test->{SourceImportData}->{MappingObjectData} eq 'ARRAY'
        && $Test->{SourceImportData}->{ImportDataSave}->{TemplateID}
        )
    {

        # delete all existing mapping data
        $ImportExportObject->MappingDelete(
            TemplateID => $Test->{SourceImportData}->{ImportDataSave}->{TemplateID},
            UserID     => 1,
        );

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

            # add a new mapping row
            my $MappingID = $ImportExportObject->MappingAdd(
                TemplateID => $Test->{SourceImportData}->{ImportDataSave}->{TemplateID},
                UserID     => 1,
            );

            # add the mapping object data
            $ImportExportObject->MappingObjectDataSave(
                MappingID         => $MappingID,
                MappingObjectData => $MappingObjectData,
                UserID            => 1,
            );
        }
    }

    # import data save
    my ( $ConfigItemID, $RetCode ) = $ObjectBackendObject->ImportDataSave(
        %{ $Test->{SourceImportData}->{ImportDataSave} },
        Counter => $ImportTestCount,
    );

    if ( !$Test->{ReferenceImportData} ) {

        $Self->False(
            $ConfigItemID,
            "ImportTest $ImportTestCount: ImportDataSave() - return no ConfigItemID"
        );
        $Self->False(
            $RetCode,
            "ImportTest $ImportTestCount: ImportDataSave() - return no RetCode"
        );

        next TEST;
    }

    $Self->True(
        $ConfigItemID,
        "ImportTest $ImportTestCount: ImportDataSave() - return ConfigItemID"
    );
    $Self->True(
        $RetCode,
        "ImportTest $ImportTestCount: ImportDataSave() - return RetCode"
    );

    # get the version list
    my $VersionList = $ConfigItemObject->VersionList(
        ConfigItemID => $ConfigItemID,
    ) // [];

    # check number of versions
    $Self->Is(
        scalar @{$VersionList},
        $Test->{ReferenceImportData}->{VersionNumber} || 0,
        "ImportTest $ImportTestCount: ImportDataSave() - correct number of versions",
    );

    # get the last version
    my $VersionData = $ConfigItemObject->VersionGet(
        ConfigItemID => $ConfigItemID,
        XMLDataGet   => 1,
    );

    # translate xmldata in a 2d hash
    my %XMLHash = $XMLObject->XMLHash2D(
        XMLHash => $VersionData->{XMLData},
    );

    # clean the xml hash
    KEY:
    for my $Key ( sort keys %XMLHash ) {

        next KEY if $Key =~ m{ \{'Content'\} \z }xms;

        delete $XMLHash{$Key};
    }

    # check general elements
    ELEMENT:
    for my $Element (qw(Number Name DeplState InciState)) {

        next ELEMENT if !exists $Test->{ReferenceImportData}->{LastVersion}->{$Element};

        # set content if values are undef
        if ( !defined $Test->{ReferenceImportData}->{LastVersion}->{$Element} ) {
            $Test->{ReferenceImportData}->{LastVersion}->{$Element} = 'UNDEF-unittest';
        }
        if ( !defined $Test->{ReferenceImportData}->{LastVersion}->{$Element} ) {
            $Test->{ReferenceImportData}->{LastVersion}->{$Element} = 'UNDEF-unittest';
        }

        # check element
        $Self->Is(
            $VersionData->{$Element},
            $Test->{ReferenceImportData}->{LastVersion}->{$Element},
            "ImportTest $ImportTestCount: ImportDataSave() $Element is identical",
        );

        delete $Test->{ReferenceImportData}->{LastVersion}->{$Element};
    }

    # check number of XML elements
    $Self->Is(
        scalar keys %XMLHash,
        scalar keys %{ $Test->{ReferenceImportData}->{LastVersion} },
        "ImportTest $ImportTestCount: ImportDataSave() - correct number of XML elements",
    );

    # check XML elements
    ELEMENT:
    for my $Key ( sort keys %{ $Test->{ReferenceImportData}->{LastVersion} } ) {

        # duplicate key
        my $XMLKey = $Key;

        # prepare key
        my $Counter = 0;
        while ( $XMLKey =~ m{ :: }xms ) {

            if ( $Counter % 2 ) {
                $XMLKey =~ s{ :: }{]\{'}xms;
            }
            else {
                $XMLKey =~ s{ :: }{'\}[}xms;
            }

            $Counter++;
        }

        next ELEMENT if !exists $XMLHash{ '[1]{\'Version\'}[1]{\'' . $XMLKey . ']{\'Content\'}' };

        # set content if values are undef
        if ( !defined $XMLHash{ '[1]{\'Version\'}[1]{\'' . $XMLKey . ']{\'Content\'}' } ) {
            $XMLHash{ '[1]{\'Version\'}[1]{\'' . $XMLKey . ']{\'Content\'}' } = 'UNDEF-unittest';
        }
        if ( !defined $Test->{ReferenceImportData}->{LastVersion}->{$Key} ) {
            $Test->{ReferenceImportData}->{LastVersion}->{$Key} = 'UNDEF-unittest';
        }

        # check XML element
        $Self->Is(
            $XMLHash{ '[1]{\'Version\'}[1]{\'' . $XMLKey . ']{\'Content\'}' },
            $Test->{ReferenceImportData}->{LastVersion}->{$Key},
            "ImportTest $ImportTestCount: ImportDataSave() $Key is identical",
        );
    }
}
continue {
    $ImportTestCount++;
}

# cleanup is done by RestoreDatabase

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

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

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

my $XMLDefinition = [
    {
        Key        => 'Vendor',
        Name       => 'Vendor',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 50,
        },
    },
    {
        Key        => 'Model',
        Name       => 'Model',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 50,
        },
    },
    {
        Key        => 'Description',
        Name       => 'Description',
        Searchable => 1,
        Input      => {
            Type => 'TextArea',
        },
    },
    {
        Key        => 'Type',
        Name       => 'Type',
        Searchable => 1,
        Input      => {
            Type        => 'GeneralCatalog',
            Class       => 'ITSM::ConfigItem::Computer::Type',
            Translation => 1,
        },
    },
    {
        Key        => 'Owner',
        Name       => 'Owner',
        Searchable => 1,
        Input      => {
            Type => 'Customer',
        },
    },
    {
        Key        => 'SerialNumber',
        Name       => 'Serial Number',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 100,
        },
    },
    {
        Key   => 'OperatingSystem',
        Name  => 'Operating System',
        Input => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 100,
        },
    },
    {
        Key   => 'CPU',
        Name  => 'CPU',
        Input => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 100,
        },
        CountMax => 16,
    },
    {
        Key   => 'Ram',
        Name  => 'Ram',
        Input => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 100,
        },
        CountMax => 10,
    },
    {
        Key   => 'HardDisk',
        Name  => 'Hard Disk',
        Input => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 100,
        },
        CountMax => 100,
        Sub      => [
            {
                Key   => 'Capacity',
                Name  => 'Capacity',
                Input => {
                    Type      => 'Text',
                    Size      => 20,
                    MaxLength => 10,
                },
                CountMax => 100,
                Sub      => [
                    {
                        Key   => 'Capacity',
                        Name  => 'Capacity',
                        Input => {
                            Type      => 'Text',
                            Size      => 20,
                            MaxLength => 10,
                        },
                        CountMax => 100,
                        Sub      => [
                            {
                                Key   => 'Capacity',
                                Name  => 'Capacity',
                                Input => {
                                    Type      => 'Text',
                                    Size      => 20,
                                    MaxLength => 10,
                                },
                                CountMax => 100,
                            },
                        ],
                    },
                ],
            },
        ],
    },
    {
        Key        => 'FQDN',
        Name       => 'FQDN',
        Searchable => 1,
        Input      => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 100,
        },
    },
    {
        Key   => 'NIC',
        Name  => 'Network Adapter',
        Input => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 100,
            Required  => 1,
        },
        CountMin     => 0,
        CountMax     => 10,
        CountDefault => 1,
        Sub          => [
            {
                Key   => 'IPoverDHCP',
                Name  => 'IP over DHCP',
                Input => {
                    Type        => 'GeneralCatalog',
                    Class       => 'ITSM::ConfigItem::YesNo',
                    Translation => 1,
                    Required    => 1,
                },
            },
            {
                Key        => 'IPAddress',
                Name       => 'IP Address',
                Searchable => 1,
                Input      => {
                    Type      => 'Text',
                    Size      => 40,
                    MaxLength => 40,
                    Required  => 1,
                },
                CountMin     => 0,
                CountMax     => 20,
                CountDefault => 0,
            },
        ],
    },
    {
        Key   => 'GraphicAdapter',
        Name  => 'Graphic Adapter',
        Input => {
            Type      => 'Text',
            Size      => 50,
            MaxLength => 100,
        },
    },
    {
        Key   => 'OtherEquipment',
        Name  => 'Other Equipment',
        Input => {
            Type     => 'TextArea',
            Required => 1,
        },
        CountMin     => 0,
        CountDefault => 0,
    },
    {
        Key        => 'WarrantyExpirationDate',
        Name       => 'Warranty Expiration Date',
        Searchable => 1,
        Input      => {
            Type             => 'Date',
            YearPeriodPast   => 20,
            YearPeriodFuture => 10,
        },
    },
    {
        Key        => 'InstallDate',
        Name       => 'Install Date',
        Searchable => 1,
        Input      => {
            Type             => 'Date',
            Required         => 1,
            YearPeriodPast   => 20,
            YearPeriodFuture => 10,
        },
        CountMin     => 0,
        CountDefault => 0,
    },
    {
        Key        => 'Note',
        Name       => 'Note',
        Searchable => 1,
        Input      => {
            Type     => 'TextArea',
            Required => 1,
        },
        CountMin     => 0,
        CountDefault => 0,
    },
];

my $XMLData = {
    HardDisk => [
        undef,
        {
            Content  => 'HD1',
            Capacity => [
                undef,
                {
                    Content => '42 GB',
                }
            ]
        },
        {
            Content  => 'HD2',
            Capacity => [
                undef,
                {
                    Content => '780 GB',
                }
            ]
        },
    ],
};

my @ShowColumnsByClass = (
    'Name',
    'CurDeplState',
    'CPU::1',
    'HardDisk::1::Capacity::1',
    'HardDisk::2',
    'HardDisk::2::Capacity::1',
    'WarrantyExpirationDate::1',
    'CreateTime',
);

my $Result = $LayoutObject->XMLData2Hash(
    XMLDefinition => $XMLDefinition,
    XMLData       => $XMLData,
    Attributes    => \@ShowColumnsByClass,
);

my $Expected = {
    'CPU::1' => {
        Value => '',
        Name  => 'CPU',
    },
    'HardDisk::1::Capacity::1' => {
        Value => '42 GB',
        Name  => 'Capacity',
    },
    'HardDisk::2' => {
        Value => 'HD2',
        Name  => 'Hard Disk',
    },
    'HardDisk::2::Capacity::1' => {
        Value => '780 GB',
        Name  => 'Capacity',
    },
    'WarrantyExpirationDate::1' => {
        Name  => 'Warranty Expiration Date',
        Value => ''
    }
};

$Self->IsDeeply(
    $Result,
    $Expected,
    'XML Data converts into Hash structure',
);

# Test ITSMConfigItemInputCreate() to check bug#14174 (https://bugs.otrs.org/show_bug.cgi?id=14174).
# Set OTRSTimeZone to UTC.
$Helper->ConfigSettingChange(
    Valid => 1,
    Key   => 'OTRSTimeZone',
    Value => 'UTC',
);

# Get Layout object with EST user time zone.
$Kernel::OM->ObjectsDiscard(
    Objects => ['Kernel::Output::HTML::Layout'],
);
$Kernel::OM->ObjectParamAdd(
    'Kernel::Output::HTML::Layout' => {
        Lang         => 'en',
        UserTimeZone => 'EST',
    },
);
$LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

my %Param = (
    'Item' => {
        'Searchable' => 1,
        'Form'       => {
            'InstallDate::1' => {}
        },
        'Key'      => 'InstallDate',
        'CountMax' => 1,
        'Name'     => 'Install Date',
        'CountMin' => 0,
        'Input'    => {
            'Type'             => 'Date',
            'YearPeriodFuture' => 10,
            'Required'         => 1,
            'YearPeriodPast'   => 20
        },
        'CountDefault' => 0
    },
    'Invalid'  => 0,
    'Value'    => '2018-01-16',
    'Key'      => 'InstallDate::1',
    'Required' => 1,
    'ItemId'   => 'Item140'
);

my @Tests = (
    {
        OverrideTimeZone => 0,
        Expected         => '<option \s value="15" \s selected="selected">15</option>',
        Comment          => 'date is changed',
    },
    {
        OverrideTimeZone => 1,
        Expected         => '<option \s value="16" \s selected="selected">16<\/option>',
        Comment          => 'date is not changed',
    },
);

for my $Test (@Tests) {
    my $String = $LayoutObject->ITSMConfigItemInputCreate(
        %Param,
        OverrideTimeZone => $Test->{OverrideTimeZone},
    );

    my $Success;
    if ( $String =~ m{ $Test->{Expected} }msx ) {
        $Success = 1;
    }

    $Self->True(
        $Success,
        "OverrideTimeZone = $Test->{OverrideTimeZone} - $Test->{Comment}",
    );
}

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

my $Selenium = $Kernel::OM->Get('Kernel::System::UnitTest::Selenium');

$Selenium->RunTest(
    sub {

        my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

        my $RandomID = $Helper->GetRandomID();

        # Create test user and login.
        my $TestUserLogin = $Helper->TestUserCreate(
            Groups => [ 'admin', 'itsm-configitem' ],
        ) || die "Did not get test user";

        $Selenium->Login(
            Type     => 'Agent',
            User     => $TestUserLogin,
            Password => $TestUserLogin,
        );

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

        # Navigate to AdminITSMConfigItem.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AdminITSMConfigItem");

        # Test default ITSMConfigItem class.
        for my $Item (qw( Computer Hardware Location Network Software )) {
            my $Element = $Selenium->find_element( $Item, 'link_text' );
            $Element->is_enabled();
            $Element->is_displayed();
            $Element->VerifiedClick();

            # Check for table structure.
            $Selenium->find_element( "table",             'css' );
            $Selenium->find_element( "table thead tr th", 'css' );
            $Selenium->find_element( "table tbody tr td", 'css' );

            # Click on 'Change class definition'.
            $Selenium->find_element("//button[\@value='Add'][\@type='submit']")->VerifiedClick();

            # Check for input area.
            my $InputField = $Selenium->find_element( "#Definition", 'css' );
            $InputField->is_enabled();
            $InputField->is_displayed();

            # Return back to overview screen.
            $Selenium->find_element("//a[contains(\@href, \'Action=AdminITSMConfigItem' )]")->VerifiedClick();
        }

        # Add integer field to new ITSMConfigItem class to verify existence of 0.
        # For more information, see bug#6005 - https://bugs.otrs.org/show_bug.cgi?id=6005.
        my $Class   = "Class$RandomID";
        my $ClassID = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemAdd(
            Class   => 'ITSM::ConfigItem::Class',
            Name    => $Class,
            ValidID => 1,
            Comment => 'Comment',
            UserID  => 1,
        );

        # Get 'itsm-configitem' group ID.
        my $ITSMConfigItemGroupID = $Kernel::OM->Get('Kernel::System::Group')->GroupLookup(
            Group => 'itsm-configitem',
        );

        # Set permission for test class.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AdminGeneralCatalog;Subaction=ItemEdit;ItemID=$ClassID");
        $Selenium->execute_script(
            "\$('#Permission').val('$ITSMConfigItemGroupID').trigger('redraw.InputField').trigger('change');"
        );
        $Selenium->find_element("//button[\@value='Submit'][\@type='submit']")->VerifiedClick();

        # Navigate to AdminITSMConfigItem.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AdminITSMConfigItem");

        # Click on test class.
        $Selenium->find_element( $Class, 'link_text' )->VerifiedClick();

        # Click on 'Change class definition'.
        $Selenium->find_element("//button[\@value='Add'][\@type='submit']")->VerifiedClick();

        my $IntegerKey        = "TestInteger$RandomID";
        my $IntegerDefinition = << "EOF";
---
- Key: $IntegerKey
  Name: Test Integer
  Searchable: 1
  Input:
    Type:  Integer
    ValueMin: 0
    ValueMax: 10
    ValueDefault: 0
EOF

        $Selenium->find_element( "#Definition", 'css' )->send_keys($IntegerDefinition);
        $Selenium->find_element("//button[\@value='Submit'][\@type='submit']")->VerifiedClick();

        # Verify option with 0 in add config item screen.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentITSMConfigItemEdit;ClassID=$ClassID");
        $Self->True(
            $Selenium->find_element("//select[contains(\@id, \'$IntegerKey')]/option[\@value='0']"),
            "Add screen - Option with '0' value is found",
        );

        # Verify option with 0 in search config item screen.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentITSMConfigItemSearch");
        $Selenium->WaitFor( JavaScript => "return typeof(\$) === 'function' && \$('#SearchClassID').length;" );
        sleep 1;

        # Select test class.
        $Selenium->execute_script(
            "\$('#SearchClassID').val('$ClassID').trigger('redraw.InputField').trigger('change');"
        );
        $Selenium->WaitFor(
            JavaScript => 'return typeof($) === "function" && $("#Attribute").length;',
        );
        sleep 1;

        # Choose test integer field.
        $Selenium->execute_script(
            "\$('#Attribute').val('$IntegerKey').trigger('redraw.InputField').trigger('change');"
        );
        $Selenium->WaitFor(
            JavaScript => "return typeof(\$) === 'function' && \$('#SearchInsert #$IntegerKey').length;"
        );

        # Verify option with 0 exists.
        $Self->True(
            $Selenium->find_element("//select[\@id='$IntegerKey']/option[\@value='0']"),
            "Search screen - Option with '0' value is found",
        );

        # Cleanup.
        my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

        my $Success = $DBObject->Do(
            SQL  => "DELETE FROM configitem_definition WHERE class_id = ?",
            Bind => [ \$ClassID ],
        );
        $Self->True(
            $Success,
            "ConfigItem definitions from ClassID $ClassID is deleted",
        );

        $Success = $DBObject->Do(
            SQL  => "DELETE FROM general_catalog_preferences WHERE general_catalog_id = ?",
            Bind => [ \$ClassID ],
        );
        $Self->True(
            $Success,
            "CatalogItemID $ClassID preference is deleted",
        );

        $Success = $DBObject->Do(
            SQL  => "DELETE FROM general_catalog WHERE id = ?",
            Bind => [ \$ClassID ],
        );
        $Self->True(
            $Success,
            "CatalogItemID $ClassID is deleted",
        );

        # Make sure the cache is correct.
        my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');
        for my $Cache (qw(ConfigItem GeneralCatalog)) {
            $CacheObject->CleanUp( Type => $Cache );
        }

    }
);

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

my $Selenium = $Kernel::OM->Get('Kernel::System::UnitTest::Selenium');

$Selenium->RunTest(
    sub {

        my $Helper               = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
        my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
        my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');

        my %ShowColumnsByClassSysConfig = $Kernel::OM->Get('Kernel::System::SysConfig')->SettingGet(
            Name => 'ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumnsByClass',
        );

        push @{ $ShowColumnsByClassSysConfig{EffectiveValue} }, 'Computer::Owner::1';

        # Enable AgentITSMConfigItem###ShowColumnsByClass sysconfig item.
        $Helper->ConfigSettingChange(
            Valid => 1,
            Key   => 'ITSMConfigItem::Frontend::AgentITSMConfigItem###ShowColumnsByClass',
            Value => $ShowColumnsByClassSysConfig{EffectiveValue},
        );

        # Get catalog class IDs.
        my @ConfigItemClassIDs;
        for my $ConfigItemClass (qw(Computer Hardware Location Network Software)) {
            my $ConfigItemDataRef = $GeneralCatalogObject->ItemGet(
                Class => 'ITSM::ConfigItem::Class',
                Name  => $ConfigItemClass,
            );
            push @ConfigItemClassIDs, $ConfigItemDataRef->{ItemID};
        }

        # Get 'Production' deployment state ID.
        my $DeplStateDataRef = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::DeploymentState',
            Name  => 'Production',
        );
        my $DeplStateID = $DeplStateDataRef->{ItemID};

        my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

        my $Owner           = '"Test Test" <test@test.com>';
        my $ComputerClassID = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::Class',
            Name  => 'Computer',
        );

        # Create ConfigItem for each ConfigItem class.
        my @ConfigItemNumbers;
        my @ConfigItemIDs;
        for my $ITSMConfigItem (@ConfigItemClassIDs) {

            # Create ConfigItem number.
            my $ConfigItemNumber = $ConfigItemObject->ConfigItemNumberCreate(
                Type    => $ConfigObject->Get('ITSMConfigItem::NumberGenerator'),
                ClassID => $ITSMConfigItem,
            );
            $Self->True(
                $ConfigItemNumber,
                "ConfigItem number is created - $ConfigItemNumber"
            );
            push @ConfigItemNumbers, $ConfigItemNumber;

            # Add the new ConfigItem.
            my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
                Number  => $ConfigItemNumber,
                ClassID => $ITSMConfigItem,
                UserID  => 1,
            );
            $Self->True(
                $ConfigItemID,
                "ConfigItem is created - ID $ConfigItemID"
            );

            my %XMLData;
            if ( $ComputerClassID->{ItemID} == $ITSMConfigItem ) {
                %XMLData = (
                    XMLData => [
                        undef,
                        {
                            Version => [
                                undef,
                                {
                                    Owner => [
                                        undef,
                                        {
                                            Content => $Owner,
                                        },
                                    ],
                                },
                            ],
                        },
                    ],
                );
            }

            # Add a new version.
            my $VersionID = $ConfigItemObject->VersionAdd(
                Name         => 'SeleniumTest',
                DefinitionID => 1,
                DeplStateID  => $DeplStateID,
                InciStateID  => 1,
                UserID       => 1,
                ConfigItemID => $ConfigItemID,
                %XMLData,
            );
            $Self->True(
                $VersionID,
                "Version is created - ID $VersionID"
            );
            push @ConfigItemIDs, $ConfigItemID;
        }

        # Create test user and login.
        my $TestUserLogin = $Helper->TestUserCreate(
            Groups => [ 'admin', 'itsm-configitem' ],
        ) || die "Did not get test user";

        $Selenium->Login(
            Type     => 'Agent',
            User     => $TestUserLogin,
            Password => $TestUserLogin,
        );

        my $ScriptAlias = $ConfigObject->Get('ScriptAlias');

        # Navigate to AgentITSMConfigItem, sorted by created time.
        $Selenium->VerifiedGet(
            "${ScriptAlias}index.pl?Action=AgentITSMConfigItem;Filter=All;View=;;SortBy=ChangeTime;OrderBy=Down"
        );

        # Check for created test ConfigItems with 'All' filter active
        for my $AllConfigItem (@ConfigItemNumbers) {
            $Self->True(
                $Selenium->find_element("//div[contains(\@title, \'$AllConfigItem\' )]"),
                "Test ConfigItem number $AllConfigItem - found",
            );
        }

        # Check each of ConfigItem class filters for there respective test ConfigItem
        my $Count = 0;
        for my $CheckConfigItem (@ConfigItemIDs) {

            # Click on ConfigItem class
            $Selenium->find_element(
                "//a[contains(\@href, \'Action=AgentITSMConfigItem;SortBy=ChangeTime;OrderBy=Down;View=;Filter=$ConfigItemClassIDs[$Count]' )]"
            )->VerifiedClick();

            # Check for table structure
            $Selenium->find_element( "#OverviewBody .TableSmall", 'css' );

            # Check for ConfigItem number
            $Self->True(
                $Selenium->find_element("//div[contains(\@title, \'$ConfigItemNumbers[$Count]\' )]"),
                "Test ConfigItem number $ConfigItemNumbers[$Count] - found",
            );
            $Count++;

            # Check if there is column Create Time for Computer class
            # See bug#14049
            my $ConfigItemData = $ConfigItemObject->ConfigItemGet(
                ConfigItemID => $CheckConfigItem,
            );
            if ( $ConfigItemData->{Class} eq 'Computer' ) {

                # Check if AgentITSMConfigItem Owner is displayed correctly. See bug#14633.
                $Self->Is(
                    $Selenium->execute_script(
                        "return \$('#ConfigItemID_$CheckConfigItem td:contains($Owner)').text().trim();"
                    ),
                    $Owner,
                    'Owner name and address is displayed correctly',
                );

                $Self->True(
                    $Selenium->find_element("//a[contains(.,'Create Time')]"),
                    "There is column 'CreateTime', enabled by sysconfig item  AgentITSMConfigItem###ShowColumnsByClass",
                );
            }
        }

        # Verify there are links to AgentITSMConfigItemZoom on ConfigItem Number and Name columns.
        $Self->True(
            $Selenium->execute_script(
                "return \$('tr:eq(\"1\") div[title*=\"$ConfigItemNumbers[4]\"] a[href*=\"ItemID=$ConfigItemIDs[4]\"]').length;"
            ),
            "Link on ConfigItem 'Number' column correct."
        );
        $Self->True(
            $Selenium->execute_script(
                "return \$('tr:eq(\"1\") div[title*=\"SeleniumTest\"] a[href*=\"ItemID=$ConfigItemIDs[4]\"]').length;"
            ),
            "Link on ConfigItem 'Name' column correct."
        );

        # Delete created test ConfigItems.
        for my $ConfigItemDelete (@ConfigItemIDs) {
            my $Success = $ConfigItemObject->ConfigItemDelete(
                ConfigItemID => $ConfigItemDelete,
                UserID       => 1,
            );
            $Self->True(
                $Success,
                "ConfigItem is deleted - ID $ConfigItemDelete",
            );
        }
    }
);

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwp1c2UgdXRmODsKCnVzZSB2YXJzIChxdygkU2VsZikpOwoKIyBnZXQgc2VsZW5pdW0gb2JqZWN0Cm15ICRTZWxlbml1bSA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6U2VsZW5pdW0nKTsKCiRTZWxlbml1bS0+UnVuVGVzdCgKICAgIHN1YiB7CgogICAgICAgICMgZ2V0IGhlbHBlciBvYmplY3QKICAgICAgICBteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCiAgICAgICAgIyBnZXQgY2F0YWxvZyBjbGFzcyBJRHMKICAgICAgICBteSBAQ29uZmlnSXRlbUNsYXNzSURzOwogICAgICAgIG15IEBDb25maWdJdGVtQ2xhc3NOYW1lczsKICAgICAgICBmb3IgbXkgJENvbmZpZ0l0ZW1DbGFzcyAocXcoQ29tcHV0ZXIgSGFyZHdhcmUgTG9jYXRpb24gTmV0d29yayBTb2Z0d2FyZSkpIHsKICAgICAgICAgICAgbXkgJENvbmZpZ0l0ZW1EYXRhUmVmID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkdlbmVyYWxDYXRhbG9nJyktPkl0ZW1HZXQoCiAgICAgICAgICAgICAgICBDbGFzcyA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgICAgICAgICAgICAgTmFtZSAgPT4gJENvbmZpZ0l0ZW1DbGFzcywKICAgICAgICAgICAgKTsKICAgICAgICAgICAgcHVzaCBAQ29uZmlnSXRlbUNsYXNzSURzLCAgICRDb25maWdJdGVtRGF0YVJlZi0+e0l0ZW1JRH07CiAgICAgICAgICAgIHB1c2ggQENvbmZpZ0l0ZW1DbGFzc05hbWVzLCAkQ29uZmlnSXRlbURhdGFSZWYtPntOYW1lfTsKICAgICAgICB9CgogICAgICAgICMgY3JlYXRlIHRlc3QgdXNlciBhbmQgbG9naW4KICAgICAgICBteSAkVGVzdFVzZXJMb2dpbiA9ICRIZWxwZXItPlRlc3RVc2VyQ3JlYXRlKAogICAgICAgICAgICBHcm91cHMgPT4gWyAnYWRtaW4nLCAnaXRzbS1jb25maWdpdGVtJyBdLAogICAgICAgICkgfHwgZGllICJEaWQgbm90IGdldCB0ZXN0IHVzZXIiOwoKICAgICAgICAkU2VsZW5pdW0tPkxvZ2luKAogICAgICAgICAgICBUeXBlICAgICA9PiAnQWdlbnQnLAogICAgICAgICAgICBVc2VyICAgICA9PiAkVGVzdFVzZXJMb2dpbiwKICAgICAgICAgICAgUGFzc3dvcmQgPT4gJFRlc3RVc2VyTG9naW4sCiAgICAgICAgKTsKCiAgICAgICAgIyBnZXQgc2NyaXB0IGFsaWFzCiAgICAgICAgbXkgJFNjcmlwdEFsaWFzID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpDb25maWcnKS0+R2V0KCdTY3JpcHRBbGlhcycpOwoKICAgICAgICAjIG5hdmlnYXRlIHRvIEFnZW50SVRTTUNvbmZpZ0l0ZW1BZGQKICAgICAgICAkU2VsZW5pdW0tPlZlcmlmaWVkR2V0KCIke1NjcmlwdEFsaWFzfWluZGV4LnBsP0FjdGlvbj1BZ2VudElUU01Db25maWdJdGVtQWRkIik7CgogICAgICAgICMgY2hlY2sgZm9yIElUU01Db25maWdJdGVtIGNsYXNzZXMKICAgICAgICBmb3IgbXkgJENvbmZpZ0l0ZW1DbGFzc05hbWUgKEBDb25maWdJdGVtQ2xhc3NOYW1lcykgewogICAgICAgICAgICAkU2VsZi0+VHJ1ZSgKICAgICAgICAgICAgICAgIGluZGV4KCAkU2VsZW5pdW0tPmdldF9wYWdlX3NvdXJjZSgpLCAkQ29uZmlnSXRlbUNsYXNzTmFtZSApID4gLTEsCiAgICAgICAgICAgICAgICAiQ29uZmlnSXRlbSBjbGFzcyAkQ29uZmlnSXRlbUNsYXNzTmFtZSAtIGZvdW5kIiwKICAgICAgICAgICAgKTsKICAgICAgICB9CgogICAgICAgICMgY2hlY2sgZm9yIGxpbmtzIHRvICdBZGQnIG5ldyBpdGVtIGZvciBlYWNoIGNsYXNzCiAgICAgICAgZm9yIG15ICRDb25maWdJdGVtQ2xhc3NBZGQgKEBDb25maWdJdGVtQ2xhc3NJRHMpIHsKICAgICAgICAgICAgbXkgJFN1Y2Nlc3MgPSAkU2VsZW5pdW0tPmZpbmRfZWxlbWVudCgKICAgICAgICAgICAgICAgICIvL2FbY29udGFpbnMoXEBocmVmLCBcJ0FnZW50SVRTTUNvbmZpZ0l0ZW1FZGl0O0NsYXNzSUQ9JENvbmZpZ0l0ZW1DbGFzc0FkZCcgKV0iCiAgICAgICAgICAgICk7CiAgICAgICAgICAgICRTZWxmLT5UcnVlKAogICAgICAgICAgICAgICAgJFN1Y2Nlc3MsCiAgICAgICAgICAgICAgICAiQ29uZmlnSXRlbSBJRCAkQ29uZmlnSXRlbUNsYXNzQWRkIGFkZCBsaW5rIC0gZm91bmQiLAogICAgICAgICAgICApOwogICAgICAgIH0KICAgIH0KKTsKCjE7Cg==
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

my $Selenium = $Kernel::OM->Get('Kernel::System::UnitTest::Selenium');

$Selenium->RunTest(
    sub {

        my $Helper               = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
        my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
        my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
        my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');

        # Get 'Computer' catalog class IDs.
        my $ConfigItemDataRef = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::Class',
            Name  => 'Computer',
        );
        my $ComputerConfigItemID = $ConfigItemDataRef->{ItemID};

        # Get 'Production' and 'Repair' deployment state IDs.
        my @DeplStateIDs;
        for my $DeplState (qw (Production Repair)) {
            my $DeplStateDataRef = $GeneralCatalogObject->ItemGet(
                Class => 'ITSM::ConfigItem::DeploymentState',
                Name  => $DeplState,
            );
            push @DeplStateIDs, $DeplStateDataRef->{ItemID};
        }

        # Create three test ConfigItems for 'Computer' ConfigItem class.
        my @ConfigItems;
        for my $ITSMConfigItem ( 1 .. 3 ) {

            # Create ConfigItem number.
            my $ConfigItemNumber = $ConfigItemObject->ConfigItemNumberCreate(
                Type    => $ConfigObject->Get('ITSMConfigItem::NumberGenerator'),
                ClassID => $ComputerConfigItemID,
            );
            $Self->True(
                $ConfigItemNumber,
                "ConfigItem number is created - $ConfigItemNumber",
            );

            # Add the new ConfigItem.
            my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
                Number  => $ConfigItemNumber,
                ClassID => $ComputerConfigItemID,
                UserID  => 1,
            );
            $Self->True(
                $ConfigItemID,
                "ConfigItem is created - ID $ConfigItemID"
            );

            # Add a new version.
            my $VersionID = $ConfigItemObject->VersionAdd(
                Name         => 'SeleniumTest',
                DefinitionID => 1,
                DeplStateID  => $DeplStateIDs[0],
                InciStateID  => 1,
                UserID       => 1,
                ConfigItemID => $ConfigItemID,
            );
            $Self->True(
                $VersionID,
                "Version is created - ID $VersionID"
            );

            push @ConfigItems,
                {
                ID               => $ConfigItemID,
                ConfigItemNumber => $ConfigItemNumber,
                };
        }

        # Create test user and login.
        my $TestUserLogin = $Helper->TestUserCreate(
            Groups => [ 'admin', 'itsm-configitem' ],
        ) || die "Did not get test user";

        $Selenium->Login(
            Type     => 'Agent',
            User     => $TestUserLogin,
            Password => $TestUserLogin,
        );

        my $ScriptAlias = $ConfigObject->Get('ScriptAlias');

        # Navigate to AgentITSMConfigItem, sorted by created time.
        $Selenium->VerifiedGet(
            "${ScriptAlias}index.pl?Action=AgentITSMConfigItem;Filter=All;View=;;SortBy=ChangeTime;OrderBy=Down"
        );

        # Select two created test ConfigItems.
        for my $SelectConfigItem (@ConfigItems) {

            # Don't click on third test ConfigItem.
            if ( $SelectConfigItem->{ID} ne $ConfigItems[2]->{ID} ) {
                $Selenium->find_element("//input[\@value='$SelectConfigItem->{ID}']")->click();
            }
        }

        # Click on bulk action.
        $Selenium->find_element("//*[text()='Bulk']")->click();

        # Switch to 'Bulk' window.
        $Selenium->WaitFor( WindowCount => 2 );
        my $Handles = $Selenium->get_window_handles();
        $Selenium->switch_to_window( $Handles->[1] );

        # wait until page has loaded, if necessary
        $Selenium->WaitFor( JavaScript => 'return typeof($) === "function" && $("#DeplStateID").length;' );

        # Check screen
        for my $ID (
            qw(DeplStateID InciStateID LinkTogether LinkTogetherLinkType LinkTogetherAnother LinkType submitRichText)
            )
        {
            my $Element = $Selenium->find_element( "#$ID", 'css' );
            $Element->is_enabled();
            $Element->is_displayed();
        }

        # Change deployment state to 'Repair' for test ConfigItems
        $Selenium->execute_script("\$('#DeplStateID').val('$DeplStateIDs[1]');");

        # link 'Alternative to' test ConfigItems together
        $Selenium->find_element( "#LinkTogether option[value='1']",                           'css' )->click();
        $Selenium->find_element( "#LinkTogetherLinkType option[value='ConnectedTo::Source']", 'css' )->click();

        # link third test ConfigItem as part of first two
        $Selenium->find_element( "#LinkTogetherAnother", 'css' )->send_keys( $ConfigItems[2]->{ConfigItemNumber} );
        $Selenium->find_element( "#LinkType option[value='Includes::Target']", 'css' )->click();

        # Submit bulk changes
        $Selenium->find_element( "#submitRichText", 'css' )->click();

        # Switch window
        $Selenium->WaitFor( WindowCount => 1 );
        $Selenium->switch_to_window( $Handles->[0] );

        # Navigate to first created test ConfigItems zoom view.
        $Selenium->VerifiedGet(
            "${ScriptAlias}index.pl?Action=AgentITSMConfigItemZoom;ConfigItemID=$ConfigItems[1]->{ID}"
        );

        # Check for other two created test ConfigItems.
        # Verify that link action in bulk screen was success.
        my $Count = 1;
        ITEM:
        for my $CheckConfigItem (@ConfigItems) {

            next ITEM if $ConfigItems[1]->{ConfigItemNumber} == $CheckConfigItem->{ConfigItemNumber};
            $Self->True(
                $Selenium->find_element(
                    "//a[\@class=\'AsBlock LinkObjectLink\'][contains(.,\'$CheckConfigItem->{ConfigItemNumber}\')]"
                ),
                "Test ConfigItem number $CheckConfigItem->{ConfigItemNumber} - found",
            );
            $Count++;
        }

        $Self->Is(
            $Selenium->execute_script("return \$('.MasterAction td:eq(1) span').text();"),
            'Repair',
            "Deployment state is Repair.",
        );

        # Delete created test ConfigItems.
        for my $ConfigItemDelete (@ConfigItems) {
            my $Success = $ConfigItemObject->ConfigItemDelete(
                ConfigItemID => $ConfigItemDelete->{ID},
                UserID       => 1,
            );
            $Self->True(
                $Success,
                "Config item is deleted - ID $ConfigItemDelete->{ID}",
            );
        }
    }
);

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

my $Selenium = $Kernel::OM->Get('Kernel::System::UnitTest::Selenium');

$Selenium->RunTest(
    sub {

        my $Helper               = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
        my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
        my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
        my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');

        # Get 'Hardware' catalog class IDs.
        my $ConfigItemDataRef = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::Class',
            Name  => 'Hardware',
        );
        my $HardwareConfigItemID = $ConfigItemDataRef->{ItemID};

        # Get 'Production' deployment state IDs.
        my $ProductionDeplStateDataRef = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::DeploymentState',
            Name  => 'Production',
        );
        my $ProductionDeplStateID = $ProductionDeplStateDataRef->{ItemID};

        # Create ConfigItem number.
        my $ConfigItemNumber = $ConfigItemObject->ConfigItemNumberCreate(
            Type    => $ConfigObject->Get('ITSMConfigItem::NumberGenerator'),
            ClassID => $HardwareConfigItemID,
        );
        $Self->True(
            $ConfigItemNumber,
            "ConfigItem number is created - $ConfigItemNumber"
        );

        # Add the new ConfigItem.
        my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
            Number  => $ConfigItemNumber,
            ClassID => $HardwareConfigItemID,
            UserID  => 1,
        );
        $Self->True(
            $ConfigItemID,
            "ConfigItem is created - ID $ConfigItemID"
        );

        # Add a new version.
        my $ConfigItemName = 'Hardware' . $Helper->GetRandomID();
        my $VersionID      = $ConfigItemObject->VersionAdd(
            Name         => $ConfigItemName,
            DefinitionID => 1,
            DeplStateID  => $ProductionDeplStateID,
            InciStateID  => 1,
            UserID       => 1,
            ConfigItemID => $ConfigItemID,
        );
        $Self->True(
            $VersionID,
            "Version is created - ID $VersionID"
        );

        # Create test user and login.
        my $TestUserLogin = $Helper->TestUserCreate(
            Groups => [ 'admin', 'itsm-configitem' ],
        ) || die "Did not get test user";

        $Selenium->Login(
            Type     => 'Agent',
            User     => $TestUserLogin,
            Password => $TestUserLogin,
        );

        my $ScriptAlias = $ConfigObject->Get('ScriptAlias');

        # Navigate to AgentITSMConfigItemZoom screen.
        $Selenium->VerifiedGet(
            "${ScriptAlias}index.pl?Action=AgentITSMConfigItemZoom;ConfigItemID=$ConfigItemID;Version=$VersionID"
        );

        # Get ConfigItem value params.
        my @ConfigItemValues = (
            {
                Value => 'Hardware',
                Check => 'p.Value:contains(Hardware)',
            },
            {
                Value => $ConfigItemName,
                Check => "h1:contains($ConfigItemName)",
            },
        );

        # Check ConfigItem values on screen
        for my $CheckConfigItemValue (@ConfigItemValues) {
            $Self->True(
                $Selenium->execute_script(
                    "return \$('$CheckConfigItemValue->{Check}').length;"
                ),
                "Test ConfigItem value $CheckConfigItemValue->{Value} - found",
            );
        }

        # Click on 'Duplicate' menu item and switch window
        $Selenium->find_element(
            "//a[contains(\@href, \'Action=AgentITSMConfigItemEdit;DuplicateID=$ConfigItemID;VersionID=$VersionID\' )]"
        )->click();

        # switch window
        $Selenium->WaitFor( WindowCount => 2 );
        my $Handles = $Selenium->get_window_handles();
        $Selenium->switch_to_window( $Handles->[1] );

        # wait until page has loaded, if necessary
        $Selenium->WaitFor( JavaScript => 'return typeof($) === "function" && $("#DeplStateID").length;' );

        # Check for created ConfigItem values
        $Self->Is(
            $Selenium->find_element( '#Name', 'css' )->get_value(),
            $ConfigItemName,
            "#Name stored value",
        );
        $Self->Is(
            $Selenium->find_element( '#DeplStateID', 'css' )->get_value(),
            $ProductionDeplStateID,
            "#DeplStateID stored value",
        );
        $Self->Is(
            $Selenium->find_element( '#InciStateID', 'css' )->get_value(),
            1,
            "#InciStateID stored value",
        );

        # Edit name for duplicate test ConfigItem.
        my $DuplicateConfigItemName = "Duplicate" . $ConfigItemName;
        $Selenium->find_element( "#Name", 'css' )->clear();
        $Selenium->find_element( "#Name", 'css' )->send_keys($DuplicateConfigItemName);
        sleep 2;
        $Selenium->find_element("//button[\@value='Submit'][\@type='submit']")->click();

        # Switch back to zoom view.
        $Selenium->WaitFor( WindowCount => 1 );
        $Selenium->switch_to_window( $Handles->[0] );

        # Wait until page has loaded, if necessary.
        $Selenium->WaitFor(
            JavaScript =>
                'return typeof(Core) == "object" && typeof(Core.App) == "object" && Core.App.PageLoadComplete;'
        );
        $Selenium->WaitFor(
            JavaScript => "return typeof(\$) === 'function' && \$('h1:contains($DuplicateConfigItemName)').length;"
        );

        # Check for duplicated ConfigItem value.
        $Self->True(
            $Selenium->execute_script(
                "return \$('h1:contains($DuplicateConfigItemName)').length;"
            ),
            "Test ConfigItem value $DuplicateConfigItemName - found",
        );

        # Delete created test ConfigItems.
        for my $DeleteConfigItem ( 1 .. 2 ) {
            my $Success = $ConfigItemObject->ConfigItemDelete(
                ConfigItemID => $ConfigItemID,
                UserID       => 1,
            );
            $Self->True(
                $Success,
                "ConfigItem is deleted - ID $ConfigItemID",
            );
            $ConfigItemID++;
        }
    }
);

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

my $Selenium = $Kernel::OM->Get('Kernel::System::UnitTest::Selenium');

$Selenium->RunTest(
    sub {

        my $Helper               = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
        my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
        my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

        my @Test = (
            {
                ConfigItemClass => 'Computer',
                CheckEditFields => [
                    'Name', 'DeplStateID', 'InciStateID', 'Vendor', 'Model', 'Description', 'Type', 'Owner',
                    'SerialNumber',
                    'OperatingSystem', 'CPU', 'Ram', 'HardDisk', 'Capacity', 'FQDN', 'NIC', 'PoverDHCP',
                    'GraphicAdapter',
                    'OtherEquipment', 'WarrantyExpirationDate', 'InstallDate', 'Note', 'FileUpload', 'SubmitSave'
                ],
            },
            {
                ConfigItemClass => 'Hardware',
                CheckEditFields => [
                    'Name', 'DeplStateID', 'InciStateID', 'Vendor', 'Model', 'Description', 'Type', 'Owner',
                    'SerialNumber',
                    'WarrantyExpirationDate', 'InstallDate', 'Note', 'FileUpload', 'SubmitSave'
                ],
            },
            {
                ConfigItemClass => 'Location',
                CheckEditFields => [
                    'Name', 'DeplStateID', 'InciStateID', 'Type', 'Phone1', 'Phone2', 'Fax', 'E-Mail', 'Address',
                    'Note', 'FileUpload', 'SubmitSave'
                ],
            },
            {
                ConfigItemClass => 'Network',
                CheckEditFields => [
                    'Name', 'DeplStateID', 'InciStateID', 'Description', 'Type', 'NetworkAddress', 'SubnetMask',
                    'Gateway',
                    'Note', 'FileUpload', 'SubmitSave'
                ],
            },
            {
                ConfigItemClass => 'Software',
                CheckEditFields => [
                    'Name', 'DeplStateID', 'InciStateID', 'Vendor', 'Version', 'Description', 'Type', 'Owner',
                    'SerialNumber',
                    'LicenceType', 'LicenceKey', 'Media', 'Note', 'SubmitSave'
                ],
            },
        );

        # Get 'Production' deployment state ID.
        my $DeplStateDataRef = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::DeploymentState',
            Name  => 'Production',
        );
        my $DeplStateID = $DeplStateDataRef->{ItemID};

        # Create test user and login.
        my $TestUserLogin = $Helper->TestUserCreate(
            Groups => [ 'admin', 'itsm-configitem' ],
        ) || die "Did not get test user";

        $Selenium->Login(
            Type     => 'Agent',
            User     => $TestUserLogin,
            Password => $TestUserLogin,
        );

        my $ScriptAlias = $Kernel::OM->Get('Kernel::Config')->Get('ScriptAlias');
        my $RandomID    = $Helper->GetRandomID();

        for my $ConfigItemEdit (@Test) {

            # Navigate to AgentITSMConfigItemAdd screen.
            $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentITSMConfigItemAdd");

            # Get ConfigItem class ID.
            my $ConfigItemDataRef = $GeneralCatalogObject->ItemGet(
                Class => 'ITSM::ConfigItem::Class',
                Name  => $ConfigItemEdit->{ConfigItemClass},
            );
            my $ConfigItemClassID = $ConfigItemDataRef->{ItemID};

            $Selenium->WaitFor(
                JavaScript =>
                    "return typeof(\$) === 'function' && \$('a[href*=\"Action=AgentITSMConfigItemEdit;ClassID=$ConfigItemClassID\"]').length"
            );

            # Click on ConfigItem class.
            $Selenium->find_element(
                "//a[contains(\@href, \'Action=AgentITSMConfigItemEdit;ClassID=$ConfigItemClassID\' )]"
            )->VerifiedClick();

            $Selenium->WaitFor(
                JavaScript => "return typeof(\$) === 'function' && \$('#Name').length && \$('#SubmitButton').length"
            );

            # Check for ConfigItemEdit fields.
            for my $CheckConfigItemField ( @{ $ConfigItemEdit->{CheckEditFields} } ) {

                my $Element = $Selenium->find_element("//*[contains(\@name, \'$CheckConfigItemField\' )]");
                $Element->is_enabled();
                $Element->is_displayed();
            }

            # Create test ConfigItem.
            my $ConfigItemName = $ConfigItemEdit->{ConfigItemClass} . $RandomID;
            $Selenium->find_element( "#Name", 'css' )->send_keys($ConfigItemName);

            $Selenium->execute_script(
                "\$('#DeplStateID').val('$DeplStateID').trigger('redraw.InputField').trigger('change')"
            );
            $Selenium->WaitFor(
                JavaScript => "return typeof(\$) === 'function' && \$('#DeplStateID').val() === '$DeplStateID'"
            );

            $Selenium->execute_script("\$('#InciStateID').val('1').trigger('redraw.InputField').trigger('change')");
            $Selenium->WaitFor( JavaScript => "return typeof(\$) === 'function' && \$('#InciStateID').val() === '1'" );

            if ( $ConfigItemEdit->{ConfigItemClass} eq 'Computer' ) {

                # Get General Catalog ID for 'Yes'.
                my $YesDataRef = $GeneralCatalogObject->ItemGet(
                    Class => 'ITSM::ConfigItem::YesNo',
                    Name  => 'Yes',
                );
                my $YesID = $YesDataRef->{ItemID};

                # Enter NIC name.
                $Selenium->find_element("//*[contains(\@name, \'NIC::1\' )]")->send_keys('SeleniumNetwork');

                # Select Yes for DHCPOverIP.
                $Selenium->execute_script(
                    "\$('#' + Core.App.EscapeSelector('Item1NIC::11')).val('$YesID').trigger('redraw.InputField').trigger('change');"
                );
            }
            if ( $ConfigItemEdit->{ConfigItemClass} eq 'Network' ) {
                $Selenium->find_element("//*[contains(\@name, \'NetworkAddress\' )]")->send_keys('SeleniumNetwork');
            }

            $Selenium->find_element("//button[\@value='Submit'][\@type='submit']")->VerifiedClick();

            $Selenium->WaitFor(
                JavaScript => "return typeof(\$) === 'function' && \$('h1:contains($ConfigItemName)').length"
            );

            # Get ConfigItem value.
            my @ConfigItemValues = (
                {
                    Value       => $ConfigItemName,
                    Check       => "h1:contains($ConfigItemName)",
                    CheckResult => 1,
                },
                {
                    Value       => $ConfigItemEdit->{ConfigItemClass},
                    Check       => "p.Value:contains($ConfigItemEdit->{ConfigItemClass})",
                    CheckResult => 2,
                },
            );

            # Check submitted values in AgentITSMConfigItemZoom screen.
            for my $CheckConfigItemValue (@ConfigItemValues) {
                $Self->True(
                    $Selenium->execute_script(
                        "return \$('$CheckConfigItemValue->{Check}').length === $CheckConfigItemValue->{CheckResult}"
                    ),
                    "Test ConfigItem value $CheckConfigItemValue->{Value} - found",
                );
            }

            # Get ConfigItemID.
            my $ConfigItemID = $ConfigItemObject->VersionSearch(
                Name => $ConfigItemName
            );

            # Delete created test ConfigItem.
            my $Success = $ConfigItemObject->ConfigItemDelete(
                ConfigItemID => $ConfigItemID->[0],
                UserID       => 1,
            );
            $Self->True(
                $Success,
                "ConfigItem is deleted - ID $ConfigItemID->[0]",
            );
        }

        # Check multiple Customer type fields (see bug#14218 - https://bugs.otrs.org/show_bug.cgi?id=14218).
        # Create test CI class.
        my $ClassName = 'Class' . $Helper->GetRandomID();
        my $ClassID   = $GeneralCatalogObject->ItemAdd(
            Class   => 'ITSM::ConfigItem::Class',
            Name    => $ClassName,
            ValidID => 1,
            UserID  => 1,
        );
        $Self->True(
            $ClassID,
            "ClassID $ClassID is created",
        );

        my $GroupID = $Kernel::OM->Get('Kernel::System::Group')->GroupLookup(
            Group  => 'itsm-configitem',
            UserID => 1,
        );

        # Set permission.
        $Kernel::OM->Get('Kernel::System::GeneralCatalog')->GeneralCatalogPreferencesSet(
            ItemID => $ClassID,
            Key    => 'Permission',
            Value  => $GroupID,
        );

        my $Definition = << "EOF";
---
- Key: CustomerID1
  Name: CustomerCompany 1
  Searchable: 1
  Input:
    Type: CustomerCompany
  Sub:
  - Key: Customer1
    Name: Customer 1
    Searchable: 1
    Input:
      Type: Customer

- Key: CustomerID2
  Name: CustomerCompany 2
  Searchable: 1
  Input:
    Type: CustomerCompany
  Sub:
  - Key: Customer2
    Name: Customer 2
    Searchable: 1
    Input:
      Type: Customer

- Key: CustomerID3
  Name: CustomerCompany 3
  Searchable: 1
  Input:
    Type: CustomerCompany
  Sub:
  - Key: Customer3
    Name: Customer 3
    Searchable: 1
    Input:
      Type: Customer
EOF

        # Add test definition to test CI class.
        my $DefinitionID = $ConfigItemObject->DefinitionAdd(
            ClassID    => $ClassID,
            Definition => $Definition,
            UserID     => 1,
        );
        $Self->True(
            $DefinitionID,
            "DefinitionID $DefinitionID is created",
        );

        # Navigate to AgentITSMConfigItemAdd screen.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentITSMConfigItemEdit;ClassID=$ClassID");

        # Verify all of three sub elements are autocomplete input fields.
        $Self->Is(
            $Selenium->execute_script(
                "return \$('.SubElement .ITSMCustomerSearch.ui-autocomplete-input').length;"
            ),
            3,
            "All sub elements work correctly - they are autocomplete input fields",
        );

        # Cleanup.
        my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

        # Delete test definition.
        my $Success = $DBObject->Do(
            SQL  => "DELETE FROM configitem_definition WHERE id = ?",
            Bind => [ \$DefinitionID ],
        );
        $Self->True(
            $Success,
            "Config item definition is deleted",
        );

        $Success = $DBObject->Do(
            SQL  => "DELETE FROM general_catalog_preferences WHERE general_catalog_id = ?",
            Bind => [ \$ClassID ],
        );
        $Self->True(
            $Success,
            "General catalog preferences for ClassID $ClassID is deleted",
        );

        $Success = $DBObject->Do(
            SQL  => "DELETE FROM general_catalog WHERE id = ?",
            Bind => [ \$ClassID ],
        );
        $Self->True(
            $Success,
            "ClassID $ClassID is deleted",
        );

    }
);

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwp1c2UgdXRmODsKCnVzZSB2YXJzIChxdygkU2VsZikpOwoKIyBnZXQgc2VsZW5pdW0gb2JqZWN0Cm15ICRTZWxlbml1bSA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6U2VsZW5pdW0nKTsKCiRTZWxlbml1bS0+UnVuVGVzdCgKICAgIHN1YiB7CgogICAgICAgICMgZ2V0IG5lZWRlZCBvYmplY3RzCiAgICAgICAgbXkgJEhlbHBlciAgICAgICAgICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKICAgICAgICBteSAkR2VuZXJhbENhdGFsb2dPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6R2VuZXJhbENhdGFsb2cnKTsKCiAgICAgICAgIyBnZXQgJ0hhcmR3YXJlJyBjYXRhbG9nIGNsYXNzIElEcwogICAgICAgIG15ICRDb25maWdJdGVtRGF0YVJlZiA9ICRHZW5lcmFsQ2F0YWxvZ09iamVjdC0+SXRlbUdldCgKICAgICAgICAgICAgQ2xhc3MgPT4gJ0lUU006OkNvbmZpZ0l0ZW06OkNsYXNzJywKICAgICAgICAgICAgTmFtZSAgPT4gJ0hhcmR3YXJlJywKICAgICAgICApOwogICAgICAgIG15ICRIYXJkd2FyZUNvbmZpZ0l0ZW1JRCA9ICRDb25maWdJdGVtRGF0YVJlZi0+e0l0ZW1JRH07CgogICAgICAgICMgZ2V0ICdQcm9kdWN0aW9uJyBkZXBsb3ltZW50IHN0YXRlIElEcwogICAgICAgIG15ICRQcm9kdWN0aW9uRGVwbFN0YXRlRGF0YVJlZiA9ICRHZW5lcmFsQ2F0YWxvZ09iamVjdC0+SXRlbUdldCgKICAgICAgICAgICAgQ2xhc3MgPT4gJ0lUU006OkNvbmZpZ0l0ZW06OkRlcGxveW1lbnRTdGF0ZScsCiAgICAgICAgICAgIE5hbWUgID0+ICdQcm9kdWN0aW9uJywKICAgICAgICApOwogICAgICAgIG15ICRQcm9kdWN0aW9uRGVwbFN0YXRlSUQgPSAkUHJvZHVjdGlvbkRlcGxTdGF0ZURhdGFSZWYtPntJdGVtSUR9OwoKICAgICAgICAjIGdldCBuZWVkZWQgb2JqZWN0cwogICAgICAgIG15ICRDb25maWdJdGVtT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtJyk7CiAgICAgICAgbXkgJENvbmZpZ09iamVjdCAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OkNvbmZpZycpOwoKICAgICAgICAjIGNyZWF0ZSBDb25maWdJdGVtIG51bWJlcgogICAgICAgIG15ICRDb25maWdJdGVtTnVtYmVyID0gJENvbmZpZ0l0ZW1PYmplY3QtPkNvbmZpZ0l0ZW1OdW1iZXJDcmVhdGUoCiAgICAgICAgICAgIFR5cGUgICAgPT4gJENvbmZpZ09iamVjdC0+R2V0KCdJVFNNQ29uZmlnSXRlbTo6TnVtYmVyR2VuZXJhdG9yJyksCiAgICAgICAgICAgIENsYXNzSUQgPT4gJEhhcmR3YXJlQ29uZmlnSXRlbUlELAogICAgICAgICk7CiAgICAgICAgJFNlbGYtPlRydWUoCiAgICAgICAgICAgICRDb25maWdJdGVtTnVtYmVyLAogICAgICAgICAgICAiQ29uZmlnSXRlbSBudW1iZXIgaXMgY3JlYXRlZCAtICRDb25maWdJdGVtTnVtYmVyIgogICAgICAgICk7CgogICAgICAgICMgYWRkIHRoZSBuZXcgQ29uZmlnSXRlbQogICAgICAgIG15ICRDb25maWdJdGVtSUQgPSAkQ29uZmlnSXRlbU9iamVjdC0+Q29uZmlnSXRlbUFkZCgKICAgICAgICAgICAgTnVtYmVyICA9PiAkQ29uZmlnSXRlbU51bWJlciwKICAgICAgICAgICAgQ2xhc3NJRCA9PiAkSGFyZHdhcmVDb25maWdJdGVtSUQsCiAgICAgICAgICAgIFVzZXJJRCAgPT4gMSwKICAgICAgICApOwogICAgICAgICRTZWxmLT5UcnVlKAogICAgICAgICAgICAkQ29uZmlnSXRlbUlELAogICAgICAgICAgICAiQ29uZmlnSXRlbSBpcyBjcmVhdGVkIC0gSUQgJENvbmZpZ0l0ZW1JRCIKICAgICAgICApOwoKICAgICAgICAjIGFkZCBhIG5ldyB2ZXJzaW9uCiAgICAgICAgbXkgJFZlcnNpb25JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5WZXJzaW9uQWRkKAogICAgICAgICAgICBOYW1lICAgICAgICAgPT4gJ1NlbGVuaXVtVGVzdCcsCiAgICAgICAgICAgIERlZmluaXRpb25JRCA9PiAxLAogICAgICAgICAgICBEZXBsU3RhdGVJRCAgPT4gJFByb2R1Y3Rpb25EZXBsU3RhdGVJRCwKICAgICAgICAgICAgSW5jaVN0YXRlSUQgID0+IDEsCiAgICAgICAgICAgIFVzZXJJRCAgICAgICA9PiAxLAogICAgICAgICAgICBDb25maWdJdGVtSUQgPT4gJENvbmZpZ0l0ZW1JRCwKICAgICAgICApOwogICAgICAgICRTZWxmLT5UcnVlKAogICAgICAgICAgICAkVmVyc2lvbklELAogICAgICAgICAgICAiVmVyc2lvbiBpcyBjcmVhdGVkIC0gSUQgJFZlcnNpb25JRCIKICAgICAgICApOwoKICAgICAgICAjIGNyZWF0ZSB0ZXN0IHVzZXIgYW5kIGxvZ2luCiAgICAgICAgbXkgJFRlc3RVc2VyTG9naW4gPSAkSGVscGVyLT5UZXN0VXNlckNyZWF0ZSgKICAgICAgICAgICAgR3JvdXBzID0+IFsgJ2FkbWluJywgJ2l0c20tY29uZmlnaXRlbScgXSwKICAgICAgICApIHx8IGRpZSAiRGlkIG5vdCBnZXQgdGVzdCB1c2VyIjsKCiAgICAgICAgJFNlbGVuaXVtLT5Mb2dpbigKICAgICAgICAgICAgVHlwZSAgICAgPT4gJ0FnZW50JywKICAgICAgICAgICAgVXNlciAgICAgPT4gJFRlc3RVc2VyTG9naW4sCiAgICAgICAgICAgIFBhc3N3b3JkID0+ICRUZXN0VXNlckxvZ2luLAogICAgICAgICk7CgogICAgICAgICMgZ2V0IHRlc3QgdXNlciBJRAogICAgICAgIG15ICRUZXN0VXNlcklEID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVzZXInKS0+VXNlckxvb2t1cCgKICAgICAgICAgICAgVXNlckxvZ2luID0+ICRUZXN0VXNlckxvZ2luLAogICAgICAgICk7CgogICAgICAgICMgZ2V0IHNjcmlwdCBhbGlhcwogICAgICAgIG15ICRTY3JpcHRBbGlhcyA9ICRDb25maWdPYmplY3QtPkdldCgnU2NyaXB0QWxpYXMnKTsKCiAgICAgICAgIyBjaGVjayBmb3IgZXJyb3IgbWVzc2FnZSB3aGVuIG5vIENvbmZpZ0l0ZW1JRCBpcyBwcm92aWRlZCBpbiBoaXN0b3J5IHNjcmVlbgogICAgICAgICRTZWxlbml1bS0+VmVyaWZpZWRHZXQoIiR7U2NyaXB0QWxpYXN9aW5kZXgucGw/QWN0aW9uPUFnZW50SVRTTUNvbmZpZ0l0ZW1IaXN0b3J5Iik7CgogICAgICAgIG15ICRFcnJvck1lc3NhZ2VOb0lEID0gJ0NhblwndCBzaG93IGhpc3RvcnksIG5vIENvbmZpZ0l0ZW1JRCBpcyBnaXZlbiEnOwogICAgICAgICRTZWxmLT5UcnVlKAogICAgICAgICAgICBpbmRleCggJFNlbGVuaXVtLT5nZXRfcGFnZV9zb3VyY2UoKSwgJEVycm9yTWVzc2FnZU5vSUQgKSA+IC0xLAogICAgICAgICAgICAiRXJyb3IgbWVzc2FnZSAkRXJyb3JNZXNzYWdlTm9JRCAtIGZvdW5kIiwKICAgICAgICApOwoKICAgICAgICAjIGNyZWF0ZSBoaXN0b3J5IG1lc3NhZ2VzIGxpc3QKICAgICAgICBteSBASGlzdG9yeU1lc3NhZ2VzID0gKAogICAgICAgICAgICAnTmV3IENvbmZpZ0l0ZW0gKElEPScgLiAkQ29uZmlnSXRlbUlEIC4gJyknLAogICAgICAgICAgICAnTmV3IHZlcnNpb24gKElEPScgLiAkVmVyc2lvbklEIC4gJyknLAogICAgICAgICAgICAnQ29uZmlnSXRlbSBkZWZpbml0aW9uIHVwZGF0ZWQgKElEPTEpJywKICAgICAgICAgICAgJ05hbWUgdXBkYXRlZCAobmV3PVNlbGVuaXVtVGVzdCwgb2xkPSknLAogICAgICAgICAgICAnSW5jaWRlbnQgc3RhdGUgdXBkYXRlZCAobmV3PU9wZXJhdGlvbmFsLCBvbGQ9KScsCiAgICAgICAgICAgICdEZXBsb3ltZW50IHN0YXRlIHVwZGF0ZWQgKG5ldz1Qcm9kdWN0aW9uLCBvbGQ9KScsCiAgICAgICAgKTsKCiAgICAgICAgIyBjaGVjayBmb3IgaGlzdG9yeSBtZXNzYWdlcyBpbiBoaXN0b3J5IHNjcmVlbiBvZiBjcmVhdGVkIHRlc3QgQ29uZmlnSXRlbQogICAgICAgICRTZWxlbml1bS0+VmVyaWZpZWRHZXQoCiAgICAgICAgICAgICIke1NjcmlwdEFsaWFzfWluZGV4LnBsP0FjdGlvbj1BZ2VudElUU01Db25maWdJdGVtSGlzdG9yeTtDb25maWdJdGVtSUQ9JENvbmZpZ0l0ZW1JRDtWZXJzaW9uSUQ9JFZlcnNpb25JRCIKICAgICAgICApOwoKICAgICAgICBmb3IgbXkgJEhpc3RvcnlNZXNzYWdlIChASGlzdG9yeU1lc3NhZ2VzKSB7CiAgICAgICAgICAgICRTZWxmLT5UcnVlKAogICAgICAgICAgICAgICAgaW5kZXgoICRTZWxlbml1bS0+Z2V0X3BhZ2Vfc291cmNlKCksICRIaXN0b3J5TWVzc2FnZSApID4gLTEsCiAgICAgICAgICAgICAgICAiSGlzdG9yeSBtZXNzYWdlICRIaXN0b3J5TWVzc2FnZSAtIGZvdW5kIiwKICAgICAgICAgICAgKTsKICAgICAgICB9CgogICAgICAgICMgcmVtb3ZlIGl0c20tY29uZmlnaXRlbSAncm8nIGFjY2VzcyByaWdodCBmb3IgdGVzdCB1c2VyCiAgICAgICAgIyBnZXQgZ3JvdXAgb2JqZWN0CiAgICAgICAgbXkgJEdyb3VwT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06Okdyb3VwJyk7CgogICAgICAgICMgZ2V0ICdpdHNtLWNvbmZpZ2l0ZW0nIGdyb3VwIElECiAgICAgICAgbXkgJElUU01Db25maWdJdGVtR3JvdXBJRCA9ICRHcm91cE9iamVjdC0+R3JvdXBMb29rdXAoCiAgICAgICAgICAgIEdyb3VwID0+ICdpdHNtLWNvbmZpZ2l0ZW0nLAogICAgICAgICk7CgogICAgICAgICMgcmVtb3ZlIGFjY2VzcyByaWdodHMgdG8gdGVzdCB1c2VyCiAgICAgICAgJEdyb3VwT2JqZWN0LT5Hcm91cE1lbWJlckFkZCgKICAgICAgICAgICAgR0lEICAgICAgICA9PiAkSVRTTUNvbmZpZ0l0ZW1Hcm91cElELAogICAgICAgICAgICBVSUQgICAgICAgID0+ICRUZXN0VXNlcklELAogICAgICAgICAgICBQZXJtaXNzaW9uID0+IHsKICAgICAgICAgICAgICAgIHJvID0+IDAsCiAgICAgICAgICAgIH0sCiAgICAgICAgICAgIFVzZXJJRCA9PiAxLAogICAgICAgICk7CgogICAgICAgICMgY2hlY2sgZm9yIGVycm9yIG1lc3NhZ2Ugd2hlbiB1c2VyIGhhdmUgbm8gYWNjZXNzIHJpZ2h0cyBpbiBoaXN0b3J5IHNjcmVlbgogICAgICAgICRTZWxlbml1bS0+VmVyaWZpZWRHZXQoCiAgICAgICAgICAgICIke1NjcmlwdEFsaWFzfWluZGV4LnBsP0FjdGlvbj1BZ2VudElUU01Db25maWdJdGVtSGlzdG9yeTtDb25maWdJdGVtSUQ9JENvbmZpZ0l0ZW1JRDtWZXJzaW9uSUQ9JFZlcnNpb25JRCIKICAgICAgICApOwoKICAgICAgICBteSAkRXJyb3JNZXNzYWdlTm9QZXJtaXNzaW9uID0gJ05vIFBlcm1pc3Npb24gdG8gdXNlIHRoaXMgZnJvbnRlbmQgbW9kdWxlISc7CiAgICAgICAgJFNlbGYtPlRydWUoCiAgICAgICAgICAgIGluZGV4KCAkU2VsZW5pdW0tPmdldF9wYWdlX3NvdXJjZSgpLCAkRXJyb3JNZXNzYWdlTm9QZXJtaXNzaW9uICkgPiAtMSwKICAgICAgICAgICAgIkVycm9yIG1lc3NhZ2UgJEVycm9yTWVzc2FnZU5vUGVybWlzc2lvbiAtIGZvdW5kIiwKICAgICAgICApOwoKICAgICAgICAjIGRlbGV0ZSBjcmVhdGVkIHRlc3QgQ29uZmlnSXRlbQogICAgICAgIG15ICRTdWNjZXNzID0gJENvbmZpZ0l0ZW1PYmplY3QtPkNvbmZpZ0l0ZW1EZWxldGUoCiAgICAgICAgICAgIENvbmZpZ0l0ZW1JRCA9PiAkQ29uZmlnSXRlbUlELAogICAgICAgICAgICBVc2VySUQgICAgICAgPT4gMSwKICAgICAgICApOwogICAgICAgICRTZWxmLT5UcnVlKAogICAgICAgICAgICAkU3VjY2VzcywKICAgICAgICAgICAgIkNvbmZpZ0l0ZW0gaXMgZGVsZXRlZCAtIElEICRDb25maWdJdGVtSUQiLAogICAgICAgICk7CiAgICB9Cik7CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDIxIE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KIyAtLQoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwp1c2UgdXRmODsKCnVzZSB2YXJzIChxdygkU2VsZikpOwoKbXkgJFNlbGVuaXVtID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpTZWxlbml1bScpOwoKaWYgKCAkU2VsZW5pdW0tPnticm93c2VyX25hbWV9IG5lICdmaXJlZm94JyApIHsKICAgICRTZWxmLT5UcnVlKAogICAgICAgIDEsCiAgICAgICAgIlBERiB0ZXN0IGN1cnJlbnRseSBvbmx5IHN1cHBvcnRzIEZpcmVmb3giLAogICAgKTsKICAgIHJldHVybiAxOwp9CgokU2VsZW5pdW0tPlJ1blRlc3QoCiAgICBzdWIgewoKICAgICAgICBteSAkSGVscGVyICAgICAgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VW5pdFRlc3Q6OkhlbHBlcicpOwogICAgICAgIG15ICRHZW5lcmFsQ2F0YWxvZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpOwogICAgICAgIG15ICRDb25maWdJdGVtT2JqZWN0ICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbScpOwogICAgICAgIG15ICRDb25maWdPYmplY3QgICAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyk7CgogICAgICAgICMgR2V0ICdIYXJkd2FyZScgY2F0YWxvZyBjbGFzcyBJRHMuCiAgICAgICAgbXkgJENvbmZpZ0l0ZW1EYXRhUmVmID0gJEdlbmVyYWxDYXRhbG9nT2JqZWN0LT5JdGVtR2V0KAogICAgICAgICAgICBDbGFzcyA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgICAgICAgICBOYW1lICA9PiAnSGFyZHdhcmUnLAogICAgICAgICk7CiAgICAgICAgbXkgJEhhcmR3YXJlQ29uZmlnSXRlbUlEID0gJENvbmZpZ0l0ZW1EYXRhUmVmLT57SXRlbUlEfTsKCiAgICAgICAgIyBHZXQgJ1Byb2R1Y3Rpb24nIGRlcGxveW1lbnQgc3RhdGUgSURzLgogICAgICAgIG15ICRQcm9kdWN0aW9uRGVwbFN0YXRlRGF0YVJlZiA9ICRHZW5lcmFsQ2F0YWxvZ09iamVjdC0+SXRlbUdldCgKICAgICAgICAgICAgQ2xhc3MgPT4gJ0lUU006OkNvbmZpZ0l0ZW06OkRlcGxveW1lbnRTdGF0ZScsCiAgICAgICAgICAgIE5hbWUgID0+ICdQcm9kdWN0aW9uJywKICAgICAgICApOwogICAgICAgIG15ICRQcm9kdWN0aW9uRGVwbFN0YXRlSUQgPSAkUHJvZHVjdGlvbkRlcGxTdGF0ZURhdGFSZWYtPntJdGVtSUR9OwoKICAgICAgICAjIENyZWF0ZSBDb25maWdJdGVtIG51bWJlci4KICAgICAgICBteSAkQ29uZmlnSXRlbU51bWJlciA9ICRDb25maWdJdGVtT2JqZWN0LT5Db25maWdJdGVtTnVtYmVyQ3JlYXRlKAogICAgICAgICAgICBUeXBlICAgID0+ICRDb25maWdPYmplY3QtPkdldCgnSVRTTUNvbmZpZ0l0ZW06Ok51bWJlckdlbmVyYXRvcicpLAogICAgICAgICAgICBDbGFzc0lEID0+ICRIYXJkd2FyZUNvbmZpZ0l0ZW1JRCwKICAgICAgICApOwogICAgICAgICRTZWxmLT5UcnVlKAogICAgICAgICAgICAkQ29uZmlnSXRlbU51bWJlciwKICAgICAgICAgICAgIkNvbmZpZ0l0ZW0gbnVtYmVyIGlzIGNyZWF0ZWQgLSAkQ29uZmlnSXRlbU51bWJlciIKICAgICAgICApOwoKICAgICAgICAjIEFkZCB0aGUgbmV3IENvbmZpZ0l0ZW0uCiAgICAgICAgbXkgJENvbmZpZ0l0ZW1JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5Db25maWdJdGVtQWRkKAogICAgICAgICAgICBOdW1iZXIgID0+ICRDb25maWdJdGVtTnVtYmVyLAogICAgICAgICAgICBDbGFzc0lEID0+ICRIYXJkd2FyZUNvbmZpZ0l0ZW1JRCwKICAgICAgICAgICAgVXNlcklEICA9PiAxLAogICAgICAgICk7CiAgICAgICAgJFNlbGYtPlRydWUoCiAgICAgICAgICAgICRDb25maWdJdGVtSUQsCiAgICAgICAgICAgICJDb25maWdJdGVtIGlzIGNyZWF0ZWQgLSBJRCAkQ29uZmlnSXRlbUlEIgogICAgICAgICk7CgogICAgICAgICMgQWRkIGEgbmV3IHZlcnNpb24uCiAgICAgICAgbXkgJFZlcnNpb25JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5WZXJzaW9uQWRkKAogICAgICAgICAgICBOYW1lICAgICAgICAgPT4gJ1NlbGVuaXVtVGVzdCcsCiAgICAgICAgICAgIERlZmluaXRpb25JRCA9PiAxLAogICAgICAgICAgICBEZXBsU3RhdGVJRCAgPT4gJFByb2R1Y3Rpb25EZXBsU3RhdGVJRCwKICAgICAgICAgICAgSW5jaVN0YXRlSUQgID0+IDEsCiAgICAgICAgICAgIFVzZXJJRCAgICAgICA9PiAxLAogICAgICAgICAgICBDb25maWdJdGVtSUQgPT4gJENvbmZpZ0l0ZW1JRCwKICAgICAgICApOwogICAgICAgICRTZWxmLT5UcnVlKAogICAgICAgICAgICAkVmVyc2lvbklELAogICAgICAgICAgICAiVmVyc2lvbiBpcyBjcmVhdGVkIC0gSUQgJFZlcnNpb25JRCIKICAgICAgICApOwoKICAgICAgICAjIENyZWF0ZSB0ZXN0IHVzZXIgYW5kIGxvZ2luLgogICAgICAgIG15ICRUZXN0VXNlckxvZ2luID0gJEhlbHBlci0+VGVzdFVzZXJDcmVhdGUoCiAgICAgICAgICAgIEdyb3VwcyA9PiBbICdhZG1pbicsICdpdHNtLWNvbmZpZ2l0ZW0nIF0sCiAgICAgICAgKSB8fCBkaWUgIkRpZCBub3QgZ2V0IHRlc3QgdXNlciI7CgogICAgICAgICRTZWxlbml1bS0+TG9naW4oCiAgICAgICAgICAgIFR5cGUgICAgID0+ICdBZ2VudCcsCiAgICAgICAgICAgIFVzZXIgICAgID0+ICRUZXN0VXNlckxvZ2luLAogICAgICAgICAgICBQYXNzd29yZCA9PiAkVGVzdFVzZXJMb2dpbiwKICAgICAgICApOwoKICAgICAgICBteSAkU2NyaXB0QWxpYXMgPSAkQ29uZmlnT2JqZWN0LT5HZXQoJ1NjcmlwdEFsaWFzJyk7CgogICAgICAgICMgTmF2aWdhdGUgdG8gQWdlbnRJVFNDb25maWdJdGVtWm9vbSBzY3JlZW4uCiAgICAgICAgJFNlbGVuaXVtLT5WZXJpZmllZEdldCgiJHtTY3JpcHRBbGlhc31pbmRleC5wbD9BY3Rpb249QWdlbnRJVFNNQ29uZmlnSXRlbVpvb207Q29uZmlnSXRlbUlEPSRDb25maWdJdGVtSUQiKTsKCiAgICAgICAgIyBDaGVjayBpZiB0aGVyZSBpcyB0aGUgbGluayB0byBBZ2VudElUU01Db25maWdJdGVtUHJpbnQgc2NyZWVuLgogICAgICAgICRTZWxlbml1bS0+V2FpdEZvcigKICAgICAgICAgICAgSmF2YVNjcmlwdCA9PgogICAgICAgICAgICAgICAgInJldHVybiB0eXBlb2YoXCQpID09PSAnZnVuY3Rpb24nICYmIFwkKCdhW2hyZWYqPVwiQWN0aW9uPUFnZW50SVRTTUNvbmZpZ0l0ZW1QcmludDtDb25maWdJdGVtSUQ9JENvbmZpZ0l0ZW1JRFwiXScpLmxlbmd0aDsiCiAgICAgICAgKTsKCiAgICAgICAgIyBHbyB0byBBZ2VudElUU01Db25maWdJdGVtUHJpbnQgc2NyZWVuIGZvciB0ZXN0IENvbmZpZ0l0ZW0uCiAgICAgICAgJFNlbGVuaXVtLT5nZXQoCiAgICAgICAgICAgICIke1NjcmlwdEFsaWFzfWluZGV4LnBsP0FjdGlvbj1BZ2VudElUU01Db25maWdJdGVtUHJpbnQ7Q29uZmlnSXRlbUlEPSRDb25maWdJdGVtSUQ7VmVyc2lvbklEPSRWZXJzaW9uSUQiCiAgICAgICAgKTsKCiAgICAgICAgIyBXYWl0IHVudGlsIHByaW50IHNjcmVlbiBpcyBsb2FkZWQuCiAgICAgICAgQUNUSVZFU0xFRVA6CiAgICAgICAgZm9yIG15ICRTZWNvbmQgKCAxIC4uIDMwICkgewogICAgICAgICAgICBpZiAoIGluZGV4KCAkU2VsZW5pdW0tPmdldF9wYWdlX3NvdXJjZSgpLCAicHJpbnRlZCBieSIgKSA+IC0xLCApIHsKICAgICAgICAgICAgICAgICRTZWxmLT5UcnVlKAogICAgICAgICAgICAgICAgICAgIGluZGV4KCAkU2VsZW5pdW0tPmdldF9wYWdlX3NvdXJjZSgpLCAicHJpbnRlZCBieSIgKSA+IC0xLAogICAgICAgICAgICAgICAgICAgICJQcmludCBzY3JlZW4gaXMgbG9hZGVkIiwKICAgICAgICAgICAgICAgICkgfHwgZGllOwogICAgICAgICAgICAgICAgbGFzdCBBQ1RJVkVTTEVFUDsKICAgICAgICAgICAgfQogICAgICAgICAgICBzbGVlcCAxOwogICAgICAgIH0KCiAgICAgICAgIyBHZXQgdGVzdCBwcmludCB2YWx1ZXMuCiAgICAgICAgbXkgQENvbmZpZ0l0ZW1QcmludCA9ICgKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgVmFsdWUgICA9PiAkQ29uZmlnSXRlbU51bWJlciwKICAgICAgICAgICAgICAgIE1lc3NhZ2UgPT4gIkNvbmZpZ0l0ZW0jICRDb25maWdJdGVtTnVtYmVyIC0gZm91bmQiLAogICAgICAgICAgICB9LAogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBWYWx1ZSAgID0+ICdTZWxlbml1bVRlc3QnLAogICAgICAgICAgICAgICAgTWVzc2FnZSA9PiAiQ29uZmlnSXRlbTogU2VsZW5pdW1UZXN0IC0gZm91bmQiLAogICAgICAgICAgICB9LAogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBWYWx1ZSAgID0+ICdIYXJkd2FyZScsCiAgICAgICAgICAgICAgICBNZXNzYWdlID0+ICJDbGFzczogSGFyZHdhcmUgLSBmb3VuZCIsCiAgICAgICAgICAgIH0sCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIFZhbHVlICAgPT4gJ1Byb2R1Y3Rpb24nLAogICAgICAgICAgICAgICAgTWVzc2FnZSA9PiAiQ3VycmVudCBEZXBsb3ltZW50IFN0YXRlOiBQcm9kdWN0aW9uIC0gZm91bmQiLAogICAgICAgICAgICB9LAogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBWYWx1ZSAgID0+ICdPcGVyYXRpb25hbCcsCiAgICAgICAgICAgICAgICBNZXNzYWdlID0+ICJDdXJyZW50IEluY2lkZW50IFN0YXRlOiBPcGVyYXRpb25hbCAtIGZvdW5kIiwKICAgICAgICAgICAgfSwKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgVmFsdWUgICA9PiAnVmVyc2lvbiAxJywKICAgICAgICAgICAgICAgIE1lc3NhZ2UgPT4gIkN1cnJlbnQgVmVyc2lvbjogVmVyc2lvbiAxIC0gZm91bmQiLAogICAgICAgICAgICB9LAogICAgICAgICk7CgogICAgICAgICMgQ2hlY2sgZm9yIHByaW50ZWQgdmFsdWVzLgogICAgICAgIGZvciBteSAkQ29uZmlnSXRlbVZhbHVlIChAQ29uZmlnSXRlbVByaW50KSB7CiAgICAgICAgICAgICRTZWxmLT5UcnVlKAogICAgICAgICAgICAgICAgaW5kZXgoICRTZWxlbml1bS0+Z2V0X3BhZ2Vfc291cmNlKCksICRDb25maWdJdGVtVmFsdWUtPntWYWx1ZX0gKSA+IC0xLAogICAgICAgICAgICAgICAgIiRDb25maWdJdGVtVmFsdWUtPntNZXNzYWdlfSIsCiAgICAgICAgICAgICk7CiAgICAgICAgfQoKICAgICAgICAjIERlbGV0ZSBjcmVhdGVkIHRlc3QgQ29uZmlnSXRlbS4KICAgICAgICBteSAkU3VjY2VzcyA9ICRDb25maWdJdGVtT2JqZWN0LT5Db25maWdJdGVtRGVsZXRlKAogICAgICAgICAgICBDb25maWdJdGVtSUQgPT4gJENvbmZpZ0l0ZW1JRCwKICAgICAgICAgICAgVXNlcklEICAgICAgID0+IDEsCiAgICAgICAgKTsKICAgICAgICAkU2VsZi0+VHJ1ZSgKICAgICAgICAgICAgJFN1Y2Nlc3MsCiAgICAgICAgICAgICJDb25maWdJdGVtIGlzIGRlbGV0ZWQgLSBJRCAkQ29uZmlnSXRlbUlEIiwKICAgICAgICApOwogICAgfQopOwoKMTsK
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

my $Selenium = $Kernel::OM->Get('Kernel::System::UnitTest::Selenium');

$Selenium->RunTest(
    sub {

        my $Helper               = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
        my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

        # Get 'Computer' ConfigItem ID.
        my @ConfigItemClassIDs;
        for my $ConfigItemClass (qw(Computer)) {
            my $ConfigItemDataRef = $GeneralCatalogObject->ItemGet(
                Class => 'ITSM::ConfigItem::Class',
                Name  => $ConfigItemClass,
            );
            push @ConfigItemClassIDs, $ConfigItemDataRef->{ItemID};
        }

        # Get 'Production' deployment state IDs.
        my $ProductionDeplStateDataRef = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::DeploymentState',
            Name  => 'Production',
        );
        my $ProductionDeplStateID = $ProductionDeplStateDataRef->{ItemID};

        my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
        my $ConfigObject     = $Kernel::OM->Get('Kernel::Config');

        # Create ConfigItem numbers.
        my @ConfigItemNumbers;
        for my $ConfigNumberCreate ( 1 .. 2 ) {
            my $ConfigItemNumber = $ConfigItemObject->ConfigItemNumberCreate(
                Type    => $ConfigObject->Get('ITSMConfigItem::NumberGenerator'),
                ClassID => $ConfigItemClassIDs[0],
            );
            $Self->True(
                $ConfigItemNumber,
                "ConfigItem $ConfigItemNumber number is created"
            );
            push @ConfigItemNumbers, $ConfigItemNumber;
        }

        # Add the new ConfigItems.
        my @ConfigItemIDs;
        for my $ConfigItemCreateNumber (@ConfigItemNumbers) {
            my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
                Number  => $ConfigItemCreateNumber,
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            );
            $Self->True(
                $ConfigItemID,
                "ConfigItemID $ConfigItemID is created"
            );
            push @ConfigItemIDs, $ConfigItemID;
        }

        my $InciStateID = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::Core::IncidentState',
            Name  => 'Incident',
        );

        # Add a new version for each ConfigItem.
        my @VersionIDs;
        my $Count    = 1;
        my $RandomID = $Helper->GetRandomID();
        for my $ConfigItemVersion (@ConfigItemIDs) {
            my $VersionID = $ConfigItemObject->VersionAdd(
                Name         => $Count . $RandomID,
                DefinitionID => 1,
                DeplStateID  => $ProductionDeplStateID,
                InciStateID  => $InciStateID->{ItemID},
                UserID       => 1,
                ConfigItemID => $ConfigItemVersion,
            );
            $Self->True(
                $VersionID,
                "VersionID $VersionID is created"
            );
            push @VersionIDs, $VersionID;

            $Count++;
        }

        # Create test user and login.
        my $TestUserLogin = $Helper->TestUserCreate(
            Groups => [ 'admin', 'users', 'itsm-configitem' ],
        ) || die "Did not get test user";

        $Selenium->Login(
            Type     => 'Agent',
            User     => $TestUserLogin,
            Password => $TestUserLogin,
        );

        my $ScriptAlias = $ConfigObject->Get('ScriptAlias');

        # Navigate to AgentITSMConfigItemSearch.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentITSMConfigItemSearch");

        # Wait until form and overlay has loaded, if necessary.
        $Selenium->WaitFor( JavaScript => "return typeof(\$) === 'function' && \$('#SearchClassID').length" );

        # Check for 'Class' label.
        $Self->True(
            $Selenium->execute_script("return \$('label[for=\"Class\"]').length"),
            "Class label - found",
        );

        # Check for 'Class' select box.
        $Self->True(
            $Selenium->find_element( "#SearchClassID", 'css' ),
            "Class select box - found",
        );

        # Be sure modernized field has loaded.
        sleep 1;

        # Select 'Computer' class.
        $Selenium->execute_script(
            "\$('#SearchClassID').val('$ConfigItemClassIDs[0]').trigger('redraw.InputField').trigger('change');"
        );

        # Be sure all fields are loaded.
        $Selenium->WaitFor(
            JavaScript => 'return typeof($) === "function" && $("#SearchFormSubmit").length;',
        );

        # Check ConfigItem search page.
        for my $ID (
            qw(SearchClassID SearchProfile SearchProfileNew Attribute PreviousVersionSearch ResultForm SearchFormSubmit)
            )
        {
            sleep 1;
            $Selenium->WaitFor( JavaScript => "return typeof(\$) === 'function' && \$('#$ID').length" );
            $Self->True(
                $Selenium->execute_script("return typeof(\$) === 'function' && \$('#$ID').length;"),
                "Element is found - $ID",
            );
        }

        # Check if ITSMChange Search form contain Excel output option, see bug#12769.
        $Self->Is(
            $Selenium->execute_script(
                "return \$('#ResultForm option:contains(\"Excel\")').length"
            ),
            '1',
            'ITSMConfigItem Search contain Excel output',
        );

        # Search ConfigItems by test ConfigItem number and names.
        $Selenium->execute_script("\$('#Attribute').val('Name').trigger('redraw.InputField').trigger('change');");
        $Selenium->WaitFor(
            JavaScript => "return typeof(\$) === 'function' && \$('#SearchInsert input[name=\"Name\"]').length"
        );

        $Selenium->find_element("//input[\@name='Number']")->send_keys('*');
        $Selenium->find_element("//input[\@name='Name']")->send_keys( '*' . $RandomID );
        $Selenium->find_element( "#SearchFormSubmit", 'css' )->click();
        $Selenium->WaitFor(
            JavaScript =>
                "return typeof(\$) === 'function' && !\$('Dialog.Modal').length && \$('#OverviewBody .TableSmall').length"
        );

        # Check for expected result.
        for my $CheckConfigItem (@ConfigItemNumbers) {
            $Self->True(
                $Selenium->execute_script(
                    "return \$('#OverviewBody .TableSmall td:contains($CheckConfigItem)').length"
                ),
                "ConfigItem $CheckConfigItem number is found",
            );
        }

        # Verify sorting in table, by default sorting is done by ConfigItemNumber - sort ascending.
        # Lower ID will on the top of table.
        $Self->Is(
            $Selenium->execute_script("return \$('tbody tr:eq(0)').attr('id')"),
            'ConfigItemID_' . $ConfigItemIDs[1],
            "ConfigItemID $ConfigItemIDs[1] is on top of table sort by Number ascending"
        );
        $Self->Is(
            $Selenium->execute_script("return \$('tbody tr:eq(1)').attr('id')"),
            'ConfigItemID_' . $ConfigItemIDs[0],
            "ConfigItemID $ConfigItemIDs[0] is on bottom of table sort by Number ascending"
        );

        # Click to sort by Name.
        $Selenium->find_element( ".Name", 'css' )->VerifiedClick();

        # Check for expected result.
        for my $CheckConfigItem (@ConfigItemNumbers) {
            $Self->True(
                $Selenium->execute_script(
                    "return \$('#OverviewBody .TableSmall td:contains($CheckConfigItem)').length"
                ),
                "ConfigItem $CheckConfigItem number is found",
            );
        }

        # Sort is by Name descending.
        $Self->Is(
            $Selenium->execute_script("return \$('tbody tr:eq(0)').attr('id')"),
            'ConfigItemID_' . $ConfigItemIDs[0],
            "ConfigItemID $ConfigItemIDs[1] is on top of table sort by Name descending"
        );
        $Self->Is(
            $Selenium->execute_script("return \$('tbody tr:eq(1)').attr('id')"),
            'ConfigItemID_' . $ConfigItemIDs[1],
            "ConfigItemID $ConfigItemIDs[0] is on bottom of table sort by Name descending"
        );

        # Click to sort by Name again.
        $Selenium->find_element( ".Name", 'css' )->VerifiedClick();

        # Verify order is changed, sort by Name ascending.
        $Self->Is(
            $Selenium->execute_script("return \$('tbody tr:eq(0)').attr('id')"),
            'ConfigItemID_' . $ConfigItemIDs[1],
            "ConfigItemID $ConfigItemIDs[0] is on top of table sort by Name ascending"
        );
        $Self->Is(
            $Selenium->execute_script("return \$('tbody tr:eq(1)').attr('id')"),
            'ConfigItemID_' . $ConfigItemIDs[0],
            "ConfigItemID $ConfigItemIDs[1] is on bottom of table sort by Name ascending"
        );

        # Create ConfigItem numbers and add the new ConfigItems.
        @ConfigItemNumbers = ();
        for my $ConfigNumberCreate ( 1 .. 35 ) {
            my $ConfigItemNumber = $ConfigItemObject->ConfigItemNumberCreate(
                Type    => $ConfigObject->Get('ITSMConfigItem::NumberGenerator'),
                ClassID => $ConfigItemClassIDs[0],
            );
            $Self->True(
                $ConfigItemNumber,
                "ConfigItem $ConfigItemNumber number is created"
            );
            push @ConfigItemNumbers, $ConfigItemNumber;

            my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
                Number  => $ConfigItemNumber,
                ClassID => $ConfigItemClassIDs[0],
                UserID  => 1,
            );
            $Self->True(
                $ConfigItemID,
                "ConfigItemID $ConfigItemID is created"
            );
            push @ConfigItemIDs, $ConfigItemID;
        }

        my @XMLDataArray = [
            undef,
            {
                'Version' => [
                    undef,
                    {

                        'WarrantyExpirationDate' => [
                            undef,
                            {
                                'Content' => '2017-10-10'
                            },
                        ],
                        'NIC' => [
                            undef,
                            {
                                'IPoverDHCP' => [
                                    undef,
                                    {
                                        'Content' => '38'
                                    }
                                ],
                                'IPAddress' => [
                                    undef,
                                    {
                                        'Content' => '172.0.0.0'
                                    }
                                ],
                                'Content' => 'test 172.0.0.0'
                            }
                        ],
                    },
                ],
            },
            {
                'Version' => [
                    undef,
                    {

                        'WarrantyExpirationDate' => [
                            undef,
                            {
                                'Content' => '2017-11-11'
                            },
                        ],
                    },
                ],
            },
        ];

        # Add a new version for each ConfigItem.
        $Count = 1;
        for my $ConfigItemVersion (@ConfigItemIDs) {

            my $XMLData   = ( $Count <= 30 ) ? $XMLDataArray[0] : $XMLDataArray[1];
            my $VersionID = $ConfigItemObject->VersionAdd(
                Name         => $Count . $RandomID,
                DefinitionID => 1,
                DeplStateID  => $ProductionDeplStateID,
                InciStateID  => $InciStateID->{ItemID},
                UserID       => 1,
                XMLData      => $XMLData,
                ConfigItemID => $ConfigItemVersion,
            );
            $Self->True(
                $VersionID,
                "VersionID $VersionID is created"
            );
            push @VersionIDs, $VersionID;

            $Count++;
        }

        # Change search option.
        $Selenium->find_element( "#ITSMConfigItemSearch", 'css' )->click();

        # Wait until form has loaded, if necessary.
        $Selenium->WaitFor( JavaScript => "return \$('#Attribute').length" );

        # Add search filter by WarrantyExpirationDate and set date range (8-10-2017 - 15-10-2017).
        $Selenium->execute_script(
            "\$('#Attribute').val('WarrantyExpirationDate').trigger('redraw.InputField').trigger('change');",
        );
        $Selenium->WaitFor(
            JavaScript => "return typeof(\$) === 'function' && \$('input[name=\"WarrantyExpirationDate\"]').length"
        );
        $Selenium->execute_script(
            "\$('#SearchInsert select[id=\"WarrantyExpirationDate::TimeStart::Day\"]').val('8');"
        );
        $Selenium->execute_script(
            "\$('#SearchInsert select[id=\"WarrantyExpirationDate::TimeStart::Month\"]').val('10');"
        );
        $Selenium->execute_script(
            "\$('#SearchInsert select[id=\"WarrantyExpirationDate::TimeStart::Year\"]').val('2017');"
        );
        $Selenium->execute_script(
            "\$('#SearchInsert select[id=\"WarrantyExpirationDate::TimeStop::Day\"]').val('15');"
        );
        $Selenium->execute_script(
            "\$('#SearchInsert select[id=\"WarrantyExpirationDate::TimeStop::Month\"]').val('10');"
        );
        $Selenium->execute_script(
            "\$('#SearchInsert select[id=\"WarrantyExpirationDate::TimeStop::Year\"]').val('2017');"
        );
        $Selenium->find_element("//input[\@name='Number']")->clear();
        $Selenium->find_element("//input[\@name='Number']")->send_keys('*');
        $Selenium->find_element("//input[\@name='Name']")->clear();
        $Selenium->find_element("//input[\@name='Name']")->send_keys( '*' . $RandomID );

        $Selenium->find_element( "#SearchFormSubmit", 'css' )->click();
        $Selenium->WaitFor(
            JavaScript =>
                "return typeof(\$) === 'function' && !\$('Dialog.Modal').length && \$('#OverviewBody .TableSmall').length"
        );

        $Self->True(
            $Selenium->execute_script("return \$('.Pagination').text().trim().indexOf('1-25 of 30') > -1;"),
            "Check pagination on the first page",
        );

        $Self->Is(
            $Selenium->execute_script("return \$('#OverviewBody .TableSmall tbody tr').length;"),
            '25',
            "Number of config items on the first page is correct - 25",
        );

        # Go to the second page.
        $Selenium->find_element( "#GenericPage2", 'css' )->VerifiedClick();

        $Self->True(
            $Selenium->execute_script("return \$('.Pagination').text().trim().indexOf('26-30 of 30') > -1;"),
            "Check pagination on the second page",
        );

        $Self->Is(
            $Selenium->execute_script("return \$('#OverviewBody .TableSmall tbody tr').length;"),
            '5',
            "Number of config items on the second page is correct - 5",
        );

        # Change search option.
        $Selenium->find_element( "#ITSMConfigItemSearch", 'css' )->click();

        # Wait until form has loaded, if necessary.
        $Selenium->WaitFor(
            JavaScript => "return \$('#Attribute').length && \$('#SearchInsert input[name=\"Name\"]').length"
        );

        # Input wrong search parameters, result should be 'No data found'.
        $Selenium->find_element("//input[\@name='Name']")->clear();
        $Selenium->find_element("//input[\@name='Name']")->send_keys('asdfg');
        $Selenium->find_element( "#SearchFormSubmit", 'css' )->click();
        $Selenium->WaitFor(
            JavaScript =>
                "return typeof(\$) === 'function' && !\$('Dialog.Modal').length && \$('#OverviewBody .TableSmall').length"
        );

        # Check for expected result.
        $Self->True(
            index( $Selenium->get_page_source(), 'No data found' ) > -1,
            "'No data found' - found",
        );

        # Click on "Change search option"
        $Selenium->find_element( "#ITSMConfigItemSearch", 'css' )->click();
        $Selenium->WaitFor( JavaScript => "return typeof(\$) === 'function' && \$('#SearchProfileNew').length" );
        sleep 2;

        # Create new template search.
        my $SearchProfileName = "Search-" . $Helper->GetRandomID();
        $Selenium->find_element( "#SearchProfileNew", 'css' )->click();
        $Selenium->WaitFor( JavaScript => "return typeof(\$) === 'function' && \$('#SearchProfileAddName').length" );

        $Selenium->find_element( "#SearchProfileAddName",   'css' )->send_keys($SearchProfileName);
        $Selenium->find_element( "#SearchProfileAddAction", 'css' )->click();
        $Selenium->WaitFor(
            JavaScript => "return typeof(\$) === 'function' && \$('#SearchProfile').val() === '$SearchProfileName'"
        );

        # Execute search with new search profile to save it.
        $Selenium->find_element("//input[\@name='Number']")->clear();
        $Selenium->find_element("//input[\@name='Number']")->send_keys('*');
        $Selenium->find_element("//input[\@name='Name']")->clear();
        $Selenium->find_element("//input[\@name='Name']")->send_keys( '*' . $RandomID );

        $Selenium->find_element( "#SearchFormSubmit", 'css' )->click();
        $Selenium->WaitFor(
            JavaScript =>
                "return typeof(\$) === 'function' && !\$('Dialog.Modal').length && \$('#OverviewBody .TableSmall').length"
        );

        # Click on "Change search option"
        $Selenium->find_element( "#ITSMConfigItemSearch", 'css' )->click();

        # Select newly created search profile.
        $Selenium->execute_script(
            "\$('#SearchProfile').val('$SearchProfileName').trigger('redraw.InputField').trigger('change');",
        );

        $Selenium->WaitFor(
            JavaScript => "return typeof(\$) === 'function' && \$('#SearchProfile').val() === '$SearchProfileName';"
        );
        $Selenium->WaitFor(
            JavaScript => "return typeof(\$) === 'function' && \$('#SearchProfileAsLink:visible').length"
        );

        # Check button for profile link.
        $Selenium->find_element( "#SearchProfileAsLink", 'css' )->click();

        $Selenium->WaitFor(
            JavaScript =>
                'return typeof(Core) == "object" && typeof(Core.App) == "object" && Core.App.PageLoadComplete'
        );

        # Click on "Change search option"
        $Selenium->find_element( "#ITSMConfigItemSearch", 'css' )->click();
        $Selenium->WaitFor(
            JavaScript => "return typeof(\$) === 'function' && \$('#SearchProfile').val() === '$SearchProfileName'"
        );

        $Self->Is(
            $Selenium->execute_script("return \$('#SearchProfile').val();"),
            $SearchProfileName,
            "Check if profile is loaded well"
        );

        # Check if correct config items are shown after sub attributes are searched. See bug#12998.
        my $ConfigItemNumber2 = $ConfigItemObject->ConfigItemNumberCreate(
            Type    => $ConfigObject->Get('ITSMConfigItem::NumberGenerator'),
            ClassID => $ConfigItemClassIDs[0],
        );
        $Self->True(
            $ConfigItemNumber2,
            "ConfigItem $ConfigItemNumber2 number is created"
        );
        my $ConfigItemID2 = $ConfigItemObject->ConfigItemAdd(
            Number  => $ConfigItemNumber2,
            ClassID => $ConfigItemClassIDs[0],
            UserID  => 1,
        );
        $Self->True(
            $ConfigItemID2,
            "ConfigItemID $ConfigItemID2 is created"
        );

        push @ConfigItemIDs, $ConfigItemID2;

        # Create config that should not appear in search.
        my $VersionID2 = $ConfigItemObject->VersionAdd(
            Name         => $Count . $RandomID,
            DefinitionID => 1,
            DeplStateID  => $ProductionDeplStateID,
            InciStateID  => $InciStateID->{ItemID},
            UserID       => 1,
            ConfigItemID => $ConfigItemID2,
            XMLData      => [
                undef,
                {
                    Version => [
                        undef,
                        {
                            NIC => [
                                undef,
                                {
                                    IPoverDHCP => [
                                        undef,
                                        {
                                            Content => '38'
                                        }
                                    ],
                                    IPAddress => [
                                        undef,
                                        {
                                            Content => '222.0.0.0'
                                        }
                                    ],
                                    Content => 'test 222.0.0.0'
                                }
                            ],
                        }
                    ]
                }
            ],
        );
        $Self->True(
            $VersionID2,
            "VersionID $VersionID2 is created"
        );

        # Configure IPAddres to show in search result.
        $Helper->ConfigSettingChange(
            Valid => 1,
            Key   => 'ITSMConfigItem::Frontend::AgentITSMConfigItemSearch###ShowColumns',
            Value => {
                Class                  => 0,
                CurDeplSignal          => 1,
                CurDeplState           => 1,
                CurDeplStateType       => 0,
                CurInciSignal          => 1,
                CurInciState           => 1,
                CurInciStateType       => 0,
                LastChanged            => 1,
                'NIC::1::IPAddress::1' => 1,
                Name                   => 1,
                Number                 => 1,
            },
        );

        $Selenium->find_element( ".Dialog .Close", 'css' )->click();

        # Navigate to AgentITSMConfigItemSearch.
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentITSMConfigItemSearch");

        # Wait until form and overlay has loaded, if necessary.
        $Selenium->WaitFor( JavaScript => "return typeof(\$) === 'function' && \$('#SearchClassID').length" );
        sleep 1;

        # Select 'Computer' class.
        $Selenium->execute_script(
            "\$('#SearchClassID').val('$ConfigItemClassIDs[0]').trigger('redraw.InputField').trigger('change');"
        );

        # Wait until form has loaded, if necessary.
        $Selenium->WaitFor( JavaScript => "return \$('#Attribute').length" );

        # Add search filter by NIC::IPAddress and set 172*.
        $Selenium->execute_script(
            "\$('#Attribute').val('NIC::IPAddress').trigger('redraw.InputField').trigger('change');",
        );

        # Wait for Network Adapter::IP Address to appear.
        $Selenium->WaitFor(
            JavaScript =>
                "return typeof(\$) === 'function' && \$('#SearchInsert input[name=\"NIC::IPAddress\"]').length"
        );

        $Selenium->find_element("//input[\@name='NIC::IPAddress']")->send_keys('172*');

        $Selenium->find_element( "#SearchFormSubmit", 'css' )->click();
        $Selenium->WaitFor(
            JavaScript =>
                "return typeof(\$) === 'function' && !\$('Dialog.Modal').length && \$('#OverviewBody .TableSmall').length"
        );

        # Check if correct number of items are shown on pagination.
        $Self->True(
            $Selenium->execute_script("return \$('.Pagination').text().trim().indexOf('1-25 of 30') > -1;"),
            "Check pagination on the first page",
        );

        $Self->Is(
            $Selenium->execute_script("return \$('#OverviewBody .TableSmall tbody tr').length;"),
            '25',
            "Number of config items on the first page is correct - 25",
        );

        # Go to the second page.
        $Selenium->find_element( "#GenericPage2", 'css' )->VerifiedClick();

        # Check if correct number of items are shown on pagination.
        $Self->True(
            $Selenium->execute_script("return \$('.Pagination').text().trim().indexOf('26-30 of 30') > -1;"),
            "Check pagination on the second page",
        );

        $Self->Is(
            $Selenium->execute_script("return \$('#OverviewBody .TableSmall tbody tr').length;"),
            '5',
            "Number of config items on the second page is correct - 5",
        );

        # Verify search result remained intact after changing items per page, see bug#14717 for more details.
        # Set 10 config items per page.
        $Selenium->find_element( "a#ShowContextSettingsDialog", 'css' )->click();
        $Selenium->WaitFor(
            JavaScript => 'return $(".Dialog.Modal #UserConfigItemOverviewSmallPageShown").length'
        );
        $Selenium->InputFieldValueSet(
            Element => '#UserConfigItemOverviewSmallPageShown',
            Value   => '10',
        );
        $Selenium->find_element( "#DialogButton1", 'css' )->click();
        $Selenium->WaitFor(
            JavaScript => 'return !$(".Dialog.Modal").length'
        );

        # Check if correct number of items are shown on pagination.
        $Self->True(
            $Selenium->execute_script("return \$('.Pagination').text().trim().indexOf('1-10 of 30') > -1;"),
            "Check pagination after switch",
        );
        $Self->Is(
            $Selenium->execute_script("return \$('#OverviewBody .TableSmall tbody tr').length;"),
            '10',
            "Number of config items is correct - 10",
        );

        # Delete created test ConfigItem.
        for my $ConfigItemDeleteID (@ConfigItemIDs) {
            my $Success = $ConfigItemObject->ConfigItemDelete(
                ConfigItemID => $ConfigItemDeleteID,
                UserID       => 1,
            );
            $Self->True(
                $Success,
                "ConfigItemID $ConfigItemDeleteID is deleted",
            );
        }

        # Make sure the cache is correct.
        my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');
        for my $Cache (qw(ConfigItem Version)) {
            $CacheObject->CleanUp( Type => $Cache );
        }
    }
);

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

# get selenium object
my $Selenium = $Kernel::OM->Get('Kernel::System::UnitTest::Selenium');

$Selenium->RunTest(
    sub {

        # get needed objects
        my $Helper               = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
        my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

        # get 'Hardware' catalog class IDs
        my $ConfigItemDataRef = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::Class',
            Name  => 'Hardware',
        );
        my $HardwareConfigItemID = $ConfigItemDataRef->{ItemID};

        # get 'Production' and 'Repair' deployment state IDs
        my @DeplStateIDs;
        my @DeplStates = (qw(Production Repair));
        for my $DeplState (@DeplStates) {
            my $DeplStateDataRef = $GeneralCatalogObject->ItemGet(
                Class => 'ITSM::ConfigItem::DeploymentState',
                Name  => $DeplState,
            );
            push @DeplStateIDs, $DeplStateDataRef->{ItemID};
        }

        # get needed objects
        my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
        my $ConfigObject     = $Kernel::OM->Get('Kernel::Config');

        # create ConfigItem number
        my $ConfigItemNumber = $ConfigItemObject->ConfigItemNumberCreate(
            Type    => $ConfigObject->Get('ITSMConfigItem::NumberGenerator'),
            ClassID => $HardwareConfigItemID,
        );
        $Self->True(
            $ConfigItemNumber,
            "ConfigItem number is created - $ConfigItemNumber"
        );

        # add the new ConfigItem
        my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
            Number  => $ConfigItemNumber,
            ClassID => $HardwareConfigItemID,
            UserID  => 1,
        );
        $Self->True(
            $ConfigItemID,
            "ConfigItem is created - ID $ConfigItemID"
        );

        # add two versions
        my @VersionIDs;
        for my $Version (@DeplStateIDs) {
            my $VersionID = $ConfigItemObject->VersionAdd(
                Name         => 'SeleniumTest',
                DefinitionID => 1,
                DeplStateID  => $Version,
                InciStateID  => 1,
                UserID       => 1,
                ConfigItemID => $ConfigItemID,
            );
            $Self->True(
                $VersionID,
                "Version is created - ID $VersionID"
            );
            push @VersionIDs, $VersionID;
        }

        # create test user and login
        my $TestUserLogin = $Helper->TestUserCreate(
            Groups => [ 'admin', 'itsm-configitem' ],
        ) || die "Did not get test user";

        $Selenium->Login(
            Type     => 'Agent',
            User     => $TestUserLogin,
            Password => $TestUserLogin,
        );

        # get script alias
        my $ScriptAlias = $ConfigObject->Get('ScriptAlias');

        # Navigate to AgentITSMConfigItem screen.
        $Selenium->VerifiedGet(
            "${ScriptAlias}index.pl?Action=AgentITSMConfigItem;SortBy=Number;OrderBy=Down;View=;Filter=$HardwareConfigItemID"
        );

        # Click on created ConfigItem.
        $Selenium->find_element("//div[contains(\@title, '$ConfigItemNumber' )]")->VerifiedClick();

        # get ConfigItem value params
        my @ConfigItemValues = (
            {
                Value => 'Hardware',
                Check => 'p.Value:contains(Hardware)',
            },
            {
                Value => 'SeleniumTest',
                Check => 'p.Value:contains(SeleniumTest)',
            },
            {
                Value => 'Repair',
                Check => 'div.Value:contains(Repair)',
            },
            {
                Value => 'Operational',
                Check => 'div.Value:contains(Operational)',
            },
            {
                Value => "$ConfigItemNumber",
                Check => "h1:contains($ConfigItemNumber)",
            },
        );

        # check ConfigItem values on screen
        for my $CheckConfigItemValue (@ConfigItemValues) {
            $Self->True(
                $Selenium->execute_script(
                    "return \$('$CheckConfigItemValue->{Check}').length"
                ),
                "Test ConfigItem value $CheckConfigItemValue->{Value} - found",
            );
        }

        # click to show all versions
        $Selenium->find_element( ".AllITSMItems", 'css' )->click();

        # verify both versions are present on screen
        for my $VersionsCheck (@DeplStates) {
            $Self->True(
                index( $Selenium->get_page_source(), "SeleniumTest ($VersionsCheck)" ) > -1,
                "SeleniumTest ($VersionsCheck) - found",
            );
        }

        # get test menu buttons params
        my @MenuButtons = (
            {
                Name => 'Back',
                Link => "Action=AgentITSMConfigItem;SortBy=Number;OrderBy=Down;View=;Filter=$HardwareConfigItemID",
            },
            {
                Name => 'History',
                Link => "AgentITSMConfigItemHistory;ConfigItemID=$ConfigItemID;VersionID=$VersionIDs[1]",
            },
            {
                Name => 'Edit',
                Link => "AgentITSMConfigItemEdit;ConfigItemID=$ConfigItemID",
            },
            {
                Name => 'Print',
                Link => "AgentITSMConfigItemPrint;ConfigItemID=$ConfigItemID;VersionID=$VersionIDs[1]",
            },
            {
                Name => 'Link',
                Link => "AgentLinkObject;SourceObject=ITSMConfigItem;SourceKey=$ConfigItemID",
            },
            {
                Name => 'Duplicate',
                Link => "Action=AgentITSMConfigItemEdit;DuplicateID=$ConfigItemID;VersionID=$VersionIDs[1]",
            },
        );

        # check menu buttons
        for my $MenuButtonCheck (@MenuButtons) {
            my $Success = $Selenium->find_element("//a[contains(\@href, \'$MenuButtonCheck->{Link}' )]");
            $Self->True(
                $Success,
                "Menu button $MenuButtonCheck->{Name} - found",
            );
        }

        # delete created test ConfigItem
        my $Success = $ConfigItemObject->ConfigItemDelete(
            ConfigItemID => $ConfigItemID,
            UserID       => 1,
        );
        $Self->True(
            $Success,
            "ConfigItem is deleted - ID $ConfigItemID",
        );
    }

);

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

my $Selenium = $Kernel::OM->Get('Kernel::System::UnitTest::Selenium');

$Selenium->RunTest(
    sub {

        my $Helper               = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
        my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
        my $TicketObject         = $Kernel::OM->Get('Kernel::System::Ticket');
        my $ConfigItemObject     = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
        my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');

        # Get Computer and Hardware catalog class IDs.
        my @ConfigItemClassIDs;
        for my $ConfigItem (qw(Computer Hardware)) {
            my $ConfigItemDataRef = $GeneralCatalogObject->ItemGet(
                Class => 'ITSM::ConfigItem::Class',
                Name  => $ConfigItem,
            );
            push @ConfigItemClassIDs, $ConfigItemDataRef->{ItemID};
        }

        # Get 'Production' deployment state ID.
        my $DeplStateDataRef = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::DeploymentState',
            Name  => 'Production',
        );
        my $DeplStateID = $DeplStateDataRef->{ItemID};

        # Create two test ConfigItems for Computer and Hardware ConfigItem class.
        my @ConfigItemNumbers;
        my @ConfigItemIDs;
        my @VersionIDs;
        for my $ITSMConfigItem (@ConfigItemClassIDs) {

            # Create ConfigItem number.
            my $ConfigItemNumber = $ConfigItemObject->ConfigItemNumberCreate(
                Type    => $ConfigObject->Get('ITSMConfigItem::NumberGenerator'),
                ClassID => $ITSMConfigItem,
            );
            $Self->True(
                $ConfigItemNumber,
                "ConfigItem number is created - $ConfigItemNumber"
            );
            push @ConfigItemNumbers, $ConfigItemNumber;

            # Add the new ConfigItem.
            my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
                Number  => $ConfigItemNumber,
                ClassID => $ITSMConfigItem,
                UserID  => 1,
            );
            $Self->True(
                $ConfigItemID,
                "ConfigItem is created - ID $ConfigItemID"
            );
            push @ConfigItemIDs, $ConfigItemID;

            # Add a new version.
            my $VersionID = $ConfigItemObject->VersionAdd(
                Name         => 'SeleniumTest',
                DefinitionID => 1,
                DeplStateID  => $DeplStateID,
                InciStateID  => 1,
                UserID       => 1,
                ConfigItemID => $ConfigItemID,
            );
            $Self->True(
                $VersionID,
                "Version is created - ID $VersionID"
            );
            push @VersionIDs, $VersionID;

        }

        # Create test service.
        my $ServiceName = "Service" . $Helper->GetRandomID();
        my $ServiceID   = $Kernel::OM->Get('Kernel::System::Service')->ServiceAdd(
            Name        => $ServiceName,
            ValidID     => 1,
            Comment     => 'Selenium Test Service',
            UserID      => 1,
            TypeID      => 2,
            Criticality => '3 normal',
        );
        $Self->True(
            $ServiceID,
            "Service is created - ID $ServiceID"
        );

        # Create test ticket.
        my $TicketNumber = $TicketObject->TicketCreateNumber();
        my $TicketID     = $TicketObject->TicketCreate(
            TN           => $TicketNumber,
            Title        => 'SeleniumTestTicket',
            Queue        => 'Raw',
            Lock         => 'unlock',
            Priority     => '3 normal',
            State        => 'new',
            CustomerID   => '123465',
            CustomerUser => 'customer@example.com',
            OwnerID      => 1,
            UserID       => 1,
        );
        $Self->True(
            $TicketID,
            "Ticket is created - ID $TicketID"
        );

        # Create test user and login.
        my $TestUserLogin = $Helper->TestUserCreate(
            Groups => [ 'admin', 'users', 'itsm-configitem', 'itsm-service' ],
        ) || die "Did not get test user";

        $Selenium->Login(
            Type     => 'Agent',
            User     => $TestUserLogin,
            Password => $TestUserLogin,
        );

        my $ScriptAlias = $ConfigObject->Get('ScriptAlias');

        # Navigate to AgentITSMConfigItemZoom screen.
        $Selenium->VerifiedGet(
            "${ScriptAlias}index.pl?Action=AgentITSMConfigItemZoom;ConfigItemID=$ConfigItemIDs[0];Version=$VersionIDs[0]"
        );

        # Click on 'Link' menu.
        $Selenium->find_element(
            "//a[contains(\@href, \'Action=AgentLinkObject;SourceObject=ITSMConfigItem;SourceKey=$ConfigItemIDs[0]\' )]"
        )->click();

        # Switch to 'Link' window.
        $Selenium->WaitFor( WindowCount => 2 );
        my $Handles = $Selenium->get_window_handles();
        $Selenium->switch_to_window( $Handles->[1] );

        # Wait until page has loaded, if necessary.
        $Selenium->WaitFor(
            JavaScript => 'return typeof(Core) == "object" && typeof(Core.App) == "object" && Core.App.PageLoadComplete'
        );

        # Select to link with test Hardware ConfigItem.
        $Selenium->execute_script(
            "\$('#TargetIdentifier').val('ITSMConfigItem::$ConfigItemClassIDs[1]').trigger('redraw.InputField').trigger('change');"
        );

        sleep 2;
        $Selenium->WaitFor(
            JavaScript => 'return typeof($) === "function" && $("#SEARCH\\\\:\\\\:Number").length;'
        );

        # Check 'Link' screen for two ConfigItems.
        for my $ID (
            qw(TargetIdentifier SubmitSearch)
            )
        {
            my $Element = $Selenium->find_element( "#$ID", 'css' );
            $Element->is_enabled();
            $Element->is_displayed();
        }

        # Search ConfigItem by its number and select result.
        $Selenium->find_element("//input[\@id='SEARCH::Number']")->send_keys( $ConfigItemNumbers[1] );
        $Selenium->find_element( "#SubmitSearch", 'css' )->VerifiedClick();

        $Selenium->find_element( '#LinkTargetKeys', 'css' )->click();
        sleep 1;

        # Select 'AlternativeTo' link type.
        $Selenium->find_element("//button[\@id='AddLinks']")->VerifiedClick();

        # Select to link with test Service.
        $Selenium->execute_script(
            "\$('#TargetIdentifier').val('Service').trigger('redraw.InputField').trigger('change');"
        );

        sleep 2;
        $Selenium->WaitFor(
            JavaScript => 'return typeof($) === "function" && $("#SEARCH\\\\:\\\\:Name").length;'
        );

        # Search Service by name and select result.
        $Selenium->find_element("//input[\@id='SEARCH::Name']")->send_keys($ServiceName);
        $Selenium->find_element( "#SubmitSearch", 'css' )->VerifiedClick();
        $Selenium->find_element("//input[\@id='LinkTargetKeys'][\@value='$ServiceID']")->click();

        # Select 'Relevant to' link type.
        $Selenium->execute_script(
            "\$('#TypeIdentifier').val('RelevantTo::Source').trigger('redraw.InputField').trigger('change');"
        );
        $Selenium->find_element("//button[\@id='AddLinks']")->VerifiedClick();

        # Select to link with test Ticket.
        $Selenium->execute_script(
            "\$('#TargetIdentifier').val('Ticket').trigger('redraw.InputField').trigger('change');"
        );

        sleep 2;
        $Selenium->WaitFor(
            JavaScript => 'return typeof($) === "function" && $("#SEARCH\\\\:\\\\:TicketNumber").length;'
        );

        # Search Ticket by number and select result.
        $Selenium->find_element("//input[\@id='SEARCH::TicketNumber']")->send_keys($TicketNumber);
        $Selenium->find_element( "#SubmitSearch", 'css' )->VerifiedClick();

        $Selenium->find_element("//input[\@id='LinkTargetKeys'][\@value='$TicketID']")->click();

        # Select 'Depends to' link type.
        $Selenium->execute_script(
            "\$('#TypeIdentifier').val('DependsOn::Source').trigger('redraw.InputField').trigger('change');"
        );
        $Selenium->find_element("//button[\@id='AddLinks']")->VerifiedClick();

        # Close link window, return to Zoom view and refresh page.
        $Selenium->find_element( "#LinkAddCloseLink", 'css' )->click();

        $Selenium->WaitFor( WindowCount => 1 );
        $Selenium->switch_to_window( $Handles->[0] );
        $Selenium->VerifiedRefresh();

        # Check for linked values in AgentITSMConfigItemZoom screen.
        for my $CheckValues ( $ConfigItemNumbers[1], $ServiceName, $TicketNumber ) {
            $Self->True(
                index( $Selenium->get_page_source(), $CheckValues ) > -1,
                "Linked value $CheckValues - found",
            );
        }

        # Click on 'Link' menu and switch window.
        $Selenium->find_element(
            "//a[contains(\@href, \'Action=AgentLinkObject;SourceObject=ITSMConfigItem;SourceKey=$ConfigItemIDs[0]\' )]"
        )->click();

        $Selenium->WaitFor( WindowCount => 2 );
        $Handles = $Selenium->get_window_handles();
        $Selenium->switch_to_window( $Handles->[1] );

        # Wait until page has loaded, if necessary.
        $Selenium->WaitFor(
            JavaScript => 'return typeof(Core) == "object" && typeof(Core.App) == "object" && Core.App.PageLoadComplete'
        );

        # Click 'go to manage links screen'.
        $Selenium->find_element("//a[contains(\@href, \'#ManageLinks\' )]")->click();

        # Wait for the manage links tab to show up.
        $Selenium->WaitFor(
            JavaScript =>
                'return typeof($) === "function" && $("div[data-id=ManageLinks]:visible").length && parseInt($("div[data-id=ManageLinks]").css("opacity"), 10) == 1'
        );

        # Select all linked items and delete links.
        $Selenium->find_element("//input[\@value='ITSMConfigItem::$ConfigItemIDs[1]::AlternativeTo']")->click();
        $Selenium->find_element("//button[\@title='Delete links']")->VerifiedClick();

        $Selenium->find_element("//input[\@value='Service::$ServiceID\::RelevantTo']")->click();
        $Selenium->find_element("//button[\@title='Delete links']")->VerifiedClick();

        $Selenium->find_element("//input[\@value='Ticket::$TicketID\::DependsOn']")->click();
        $Selenium->find_element("//button[\@title='Delete links']")->VerifiedClick();

        # Close link window, return to Zoom view and refresh page.
        $Selenium->find_element( "#LinkAddCloseLink", 'css' )->click();

        $Selenium->WaitFor( WindowCount => 1 );
        $Selenium->switch_to_window( $Handles->[0] );
        $Selenium->VerifiedRefresh();

        # Check for no linked values in AgentITSMConfigItemZoom screen.
        for my $CheckValues ( $ConfigItemNumbers[1], $ServiceName, $TicketNumber ) {
            $Self->True(
                index( $Selenium->get_page_source(), $CheckValues ) == -1,
                "Linked value $CheckValues - not found",
            );
        }

        # Delete test Ticket.
        my $Success = $TicketObject->TicketDelete(
            TicketID => $TicketID,
            UserID   => 1,
        );
        $Self->True(
            $Success,
            "Ticket is deleted - ID $TicketID "
        );

        # Delete test ConfigItem.
        for my $ConfigItemDelete (@ConfigItemIDs) {
            $Success = $ConfigItemObject->ConfigItemDelete(
                ConfigItemID => $ConfigItemDelete,
                UserID       => 1,
            );
            $Self->True(
                $Success,
                "ConfigItem is deleted - ID $ConfigItemDelete",
            );
        }

        my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

        # Delete test service and its preferences.
        $Success = $DBObject->Do(
            SQL => "DELETE FROM service_preferences WHERE service_id = $ServiceID",
        );
        $Self->True(
            $Success,
            "Service preferences is deleted",
        );
        $Success = $DBObject->Do(
            SQL => "DELETE FROM service WHERE id = $ServiceID",
        );
        $Self->True(
            $Success,
            "Service is deleted - ID $ServiceID",
        );
    }
);

1;

# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

my $Selenium = $Kernel::OM->Get('Kernel::System::UnitTest::Selenium');

$Selenium->RunTest(
    sub {

        my $Helper               = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
        my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

        # Create test customer user login.
        my $TestCustomerUser = $Helper->TestCustomerUserCreate() || die "Did not get test customer user";

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

        # Get customer user data.
        my %CustomerUser = $CustomerUserObject->CustomerUserDataGet(
            User => $TestCustomerUser,
        );

        # Create random id.
        my $RandomID = $Helper->GetRandomID();

        my $CustomerCompanyID = $CustomerCompanyObject->CustomerCompanyAdd(
            CustomerID          => "$TestCustomerUser-$RandomID",
            CustomerCompanyName => "$TestCustomerUser-$RandomID",
            ValidID             => 1,
            UserID              => 1,
        );
        $Self->True(
            $CustomerCompanyID,
            "Created test customer company $CustomerCompanyID",
        );

        # Define test array.
        my @Test = (
            {
                ClassName  => 'Customer' . $RandomID,
                Key        => 'Customer',
                Definition => << "EOF",
---
- Key: Customer
  Name: Customer
  Searchable: 1,
  Input:
    Type: Customer
EOF
            },
            {
                ClassName  => 'CustomerCompany' . $RandomID,
                Key        => 'CustomerCompany',
                Definition => << "EOF",
---
- Key: CustomerCompany
  Name: CustomerCompany
  Searchable: 1,
  Input:
    Type: CustomerCompany
EOF
            },
        );

        # Change config settings.
        my %ITSMConfigItemCustomerUserSysConfig = $Kernel::OM->Get('Kernel::System::SysConfig')->SettingGet(
            Name => 'AgentCustomerUserInformationCenter::Backend###0060-CUIC-ITSMConfigItemCustomerUser',
        );

        delete $ITSMConfigItemCustomerUserSysConfig{EffectiveValue}->{ConfigItemKey};
        $ITSMConfigItemCustomerUserSysConfig{EffectiveValue}->{ConfigItemKey} = {
            'Customer' . $RandomID => 'Customer',
        };

        $Helper->ConfigSettingChange(
            Valid => 1,
            Key   => 'AgentCustomerUserInformationCenter::Backend###0060-CUIC-ITSMConfigItemCustomerUser',
            Value => {
                %{ $ITSMConfigItemCustomerUserSysConfig{EffectiveValue} },
            },
        );

        my %ITSMConfigItemCustomerCompanySysConfig = $Kernel::OM->Get('Kernel::System::SysConfig')->SettingGet(
            Name => 'AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany',
        );

        delete $ITSMConfigItemCustomerCompanySysConfig{EffectiveValue}->{ConfigItemKey};
        $ITSMConfigItemCustomerCompanySysConfig{EffectiveValue}->{ConfigItemKey} = {
            'CustomerCompany' . $RandomID => 'CustomerCompany',
        };

        # Change config setting.
        $Helper->ConfigSettingChange(
            Valid => 1,
            Key   => 'AgentCustomerInformationCenter::Backend###0060-CIC-ITSMConfigItemCustomerCompany',
            Value => {
                %{ $ITSMConfigItemCustomerCompanySysConfig{EffectiveValue} },
            },
        );

        my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');

        # Add the test classes.
        my @ConfigItemClassIDs;
        my @ConfigItemClasses;
        my @ConfigItemDefinitionIDs;

        for my $Definition (@Test) {

            # Add an unittest config item class.
            my $ClassID = $GeneralCatalogObject->ItemAdd(
                Class   => 'ITSM::ConfigItem::Class',
                Name    => $Definition->{ClassName},
                ValidID => 1,
                UserID  => 1,
            );

            # Check class id.
            if ( !$ClassID ) {

                $Self->True(
                    0,
                    "Can't add new config item class.",
                );
            }

            push @ConfigItemClassIDs, $ClassID;
            push @ConfigItemClasses,  $Definition->{ClassName};

            # Check if group already exists.
            my $GroupID = $Kernel::OM->Get('Kernel::System::Group')->GroupLookup(
                Group  => 'itsm-configitem',
                UserID => 1,
            );

            # Set permission.
            $Kernel::OM->Get('Kernel::System::GeneralCatalog')->GeneralCatalogPreferencesSet(
                ItemID => $ClassID,
                Key    => 'Permission',
                Value  => $GroupID,
            );

            # Add a definition to the class.
            my $DefinitionID = $ConfigItemObject->DefinitionAdd(
                ClassID    => $ClassID,
                Definition => $Definition->{Definition},
                UserID     => 1,
            );

            # Check definition id.
            if ( !$DefinitionID ) {

                $Self->True(
                    0,
                    "Can't add new config item definition.",
                );
            }

            push @ConfigItemDefinitionIDs, $DefinitionID;

            # Add data to test array.
            $Definition->{ClassID} = $ClassID;
        }

        # Get 'Production' deployment state ID.
        my $DeplStateDataRef = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::DeploymentState',
            Name  => 'Production',
        );
        my $DeplStateID = $DeplStateDataRef->{ItemID};

        # Create test user and login.
        my $TestUserLogin = $Helper->TestUserCreate(
            Groups => [ 'admin', 'itsm-configitem' ],
        ) || die "Did not get test user";

        $Selenium->Login(
            Type     => 'Agent',
            User     => $TestUserLogin,
            Password => $TestUserLogin,
        );

        # Get script alias.
        my $ScriptAlias = $Kernel::OM->Get('Kernel::Config')->Get('ScriptAlias');

        for my $Test (@Test) {

            # Navigate to AgentITSMConfigItemAdd.
            $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentITSMConfigItemAdd");

            $Selenium->WaitFor(
                JavaScript =>
                    "return typeof(\$) === 'function' && \$('a[href*=\"Action=AgentITSMConfigItemEdit;ClassID=$Test->{ClassID}\"]').length"
            );

            # Click on ConfigItem class.
            $Selenium->find_element(
                "//a[contains(\@href, \'Action=AgentITSMConfigItemEdit;ClassID=$Test->{ClassID}\' )]"
            )->VerifiedClick();

            $Selenium->WaitFor(
                JavaScript => "return typeof(\$) === 'function' && \$('#Name').length && \$('#SubmitButton').length"
            );

            # Create test ConfigItem.
            my $ConfigItemName = 'SeleniumTest-' . $Helper->GetRandomID();
            $Selenium->find_element( "#Name", 'css' )->send_keys($ConfigItemName);

            $Selenium->execute_script(
                "\$('#DeplStateID').val('$DeplStateID').trigger('redraw.InputField').trigger('change')"
            );
            $Selenium->WaitFor(
                JavaScript => "return typeof(\$) === 'function' && \$('#DeplStateID').val() === '$DeplStateID'"
            );

            $Selenium->execute_script("\$('#InciStateID').val('1').trigger('redraw.InputField').trigger('change')");
            $Selenium->WaitFor( JavaScript => "return typeof(\$) === 'function' && \$('#InciStateID').val() === '1'" );

            if ( $Test->{Key} eq 'Customer' ) {
                $Selenium->find_element( "#Item10", 'css' )->clear();
                $Selenium->find_element( "#Item10", 'css' )->send_keys( $CustomerUser{UserID} );
                $Selenium->WaitFor(
                    JavaScript => 'return typeof($) === "function" && $("li.ui-menu-item:visible").length'
                );
                $Selenium->execute_script("\$('li.ui-menu-item:contains($CustomerUser{UserID})').click()");
            }
            else {
                $Selenium->execute_script(
                    "\$('#Item10').val('$CustomerCompanyID').trigger('redraw.InputField').trigger('change')"
                );
                $Selenium->WaitFor(
                    JavaScript => "return typeof(\$) === 'function' && \$('#Item10').val() === '$CustomerCompanyID'"
                );
            }

            $Selenium->find_element("//button[\@value='Submit'][\@type='submit']")->VerifiedClick();

            $Selenium->WaitFor(
                JavaScript => "return typeof(\$) === 'function' && \$('h1:contains($ConfigItemName)').length"
            );

            if ( $Test->{Key} eq 'Customer' ) {

                # Navigate to AgentCustomerUserInformationCenter.
                $Selenium->VerifiedGet(
                    "${ScriptAlias}index.pl?Action=AgentCustomerUserInformationCenter;CustomerUserID=$CustomerUser{UserID}"
                );

                $Self->True(
                    index( $Selenium->get_page_source(), $ConfigItemName ) > -1,
                    "$ConfigItemName found on page",
                );
            }
            elsif ( $Test->{Key} eq 'CustomerCompany' ) {

                # Navigate to AgentCustomerInformationCenter.
                $Selenium->VerifiedGet(
                    "${ScriptAlias}index.pl?Action=AgentCustomerInformationCenter;CustomerID=$CustomerCompanyID"
                );

                $Self->True(
                    index( $Selenium->get_page_source(), $ConfigItemName ) > -1,
                    "$ConfigItemName found on page",
                );
            }
        }

        # Cleanup.
        my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

        my $Success = $DBObject->Do(
            SQL  => "DELETE FROM customer_company WHERE customer_id = ?",
            Bind => [ \$CustomerCompanyID ],
        );
        $Self->True(
            $Success,
            "Deleted CustomerCompany",
        );

        my $ConfigItemIDs = $ConfigItemObject->ConfigItemSearch( ClassIDs => \@ConfigItemClassIDs );

        # Delete created test ConfigItem.
        for my $ConfigItemID ( @{$ConfigItemIDs} ) {
            $Success = $ConfigItemObject->ConfigItemDelete(
                ConfigItemID => $ConfigItemID,
                UserID       => 1,
            );
            $Self->True(
                $Success,
                "ConfigItem is deleted - ID $ConfigItemID",
            );
        }

        for my $DefinitionID (@ConfigItemDefinitionIDs) {
            $Success = $DBObject->Do(
                SQL => "DELETE FROM configitem_definition WHERE id = $DefinitionID",
            );
            $Self->True(
                $Success,
                "Config item definition is deleted",
            );
        }

        for my $ClassID (@ConfigItemClassIDs) {
            $Success = $DBObject->Do(
                SQL => "DELETE FROM general_catalog_preferences WHERE general_catalog_id = $ClassID",
            );
            $Self->True(
                $Success,
                "Preferences deleted",
            );

            $Success = $DBObject->Do(
                SQL => "DELETE FROM general_catalog WHERE id = $ClassID",
            );
            $Self->True(
                $Success,
                "Class item is deleted",
            );
        }

        # Make sure the cache is correct.
        $Kernel::OM->Get('Kernel::System::Cache')->CleanUp();
    }
);

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

use Kernel::System::ObjectManager;

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

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);

my $ConfigObject                    = $Kernel::OM->Get('Kernel::Config');
my $HelperObject                    = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
my $LinkObject                      = $Kernel::OM->Get('Kernel::System::LinkObject');
my $UnitTestITSMConfigItemObject    = $Kernel::OM->Get('Kernel::System::UnitTest::ITSMConfigItem');
my $ITSMConfigItemCustomerCIsObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItemCustomerCIs');

my %ConfigItemKey = %{
    $ConfigObject->Get('AgentCustomerUserInformationCenter::Backend')->{'0060-CUIC-ITSMConfigItemCustomerUser'}
        ->{ConfigItemKey} || {}
};

my $TestUserLogin = $HelperObject->TestCustomerUserCreate();

my @ExpectedConfigItemIDs;
my $VersionRef;
my $RandomID;
my $ConfigItemName;

for my $Index ( 0 .. 3 ) {
    $RandomID       = $HelperObject->GetRandomID();
    $ConfigItemName = "UnitTestPC $RandomID";
    $VersionRef     = $UnitTestITSMConfigItemObject->ConfigItemCreate(
        Name          => $ConfigItemName,
        ClassName     => 'Computer',
        DeplStateName => 'Production',
        InciStateName => 'Operational',
        XMLData       => {
            OtherEquipment           => '...',
            Note                     => '...',
            SerialNumber             => '...',
            WarrantyExpirationDate   => '2016-01-01',
            InstallDate              => '2016-01-01',
            $ConfigItemKey{Computer} => $TestUserLogin,
        },
    );

    $Self->True(
        $VersionRef->{ConfigItemID},
        "ConfigItemCreate was successfull - $VersionRef->{ConfigItemID}",
    );

    push @ExpectedConfigItemIDs, $VersionRef->{ConfigItemID};

}

my @CustomerUserCIs = $ITSMConfigItemCustomerCIsObject->GetCustomerUserCIs(
    CustomerUserID => $TestUserLogin,
);

$Self->IsDeeply(
    \@CustomerUserCIs,
    \@ExpectedConfigItemIDs,
    "GetCustomerCIs was successfull for CustomerUser - $TestUserLogin",
);

# with other user
my @ExpectedConfigItemIDs2;

$RandomID = $HelperObject->GetRandomID();

my $TestUserLogin2 = $HelperObject->TestCustomerUserCreate();

$VersionRef = $UnitTestITSMConfigItemObject->ConfigItemCreate(
    Name          => $ConfigItemName,
    ClassName     => 'Computer',
    DeplStateName => 'Production',
    InciStateName => 'Operational',
    XMLData       => {
        OtherEquipment           => '...',
        Note                     => '...',
        SerialNumber             => '...',
        WarrantyExpirationDate   => '2016-01-01',
        InstallDate              => '2016-01-01',
        $ConfigItemKey{Computer} => $TestUserLogin2,
    },
);

$Self->True(
    $VersionRef->{ConfigItemID},
    "ConfigItemCreate was successfull - $VersionRef->{ConfigItemID}",
);

push @ExpectedConfigItemIDs2, $VersionRef->{ConfigItemID};

$RandomID = $HelperObject->GetRandomID();

$VersionRef = $UnitTestITSMConfigItemObject->ConfigItemCreate(
    Name          => $ConfigItemName,
    ClassName     => 'Computer',
    DeplStateName => 'Production',
    InciStateName => 'Operational',
    XMLData       => {
        OtherEquipment           => '...',
        Note                     => '...',
        SerialNumber             => '...',
        WarrantyExpirationDate   => '2016-01-01',
        InstallDate              => '2016-01-01',
        $ConfigItemKey{Computer} => "DummyUser",
    },
);

@CustomerUserCIs = $ITSMConfigItemCustomerCIsObject->GetCustomerUserCIs(
    CustomerUserID => $TestUserLogin2,
);

$Self->IsDeeply(
    \@CustomerUserCIs,
    \@ExpectedConfigItemIDs2,
    "GetCustomerCIs was successfull for CustomerUser - $TestUserLogin2",
);

# GetPossibleCustomerCIs

$RandomID       = $HelperObject->GetRandomID();
$ConfigItemName = "UnitTestPC $RandomID";

my $TestUserLogin3 = $HelperObject->TestCustomerUserCreate();
$VersionRef = $UnitTestITSMConfigItemObject->ConfigItemCreate(
    Name          => $ConfigItemName,
    ClassName     => 'Computer',
    DeplStateName => 'Production',
    InciStateName => 'Operational',
    XMLData       => {
        OtherEquipment           => '...',
        Note                     => '...',
        SerialNumber             => '...',
        WarrantyExpirationDate   => '2016-01-01',
        InstallDate              => '2016-01-01',
        $ConfigItemKey{Computer} => $TestUserLogin3,
    },
);

my @ConfigItems = $ITSMConfigItemCustomerCIsObject->GetPossibleCustomerCIs(
    CustomerUserID => $TestUserLogin3,
);

$Self->IsDeeply(
    \@ConfigItems,
    [
        {
            'Class'        => 'Computer',
            'Icon'         => '',
            'Linked'       => '0',
            'ConfigItemID' => $VersionRef->{ConfigItemID},
            'Number'       => $VersionRef->{Number},
            'Name'         => $ConfigItemName,
        }
    ],
    "GetPossibleCustomerCIs",
);

my $TicketID = $HelperObject->TicketCreate(
    CustomerUser => $TestUserLogin3,
);

$LinkObject->LinkAdd(
    SourceObject => 'Ticket',
    SourceKey    => $TicketID,
    TargetObject => 'ITSMConfigItem',
    TargetKey    => $VersionRef->{ConfigItemID},
    Type         => 'RelevantTo',
    State        => 'Valid',
    UserID       => 1,
);

@ConfigItems = $ITSMConfigItemCustomerCIsObject->GetPossibleCustomerCIs(
    CustomerUserID => $TestUserLogin3,
    TicketID       => $TicketID,
);

$Self->IsDeeply(
    \@ConfigItems,
    [
        {
            'Class'        => 'Computer',
            'Icon'         => '',
            'Linked'       => '1',
            'ConfigItemID' => $VersionRef->{ConfigItemID},
            'Number'       => $VersionRef->{Number},
            'Name'         => $ConfigItemName,
        }
    ],
    "GetPossibleCustomerCIs with Ticket link",
);

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

use Kernel::System::ObjectManager;

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

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);

my $HelperObject                = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
my $ConfigItemObject            = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
my $GeneralCatalogObject        = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $ValidObject                 = $Kernel::OM->Get('Kernel::System::Valid');
my $ITSMConfigItemInvokerObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItemInvoker');
my $ZnunyHelperObject           = $Kernel::OM->Get('Kernel::System::ZnunyHelper');

my $UserID = 1;

#
# Prepare a config item
#
my $ValidID = $ValidObject->ValidLookup(
    Valid => 'valid',
);

my $Classes = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::Class',
    Valid => $ValidID,
);
my %Classes = reverse %{ $Classes // {} };

$Self->True(
    scalar $Classes{Computer},
    'Class "Computer" must be present.',
);

my $YesNo = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::YesNo',
    Valid => $ValidID,
);
my %YesNo = reverse %{ $YesNo // {} };

my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
    ClassID => $Classes{Computer},
    UserID  => $UserID,
);

$Self->True(
    scalar $ConfigItemID,
    'Config item must have been created successfully.',
) || return;

my $ComputerNameFirstVersion  = 'UnitTestComputer' . $HelperObject->GetRandomID();
my $ComputerNameSecondVersion = 'UnitTestComputer' . $HelperObject->GetRandomID();
my $Class                     = 'Computer';
my $ClassID                   = $Classes{Computer};

my $DeplStateProduction = $GeneralCatalogObject->ItemGet(
    Class => 'ITSM::ConfigItem::DeploymentState',
    Name  => 'Production',
);

my $DeplStateRepair = $GeneralCatalogObject->ItemGet(
    Class => 'ITSM::ConfigItem::DeploymentState',
    Name  => 'Repair',
);

my $InciStateOperational = $GeneralCatalogObject->ItemGet(
    Class => 'ITSM::Core::IncidentState',
    Name  => 'Operational',
);

my $InciStateIncident = $GeneralCatalogObject->ItemGet(
    Class => 'ITSM::Core::IncidentState',
    Name  => 'Incident',
);

my $Definition = $ConfigItemObject->DefinitionGet(
    ClassID => $ClassID,
);

my $FirstVersionID = $ZnunyHelperObject->_ITSMConfigItemVersionAdd(
    ConfigItemID  => $ConfigItemID,
    Name          => $ComputerNameFirstVersion,
    ClassName     => $Class,
    DeplStateName => 'Production',
    InciStateName => 'Operational',
    XMLData       => {
        CPU => [
            {
                Content => 'AMD Ryzen',
            },
        ],
        WarrantyExpirationDate => '2021-07-16',
        InstallDate            => '2021-07-16',
        NIC                    => [
            {
                Content    => 'NIC',
                IPoverDHCP => [
                    {
                        Content => $YesNo{Yes},
                    },
                ],
                IPAddress => [
                    {
                        Content => '127.0.0.1'
                    },
                ],
            },
        ],
    },
);

#
# Invalid config item ID
#
my $ConfigItemData = $ITSMConfigItemInvokerObject->GetConfigItemData(
    ConfigItemID => 999999999999,
    Event        => undef,
);

$Self->False(
    scalar $ConfigItemData,
    'GetConfigItemData(): Config item data must match expected one for invalid config item ID.',
);

#
# Valid config item ID without event
#
$ConfigItemData = $ITSMConfigItemInvokerObject->GetConfigItemData(
    ConfigItemID => $ConfigItemID,
    Event        => undef,
);

my %ExpectedConfigItemDataFirstVersion = (
    XMLData => {
        NIC => [
            {
                IPAddress => [
                    {
                        Content => '127.0.0.1',
                    }
                ],
                IPoverDHCP => [
                    {
                        ReadableValue => 'Yes',
                        Content       => $YesNo{Yes},
                    }
                ],
                Content => 'NIC',
            },
        ],
        WarrantyExpirationDate => [
            {
                Content => '2021-07-16',
            },
        ],
        CPU => [
            {
                Content => 'AMD Ryzen',
            },
        ],
        InstallDate => [
            {
                Content => '2021-07-16',
            },
        ],
    },
    ConfigItemID => $ConfigItemID,
    DefinitionID => $Definition->{DefinitionID},
    InciStateID  => $InciStateOperational->{ItemID},
    Name         => $ComputerNameFirstVersion,
    DeplStateID  => $DeplStateProduction->{ItemID},
    DeplState    => 'Production',
    ClassID      => $ClassID,
    InciState    => 'Operational',
    Class        => $Class,
);

$Self->IsDeeply(
    $ConfigItemData,
    \%ExpectedConfigItemDataFirstVersion,
    'GetConfigItemData(): Config item data must match expected one for valid config item ID (first version) without event.',
);

my $SecondVersionID = $ZnunyHelperObject->_ITSMConfigItemVersionAdd(
    ConfigItemID  => $ConfigItemID,
    Name          => $ComputerNameSecondVersion,
    ClassName     => $Class,
    DeplStateName => 'Repair',
    InciStateName => 'Incident',
    XMLData       => {
        CPU => [
            {
                Content => 'Intel Core',
            },
        ],
        WarrantyExpirationDate => '2023-07-15',
        InstallDate            => '2021-07-16',
        NIC                    => [
            {
                Content    => 'NIC1',
                IPoverDHCP => [
                    {
                        Content => $YesNo{Yes},
                    },
                ],
                IPAddress => [
                    {
                        Content => '127.0.0.1'
                    },
                ],
            },
            {
                Content    => 'NIC2',
                IPoverDHCP => [
                    {
                        Content => $YesNo{No},
                    },
                ],
                IPAddress => [
                    {
                        Content => '127.1.0.1'
                    },
                ],
            },
        ],
    },
);

$ConfigItemData = $ITSMConfigItemInvokerObject->GetConfigItemData(
    ConfigItemID => $ConfigItemID,
    Event        => undef,
);

my %ExpectedConfigItemDataSecondVersion = (
    XMLData => {
        NIC => [
            {
                IPAddress => [
                    {
                        Content => '127.0.0.1',
                    }
                ],
                IPoverDHCP => [
                    {
                        ReadableValue => 'Yes',
                        Content       => $YesNo{Yes},
                    }
                ],
                Content => 'NIC1',
            },
            {
                IPAddress => [
                    {
                        Content => '127.1.0.1',
                    }
                ],
                IPoverDHCP => [
                    {
                        ReadableValue => 'No',
                        Content       => $YesNo{No},
                    }
                ],
                Content => 'NIC2',
            },
        ],
        WarrantyExpirationDate => [
            {
                Content => '2023-07-15',
            },
        ],
        CPU => [
            {
                Content => 'Intel Core',
            },
        ],
        InstallDate => [
            {
                Content => '2021-07-16',
            },
        ],
    },
    ConfigItemID => $ConfigItemID,
    DefinitionID => $Definition->{DefinitionID},
    InciStateID  => $InciStateIncident->{ItemID},
    Name         => $ComputerNameSecondVersion,
    DeplStateID  => $DeplStateRepair->{ItemID},
    DeplState    => 'Repair',
    ClassID      => $ClassID,
    InciState    => 'Incident',
    Class        => $Class,
);

$Self->IsDeeply(
    $ConfigItemData,
    \%ExpectedConfigItemDataSecondVersion,
    'GetConfigItemData(): Config item data must match expected one for valid config item ID (second version) without event.',
);

#
# Valid config item ID with event ConfigItemCreate
# This should not return the data of the previous config item version.
#
$ConfigItemData = $ITSMConfigItemInvokerObject->GetConfigItemData(
    ConfigItemID => $ConfigItemID,
    Event        => 'ConfigItemCreate',
);

$Self->IsDeeply(
    $ConfigItemData,
    \%ExpectedConfigItemDataSecondVersion,
    'GetConfigItemData(): Config item data must match expected one for valid config item ID (second version) with event ConfigItemCreate.',
);

#
# Valid config item ID with event ConfigItemDelete
# This should not return the data of the previous config item version.
#
$ConfigItemData = $ITSMConfigItemInvokerObject->GetConfigItemData(
    ConfigItemID => $ConfigItemID,
    Event        => 'ConfigItemDelete',
);

$Self->IsDeeply(
    $ConfigItemData,
    \%ExpectedConfigItemDataSecondVersion,
    'GetConfigItemData(): Config item data must match expected one for valid config item ID (second version) with event ConfigItemDelete.',
);

#
# Valid config item ID with event DeploymentStateUpdate
# This should return the data of the previous config item version and as separate information
# the changed deployment status.
#
$ConfigItemData = $ITSMConfigItemInvokerObject->GetConfigItemData(
    ConfigItemID => $ConfigItemID,
    Event        => 'DeploymentStateUpdate',
);

my %ExpectedConfigItemData = (
    %ExpectedConfigItemDataSecondVersion,
    PreviousConfigItemVersion => {
        %ExpectedConfigItemDataFirstVersion,
    },
    DeploymentStateUpdate => {
        Old => {
            ID      => $ExpectedConfigItemDataFirstVersion{DeplStateID},
            Content => $ExpectedConfigItemDataFirstVersion{DeplState},
        },
        New => {
            ID      => $ExpectedConfigItemDataSecondVersion{DeplStateID},
            Content => $ExpectedConfigItemDataSecondVersion{DeplState},
        },
    },
);

$Self->IsDeeply(
    $ConfigItemData,
    \%ExpectedConfigItemData,
    'GetConfigItemData(): Config item data must match expected one for valid config item ID (second version) with event DeploymentStateUpdate.',
);

#
# Valid config item ID with event IncidentStateUpdate
# This should return the data of the previous config item version and as separate information
# the changed incident status.
#
$ConfigItemData = $ITSMConfigItemInvokerObject->GetConfigItemData(
    ConfigItemID => $ConfigItemID,
    Event        => 'IncidentStateUpdate',
);

%ExpectedConfigItemData = (
    %ExpectedConfigItemDataSecondVersion,
    PreviousConfigItemVersion => {
        %ExpectedConfigItemDataFirstVersion,
    },
    IncidentStateUpdate => {
        Old => {
            ID      => $ExpectedConfigItemDataFirstVersion{InciStateID},
            Content => $ExpectedConfigItemDataFirstVersion{InciState},
        },
        New => {
            ID      => $ExpectedConfigItemDataSecondVersion{InciStateID},
            Content => $ExpectedConfigItemDataSecondVersion{InciState},
        },
    },
);

$Self->IsDeeply(
    $ConfigItemData,
    \%ExpectedConfigItemData,
    'GetConfigItemData(): Config item data must match expected one for valid config item ID (second version) with event IncidentStateUpdate.',
);

#
# Valid config item ID with event NameUpdate
# This should return the data of the previous config item version and as separate information
# the changed name.
#
$ConfigItemData = $ITSMConfigItemInvokerObject->GetConfigItemData(
    ConfigItemID => $ConfigItemID,
    Event        => 'NameUpdate',
);

%ExpectedConfigItemData = (
    %ExpectedConfigItemDataSecondVersion,
    PreviousConfigItemVersion => {
        %ExpectedConfigItemDataFirstVersion,
    },
    NameUpdate => {
        Old => {
            Content => $ExpectedConfigItemDataFirstVersion{Name},
        },
        New => {
            Content => $ExpectedConfigItemDataSecondVersion{Name},
        },
    },
);

$Self->IsDeeply(
    $ConfigItemData,
    \%ExpectedConfigItemData,
    'GetConfigItemData(): Config item data must match expected one for valid config item ID (second version) with event NameUpdate.',
);

#
# Valid config item ID with event ValueUpdate
# This should return the data of the previous config item version and as separate information
# the changed values of field CPU.
#
$ConfigItemData = $ITSMConfigItemInvokerObject->GetConfigItemData(
    ConfigItemID => $ConfigItemID,
    Event        => 'ValueUpdate',
);

%ExpectedConfigItemData = (
    %ExpectedConfigItemDataSecondVersion,
    PreviousConfigItemVersion => {
        %ExpectedConfigItemDataFirstVersion,
    },
    ValueUpdate => {
        'CPU::1' => {
            New => {
                Content => 'Intel Core',
            },
            Old => {
                Content => 'AMD Ryzen',
            },
        },
        'NIC::1' => {
            New => {
                Content => 'NIC1',
            },
            Old => {
                Content => 'NIC',
            },
        },
        'NIC::2' => {
            New => {
                Content => 'NIC2',
            },
            Old => {
                Content => '',
            },
        },
        'NIC::2::IPAddress::1' => {
            New => {
                Content => '127.1.0.1',
            },
            Old => {
                Content => '',
            },
        },
        'NIC::2::IPoverDHCP::1' => {
            New => {
                Content       => $YesNo{No},
                ReadableValue => 'No',
            },
            Old => {
                Content => '',
            },
        },
        'WarrantyExpirationDate::1' => {
            New => {
                Content => '2023-07-15',
            },
            Old => {
                Content => '2021-07-16',
            },
        },
    },
);

$Self->IsDeeply(
    $ConfigItemData,
    \%ExpectedConfigItemData,
    'GetConfigItemData(): Config item data must match expected one for valid config item ID (second version) with event ValueUpdate.',
);

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7Cgp1c2UgdmFycyAocXcoJFNlbGYpKTsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6T2JqZWN0TWFuYWdlcjsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6VmFyaWFibGVDaGVjayBxdyg6YWxsKTsKCiRLZXJuZWw6Ok9NLT5PYmplY3RQYXJhbUFkZCgKICAgICdLZXJuZWw6OlN5c3RlbTo6VW5pdFRlc3Q6OkhlbHBlcicgPT4gewogICAgICAgIFJlc3RvcmVEYXRhYmFzZSA9PiAxLAogICAgfSwKKTsKCm15ICRIZWxwZXJPYmplY3QgICAgICAgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VW5pdFRlc3Q6OkhlbHBlcicpOwpteSAkQ29uZmlnSXRlbU9iamVjdCAgICAgICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtJyk7Cm15ICRHZW5lcmFsQ2F0YWxvZ09iamVjdCAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6R2VuZXJhbENhdGFsb2cnKTsKbXkgJFZhbGlkT2JqZWN0ICAgICAgICAgICAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpWYWxpZCcpOwpteSAkSVRTTUNvbmZpZ0l0ZW1JbnZva2VyT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtSW52b2tlcicpOwpteSAkWm51bnlIZWxwZXJPYmplY3QgICAgICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlpudW55SGVscGVyJyk7Cm15ICRDdXN0b21lclVzZXJPYmplY3QgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6Q3VzdG9tZXJVc2VyJyk7Cm15ICRDdXN0b21lckNvbXBhbnlPYmplY3QgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6Q3VzdG9tZXJDb21wYW55Jyk7Cm15ICRDb25maWdPYmplY3QgICAgICAgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OkNvbmZpZycpOwoKbXkgJFVzZXJJRCA9IDE7CgokQ29uZmlnT2JqZWN0LT5TZXQoCiAgICBLZXkgICA9PiAnQ2hlY2tFbWFpbEFkZHJlc3NlcycsCiAgICBWYWx1ZSA9PiAwLAopOwoKJENvbmZpZ09iamVjdC0+U2V0KAogICAgS2V5ICAgPT4gJ0NoZWNrTVhSZWNvcmQnLAogICAgVmFsdWUgPT4gMCwKKTsKCm15ICRWYWxpZElEID0gJFZhbGlkT2JqZWN0LT5WYWxpZExvb2t1cCgKICAgIFZhbGlkID0+ICd2YWxpZCcsCik7CgojCiMgUHJlcGFyZSBjdXN0b21lciBjb21wYW55IGFuZCB1c2VyCiMKbXkgJEN1c3RvbWVySUQgPSAkQ3VzdG9tZXJDb21wYW55T2JqZWN0LT5DdXN0b21lckNvbXBhbnlBZGQoCiAgICBDdXN0b21lcklEICAgICAgICAgICAgID0+ICdVbml0VGVzdEN1c3RvbWVyQ29tcGFueScgLiAkSGVscGVyT2JqZWN0LT5HZXRSYW5kb21JRCgpLAogICAgQ3VzdG9tZXJDb21wYW55TmFtZSAgICA9PiAnVW5pdFRlc3RDdXN0b21lckNvbXBhbnkgbmFtZScsCiAgICBDdXN0b21lckNvbXBhbnlTdHJlZXQgID0+ICdVbml0VGVzdEN1c3RvbWVyQ29tcGFueSBzdHJlZXQnLAogICAgQ3VzdG9tZXJDb21wYW55WklQICAgICA9PiAnVW5pdFRlc3RDdXN0b21lckNvbXBhbnkgcG9zdGFsIGNvZGUnLAogICAgQ3VzdG9tZXJDb21wYW55Q2l0eSAgICA9PiAnVW5pdFRlc3RDdXN0b21lckNvbXBhbnkgY2l0eScsCiAgICBDdXN0b21lckNvbXBhbnlDb3VudHJ5ID0+ICdHZXJtYW55JywKICAgIEN1c3RvbWVyQ29tcGFueVVSTCAgICAgPT4gJ2h0dHA6Ly93d3cuZXhhbXBsZS5vcmcnLAogICAgQ3VzdG9tZXJDb21wYW55Q29tbWVudCA9PiAnVW5pdFRlc3RDdXN0b21lckNvbXBhbnkgY29tbWVudCcsCiAgICBWYWxpZElEICAgICAgICAgICAgICAgID0+ICRWYWxpZElELAogICAgVXNlcklEICAgICAgICAgICAgICAgICA9PiAkVXNlcklELAopOwoKbXkgJUN1c3RvbWVyQ29tcGFueSA9ICRDdXN0b21lckNvbXBhbnlPYmplY3QtPkN1c3RvbWVyQ29tcGFueUdldCgKICAgIEN1c3RvbWVySUQgPT4gJEN1c3RvbWVySUQsCik7CgpkZWxldGUgJEN1c3RvbWVyQ29tcGFueXtDb25maWd9OwpkZWxldGUgJEN1c3RvbWVyQ29tcGFueXtDb21wYW55Q29uZmlnfTsKCm15ICRDdXN0b21lclVzZXJMb2dpbiA9ICRDdXN0b21lclVzZXJPYmplY3QtPkN1c3RvbWVyVXNlckFkZCgKICAgIFNvdXJjZSAgICAgICAgID0+ICdDdXN0b21lclVzZXInLAogICAgVXNlckZpcnN0bmFtZSAgPT4gJ1VuaXRUZXN0Q3VzdG9tZXJVc2VyIGZpcnN0IG5hbWUnLAogICAgVXNlckxhc3RuYW1lICAgPT4gJ1VuaXRUZXN0Q3VzdG9tZXJDb21wYW55IGxhc3QgbmFtZScsCiAgICBVc2VyQ3VzdG9tZXJJRCA9PiAkQ3VzdG9tZXJJRCwKICAgIFVzZXJMb2dpbiAgICAgID0+ICdVbml0VGVzdEN1c3RvbWVyQ29tcGFueScgLiAkSGVscGVyT2JqZWN0LT5HZXRSYW5kb21JRCgpLAogICAgVXNlclBhc3N3b3JkICAgPT4gJzEyMzQ1Njc4JywKICAgIFVzZXJFbWFpbCAgICAgID0+ICdlbWFpbEBleGFtcGxlLmNvbScsCiAgICBWYWxpZElEICAgICAgICA9PiAkVmFsaWRJRCwKICAgIFVzZXJJRCAgICAgICAgID0+ICRVc2VySUQsCik7CgpteSAlQ3VzdG9tZXJVc2VyID0gJEN1c3RvbWVyVXNlck9iamVjdC0+Q3VzdG9tZXJVc2VyRGF0YUdldCgKICAgIFVzZXIgPT4gJEN1c3RvbWVyVXNlckxvZ2luLAopOwoKZGVsZXRlICRDdXN0b21lclVzZXJ7Q29uZmlnfTsKZGVsZXRlICRDdXN0b21lclVzZXJ7Q29tcGFueUNvbmZpZ307CgojCiMgUHJlcGFyZSBhIGNvbmZpZyBpdGVtCiMKbXkgJENsYXNzZXMgPSAkR2VuZXJhbENhdGFsb2dPYmplY3QtPkl0ZW1MaXN0KAogICAgQ2xhc3MgPT4gJ0lUU006OkNvbmZpZ0l0ZW06OkNsYXNzJywKICAgIFZhbGlkID0+ICRWYWxpZElELAopOwpteSAlQ2xhc3NlcyA9IHJldmVyc2UgJXsgJENsYXNzZXMgLy8ge30gfTsKCiRTZWxmLT5UcnVlKAogICAgc2NhbGFyICRDbGFzc2Vze0xvY2F0aW9ufSwKICAgICdDbGFzcyAiTG9jYXRpb24iIG11c3QgYmUgcHJlc2VudC4nLAopOwoKbXkgJENvbmZpZ0l0ZW1JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5Db25maWdJdGVtQWRkKAogICAgQ2xhc3NJRCA9PiAkQ2xhc3Nlc3tMb2NhdGlvbn0sCiAgICBVc2VySUQgID0+ICRVc2VySUQsCik7CgokU2VsZi0+VHJ1ZSgKICAgIHNjYWxhciAkQ29uZmlnSXRlbUlELAogICAgJ0NvbmZpZyBpdGVtIG11c3QgaGF2ZSBiZWVuIGNyZWF0ZWQgc3VjY2Vzc2Z1bGx5LicsCikgfHwgcmV0dXJuOwoKbXkgJExvY2F0aW9uVHlwZSA9ICRHZW5lcmFsQ2F0YWxvZ09iamVjdC0+SXRlbUdldCgKICAgIENsYXNzID0+ICdJVFNNOjpDb25maWdJdGVtOjpMb2NhdGlvbjo6VHlwZScsCiAgICBOYW1lICA9PiAnT2ZmaWNlJywKKTsKCm15ICREZXBsU3RhdGVQcm9kdWN0aW9uID0gJEdlbmVyYWxDYXRhbG9nT2JqZWN0LT5JdGVtR2V0KAogICAgQ2xhc3MgPT4gJ0lUU006OkNvbmZpZ0l0ZW06OkRlcGxveW1lbnRTdGF0ZScsCiAgICBOYW1lICA9PiAnUHJvZHVjdGlvbicsCik7CgpteSAkSW5jaVN0YXRlT3BlcmF0aW9uYWwgPSAkR2VuZXJhbENhdGFsb2dPYmplY3QtPkl0ZW1HZXQoCiAgICBDbGFzcyA9PiAnSVRTTTo6Q29yZTo6SW5jaWRlbnRTdGF0ZScsCiAgICBOYW1lICA9PiAnT3BlcmF0aW9uYWwnLAopOwoKbXkgJERlZmluaXRpb24gPSAkQ29uZmlnSXRlbU9iamVjdC0+RGVmaW5pdGlvbkdldCgKICAgIENsYXNzSUQgPT4gJENsYXNzZXN7TG9jYXRpb259LAopOwoKbXkgJExvY2F0aW9uTmFtZSA9ICdVbml0VGVzdExvY2F0aW9uJyAuICRIZWxwZXJPYmplY3QtPkdldFJhbmRvbUlEKCk7CgpteSAkVmVyc2lvbklEID0gJFpudW55SGVscGVyT2JqZWN0LT5fSVRTTUNvbmZpZ0l0ZW1WZXJzaW9uQWRkKAogICAgQ29uZmlnSXRlbUlEICA9PiAkQ29uZmlnSXRlbUlELAogICAgTmFtZSAgICAgICAgICA9PiAkTG9jYXRpb25OYW1lLAogICAgQ2xhc3NOYW1lICAgICA9PiAnTG9jYXRpb24nLAogICAgRGVwbFN0YXRlTmFtZSA9PiAnUHJvZHVjdGlvbicsCiAgICBJbmNpU3RhdGVOYW1lID0+ICdPcGVyYXRpb25hbCcsCiAgICBYTUxEYXRhICAgICAgID0+IHsKICAgICAgICBUeXBlICAgICAgID0+ICRMb2NhdGlvblR5cGUtPntJdGVtSUR9LAogICAgICAgIEN1c3RvbWVySUQgPT4gJEN1c3RvbWVySUQsCiAgICAgICAgT3duZXIgICAgICA9PiAkQ3VzdG9tZXJVc2VyTG9naW4sCiAgICB9LAopOwoKbXkgJENvbmZpZ0l0ZW1EYXRhID0gJElUU01Db25maWdJdGVtSW52b2tlck9iamVjdC0+R2V0Q29uZmlnSXRlbURhdGEoCiAgICBDb25maWdJdGVtSUQgPT4gJENvbmZpZ0l0ZW1JRCwKICAgIEV2ZW50ICAgICAgICA9PiB1bmRlZiwKKTsKCm15ICVFeHBlY3RlZENvbmZpZ0l0ZW1EYXRhID0gKAogICAgWE1MRGF0YSA9PiB7CiAgICAgICAgVHlwZSA9PiBbCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIENvbnRlbnQgICAgICAgPT4gJExvY2F0aW9uVHlwZS0+e0l0ZW1JRH0sCiAgICAgICAgICAgICAgICBSZWFkYWJsZVZhbHVlID0+ICdPZmZpY2UnLAogICAgICAgICAgICB9LAogICAgICAgIF0sCiAgICAgICAgQ3VzdG9tZXJJRCA9PiBbCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIENvbnRlbnQgICAgICAgPT4gJEN1c3RvbWVySUQsCiAgICAgICAgICAgICAgICBSZWFkYWJsZVZhbHVlID0+IFwlQ3VzdG9tZXJDb21wYW55LAogICAgICAgICAgICB9LAogICAgICAgIF0sCiAgICAgICAgT3duZXIgPT4gWwogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBDb250ZW50ICAgICAgID0+ICRDdXN0b21lclVzZXJMb2dpbiwKICAgICAgICAgICAgICAgIFJlYWRhYmxlVmFsdWUgPT4gXCVDdXN0b21lclVzZXIsCiAgICAgICAgICAgIH0sCiAgICAgICAgXSwKICAgIH0sCiAgICBDb25maWdJdGVtSUQgPT4gJENvbmZpZ0l0ZW1JRCwKICAgIERlZmluaXRpb25JRCA9PiAkRGVmaW5pdGlvbi0+e0RlZmluaXRpb25JRH0sCiAgICBJbmNpU3RhdGVJRCAgPT4gJEluY2lTdGF0ZU9wZXJhdGlvbmFsLT57SXRlbUlEfSwKICAgIE5hbWUgICAgICAgICA9PiAkTG9jYXRpb25OYW1lLAogICAgRGVwbFN0YXRlSUQgID0+ICREZXBsU3RhdGVQcm9kdWN0aW9uLT57SXRlbUlEfSwKICAgIERlcGxTdGF0ZSAgICA9PiAnUHJvZHVjdGlvbicsCiAgICBDbGFzc0lEICAgICAgPT4gJENsYXNzZXN7TG9jYXRpb259LAogICAgSW5jaVN0YXRlICAgID0+ICdPcGVyYXRpb25hbCcsCiAgICBDbGFzcyAgICAgICAgPT4gJ0xvY2F0aW9uJywKKTsKCiRTZWxmLT5Jc0RlZXBseSgKICAgICRDb25maWdJdGVtRGF0YSwKICAgIFwlRXhwZWN0ZWRDb25maWdJdGVtRGF0YSwKICAgICdHZXRDb25maWdJdGVtRGF0YSgpOiBDb25maWcgaXRlbSBkYXRhIG11c3QgbWF0Y2ggZXhwZWN0ZWQgb25lLicsCik7CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAyMSBabnVueSBHbWJILCBodHRwczovL3pudW55Lm9yZy8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7Cgp1c2UgdmFycyAocXcoJFNlbGYpKTsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6VmFyaWFibGVDaGVjayBxdyg6YWxsKTsKCiRLZXJuZWw6Ok9NLT5PYmplY3RQYXJhbUFkZCgKICAgICdLZXJuZWw6OlN5c3RlbTo6VW5pdFRlc3Q6OkhlbHBlcicgPT4gewogICAgICAgIFJlc3RvcmVEYXRhYmFzZSA9PiAxLAogICAgfSwKKTsKCm15ICRDb25maWdPYmplY3QgICAgICAgICAgICAgICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpDb25maWcnKTsKbXkgJEhlbHBlck9iamVjdCAgICAgICAgICAgICAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VW5pdFRlc3Q6OkhlbHBlcicpOwpteSAkVGlja2V0T2JqZWN0ICAgICAgICAgICAgICAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpUaWNrZXQnKTsKbXkgJFVuaXRUZXN0SVRTTUNvbmZpZ0l0ZW1PYmplY3QgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VW5pdFRlc3Q6OklUU01Db25maWdJdGVtJyk7Cm15ICRVbml0VGVzdFBhcmFtT2JqZWN0ICAgICAgICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpQYXJhbScpOwpteSAkSVRTTUNvbmZpZ0l0ZW1DdXN0b21lckNJc09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbUN1c3RvbWVyQ0lzJyk7CgpteSAlQ29uZmlnSXRlbUtleSA9ICV7CiAgICAkQ29uZmlnT2JqZWN0LT5HZXQoJ0FnZW50Q3VzdG9tZXJVc2VySW5mb3JtYXRpb25DZW50ZXI6OkJhY2tlbmQnKS0+eycwMDYwLUNVSUMtSVRTTUNvbmZpZ0l0ZW1DdXN0b21lclVzZXInfQogICAgICAgIC0+e0NvbmZpZ0l0ZW1LZXl9IHx8IHt9Cn07CgpteSAkUmFuZG9tSUQgICAgICAgPSAkSGVscGVyT2JqZWN0LT5HZXRSYW5kb21JRCgpOwpteSAkQ29uZmlnSXRlbU5hbWUgPSAiVW5pdFRlc3RQQyAkUmFuZG9tSUQiOwoKbXkgJFRlc3RVc2VyTG9naW4zID0gJEhlbHBlck9iamVjdC0+VGVzdEN1c3RvbWVyVXNlckNyZWF0ZSgpOwpteSAkVmVyc2lvblJlZiAgICAgPSAkVW5pdFRlc3RJVFNNQ29uZmlnSXRlbU9iamVjdC0+Q29uZmlnSXRlbUNyZWF0ZSgKICAgIE5hbWUgICAgICAgICAgPT4gJENvbmZpZ0l0ZW1OYW1lLAogICAgQ2xhc3NOYW1lICAgICA9PiAnQ29tcHV0ZXInLAogICAgRGVwbFN0YXRlTmFtZSA9PiAnUHJvZHVjdGlvbicsCiAgICBJbmNpU3RhdGVOYW1lID0+ICdPcGVyYXRpb25hbCcsCiAgICBYTUxEYXRhICAgICAgID0+IHsKICAgICAgICBPdGhlckVxdWlwbWVudCAgICAgICAgICAgPT4gJy4uLicsCiAgICAgICAgTm90ZSAgICAgICAgICAgICAgICAgICAgID0+ICcuLi4nLAogICAgICAgIFNlcmlhbE51bWJlciAgICAgICAgICAgICA9PiAnLi4uJywKICAgICAgICBXYXJyYW50eUV4cGlyYXRpb25EYXRlICAgPT4gJzIwMTYtMDEtMDEnLAogICAgICAgIEluc3RhbGxEYXRlICAgICAgICAgICAgICA9PiAnMjAxNi0wMS0wMScsCiAgICAgICAgJENvbmZpZ0l0ZW1LZXl7Q29tcHV0ZXJ9ID0+ICRUZXN0VXNlckxvZ2luMywKICAgIH0sCik7CgpteSBAQ29uZmlnSXRlbXMgPSAkSVRTTUNvbmZpZ0l0ZW1DdXN0b21lckNJc09iamVjdC0+R2V0UG9zc2libGVDdXN0b21lckNJcygKICAgIEN1c3RvbWVyVXNlcklEID0+ICRUZXN0VXNlckxvZ2luMywKKTsKCiRTZWxmLT5Jc0RlZXBseSgKICAgIFxAQ29uZmlnSXRlbXMsCiAgICBbCiAgICAgICAgewogICAgICAgICAgICAnTGlua2VkJyAgICAgICA9PiAnMCcsCiAgICAgICAgICAgICdDb25maWdJdGVtSUQnID0+ICRWZXJzaW9uUmVmLT57Q29uZmlnSXRlbUlEfSwKICAgICAgICAgICAgJ051bWJlcicgICAgICAgPT4gJFZlcnNpb25SZWYtPntOdW1iZXJ9LAogICAgICAgICAgICAnTmFtZScgICAgICAgICA9PiAkQ29uZmlnSXRlbU5hbWUsCiAgICAgICAgICAgICdDbGFzcycgICAgICAgID0+ICdDb21wdXRlcicsCiAgICAgICAgICAgICdJY29uJyAgICAgICAgID0+ICcnLAogICAgICAgIH0KICAgIF0sCiAgICAiR2V0UG9zc2libGVDdXN0b21lckNJcyIsCik7CgokVW5pdFRlc3RQYXJhbU9iamVjdC0+UGFyYW1TZXQoCiAgICBOYW1lICA9PiAnSVRTTUNvbmZpZ0l0ZW1DdXN0b21lckNJTGlzdCcsCiAgICBWYWx1ZSA9PiAkVmVyc2lvblJlZi0+e0NvbmZpZ0l0ZW1JRH0sCik7CgpteSAkVGlja2V0SUQgPSAkSGVscGVyT2JqZWN0LT5UaWNrZXRDcmVhdGUoCiAgICBDdXN0b21lclVzZXIgPT4gJFRlc3RVc2VyTG9naW4zLAopOwoKJFRpY2tldE9iamVjdC0+RXZlbnRIYW5kbGVyVHJhbnNhY3Rpb24oKTsKCkBDb25maWdJdGVtcyA9ICRJVFNNQ29uZmlnSXRlbUN1c3RvbWVyQ0lzT2JqZWN0LT5HZXRQb3NzaWJsZUN1c3RvbWVyQ0lzKAogICAgQ3VzdG9tZXJVc2VySUQgPT4gJFRlc3RVc2VyTG9naW4zLAogICAgVGlja2V0SUQgICAgICAgPT4gJFRpY2tldElELAopOwoKJFNlbGYtPklzRGVlcGx5KAogICAgXEBDb25maWdJdGVtcywKICAgIFsKICAgICAgICB7CiAgICAgICAgICAgICdMaW5rZWQnICAgICAgID0+ICcxJywKICAgICAgICAgICAgJ0NvbmZpZ0l0ZW1JRCcgPT4gJFZlcnNpb25SZWYtPntDb25maWdJdGVtSUR9LAogICAgICAgICAgICAnTnVtYmVyJyAgICAgICA9PiAkVmVyc2lvblJlZi0+e051bWJlcn0sCiAgICAgICAgICAgICdOYW1lJyAgICAgICAgID0+ICRDb25maWdJdGVtTmFtZSwKICAgICAgICAgICAgJ0NsYXNzJyAgICAgICAgPT4gJ0NvbXB1dGVyJywKICAgICAgICAgICAgJ0ljb24nICAgICAgICAgPT4gJycsCiAgICAgICAgfQogICAgXSwKICAgICJHZXRQb3NzaWJsZUN1c3RvbWVyQ0lzIHdpdGggVGlja2V0IGxpbmsiLAopOwoKMTsK
# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

use Kernel::System::ObjectManager;

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

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);

my $HelperObject                 = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
my $ConfigItemObject             = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
my $GeneralCatalogObject         = $Kernel::OM->Get('Kernel::System::GeneralCatalog');
my $DynamicFieldObject           = $Kernel::OM->Get('Kernel::System::DynamicField');
my $DynamicFieldBackendObject    = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');
my $ValidObject                  = $Kernel::OM->Get('Kernel::System::Valid');
my $TicketObject                 = $Kernel::OM->Get('Kernel::System::Ticket');
my $ZnunyHelperObject            = $Kernel::OM->Get('Kernel::System::ZnunyHelper');
my $DynamicFieldConfigItemObject = $Kernel::OM->Get('Kernel::System::DynamicField::ConfigItem');

my $ValidID = $ValidObject->ValidLookup(
    Valid => 'valid',
);

my $UserID = 1;

#
# Prepare config item
#
my $ClassListRef = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::Class',
    Valid => $ValidID,
);
my %ClassList = reverse %{ $ClassListRef || {} };

my $YesNoRef = $GeneralCatalogObject->ItemList(
    Class => 'ITSM::ConfigItem::YesNo',
    Valid => $ValidID,
);
my %YesNoList = reverse %{ $YesNoRef || {} };

my $ConfigItem1ID = $ConfigItemObject->ConfigItemAdd(
    ClassID => $ClassList{Computer},
    UserID  => $UserID,
);

$Self->True(
    scalar $ConfigItem1ID,
    'Config item must have been created successfully.',
);

my $ConfigItem1VersionID = $ZnunyHelperObject->_ITSMConfigItemVersionAdd(
    ConfigItemID  => $ConfigItem1ID,
    Name          => 'Unit test computer 1',
    ClassName     => 'Computer',
    DeplStateName => 'Production',
    InciStateName => 'Operational',
    XMLData       => {
        OtherEquipment         => 'Mouse',
        Note                   => 'Unit test computer 1 note.',
        WarrantyExpirationDate => '2040-01-01',
        InstallDate            => '2040-01-01',
        CPU                    => [
            {
                Content => 'CPU 1',
            },
            {
                Content => 'CPU 2',
            },
        ],
        NIC => [
            {
                Content    => 'NIC',
                IPoverDHCP => [
                    {
                        Content => $YesNoList{Yes},
                    },
                ],
                IPAddress => [
                    {
                        Content => '127.0.0.1'
                    },
                ],
            },
        ],
    },
);

$Self->True(
    scalar $ConfigItem1VersionID,
    'Config item version must have been created successfully.',
);

my $ConfigItem2ID = $ConfigItemObject->ConfigItemAdd(
    ClassID => $ClassList{Computer},
    UserID  => $UserID,
);

$Self->True(
    scalar $ConfigItem2ID,
    'Config item must have been created successfully.',
);

my $ConfigItem2VersionID = $ZnunyHelperObject->_ITSMConfigItemVersionAdd(
    ConfigItemID  => $ConfigItem2ID,
    Name          => 'Unit test computer 2',
    ClassName     => 'Computer',
    DeplStateName => 'Production',
    InciStateName => 'Operational',
    XMLData       => {
        OtherEquipment         => 'Mouse',
        Note                   => 'Unit test computer 2 note.',
        WarrantyExpirationDate => '2041-01-01',
        InstallDate            => '2041-01-01',
        CPU                    => [
            {
                Content => 'CPU 1',
            },
            {
                Content => 'CPU 4',
            },
        ],
        NIC => [
            {
                Content    => 'NIC',
                IPoverDHCP => [
                    {
                        Content => $YesNoList{No},
                    },
                ],
                IPAddress => [
                    {
                        Content => '127.0.0.2'
                    },
                ],
            },
        ],
    },
);

$Self->True(
    scalar $ConfigItem2VersionID,
    'Config item version must have been created successfully.',
);

#
# Prepare dynamic fields.
#
my @DynamicFields = (
    {
        Name          => 'DynamicFieldConfigItemUnitTestText1',
        Label         => 'DynamicFieldConfigItemUnitTestText1',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Text',
        Config        => {
            DefaultValue => '',
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestText2',
        Label         => 'DynamicFieldConfigItemUnitTestText2',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Text',
        Config        => {
            DefaultValue => '',
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestText3',
        Label         => 'DynamicFieldConfigItemUnitTestText3',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Text',
        Config        => {
            DefaultValue => '',
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestText4',
        Label         => 'DynamicFieldConfigItemUnitTestText4',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Text',
        Config        => {
            DefaultValue => '',
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestText5',
        Label         => 'DynamicFieldConfigItemUnitTestText5',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Text',
        Config        => {
            DefaultValue => '',
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestDateTime',
        Label         => 'DynamicFieldConfigItemUnitTestDateTime',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'DateTime',
        Config        => {
            DefaultValue => '',
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestDate',
        Label         => 'DynamicFieldConfigItemUnitTestDate',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Date',
        Config        => {
            DefaultValue => '',
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestMultiselect1',
        Label         => 'DynamicFieldConfigItemUnitTestMultiselect1',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Multiselect',
        Config        => {
            DefaultValue   => '',
            PossibleNone   => 1,
            PossibleValues => {
                'CPU 1' => 'CPU 1',
                'CPU 2' => 'CPU 2',
                'CPU 3' => 'CPU 3',
            },
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestMultiselect2',
        Label         => 'DynamicFieldConfigItemUnitTestMultiselect2',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Multiselect',
        Config        => {
            DefaultValue   => '',
            PossibleNone   => 1,
            PossibleValues => {
                'CPU 1' => 'CPU 1',
                'CPU 2' => 'CPU 2',
                'CPU 3' => 'CPU 3',
            },
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestMultiselect3',
        Label         => 'DynamicFieldConfigItemUnitTestMultiselect3',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Multiselect',
        Config        => {
            DefaultValue   => '',
            PossibleNone   => 1,
            PossibleValues => {
                'CPU 1' => 'CPU 1',
                'CPU 2' => 'CPU 2',
                'CPU 3' => 'CPU 3',
            },
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestDropdown1',
        Label         => 'DynamicFieldConfigItemUnitTestDropdown1',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Dropdown',
        Config        => {
            DefaultValue   => '',
            PossibleNone   => 1,
            PossibleValues => {
                'CPU 1' => 'CPU 1',
                'CPU 2' => 'CPU 2',
                'CPU 3' => 'CPU 3',
            },
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestDropdown2',
        Label         => 'DynamicFieldConfigItemUnitTestDropdown2',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Dropdown',
        Config        => {
            DefaultValue   => '',
            PossibleNone   => 1,
            PossibleValues => {
                'CPU 1' => 'CPU 1',
                'CPU 2' => 'CPU 2',
                'CPU 3' => 'CPU 3',
            },
        },
    },
    {
        Name          => 'DynamicFieldConfigItemUnitTestDropdown3',
        Label         => 'DynamicFieldConfigItemUnitTestDropdown3',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'Dropdown',
        Config        => {
            DefaultValue   => '',
            PossibleNone   => 1,
            PossibleValues => {
                'CPU 1' => 'CPU 1',
                'CPU 2' => 'CPU 2',
                'CPU 3' => 'CPU 3',
            },
        },
    },

    # ConfigItemDropdown (single select)
    {
        Name          => 'DynamicFieldConfigItemUnitTestConfigItem1',
        Label         => 'DynamicFieldConfigItemUnitTestConfigItem1',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'ConfigItemDropdown',
        Config        => {
            ConfigItemClass       => 'Computer',
            ConfigItemLinkType    => undef,
            ConfigItemLinkSource  => undef,
            ConfigItemLinkRemoval => 0,
            AdditionalDFStorage   => [
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText1',
                    ConfigItemKey => 'CPU',
                    Type          => 'Frontend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText2',
                    ConfigItemKey => 'CPU::2',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText3',
                    ConfigItemKey => 'Name',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText4',
                    ConfigItemKey => 'NIC::1::IPAddress::1',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText5',
                    ConfigItemKey => 'ClassID',
                    Type          => 'Backend',                               # Notice that this one is backend only
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDate',
                    ConfigItemKey => 'WarrantyExpirationDate',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDateTime',
                    ConfigItemKey => 'WarrantyExpirationDate',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDropdown1',
                    ConfigItemKey => 'CPU::1',
                    Type          => 'Frontend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDropdown2',
                    ConfigItemKey => 'CPU',
                    Type          => 'Backend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestMultiselect1',
                    ConfigItemKey => 'CPU::2',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestMultiselect2',
                    ConfigItemKey => 'CPU',
                    Type          => 'FrontendBackend',
                },
            ],
        },
    },

    # ConfigItemMultiselect
    {
        Name          => 'DynamicFieldConfigItemUnitTestConfigItem2',
        Label         => 'DynamicFieldConfigItemUnitTestConfigItem2',
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'ConfigItemMultiselect',
        Config        => {
            ConfigItemClass       => 'Computer',
            ConfigItemLinkType    => undef,
            ConfigItemLinkSource  => undef,
            ConfigItemLinkRemoval => 0,
            AdditionalDFStorage   => [
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText1',
                    ConfigItemKey => 'CPU',
                    Type          => 'Backend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText2',
                    ConfigItemKey => 'CPU::2',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText3',
                    ConfigItemKey => 'Name',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText4',
                    ConfigItemKey => 'NIC::1::IPAddress::1',
                    Type          => 'FrontendBackend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestText5',
                    ConfigItemKey => 'ClassID',
                    Type          => 'Frontend',                              # Notice that this one is frontend only
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDate',
                    ConfigItemKey => 'WarrantyExpirationDate',
                    Type          => 'Backend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDateTime',
                    ConfigItemKey => 'WarrantyExpirationDate',
                    Type          => 'Backend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDropdown1',
                    ConfigItemKey => 'CPU::1',
                    Type          => 'Backend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestDropdown2',
                    ConfigItemKey => 'CPU',
                    Type          => 'Backend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestMultiselect1',
                    ConfigItemKey => 'CPU::2',
                    Type          => 'Backend',
                },
                {
                    DynamicField  => 'DynamicFieldConfigItemUnitTestMultiselect2',
                    ConfigItemKey => 'CPU',
                    Type          => 'Backend',
                },
            ],
        },
    },
);

my $DynamicFieldsCreated = $ZnunyHelperObject->_DynamicFieldsCreate(@DynamicFields);

$Self->True(
    scalar $DynamicFieldsCreated,
    'Dynamic fields must have been created successfully.',
);

#
# ConfigItemDropdown (DynamicFieldConfigItemUnitTestConfigItem1)
#

my $TicketID = $HelperObject->TicketCreate();

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

# Triggers event Kernel::System::Ticket::Event::DynamicFieldConfigItemAdditionalDFStorage
$DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $DynamicFieldConfig,
    ObjectID           => $TicketID,
    Value              => $ConfigItem1ID,
    UserID             => $UserID,
);

# Only additional dynamic fields set to type "backend" are expected to be set by the event.
my %ExpectedDynamicFieldData = (
    DynamicFieldConfigItemUnitTestText1        => undef,                    # configured for frontend, not backend
    DynamicFieldConfigItemUnitTestText2        => 'CPU 2',
    DynamicFieldConfigItemUnitTestText3        => 'Unit test computer 1',
    DynamicFieldConfigItemUnitTestText4        => '127.0.0.1',
    DynamicFieldConfigItemUnitTestText5        => $ClassList{Computer},
    DynamicFieldConfigItemUnitTestDate         => '2040-01-01',
    DynamicFieldConfigItemUnitTestDateTime     => '2040-01-01 00:00:00',
    DynamicFieldConfigItemUnitTestDropdown1    => undef,                    # configured for frontend, not backend
    DynamicFieldConfigItemUnitTestDropdown2    => 'CPU 1',
    DynamicFieldConfigItemUnitTestMultiselect1 => [
        'CPU 2',
    ],
    DynamicFieldConfigItemUnitTestMultiselect2 => [
        'CPU 1',
        'CPU 2',
    ],
);

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

for my $DynamicFieldName ( sort keys %ExpectedDynamicFieldData ) {
    my $ExpectedValue = $ExpectedDynamicFieldData{$DynamicFieldName};

    if ( $DynamicFieldName eq 'DynamicFieldConfigItemUnitTestDate' ) {
        $ExpectedValue .= ' 00:00:00';
    }

    $Self->IsDeeply(
        $Ticket{ 'DynamicField_' . $DynamicFieldName },
        $ExpectedValue,
        "ConfigItemDropdown - Content of dynamic field $DynamicFieldName must match expected one.",
    );
}

#
# ConfigItemMultiselect (DynamicFieldConfigItemUnitTestConfigItem2)
#

$TicketID = $HelperObject->TicketCreate();

$DynamicFieldConfig = $DynamicFieldObject->DynamicFieldGet(
    Name => 'DynamicFieldConfigItemUnitTestConfigItem2',
);

# Triggers event Kernel::System::Ticket::Event::DynamicFieldConfigItemAdditionalDFStorage
$DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $DynamicFieldConfig,
    ObjectID           => $TicketID,
    Value              => [ $ConfigItem1ID, $ConfigItem2ID, ],
    UserID             => $UserID,
);

# # Only additional dynamic fields set to type "backend" are expected to be set by the event.
%ExpectedDynamicFieldData = (
    DynamicFieldConfigItemUnitTestText1 => 'CPU 1, CPU 2, CPU 1, CPU 4',
    DynamicFieldConfigItemUnitTestText2 => 'CPU 2, CPU 4',
    DynamicFieldConfigItemUnitTestText3 => 'Unit test computer 1, Unit test computer 2',
    DynamicFieldConfigItemUnitTestText4 => '127.0.0.1, 127.0.0.2',
    DynamicFieldConfigItemUnitTestText5        => undef,                   # configured for frontend, not backend
    DynamicFieldConfigItemUnitTestDate         => '2040-01-01',
    DynamicFieldConfigItemUnitTestDateTime     => '2040-01-01 00:00:00',
    DynamicFieldConfigItemUnitTestDropdown1    => 'CPU 1',
    DynamicFieldConfigItemUnitTestDropdown2    => 'CPU 1',
    DynamicFieldConfigItemUnitTestMultiselect1 => [
        'CPU 2',
        'CPU 4',
    ],
    DynamicFieldConfigItemUnitTestMultiselect2 => [
        'CPU 1',
        'CPU 2',
        'CPU 1',
        'CPU 4',
    ],
);

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

for my $DynamicFieldName ( sort keys %ExpectedDynamicFieldData ) {
    my $ExpectedValue = $ExpectedDynamicFieldData{$DynamicFieldName};

    if ( $DynamicFieldName eq 'DynamicFieldConfigItemUnitTestDate' ) {
        $ExpectedValue .= ' 00:00:00';
    }

    $Self->IsDeeply(
        $Ticket{ 'DynamicField_' . $DynamicFieldName },
        $ExpectedValue,
        "ConfigItemMultiselect - Content of dynamic field $DynamicFieldName must match expected one.",
    );
}

1;

# --
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

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

$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);

my $HelperObject                 = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
my $TicketObject                 = $Kernel::OM->Get('Kernel::System::Ticket');
my $ZnunyHelperObject            = $Kernel::OM->Get('Kernel::System::ZnunyHelper');
my $UnitTestITSMConfigItemObject = $Kernel::OM->Get('Kernel::System::UnitTest::ITSMConfigItem');
my $DynamicFieldObject           = $Kernel::OM->Get('Kernel::System::DynamicField');
my $DynamicFieldBackendObject    = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');
my $LinkObject                   = $Kernel::OM->Get('Kernel::System::LinkObject');

my @DynamicFields = (
    {
        Name          => 'ConfigItemDropdown',
        Label         => "ConfigItemDropdown",
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'ConfigItemDropdown',
        Config        => {
            ConfigItemClass       => 'Computer',
            ConfigItemLinkType    => 'DependsOn',
            ConfigItemLinkSource  => 'Ticket',
            ConfigItemLinkRemoval => 0,
        },
    },
    {
        Name          => 'ConfigItemMultiselect',
        Label         => "ConfigItemMultiselect",
        InternalField => 0,
        ObjectType    => 'Ticket',
        FieldType     => 'ConfigItemMultiselect',
        Config        => {
            ConfigItemClass       => 'Computer',
            ConfigItemLinkType    => 'AlternativeTo',
            ConfigItemLinkSource  => 'ITSMConfigItem',
            ConfigItemLinkRemoval => 1,
        },
    },
);

$ZnunyHelperObject->_DynamicFieldsCreate(@DynamicFields);

my %DynamicFieldConfig;
for my $DynamicField (qw(ConfigItemDropdown ConfigItemMultiselect)) {
    $DynamicFieldConfig{$DynamicField} = $DynamicFieldObject->DynamicFieldGet(
        Name => $DynamicField,
    );

}

my $TicketID = $HelperObject->TicketCreate();

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

$Self->Is(
    $Ticket{DynamicField_ConfigItemDropdown},
    undef,
    'ConfigItemDropdown',
);

$Self->Is(
    $Ticket{DynamicField_ConfigItemMultiselect},
    undef,
    'ConfigItemMultiselect',
);

my $Version1 = $UnitTestITSMConfigItemObject->ConfigItemCreate(
    Name          => 'DynamicField ConfigItemDropdown Test 1',
    ClassName     => 'Computer',
    DeplStateName => 'Production',
    InciStateName => 'Operational',
    XMLData       => {
        OtherEquipment         => '...',
        Note                   => '...',
        SerialNumber           => '...',
        WarrantyExpirationDate => '2016-01-01',
        InstallDate            => '2016-01-01',
    },
);

my $Version2 = $UnitTestITSMConfigItemObject->ConfigItemCreate(
    Name          => 'DynamicField ConfigItemMultiselect Test 2',
    ClassName     => 'Computer',
    DeplStateName => 'Production',
    InciStateName => 'Operational',
    XMLData       => {
        OtherEquipment         => '...',
        Note                   => '...',
        SerialNumber           => '...',
        WarrantyExpirationDate => '2016-01-01',
        InstallDate            => '2016-01-01',
    },
);

my $Version3 = $UnitTestITSMConfigItemObject->ConfigItemCreate(
    Name          => 'DynamicField ConfigItemMultiselect Test 3',
    ClassName     => 'Computer',
    DeplStateName => 'Production',
    InciStateName => 'Operational',
    XMLData       => {
        OtherEquipment         => '...',
        Note                   => '...',
        SerialNumber           => '...',
        WarrantyExpirationDate => '2016-01-01',
        InstallDate            => '2016-01-01',
    },
);

$DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $DynamicFieldConfig{'ConfigItemDropdown'},
    ObjectID           => $TicketID,
    Value              => $Version1->{ConfigItemID},
    UserID             => 1,
);

my @Values = (
    $Version1->{ConfigItemID},
    $Version2->{ConfigItemID},
);

$DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $DynamicFieldConfig{'ConfigItemMultiselect'},
    ObjectID           => $TicketID,
    UserID             => 1,
    Value              => \@Values,
);

$Kernel::OM->ObjectsDiscard(
    Objects => ['Kernel::System::Ticket'],
);

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

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

$Self->Is(
    $Ticket{DynamicField_ConfigItemDropdown},
    $Version1->{ConfigItemID},
    'ConfigItemDropdown',
);

$Self->IsDeeply(
    $Ticket{DynamicField_ConfigItemMultiselect},
    \@Values,
    'ConfigItemMultiselect',
);

my $LinkList = $LinkObject->LinkList(
    Object  => 'Ticket',
    Key     => $TicketID,
    Object2 => 'ITSMConfigItem',
    State   => 'Valid',
    UserID  => 1,
);

$Self->IsDeeply(
    $LinkList,
    {
        ITSMConfigItem => {
            DependsOn => {
                Target => {
                    $Version1->{ConfigItemID} => 1,
                },
            },
            AlternativeTo => {
                Source => {
                    $Version1->{ConfigItemID} => 1,
                    $Version2->{ConfigItemID} => 1,
                },
            },
        },
    },
    'LinkList',
);

$Kernel::OM->ObjectsDiscard(
    Objects => ['Kernel::System::Ticket'],
);

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

# change ConfigItems again
$DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $DynamicFieldConfig{'ConfigItemDropdown'},
    ObjectID           => $TicketID,
    Value              => $Version2->{ConfigItemID},
    UserID             => 1,
);

@Values = (
    $Version1->{ConfigItemID},
    $Version3->{ConfigItemID},
);

$DynamicFieldBackendObject->ValueSet(
    DynamicFieldConfig => $DynamicFieldConfig{'ConfigItemMultiselect'},
    ObjectID           => $TicketID,
    UserID             => 1,
    Value              => \@Values,
);

$Kernel::OM->ObjectsDiscard(
    Objects => ['Kernel::System::Ticket'],
);

$TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

%Ticket = $TicketObject->TicketGet(
    TicketID      => $TicketID,
    DynamicFields => 1,
    UserID        => 1,
);

$Self->Is(
    $Ticket{DynamicField_ConfigItemDropdown},
    $Version2->{ConfigItemID},
    'ConfigItemDropdown',
);

$Self->IsDeeply(
    $Ticket{DynamicField_ConfigItemMultiselect},
    \@Values,
    'ConfigItemMultiselect',
);

$LinkList = $LinkObject->LinkList(
    Object  => 'Ticket',
    Key     => $TicketID,
    Object2 => 'ITSMConfigItem',
    State   => 'Valid',
    UserID  => 1,
);

$Self->IsDeeply(
    $LinkList,
    {
        ITSMConfigItem => {
            DependsOn => {
                Target => {
                    $Version2->{ConfigItemID} => 1,

                    # Because dropdown field is configured to not remove links to removed config items.
                    $Version1->{ConfigItemID} => 1,
                },
            },
            AlternativeTo => {
                Source => {
                    $Version1->{ConfigItemID} => 1,
                    $Version3->{ConfigItemID} => 1,
                },
            },
        },
    },
    'LinkList',
);

1;

LS0tCkRlYnVnZ2VyOgogIERlYnVnVGhyZXNob2xkOiBkZWJ1ZwogIFRlc3RNb2RlOiAwCkRlc2NyaXB0aW9uOiBJVFNNIENvbmZpZ3VyYXRpb24gTWFuYWdlbWVudCBDb25uZWN0b3IgU2FtcGxlCkZyYW1ld29ya1ZlcnNpb246IDUuMC54IGdpdApQcm92aWRlcjoKICBPcGVyYXRpb246CiAgICBDb25maWdJdGVtQ3JlYXRlOgogICAgICBEZXNjcmlwdGlvbjogQ3JlYXRlcyBuZXcgQ29uZmlndXJhdGlvbiBJdGVtcyBmcm9tIElUU01Db25maWd1cmF0aW9uTWFuYWdlbWVudAogICAgICBNYXBwaW5nSW5ib3VuZDoge30KICAgICAgTWFwcGluZ091dGJvdW5kOiB7fQogICAgICBUeXBlOiBDb25maWdJdGVtOjpDb25maWdJdGVtQ3JlYXRlCiAgICBDb25maWdJdGVtRGVsZXRlOgogICAgICBEZXNjcmlwdGlvbjogRGVsZXRlcyBDb25maWd1cmF0aW9uIEl0ZW1zIGZyb20gSVRTTUNvbmZpZ3VyYXRpb25NYW5hZ2VtZW50CiAgICAgIE1hcHBpbmdJbmJvdW5kOiB7fQogICAgICBNYXBwaW5nT3V0Ym91bmQ6IHt9CiAgICAgIFR5cGU6IENvbmZpZ0l0ZW06OkNvbmZpZ0l0ZW1EZWxldGUKICAgIENvbmZpZ0l0ZW1HZXQ6CiAgICAgIERlc2NyaXB0aW9uOiBHZXQgdGhlIGRldGFpbHMgZm9yIENvbmZpZ3VyYXRpb24gSXRlbXMgZnJvbSBJVFNNQ29uZmlndXJhdGlvbk1hbmFnZW1lbnQKICAgICAgTWFwcGluZ0luYm91bmQ6IHt9CiAgICAgIE1hcHBpbmdPdXRib3VuZDoge30KICAgICAgVHlwZTogQ29uZmlnSXRlbTo6Q29uZmlnSXRlbUdldAogICAgQ29uZmlnSXRlbVNlYXJjaDoKICAgICAgRGVzY3JpcHRpb246IFNlYXJjaCBDb25maWd1cmF0aW9uIEl0ZW1zIGZyb20gSVRTTUNvbmZpZ3VyYXRpb25NYW5hZ2VtZW50CiAgICAgIE1hcHBpbmdJbmJvdW5kOiB7fQogICAgICBNYXBwaW5nT3V0Ym91bmQ6IHt9CiAgICAgIFR5cGU6IENvbmZpZ0l0ZW06OkNvbmZpZ0l0ZW1TZWFyY2gKICAgIENvbmZpZ0l0ZW1VcGRhdGU6CiAgICAgIERlc2NyaXB0aW9uOiBVcGRhdGVzIENvbmZpZ3VyYXRpb24gSXRlbXMgZnJvbSBJVFNNQ29uZmlndXJhdGlvbk1hbmFnZW1lbnQKICAgICAgTWFwcGluZ0luYm91bmQ6IHt9CiAgICAgIE1hcHBpbmdPdXRib3VuZDoge30KICAgICAgVHlwZTogQ29uZmlnSXRlbTo6Q29uZmlnSXRlbVVwZGF0ZQogIFRyYW5zcG9ydDoKICAgIENvbmZpZzoKICAgICAgTWF4TGVuZ3RoOiAxMDAwMDAwMDAKICAgICAgTmFtZVNwYWNlOiBodHRwOi8vd3d3Lm90cnMub3JnL0NvbmZpZ0l0ZW1Db25uZWN0b3IKICAgICAgUmVxdWVzdE5hbWVGcmVlVGV4dDogJycKICAgICAgUmVxdWVzdE5hbWVTY2hlbWU6IFBsYWluCiAgICAgIFJlc3BvbnNlTmFtZUZyZWVUZXh0OiAnJwogICAgICBSZXNwb25zZU5hbWVTY2hlbWU6IFJlc3BvbnNlCiAgICBUeXBlOiBIVFRQOjpTT0FQClJlbW90ZVN5c3RlbTogJycKUmVxdWVzdGVyOgogIFRyYW5zcG9ydDoKICAgIFR5cGU6ICcnCg==
LS0tCkRlYnVnZ2VyOgogIERlYnVnVGhyZXNob2xkOiBkZWJ1ZwogIFRlc3RNb2RlOiAwCkRlc2NyaXB0aW9uOiBJVFNNIENvbmZpZ3VyYXRpb24gTWFuYWdlbWVudCBDb25uZWN0b3IgU2FtcGxlCkZyYW1ld29ya1ZlcnNpb246IDUuMC54IGdpdApQcm92aWRlcjoKICBPcGVyYXRpb246CiAgICBDb25maWdJdGVtQ3JlYXRlOgogICAgICBEZXNjcmlwdGlvbjogQ3JlYXRlcyBuZXcgQ29uZmlndXJhdGlvbiBJdGVtcyBmcm9tIElUU01Db25maWd1cmF0aW9uTWFuYWdlbWVudAogICAgICBNYXBwaW5nSW5ib3VuZDoge30KICAgICAgTWFwcGluZ091dGJvdW5kOiB7fQogICAgICBUeXBlOiBDb25maWdJdGVtOjpDb25maWdJdGVtQ3JlYXRlCiAgICBDb25maWdJdGVtRGVsZXRlOgogICAgICBEZXNjcmlwdGlvbjogRGVsZXRlcyBDb25maWd1cmF0aW9uIEl0ZW1zIGZyb20gSVRTTUNvbmZpZ3VyYXRpb25NYW5hZ2VtZW50CiAgICAgIE1hcHBpbmdJbmJvdW5kOiB7fQogICAgICBNYXBwaW5nT3V0Ym91bmQ6IHt9CiAgICAgIFR5cGU6IENvbmZpZ0l0ZW06OkNvbmZpZ0l0ZW1EZWxldGUKICAgIENvbmZpZ0l0ZW1HZXQ6CiAgICAgIERlc2NyaXB0aW9uOiBHZXQgdGhlIGRldGFpbHMgZm9yIENvbmZpZ3VyYXRpb24gSXRlbXMgZnJvbSBJVFNNQ29uZmlndXJhdGlvbk1hbmFnZW1lbnQKICAgICAgTWFwcGluZ0luYm91bmQ6IHt9CiAgICAgIE1hcHBpbmdPdXRib3VuZDoge30KICAgICAgVHlwZTogQ29uZmlnSXRlbTo6Q29uZmlnSXRlbUdldAogICAgQ29uZmlnSXRlbVNlYXJjaDoKICAgICAgRGVzY3JpcHRpb246IFNlYXJjaCBDb25maWd1cmF0aW9uIEl0ZW1zIGZyb20gSVRTTUNvbmZpZ3VyYXRpb25NYW5hZ2VtZW50CiAgICAgIE1hcHBpbmdJbmJvdW5kOiB7fQogICAgICBNYXBwaW5nT3V0Ym91bmQ6IHt9CiAgICAgIFR5cGU6IENvbmZpZ0l0ZW06OkNvbmZpZ0l0ZW1TZWFyY2gKICAgIENvbmZpZ0l0ZW1VcGRhdGU6CiAgICAgIERlc2NyaXB0aW9uOiBVcGRhdGVzIENvbmZpZ3VyYXRpb24gSXRlbXMgZnJvbSBJVFNNQ29uZmlndXJhdGlvbk1hbmFnZW1lbnQKICAgICAgTWFwcGluZ0luYm91bmQ6IHt9CiAgICAgIE1hcHBpbmdPdXRib3VuZDoge30KICAgICAgVHlwZTogQ29uZmlnSXRlbTo6Q29uZmlnSXRlbVVwZGF0ZQogIFRyYW5zcG9ydDoKICAgIENvbmZpZzoKICAgICAgTWF4TGVuZ3RoOiAxMDAwMDAwMDAKICAgICAgTmFtZVNwYWNlOiBodHRwOi8vd3d3Lm90cnMub3JnL0NvbmZpZ0l0ZW1Db25uZWN0b3IKICAgICAgUmVxdWVzdE5hbWVGcmVlVGV4dDogJycKICAgICAgUmVxdWVzdE5hbWVTY2hlbWU6IFBsYWluCiAgICAgIFJlc3BvbnNlTmFtZUZyZWVUZXh0OiAnJwogICAgICBSZXNwb25zZU5hbWVTY2hlbWU6IFJlc3BvbnNlCiAgICBUeXBlOiBIVFRQOjpTT0FQClJlbW90ZVN5c3RlbTogJycKUmVxdWVzdGVyOgogIFRyYW5zcG9ydDoKICAgIFR5cGU6ICcnCg==
LS0tCkRlYnVnZ2VyOgogIERlYnVnVGhyZXNob2xkOiBkZWJ1ZwogIFRlc3RNb2RlOiAwCkRlc2NyaXB0aW9uOiBJVFNNIENvbmZpZ3VyYXRpb24gTWFuYWdlbWVudCBDb25uZWN0b3IgU2FtcGxlCkZyYW1ld29ya1ZlcnNpb246IDUuMC54IGdpdApQcm92aWRlcjoKICBPcGVyYXRpb246CiAgICBDb25maWdJdGVtQ3JlYXRlOgogICAgICBEZXNjcmlwdGlvbjogQ3JlYXRlcyBuZXcgQ29uZmlndXJhdGlvbiBJdGVtcyBmcm9tIElUU01Db25maWd1cmF0aW9uTWFuYWdlbWVudAogICAgICBNYXBwaW5nSW5ib3VuZDoge30KICAgICAgTWFwcGluZ091dGJvdW5kOiB7fQogICAgICBUeXBlOiBDb25maWdJdGVtOjpDb25maWdJdGVtQ3JlYXRlCiAgICBDb25maWdJdGVtRGVsZXRlOgogICAgICBEZXNjcmlwdGlvbjogRGVsZXRlcyBDb25maWd1cmF0aW9uIEl0ZW1zIGZyb20gSVRTTUNvbmZpZ3VyYXRpb25NYW5hZ2VtZW50CiAgICAgIE1hcHBpbmdJbmJvdW5kOiB7fQogICAgICBNYXBwaW5nT3V0Ym91bmQ6IHt9CiAgICAgIFR5cGU6IENvbmZpZ0l0ZW06OkNvbmZpZ0l0ZW1EZWxldGUKICAgIENvbmZpZ0l0ZW1HZXQ6CiAgICAgIERlc2NyaXB0aW9uOiBHZXQgdGhlIGRldGFpbHMgZm9yIENvbmZpZ3VyYXRpb24gSXRlbXMgZnJvbSBJVFNNQ29uZmlndXJhdGlvbk1hbmFnZW1lbnQKICAgICAgTWFwcGluZ0luYm91bmQ6IHt9CiAgICAgIE1hcHBpbmdPdXRib3VuZDoge30KICAgICAgVHlwZTogQ29uZmlnSXRlbTo6Q29uZmlnSXRlbUdldAogICAgQ29uZmlnSXRlbVNlYXJjaDoKICAgICAgRGVzY3JpcHRpb246IFNlYXJjaCBDb25maWd1cmF0aW9uIEl0ZW1zIGZyb20gSVRTTUNvbmZpZ3VyYXRpb25NYW5hZ2VtZW50CiAgICAgIE1hcHBpbmdJbmJvdW5kOiB7fQogICAgICBNYXBwaW5nT3V0Ym91bmQ6IHt9CiAgICAgIFR5cGU6IENvbmZpZ0l0ZW06OkNvbmZpZ0l0ZW1TZWFyY2gKICAgIENvbmZpZ0l0ZW1VcGRhdGU6CiAgICAgIERlc2NyaXB0aW9uOiBVcGRhdGVzIENvbmZpZ3VyYXRpb24gSXRlbXMgZnJvbSBJVFNNQ29uZmlndXJhdGlvbk1hbmFnZW1lbnQKICAgICAgTWFwcGluZ0luYm91bmQ6IHt9CiAgICAgIE1hcHBpbmdPdXRib3VuZDoge30KICAgICAgVHlwZTogQ29uZmlnSXRlbTo6Q29uZmlnSXRlbVVwZGF0ZQogIFRyYW5zcG9ydDoKICAgIENvbmZpZzoKICAgICAgTWF4TGVuZ3RoOiAxMDAwMDAwMDAKICAgICAgTmFtZVNwYWNlOiBodHRwOi8vd3d3Lm90cnMub3JnL0NvbmZpZ0l0ZW1Db25uZWN0b3IKICAgICAgUmVxdWVzdE5hbWVGcmVlVGV4dDogJycKICAgICAgUmVxdWVzdE5hbWVTY2hlbWU6IFBsYWluCiAgICAgIFJlc3BvbnNlTmFtZUZyZWVUZXh0OiAnJwogICAgICBSZXNwb25zZU5hbWVTY2hlbWU6IFJlc3BvbnNlCiAgICBUeXBlOiBIVFRQOjpTT0FQClJlbW90ZVN5c3RlbTogJycKUmVxdWVzdGVyOgogIFRyYW5zcG9ydDoKICAgIFR5cGU6ICcnCg==
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91Ci8vIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCi8vIC0tCgoidXNlIHN0cmljdCI7Cgp2YXIgQ29yZSAgID0gQ29yZSB8fCB7fTsKCkNvcmUuQWdlbnQgPSBDb3JlLkFnZW50IHx8IHt9OwpDb3JlLkFnZW50LkFkbWluID0gQ29yZS5BZ2VudC5BZG1pbiB8fCB7fTsKCi8qKgogKiBAbmFtZXNwYWNlCiAqIEBleHBvcnRzIFRhcmdldE5TIGFzIENvcmUuQWdlbnQuQWRtaW4uRHluYW1pY0ZpZWxkQ29uZmlnSXRlbQogKiBAZGVzY3JpcHRpb24KICogICAgICBUaGlzIG5hbWVzcGFjZSBjb250YWlucyB0aGUgc3BlY2lhbCBmdW5jdGlvbnMgZm9yIER5bmFtaWNGaWVsZENvbmZpZ0l0ZW0uCiAqLwpDb3JlLkFnZW50LkFkbWluLkR5bmFtaWNGaWVsZENvbmZpZ0l0ZW0gPSAoZnVuY3Rpb24gKFRhcmdldE5TKSB7CgogICAgVGFyZ2V0TlMuSW5pdCA9IGZ1bmN0aW9uICgpIHsKICAgICAgICBJbml0TGlua1R5cGVTZWxlY3Rpb24oKTsKICAgIH0KCiAgICBmdW5jdGlvbiBJbml0TGlua1R5cGVTZWxlY3Rpb24oKSB7CiAgICAgICAgJCgnI0NvbmZpZ0l0ZW1MaW5rVHlwZScpLm9uKCdjaGFuZ2UnLCBmdW5jdGlvbigpIHsKICAgICAgICAgICAgdmFyIENvbmZpZ0l0ZW1MaW5rVHlwZSA9ICQodGhpcykudmFsKCk7CgogICAgICAgICAgICBpZiAoQ29uZmlnSXRlbUxpbmtUeXBlKSB7CiAgICAgICAgICAgICAgICBTaG93TGlua1NvdXJjZVNlbGVjdGlvbigpOwogICAgICAgICAgICAgICAgcmV0dXJuOwogICAgICAgICAgICB9CgogICAgICAgICAgICBIaWRlTGlua1NvdXJjZVNlbGVjdGlvbigpOwogICAgICAgIH0pCiAgICAgICAgLnRyaWdnZXIoJ2NoYW5nZScpOwogICAgfQoKICAgIGZ1bmN0aW9uIFNob3dMaW5rU291cmNlU2VsZWN0aW9uKCkgewogICAgICAgICQoJyNDb25maWdJdGVtTGlua1NvdXJjZScpLmNsb3Nlc3QoJ2RpdicpLnNob3coKTsKICAgICAgICAkKCdsYWJlbFtmb3I9IkNvbmZpZ0l0ZW1MaW5rU291cmNlIl0nKS5zaG93KCk7CgogICAgICAgICQoJyNDb25maWdJdGVtTGlua1JlbW92YWwnKS5jbG9zZXN0KCdkaXYnKS5zaG93KCk7CiAgICAgICAgJCgnbGFiZWxbZm9yPSJDb25maWdJdGVtTGlua1JlbW92YWwiXScpLnNob3coKTsKCiAgICAgICAgQ29yZS5VSS5JbnB1dEZpZWxkcy5BY3RpdmF0ZSgpOwogICAgfQoKICAgIGZ1bmN0aW9uIEhpZGVMaW5rU291cmNlU2VsZWN0aW9uKCkgewogICAgICAgICQoJyNDb25maWdJdGVtTGlua1NvdXJjZScpLmNsb3Nlc3QoJ2RpdicpLmhpZGUoKTsKICAgICAgICAkKCdsYWJlbFtmb3I9IkNvbmZpZ0l0ZW1MaW5rU291cmNlIl0nKS5oaWRlKCk7CgogICAgICAgICQoJyNDb25maWdJdGVtTGlua1JlbW92YWwnKS5jbG9zZXN0KCdkaXYnKS5oaWRlKCk7CiAgICAgICAgJCgnbGFiZWxbZm9yPSJDb25maWdJdGVtTGlua1JlbW92YWwiXScpLmhpZGUoKTsKICAgIH0KCiAgICByZXR1cm4gVGFyZ2V0TlM7Cn0oQ29yZS5BZ2VudC5BZG1pbi5EeW5hbWljRmllbGRDb25maWdJdGVtIHx8IHt9KSk7Cg==
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMjEgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKLy8gZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQuCi8vIC0tCgoidXNlIHN0cmljdCI7Cgp2YXIgSVRTTSA9IElUU00gfHwge307CklUU00uQWdlbnQgPSBJVFNNLkFnZW50IHx8IHt9OwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gPSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gfHwge307CgovKioKICogQG5hbWVzcGFjZSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uQWRkCiAqIEBtZW1iZXJvZiBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0KICogQGF1dGhvciBPVFJTIEFHCiAqIEBkZXNjcmlwdGlvbgogKiAgICAgIFRoaXMgbmFtZXNwYWNlIGNvbnRhaW5zIHRoZSBzcGVjaWFsIG1vZHVsZSBmdW5jdGlvbnMgZm9yIHRoZSBjb25maWcgaXRlbSBhZGQuCiAqLwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uQWRkID0gKGZ1bmN0aW9uIChUYXJnZXROUykgewoKICAgIC8qCiAgICAqIEBuYW1lIEluaXQKICAgICogQG1lbWJlcm9mIElUU00uQWdlbnQuQ29uZmlnSXRlbS5BZGQKICAgICogQGZ1bmN0aW9uCiAgICAqIEBkZXNjcmlwdGlvbgogICAgKiAgICAgIFRoaXMgZnVuY3Rpb24gaW5pdGlhbGl6ZXMgdGFibGUgZmlsdGVyLgogICAgKi8KICAgIFRhcmdldE5TLkluaXQgPSBmdW5jdGlvbiAoKSB7CiAgICAgICAgQ29yZS5VSS5UYWJsZS5Jbml0VGFibGVGaWx0ZXIoJCgnI0ZpbHRlckNsYXNzZXMnKSwgJCgnI0NsYXNzZXMnKSk7CiAgICB9OwoKICAgIENvcmUuSW5pdC5SZWdpc3Rlck5hbWVzcGFjZShUYXJnZXROUywgJ0FQUF9NT0RVTEUnKTsKCiAgICByZXR1cm4gVGFyZ2V0TlM7Cn0oSVRTTS5BZ2VudC5Db25maWdJdGVtLkFkZCB8fCB7fSkpOwo=
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMjEgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKLy8gZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQuCi8vIC0tCgoidXNlIHN0cmljdCI7Cgp2YXIgSVRTTSA9IElUU00gfHwge307CklUU00uQWdlbnQgPSBJVFNNLkFnZW50IHx8IHt9OwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gPSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gfHwge307CgovKioKICogQG5hbWVzcGFjZSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uQWRkCiAqIEBtZW1iZXJvZiBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0KICogQGF1dGhvciBPVFJTIEFHCiAqIEBkZXNjcmlwdGlvbgogKiAgICAgIFRoaXMgbmFtZXNwYWNlIGNvbnRhaW5zIHRoZSBzcGVjaWFsIG1vZHVsZSBmdW5jdGlvbnMgZm9yIHRoZSBjb25maWcgaXRlbSBhZGQuCiAqLwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uQWRkID0gKGZ1bmN0aW9uIChUYXJnZXROUykgewoKICAgIC8qCiAgICAqIEBuYW1lIEluaXQKICAgICogQG1lbWJlcm9mIElUU00uQWdlbnQuQ29uZmlnSXRlbS5BZGQKICAgICogQGZ1bmN0aW9uCiAgICAqIEBkZXNjcmlwdGlvbgogICAgKiAgICAgIFRoaXMgZnVuY3Rpb24gaW5pdGlhbGl6ZXMgdGFibGUgZmlsdGVyLgogICAgKi8KICAgIFRhcmdldE5TLkluaXQgPSBmdW5jdGlvbiAoKSB7CiAgICAgICAgQ29yZS5VSS5UYWJsZS5Jbml0VGFibGVGaWx0ZXIoJCgnI0ZpbHRlckNsYXNzZXMnKSwgJCgnI0NsYXNzZXMnKSk7CiAgICB9OwoKICAgIENvcmUuSW5pdC5SZWdpc3Rlck5hbWVzcGFjZShUYXJnZXROUywgJ0FQUF9NT0RVTEUnKTsKCiAgICByZXR1cm4gVGFyZ2V0TlM7Cn0oSVRTTS5BZ2VudC5Db25maWdJdGVtLkFkZCB8fCB7fSkpOwo=
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMjEgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKLy8gZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQuCi8vIC0tCgoidXNlIHN0cmljdCI7Cgp2YXIgSVRTTSA9IElUU00gfHwge307CklUU00uQWdlbnQgPSBJVFNNLkFnZW50IHx8IHt9OwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gPSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gfHwge307CgovKioKICogQG5hbWVzcGFjZSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uQWRkCiAqIEBtZW1iZXJvZiBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0KICogQGF1dGhvciBPVFJTIEFHCiAqIEBkZXNjcmlwdGlvbgogKiAgICAgIFRoaXMgbmFtZXNwYWNlIGNvbnRhaW5zIHRoZSBzcGVjaWFsIG1vZHVsZSBmdW5jdGlvbnMgZm9yIHRoZSBjb25maWcgaXRlbSBhZGQuCiAqLwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uQWRkID0gKGZ1bmN0aW9uIChUYXJnZXROUykgewoKICAgIC8qCiAgICAqIEBuYW1lIEluaXQKICAgICogQG1lbWJlcm9mIElUU00uQWdlbnQuQ29uZmlnSXRlbS5BZGQKICAgICogQGZ1bmN0aW9uCiAgICAqIEBkZXNjcmlwdGlvbgogICAgKiAgICAgIFRoaXMgZnVuY3Rpb24gaW5pdGlhbGl6ZXMgdGFibGUgZmlsdGVyLgogICAgKi8KICAgIFRhcmdldE5TLkluaXQgPSBmdW5jdGlvbiAoKSB7CiAgICAgICAgQ29yZS5VSS5UYWJsZS5Jbml0VGFibGVGaWx0ZXIoJCgnI0ZpbHRlckNsYXNzZXMnKSwgJCgnI0NsYXNzZXMnKSk7CiAgICB9OwoKICAgIENvcmUuSW5pdC5SZWdpc3Rlck5hbWVzcGFjZShUYXJnZXROUywgJ0FQUF9NT0RVTEUnKTsKCiAgICByZXR1cm4gVGFyZ2V0TlM7Cn0oSVRTTS5BZ2VudC5Db25maWdJdGVtLkFkZCB8fCB7fSkpOwo=
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91Ci8vIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCi8vIC0tCgoidXNlIHN0cmljdCI7Cgp2YXIgQ29yZSAgID0gQ29yZSB8fCB7fSwKSVRTTSA9IElUU00gfHwge307CgpDb3JlLkFnZW50ID0gQ29yZS5BZ2VudCB8fCB7fTsKSVRTTS5BZ2VudCA9IElUU00uQWdlbnQgfHwge307CklUU00uQWdlbnQuQ29uZmlnSXRlbSA9IElUU00uQWdlbnQuQ29uZmlnSXRlbSB8fCB7fTsKCi8qKgogKiBAbmFtZXNwYWNlCiAqIEBleHBvcnRzIFRhcmdldE5TIGFzIElUU00uQWdlbnQuQ29uZmlnSXRlbS5DdXN0b21lckNJc1dpZGdldAogKiBAZGVzY3JpcHRpb24KICogICAgICBUaGlzIG5hbWVzcGFjZSBjb250YWlucyB0aGUgc3BlY2lhbCBmdW5jdGlvbnMgZm9yIEFnZW50Q29uZmlnSXRlbUN1c3RvbWVyQ0lzV2lkZ2V0LgogKi8KSVRTTS5BZ2VudC5Db25maWdJdGVtLkN1c3RvbWVyQ0lzV2lkZ2V0ID0gKGZ1bmN0aW9uIChUYXJnZXROUykgewogICAgdmFyIEdldEN1c3RvbWVySW5mb0hhbmRsZXIsCiAgICAgICAgTGFzdEN1c3RvbWVyVXNlckNvdW50OwoKICAgIFRhcmdldE5TLkluaXQgPSBmdW5jdGlvbiAoUGFyYW0pIHsKICAgICAgICBUYXJnZXROUy5DcmVhdGVXaWRnZXQoUGFyYW0pOwogICAgICAgIFRhcmdldE5TLlNldEV2ZW50SGFuZGxlcigpOwoKICAgICAgICByZXR1cm4gdHJ1ZTsKICAgIH0KCiAgICBUYXJnZXROUy5DcmVhdGVXaWRnZXQgPSBmdW5jdGlvbiAoUGFyYW0pIHsKICAgICAgICB2YXIgQWN0aW9uID0gQ29yZS5Db25maWcuR2V0KCdBY3Rpb24nKTsKCiAgICAgICAgLy8gY2hlY2sgaWYgd2lkZ2V0IGV4aXN0cwogICAgICAgIGlmICgkKCcjQWdlbnRJVFNNQ29uZmlnSXRlbUN1c3RvbWVyQ0lzV2lkZ2V0JykubGVuZ3RoID4gMCkgcmV0dXJuOwoKICAgICAgICAvLyBhcHBlbmQgd2lkZ2V0IHRvIHNpZGViYXIKICAgICAgICAkKCcuU2lkZWJhckNvbHVtbicpLmFwcGVuZChQYXJhbS5XaWRnZXQpOwoKICAgICAgICBpZiAoUGFyYW0uQ29uZmlnSXRlbXMpIHsKICAgICAgICAgICAgJCgnI0FnZW50SVRTTUNvbmZpZ0l0ZW1DdXN0b21lckNJcycpLmVtcHR5KCk7CiAgICAgICAgICAgICQoJyNBZ2VudElUU01Db25maWdJdGVtQ3VzdG9tZXJDSXMnKS5hcHBlbmQoUGFyYW0uQ29uZmlnSXRlbXMpOwogICAgICAgIH0KCiAgICAgICAgaWYgKEFjdGlvbiA9PSAnQWdlbnRUaWNrZXRQaG9uZScgfHwgQWN0aW9uID09ICdBZ2VudFRpY2tldEVtYWlsJykgewogICAgICAgICAgICAkKCdmb3JtW25hbWU9Y29tcG9zZV0nKS5hcHBlbmQoJzxpbnB1dCB0eXBlPSJoaWRkZW4iIGlkPSJJVFNNQ29uZmlnSXRlbUN1c3RvbWVyQ0lMaXN0IiBuYW1lPSJJVFNNQ29uZmlnSXRlbUN1c3RvbWVyQ0lMaXN0IiAvPicpOwogICAgICAgIH0KCiAgICAgICAgcmV0dXJuIHRydWU7CiAgICB9OwoKICAgIFRhcmdldE5TLlNldEV2ZW50SGFuZGxlciA9IGZ1bmN0aW9uICgpIHsKICAgICAgICB2YXIgQWN0aW9uID0gQ29yZS5Db25maWcuR2V0KCdBY3Rpb24nKTsKCiAgICAgICAgaWYgKEFjdGlvbiA9PSAnQWdlbnRUaWNrZXRab29tJykgewogICAgICAgICAgICAkKCcuQ29uZmlnSXRlbUxpbmsnKS5vZmYoJ2NoYW5nZScpLm9uKCdjaGFuZ2UnLCBmdW5jdGlvbigpIHsKICAgICAgICAgICAgICAgIHZhciBUaWNrZXRJRCAgICAgPSBDb3JlLkNvbmZpZy5HZXQoJ1RpY2tldElEJyksCiAgICAgICAgICAgICAgICAgICAgQ29uZmlnSXRlbUlEID0gJCh0aGlzKS52YWwoKSwKICAgICAgICAgICAgICAgICAgICBTdWJhY3Rpb24gICAgPSAnTGlua0RlbGV0ZScsCiAgICAgICAgICAgICAgICAgICAgRGF0YTsKCiAgICAgICAgICAgICAgICBpZiAoJCh0aGlzKS5pcygiOmNoZWNrZWQiKSkgewogICAgICAgICAgICAgICAgICAgIFN1YmFjdGlvbiA9ICdMaW5rQWRkJzsKICAgICAgICAgICAgICAgIH0KCiAgICAgICAgICAgICAgICBEYXRhID0gewogICAgICAgICAgICAgICAgICAgIEFjdGlvbjogICAgICAgJ0FnZW50SVRTTUNvbmZpZ0l0ZW1DdXN0b21lckNJc1dpZGdldCcsCiAgICAgICAgICAgICAgICAgICAgU3ViYWN0aW9uOiAgICBTdWJhY3Rpb24sCiAgICAgICAgICAgICAgICAgICAgVGlja2V0SUQ6ICAgICBUaWNrZXRJRCwKICAgICAgICAgICAgICAgICAgICBDb25maWdJdGVtSUQ6IENvbmZpZ0l0ZW1JRCwKICAgICAgICAgICAgICAgIH07CgogICAgICAgICAgICAgICAgLy8gc2hvdyBhamF4IGxvYWRlcgogICAgICAgICAgICAgICAgJCh0aGlzKS5wYXJlbnQoKS5hcHBlbmQoJzxzcGFuIGlkPSJBSkFYTG9hZGVyIiBjbGFzcz0iQUpBWExvYWRlciI+PC9zcGFuPicpOwoKICAgICAgICAgICAgICAgIC8vIGRpc2FibGUgYWxsIGNoZWNrYm94ZXMKICAgICAgICAgICAgICAgICQoJy5Db25maWdJdGVtTGluaycpLmF0dHIoJ2Rpc2FibGVkJywgJ2Rpc2FibGVkJyk7CgogICAgICAgICAgICAgICAgQ29yZS5BSkFYLkZ1bmN0aW9uQ2FsbCgKICAgICAgICAgICAgICAgICAgICBDb3JlLkNvbmZpZy5HZXQoJ0NHSUhhbmRsZScpLAogICAgICAgICAgICAgICAgICAgIERhdGEsCiAgICAgICAgICAgICAgICAgICAgZnVuY3Rpb24gKCkgewogICAgICAgICAgICAgICAgICAgICAgICBsb2NhdGlvbi5yZWxvYWQoKTsKICAgICAgICAgICAgICAgICAgICB9LAogICAgICAgICAgICAgICAgICAgICdqc29uJwogICAgICAgICAgICAgICAgKTsKICAgICAgICAgICAgfSk7CiAgICAgICAgfQogICAgICAgIGVsc2UgewogICAgICAgICAgICBpZiAoIUdldEN1c3RvbWVySW5mb0hhbmRsZXIpIHsKICAgICAgICAgICAgICAgIEdldEN1c3RvbWVySW5mb0hhbmRsZXIgPSBDb3JlLkFwcC5TdWJzY3JpYmUoCiAgICAgICAgICAgICAgICAgICAgJ0V2ZW50LkFnZW50LkN1c3RvbWVyU2VhcmNoLkdldEN1c3RvbWVySW5mby5DYWxsYmFjaycsCiAgICAgICAgICAgICAgICAgICAgZnVuY3Rpb24gKCkgewogICAgICAgICAgICAgICAgICAgICAgICAkKCcjSVRTTUNvbmZpZ0l0ZW1DdXN0b21lckNJTGlzdCcpLnZhbCgnJyk7CiAgICAgICAgICAgICAgICAgICAgICAgIFRhcmdldE5TLkdldEN1c3RvbWVyQ0lzKCk7CiAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgKTsKCiAgICAgICAgICAgICAgICB3aW5kb3cuc2V0VGltZW91dCgKICAgICAgICAgICAgICAgICAgICBVcGRhdGVDdXN0b21lckNJc09uUmVtb3ZhbE9mTGFzdEN1c3RvbWVyVXNlciwKICAgICAgICAgICAgICAgICAgICAxMDAwCiAgICAgICAgICAgICAgICApOwogICAgICAgICAgICB9CgogICAgICAgICAgICAkKCcuQ29uZmlnSXRlbUxpbmsnKS5vZmYoJ2NoYW5nZScpLm9uKCdjaGFuZ2UnLCBmdW5jdGlvbigpIHsKICAgICAgICAgICAgICAgIHZhciBDb25maWdJdGVtSURzID0gJyc7CgogICAgICAgICAgICAgICAgJCgnI0lUU01Db25maWdJdGVtQ3VzdG9tZXJDSUxpc3QnKS52YWwoJycpOwoKICAgICAgICAgICAgICAgICQuZWFjaCgkKCIuQ29uZmlnSXRlbUxpbms6Y2hlY2tlZCIpLCBmdW5jdGlvbigpIHsKICAgICAgICAgICAgICAgICAgICBpZiAoQ29uZmlnSXRlbUlEcykgewogICAgICAgICAgICAgICAgICAgICAgICBDb25maWdJdGVtSURzICs9ICcsJzsKICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICAgQ29uZmlnSXRlbUlEcyArPSAkKHRoaXMpLnZhbCgpOwogICAgICAgICAgICAgICAgfSk7CgogICAgICAgICAgICAgICAgJCgnI0lUU01Db25maWdJdGVtQ3VzdG9tZXJDSUxpc3QnKS52YWwoQ29uZmlnSXRlbUlEcyk7CiAgICAgICAgICAgIH0pOwogICAgICAgIH0KCiAgICAgICAgcmV0dXJuIHRydWU7CiAgICB9OwoKICAgIFRhcmdldE5TLkdldEN1c3RvbWVyQ0lzID0gZnVuY3Rpb24gKCkgewogICAgICAgIHZhciBDdXN0b21lclVzZXJJRCA9ICQoJyNTZWxlY3RlZEN1c3RvbWVyVXNlcicpLnZhbCgpOwogICAgICAgIHZhciBDdXN0b21lcklEID0gJCgnI0N1c3RvbWVySUQnKS52YWwoKSwKICAgICAgICAgICAgRGF0YSA9IHsKICAgICAgICAgICAgQWN0aW9uOiAgICAgICAgICdBZ2VudElUU01Db25maWdJdGVtQ3VzdG9tZXJDSXNXaWRnZXQnLAogICAgICAgICAgICBTdWJhY3Rpb246ICAgICAgJ0N1c3RvbWVyVXBkYXRlJywKICAgICAgICAgICAgQ3VzdG9tZXJVc2VySUQ6IEN1c3RvbWVyVXNlcklELAogICAgICAgICAgICBDdXN0b21lcklEOiAgICAgQ3VzdG9tZXJJRCwKICAgICAgICB9OwoKICAgICAgICBDb3JlLkFKQVguRnVuY3Rpb25DYWxsKAogICAgICAgICAgICBDb3JlLkNvbmZpZy5HZXQoJ0NHSUhhbmRsZScpLAogICAgICAgICAgICBEYXRhLAogICAgICAgICAgICBmdW5jdGlvbiAoUGFyYW0pIHsKICAgICAgICAgICAgICAgICQoJyNBZ2VudElUU01Db25maWdJdGVtQ3VzdG9tZXJDSXMnKS5lbXB0eSgpOwoKICAgICAgICAgICAgICAgIGlmIChQYXJhbS5Db25maWdJdGVtcykgewogICAgICAgICAgICAgICAgICAgICQoJyNBZ2VudElUU01Db25maWdJdGVtQ3VzdG9tZXJDSXMnKS5hcHBlbmQoUGFyYW0uQ29uZmlnSXRlbXMpOwogICAgICAgICAgICAgICAgICAgIFRhcmdldE5TLlNldEV2ZW50SGFuZGxlcigpOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgZWxzZSB7CiAgICAgICAgICAgICAgICAgICAgJCgnI0FnZW50SVRTTUNvbmZpZ0l0ZW1DdXN0b21lckNJcycpLmFwcGVuZChDb3JlLkxhbmd1YWdlLlRyYW5zbGF0ZSgnbm9uZScpKTsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfSwKICAgICAgICAgICAgJ2pzb24nCiAgICAgICAgKTsKCiAgICAgICAgcmV0dXJuIHRydWU7CiAgICB9OwoKICAgIC8vIFdvcmthcm91bmQgZm9yIGNsZWFyaW5nIGNvbmZpZyBpdGVtIGxpc3QgaWYgdGhlIGxhc3QgY3VzdG9tZXIgdXNlciB3YXMgcmVtb3ZlZAogICAgLy8gZnJvbSB0aGUgZGlhbG9nLiBUaGUgb25lIHJlbWFpbmluZyAiQ3VzdG9tZXJTZWxlY3RlZCIgZWxlbWVudCBpcyBmcm9tIHRoZQogICAgLy8gdGVtcGxhdGUgdGhhdCBpcyBiZWluZyB1c2VkIHRvIGFkZCBjdXN0b21lciB1c2VycyB0byB0aGUgZGlhbG9nLgogICAgZnVuY3Rpb24gVXBkYXRlQ3VzdG9tZXJDSXNPblJlbW92YWxPZkxhc3RDdXN0b21lclVzZXIoKSB7CiAgICAgICAgdmFyIEN1cnJlbnRDdXN0b21lclVzZXJDb3VudCA9ICQoJ2lucHV0W25hbWU9IkN1c3RvbWVyU2VsZWN0ZWQiXScpLmxlbmd0aDsKCiAgICAgICAgaWYgKEN1cnJlbnRDdXN0b21lclVzZXJDb3VudCA9PSAxICYmIEN1cnJlbnRDdXN0b21lclVzZXJDb3VudCAhPSBMYXN0Q3VzdG9tZXJVc2VyQ291bnQpIHsKICAgICAgICAgICAgJCgnI0lUU01Db25maWdJdGVtQ3VzdG9tZXJDSUxpc3QnKS52YWwoJycpOwogICAgICAgICAgICBUYXJnZXROUy5HZXRDdXN0b21lckNJcygpOwogICAgICAgIH0KCiAgICAgICAgTGFzdEN1c3RvbWVyVXNlckNvdW50ID0gQ3VycmVudEN1c3RvbWVyVXNlckNvdW50OwoKICAgICAgICB3aW5kb3cuc2V0VGltZW91dCgKICAgICAgICAgICAgVXBkYXRlQ3VzdG9tZXJDSXNPblJlbW92YWxPZkxhc3RDdXN0b21lclVzZXIsCiAgICAgICAgICAgIDEwMDAKICAgICAgICApOwogICAgfQoKICAgIHJldHVybiBUYXJnZXROUzsKfShJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uQ3VzdG9tZXJDSXNXaWRnZXQgfHwge30pKTsK
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMjEgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKLy8gZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQuCi8vIC0tCgoidXNlIHN0cmljdCI7Cgp2YXIgSVRTTSA9IElUU00gfHwge307CklUU00uQWdlbnQgPSBJVFNNLkFnZW50IHx8IHt9OwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gPSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gfHwge307CgovKioKICogQG5hbWVzcGFjZQogKiBAZXhwb3J0cyBUYXJnZXROUyBhcyBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uU2VhcmNoCiAqIEBkZXNjcmlwdGlvbgogKiAgICAgIFRoaXMgbmFtZXNwYWNlIGNvbnRhaW5zIHRoZSBzcGVjaWFsIG1vZHVsZSBmdW5jdGlvbnMgZm9yIHRoZSBzZWFyY2guCiAqLwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uQ3VzdG9tZXJTZWFyY2ggPSAoZnVuY3Rpb24gKFRhcmdldE5TKSB7CgogICAgLyoqCiAgICAgKiBAbmFtZSBJbml0CiAgICAgKiBAbWVtYmVyb2YgSVRTTS5BZ2VudC5DdXN0b21lclNlYXJjaAogICAgICogQGZ1bmN0aW9uCiAgICAgKiBAcGFyYW0ge2pRdWVyeU9iamVjdH0gJEVsZW1lbnQgLSBUaGUgalF1ZXJ5IG9iamVjdCBvZiB0aGUgaW5wdXQgZmllbGQgd2l0aCBhdXRvY29tcGxldGUuCiAgICAgKiBAZGVzY3JpcHRpb24KICAgICAqICAgICAgSW5pdGlhbGl6ZXMgdGhlIHNwZWNpYWwgbW9kdWxlIGZ1bmN0aW9ucy4KICAgICAqLwogICAgVGFyZ2V0TlMuSW5pdCA9IGZ1bmN0aW9uICgpIHsKCiAgICAgICAgdmFyIEN1c3RvbWVyU2VhcmNoSXRlbUlEcyA9IENvcmUuQ29uZmlnLkdldCgnQ3VzdG9tZXJTZWFyY2hJdGVtSURzJyksCiAgICAgICAgICAgIGk7CgogICAgICAgIGlmICh0eXBlb2YgQ3VzdG9tZXJTZWFyY2hJdGVtSURzICE9PSAndW5kZWZpbmVkJyAmJiBBcnJheS5pc0FycmF5KEN1c3RvbWVyU2VhcmNoSXRlbUlEcykgJiYgQ3VzdG9tZXJTZWFyY2hJdGVtSURzLmxlbmd0aCkgewoKICAgICAgICAgICAgZm9yIChpID0gMDsgaSA8IEN1c3RvbWVyU2VhcmNoSXRlbUlEcy5sZW5ndGg7IGkrKykgewogICAgICAgICAgICAgICAgLy8gZXNjYXBlIHBvc3NpYmxlIGNvbG9ucyAoOikgaW4gZWxlbWVudCBpZCBiZWNhdXNlIGpRdWVyeSBjYW4gbm90IGhhbmRsZSBpdCBpbiBpZCBhdHRyaWJ1dGUgc2VsZWN0b3JzCiAgICAgICAgICAgICAgICBJVFNNLkFnZW50LkN1c3RvbWVyU2VhcmNoLkluaXQoJCgiIyIgKyBDb3JlLkFwcC5Fc2NhcGVTZWxlY3RvcihDdXN0b21lclNlYXJjaEl0ZW1JRHNbaV0pKSk7CiAgICAgICAgICAgIH0KICAgICAgICB9CiAgICB9OwoKICAgIENvcmUuSW5pdC5SZWdpc3Rlck5hbWVzcGFjZShUYXJnZXROUywgJ0FQUF9NT0RVTEUnKTsKCiAgICByZXR1cm4gVGFyZ2V0TlM7Cn0oSVRTTS5BZ2VudC5Db25maWdJdGVtLkN1c3RvbWVyU2VhcmNoIHx8IHt9KSk7Cg==
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMjEgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKLy8gZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQuCi8vIC0tCgoidXNlIHN0cmljdCI7Cgp2YXIgSVRTTSA9IElUU00gfHwge307CklUU00uQWdlbnQgPSBJVFNNLkFnZW50IHx8IHt9OwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gPSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gfHwge307CgovKioKICogQG5hbWVzcGFjZQogKiBAZXhwb3J0cyBUYXJnZXROUyBhcyBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uU2VhcmNoCiAqIEBkZXNjcmlwdGlvbgogKiAgICAgIFRoaXMgbmFtZXNwYWNlIGNvbnRhaW5zIHRoZSBzcGVjaWFsIG1vZHVsZSBmdW5jdGlvbnMgZm9yIHRoZSBzZWFyY2guCiAqLwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uQ3VzdG9tZXJTZWFyY2ggPSAoZnVuY3Rpb24gKFRhcmdldE5TKSB7CgogICAgLyoqCiAgICAgKiBAbmFtZSBJbml0CiAgICAgKiBAbWVtYmVyb2YgSVRTTS5BZ2VudC5DdXN0b21lclNlYXJjaAogICAgICogQGZ1bmN0aW9uCiAgICAgKiBAcGFyYW0ge2pRdWVyeU9iamVjdH0gJEVsZW1lbnQgLSBUaGUgalF1ZXJ5IG9iamVjdCBvZiB0aGUgaW5wdXQgZmllbGQgd2l0aCBhdXRvY29tcGxldGUuCiAgICAgKiBAZGVzY3JpcHRpb24KICAgICAqICAgICAgSW5pdGlhbGl6ZXMgdGhlIHNwZWNpYWwgbW9kdWxlIGZ1bmN0aW9ucy4KICAgICAqLwogICAgVGFyZ2V0TlMuSW5pdCA9IGZ1bmN0aW9uICgpIHsKCiAgICAgICAgdmFyIEN1c3RvbWVyU2VhcmNoSXRlbUlEcyA9IENvcmUuQ29uZmlnLkdldCgnQ3VzdG9tZXJTZWFyY2hJdGVtSURzJyksCiAgICAgICAgICAgIGk7CgogICAgICAgIGlmICh0eXBlb2YgQ3VzdG9tZXJTZWFyY2hJdGVtSURzICE9PSAndW5kZWZpbmVkJyAmJiBBcnJheS5pc0FycmF5KEN1c3RvbWVyU2VhcmNoSXRlbUlEcykgJiYgQ3VzdG9tZXJTZWFyY2hJdGVtSURzLmxlbmd0aCkgewoKICAgICAgICAgICAgZm9yIChpID0gMDsgaSA8IEN1c3RvbWVyU2VhcmNoSXRlbUlEcy5sZW5ndGg7IGkrKykgewogICAgICAgICAgICAgICAgLy8gZXNjYXBlIHBvc3NpYmxlIGNvbG9ucyAoOikgaW4gZWxlbWVudCBpZCBiZWNhdXNlIGpRdWVyeSBjYW4gbm90IGhhbmRsZSBpdCBpbiBpZCBhdHRyaWJ1dGUgc2VsZWN0b3JzCiAgICAgICAgICAgICAgICBJVFNNLkFnZW50LkN1c3RvbWVyU2VhcmNoLkluaXQoJCgiIyIgKyBDb3JlLkFwcC5Fc2NhcGVTZWxlY3RvcihDdXN0b21lclNlYXJjaEl0ZW1JRHNbaV0pKSk7CiAgICAgICAgICAgIH0KICAgICAgICB9CiAgICB9OwoKICAgIENvcmUuSW5pdC5SZWdpc3Rlck5hbWVzcGFjZShUYXJnZXROUywgJ0FQUF9NT0RVTEUnKTsKCiAgICByZXR1cm4gVGFyZ2V0TlM7Cn0oSVRTTS5BZ2VudC5Db25maWdJdGVtLkN1c3RvbWVyU2VhcmNoIHx8IHt9KSk7Cg==
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMjEgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKLy8gZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQuCi8vIC0tCgoidXNlIHN0cmljdCI7Cgp2YXIgSVRTTSA9IElUU00gfHwge307CklUU00uQWdlbnQgPSBJVFNNLkFnZW50IHx8IHt9OwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gPSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gfHwge307CgovKioKICogQG5hbWVzcGFjZSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uRGFzaGJvYXJkCiAqIEBtZW1iZXJvZiBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0KICogQGF1dGhvciBPVFJTIEFHCiAqIEBkZXNjcmlwdGlvbgogKiAgICAgIFRoaXMgbmFtZXNwYWNlIGNvbnRhaW5zIHRoZSBzcGVjaWFsIG1vZHVsZSBmdW5jdGlvbnMgZm9yIHRoZSBjb25maWcgaXRlbSBvdmVydmlldyBuYXZiYXIuCiAqLwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uRGFzaGJvYXJkID0gKGZ1bmN0aW9uIChUYXJnZXROUykgewoKICAgIC8qCiAgICAqIEBuYW1lIEluaXQKICAgICogQG1lbWJlcm9mIElUU00uQWdlbnQuQ29uZmlnSXRlbS5EYXNoYm9hcmQKICAgICogQGZ1bmN0aW9uCiAgICAqIEBkZXNjcmlwdGlvbgogICAgKiAgICAgIFRoaXMgZnVuY3Rpb24gaW5pdGlhbGl6ZXMgdGhlIG92ZXJ2aWV3IGJlaGF2aW91cnMuCiAgICAqLwogICAgVGFyZ2V0TlMuSW5pdCA9IGZ1bmN0aW9uICgpIHsKCiAgICAgICAgdmFyIERhc2hsZXREYXRhID0gQ29yZS5Db25maWcuR2V0KCdJVFNNQ29uZmlnSXRlbUdlbmVyaWMnKTsKCiAgICAgICAgaWYgKHR5cGVvZiBEYXNobGV0RGF0YSAhPT0gJ3VuZGVmaW5lZCcpIHsKICAgICAgICAgICAgSVRTTUNvbmZpZ0l0ZW1HZW5lcmljKERhc2hsZXREYXRhKTsKCiAgICAgICAgICAgIC8vIFN1YnNjcmliZSB0byBDb250ZW50VXBkYXRlIGV2ZW50IHRvIGluaXRpYXRlIHRpY2tldCBnZW5lcmljIGV2ZW50cyBvbiB3aWRnZXQgdXBkYXRlCiAgICAgICAgICAgIENvcmUuQXBwLlN1YnNjcmliZSgnRXZlbnQuQUpBWC5Db250ZW50VXBkYXRlLkNhbGxiYWNrJywgZnVuY3Rpb24oJFdpZGdldEVsZW1lbnQpIHsKICAgICAgICAgICAgICAgIGlmICh0eXBlb2YgJFdpZGdldEVsZW1lbnQgIT09ICd1bmRlZmluZWQnICYmICRXaWRnZXRFbGVtZW50LnNlYXJjaChEYXNobGV0RGF0YS5OYW1lKSAhPT0gcGFyc2VJbnQoJy0xJywgMTApKSB7CiAgICAgICAgICAgICAgICAgICAgSVRTTUNvbmZpZ0l0ZW1HZW5lcmljKERhc2hsZXREYXRhKTsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfSk7CiAgICAgICAgfQogICAgfTsKCiAgICAvKioKICAgICAqIEBwcml2YXRlCiAgICAgKiBAbmFtZSBJVFNNQ29uZmlnSXRlbUdlbmVyaWMKICAgICAqIEBtZW1iZXJvZiBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uRGFzaGJvYXJkCiAgICAgKiBAZnVuY3Rpb24KICAgICAqIEBwYXJhbSB7T2JqZWN0fSBEYXNobGV0RGF0YSAtIEhhc2ggd2l0aCBjb250YWluZXIgbmFtZSBhbmQgSFRNTCBuYW1lCiAgICAgKiBAZGVzY3JpcHRpb24KICAgICAqICAgICAgSW5pdGlhbGl6ZXMgZGFzaGJvYXJkIHdpZGdldCBJVFNNQ29uZmlnSXRlbUdlbmVyaWMKICAgICAqLwogICAgZnVuY3Rpb24gSVRTTUNvbmZpZ0l0ZW1HZW5lcmljIChEYXNobGV0RGF0YSkgewoKICAgICAgICAgICAgJCgnI0Rhc2hib2FyZCcgKyBDb3JlLkFwcC5Fc2NhcGVTZWxlY3RvcihEYXNobGV0RGF0YS5OYW1lKSArICctYm94JykuZmluZCgnLlRhYi5BY3Rpb25zIGxpIGEnKS5vZmYoJ2NsaWNrJykub24oJ2NsaWNrJywgZnVuY3Rpb24oKSB7CiAgICAgICAgICAgICAgICB2YXIgQ3VzdG9tZXJJRCwKICAgICAgICAgICAgICAgICAgICBDdXN0b21lclVzZXJJRCwKICAgICAgICAgICAgICAgICAgICBGaWx0ZXI7CgogICAgICAgICAgICAgICAgQ3VzdG9tZXJJRCAgICAgID0gJCgnaW5wdXRbbmFtZT1DdXN0b21lcklEXScpLnZhbCgpIHx8ICcnOwogICAgICAgICAgICAgICAgQ3VzdG9tZXJVc2VySUQgID0gJCgnaW5wdXRbbmFtZT1DdXN0b21lclVzZXJJRF0nKS52YWwoKSB8fCAnJzsKICAgICAgICAgICAgICAgIEZpbHRlciAgICAgICAgICA9ICQodGhpcykuYXR0cignZGF0YS1maWx0ZXInKTsKCiAgICAgICAgICAgICAgICAkKCcjRGFzaGJvYXJkJyArIENvcmUuQXBwLkVzY2FwZVNlbGVjdG9yKERhc2hsZXREYXRhLk5hbWUpICsgJy1ib3gnKS5hZGRDbGFzcygnTG9hZGluZycpOwogICAgICAgICAgICAgICAgQ29yZS5BSkFYLkNvbnRlbnRVcGRhdGUoJCgnI0Rhc2hib2FyZCcgKyBDb3JlLkFwcC5Fc2NhcGVTZWxlY3RvcihEYXNobGV0RGF0YS5OYW1lKSksIENvcmUuQ29uZmlnLkdldCgnQmFzZWxpbmsnKSArICdBY3Rpb249JyArIENvcmUuQ29uZmlnLkdldCgnQWN0aW9uJykgKyAnO1N1YmFjdGlvbj1FbGVtZW50O05hbWU9JyArIERhc2hsZXREYXRhLk5hbWUgKyAnO0N1c3RvbWVySUQ9JyArIGVuY29kZVVSSUNvbXBvbmVudChDdXN0b21lcklEKSArICc7Q3VzdG9tZXJVc2VySUQ9JyArIGVuY29kZVVSSUNvbXBvbmVudChDdXN0b21lclVzZXJJRCkgKyAnO0ZpbHRlcj0nICsgRmlsdGVyLCBmdW5jdGlvbiAoKSB7CiAgICAgICAgICAgICAgICAgICAgJCgnI0Rhc2hib2FyZCcgKyBDb3JlLkFwcC5Fc2NhcGVTZWxlY3RvcihEYXNobGV0RGF0YS5OYW1lKSArICctYm94JykucmVtb3ZlQ2xhc3MoJ0xvYWRpbmcnKTsKICAgICAgICAgICAgICAgIH0pOwogICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwogICAgICAgICAgICB9KTsKCiAgICAgICAgICAgICQoJyNEYXNoYm9hcmQnICsgQ29yZS5BcHAuRXNjYXBlU2VsZWN0b3IoRGFzaGxldERhdGEuTmFtZSkgKyAnLWJveCcpLmZpbmQoJy5EYXRhVGFibGUgdGggYScpLm9mZignY2xpY2snKS5vbignY2xpY2snLCBmdW5jdGlvbigpIHsKICAgICAgICAgICAgICAgIHZhciBDdXN0b21lcklELAogICAgICAgICAgICAgICAgICAgIEN1c3RvbWVyVXNlcklELAogICAgICAgICAgICAgICAgICAgIFNvcnRCeSAgICAgICAgICA9ICQodGhpcykuYXR0cignZGF0YS1jb2x1bW4nKSwKICAgICAgICAgICAgICAgICAgICBPcmRlckJ5ICAgICAgICAgPSAnJywKICAgICAgICAgICAgICAgICAgICAkT3JkZXJCeU9iaiAgICAgPSAkKCcjRGFzaGJvYXJkJyArIENvcmUuQXBwLkVzY2FwZVNlbGVjdG9yKERhc2hsZXREYXRhLk5hbWUpICsgJy1ib3gnKS5maW5kKCd0aC5Tb3J0RGVzY2VuZGluZ0xhcmdlLCB0aC5Tb3J0QXNjZW5kaW5nTGFyZ2UnKTsKCiAgICAgICAgICAgICAgICBDdXN0b21lcklEICAgICAgPSAkKCdpbnB1dFtuYW1lPUN1c3RvbWVySURdJykudmFsKCkgfHwgJyc7CiAgICAgICAgICAgICAgICBDdXN0b21lclVzZXJJRCAgPSAkKCdpbnB1dFtuYW1lPUN1c3RvbWVyVXNlcklEXScpLnZhbCgpIHx8ICcnOwoKICAgICAgICAgICAgICAgIGlmICgkT3JkZXJCeU9iaiAmJiAkT3JkZXJCeU9iai5oYXNDbGFzcygnU29ydERlc2NlbmRpbmdMYXJnZScpKSB7CiAgICAgICAgICAgICAgICAgICAgT3JkZXJCeSA9ICdEb3duJzsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgIGVsc2UgaWYgKCRPcmRlckJ5T2JqICYmICRPcmRlckJ5T2JqLmhhc0NsYXNzKCdTb3J0QXNjZW5kaW5nTGFyZ2UnKSkgewogICAgICAgICAgICAgICAgICAgIE9yZGVyQnkgPSAnVXAnOwogICAgICAgICAgICAgICAgfQoKICAgICAgICAgICAgICAgICQoJyNEYXNoYm9hcmQnICsgQ29yZS5BcHAuRXNjYXBlU2VsZWN0b3IoRGFzaGxldERhdGEuTmFtZSkgKyAnLWJveCcpLmFkZENsYXNzKCdMb2FkaW5nJyk7CiAgICAgICAgICAgICAgICBDb3JlLkFKQVguQ29udGVudFVwZGF0ZSgkKCcjRGFzaGJvYXJkJyArIENvcmUuQXBwLkVzY2FwZVNlbGVjdG9yKERhc2hsZXREYXRhLk5hbWUpKSwgQ29yZS5Db25maWcuR2V0KCdCYXNlbGluaycpICsgJ0FjdGlvbj0nICsgQ29yZS5Db25maWcuR2V0KCdBY3Rpb24nKSArICc7U3ViYWN0aW9uPUVsZW1lbnQ7TmFtZT0nICsgRGFzaGxldERhdGEuTmFtZSArICc7Q3VzdG9tZXJJRD0nICsgZW5jb2RlVVJJQ29tcG9uZW50KEN1c3RvbWVySUQpICsgJztDdXN0b21lclVzZXJJRD0nICsgZW5jb2RlVVJJQ29tcG9uZW50KEN1c3RvbWVyVXNlcklEKSArICc7U29ydEJ5PScgKyBTb3J0QnkgKyAnO09yZGVyQnk9JyArIE9yZGVyQnksIGZ1bmN0aW9uICgpIHsKICAgICAgICAgICAgICAgICAgICAkKCcjRGFzaGJvYXJkJyArIENvcmUuQXBwLkVzY2FwZVNlbGVjdG9yKERhc2hsZXREYXRhLk5hbWUpICsgJy1ib3gnKS5yZW1vdmVDbGFzcygnTG9hZGluZycpOwogICAgICAgICAgICAgICAgfSk7CiAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7CiAgICAgICAgICAgIH0pOwogICAgfQoKICAgIENvcmUuSW5pdC5SZWdpc3Rlck5hbWVzcGFjZShUYXJnZXROUywgJ0FQUF9NT0RVTEUnKTsKCiAgICByZXR1cm4gVGFyZ2V0TlM7Cn0oSVRTTS5BZ2VudC5Db25maWdJdGVtLkRhc2hib2FyZCB8fCB7fSkpOwo=
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMjEgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKLy8gZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQuCi8vIC0tCgoidXNlIHN0cmljdCI7Cgp2YXIgSVRTTSA9IElUU00gfHwge307CklUU00uQWdlbnQgPSBJVFNNLkFnZW50IHx8IHt9OwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gPSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gfHwge307CgovKioKICogQG5hbWVzcGFjZSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uRWRpdAogKiBAbWVtYmVyb2YgSVRTTS5BZ2VudAogKiBAYXV0aG9yIE9UUlMgQUcKICogQGRlc2NyaXB0aW9uCiAqICAgICAgVGhpcyBuYW1lc3BhY2UgY29udGFpbnMgdGhlIHNwZWNpYWwgbW9kdWxlIGZ1bmN0aW9ucyBmb3IgdGhlIGNvbmZpZyBpdGVtIGVkaXQuCiAqLwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uRWRpdCA9IChmdW5jdGlvbiAoVGFyZ2V0TlMpIHsKCiAgICAvKgogICAgKiBAbmFtZSBJbml0CiAgICAqIEBtZW1iZXJvZiBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uRWRpdAogICAgKiBAZnVuY3Rpb24KICAgICogQGRlc2NyaXB0aW9uCiAgICAqICAgICAgVGhpcyBmdW5jdGlvbiBpbml0aWFsaXplcyB0aGUgZm9ybSBzdWJtaXR0aW9uLgogICAgKi8KICAgIFRhcmdldE5TLkluaXQgPSBmdW5jdGlvbiAoKSB7CiAgICAgICAgJCgnI1N1Ym1pdEJ1dHRvbicpLm9uKCdjbGljaycsIGZ1bmN0aW9uICgpIHsKICAgICAgICAgICAgJCgnaW5wdXRbbmFtZT1TdWJtaXRTYXZlXScpLnZhbCgnMScpOwogICAgICAgIH0pOwoKICAgICAgICAkKCcjQ2FuY2VsQnV0dG9uJykub24oJ2NsaWNrJywgZnVuY3Rpb24gKCkgewogICAgICAgICAgICBpZiAoQ29yZS5VSS5Qb3B1cC5DdXJyZW50SXNQb3B1cFdpbmRvdygpKSB7CiAgICAgICAgICAgICAgICBDb3JlLlVJLlBvcHVwLkNsb3NlUG9wdXAoKTsKICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKICAgICAgICAgICAgfQogICAgICAgIH0pOwoKICAgICAgICAkKCcuRGlzYWJsZVZhbGlkYXRpb24nKS5vbignY2xpY2snLCBmdW5jdGlvbiAoKSB7CiAgICAgICAgICAgICQoJ2lucHV0W25hbWU9U3VibWl0U2F2ZV0nKS52YWwoJzAnKTsKICAgICAgICAgICAgQ29yZS5Gb3JtLlZhbGlkYXRlLkRpc2FibGVWYWxpZGF0aW9uKCQoJyNDbGFzc0l0ZW0nKSk7CiAgICAgICAgICAgIC8vIGZpeCBmb3IgU2FmYXJpOiByYWNlIGNvbmRpdGlvbiB3aXRoIHN1Ym1pdCBpbiBDb3JlLkZvcm0uVmFsaWRhdGUuanMKICAgICAgICAgICAgLy8gZml4ZWQgaW4gT1RSUyAzLjAuNSwgdGhpcyBsaW5lIHJlbWFpbnMgaGVyZSBmb3IgYmFja3dhcmRzIGNvbXBhdGliaWxpdHkgcmVhc29ucwogICAgICAgICAgICAkKCcjQ2xhc3NJdGVtJykucmVtb3ZlQ2xhc3MoJ1ByZXZlbnRNdWx0aXBsZVN1Ym1pdHMnKTsKICAgICAgICB9KTsKCiAgICAgICAgLy8gUmVnaXN0ZXIgcmV0dXJuIGtleS4gQnV0IG5vdCBpbiB0ZXh0YXJlYXMuCiAgICAgICAgJCgnI0NsYXNzSXRlbScpLm9mZigna2V5cHJlc3MuRmlsdGVySW5wdXQnKS5vbigna2V5cHJlc3MuRmlsdGVySW5wdXQnLCBmdW5jdGlvbiAoRXZlbnQpIHsKICAgICAgICAgICAgaWYgKChFdmVudC5jaGFyQ29kZSB8fCBFdmVudC5rZXlDb2RlKSA9PT0gMTMgJiYgKCQoRXZlbnQudGFyZ2V0KS5wcm9wKCd0YWdOYW1lJykgIT09ICdURVhUQVJFQScpKSB7CiAgICAgICAgICAgICAgICAkKCcjU3VibWl0QnV0dG9uJykuY2xpY2soKTsKICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKICAgICAgICAgICAgfQogICAgICAgIH0pOwogICAgfTsKCiAgICBDb3JlLkluaXQuUmVnaXN0ZXJOYW1lc3BhY2UoVGFyZ2V0TlMsICdBUFBfTU9EVUxFJyk7CgogICAgcmV0dXJuIFRhcmdldE5TOwp9KElUU00uQWdlbnQuQ29uZmlnSXRlbS5FZGl0IHx8IHt9KSk7Cg==
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMjEgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKLy8gZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQuCi8vIC0tCgovKmVzbGludC1kaXNhYmxlIG5vLXdpbmRvdyovCgoidXNlIHN0cmljdCI7Cgp2YXIgSVRTTSA9IElUU00gfHwge307CklUU00uQWdlbnQgPSBJVFNNLkFnZW50IHx8IHt9OwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gPSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gfHwge307CgovKioKICogQG5hbWVzcGFjZSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uSGlzdG9yeQogKiBAbWVtYmVyb2YgSVRTTS5BZ2VudC5Db25maWdJdGVtCiAqIEBhdXRob3IgT1RSUyBBRwogKiBAZGVzY3JpcHRpb24KICogICAgICBUaGlzIG5hbWVzcGFjZSBjb250YWlucyB0aGUgc3BlY2lhbCBtb2R1bGUgZnVuY3Rpb25zIGZvciB0aGUgY29uZmlnIGl0ZW0gaGlzdG9yeS4KICovCklUU00uQWdlbnQuQ29uZmlnSXRlbS5IaXN0b3J5ID0gKGZ1bmN0aW9uIChUYXJnZXROUykgewoKICAgIC8qCiAgICAqIEBuYW1lIEluaXQKICAgICogQG1lbWJlcm9mIElUU00uQWdlbnQuQ29uZmlnSXRlbS5IaXN0b3J5CiAgICAqIEBmdW5jdGlvbgogICAgKiBAZGVzY3JpcHRpb24KICAgICogICAgICBUaGlzIGZ1bmN0aW9uIGluaXRpYWxpemVzIHRoZSBwb3B1cC4KICAgICovCiAgICBUYXJnZXROUy5Jbml0ID0gZnVuY3Rpb24gKCkgewogICAgICAgICQoJ2EuTGlua1pvb21WaWV3Jykub24oJ2NsaWNrJywgZnVuY3Rpb24gKCkgewogICAgICAgICAgICB3aW5kb3cub3BlbmVyLkNvcmUuVUkuUG9wdXAuRmlyZVBvcHVwRXZlbnQoJ1VSTCcsIHsgVVJMOiAkKHRoaXMpLmF0dHIoJ2hyZWYnKX0pOwogICAgICAgICAgICBDb3JlLlVJLlBvcHVwLkNsb3NlUG9wdXAoKTsKICAgICAgICB9KTsKICAgIH07CgogICAgQ29yZS5Jbml0LlJlZ2lzdGVyTmFtZXNwYWNlKFRhcmdldE5TLCAnQVBQX01PRFVMRScpOwoKICAgIHJldHVybiBUYXJnZXROUzsKfShJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uSGlzdG9yeSB8fCB7fSkpOwo=
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMjEgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKLy8gZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQuCi8vIC0tCgoidXNlIHN0cmljdCI7Cgp2YXIgSVRTTSA9IElUU00gfHwge307CklUU00uQWdlbnQgPSBJVFNNLkFnZW50IHx8IHt9OwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gPSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gfHwge307CgovKioKICogQG5hbWVzcGFjZSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uT3ZlcnZpZXcKICogQG1lbWJlcm9mIElUU00uQWdlbnQuQ29uZmlnSXRlbQogKiBAYXV0aG9yIE9UUlMgQUcKICogQGRlc2NyaXB0aW9uCiAqICAgICAgVGhpcyBuYW1lc3BhY2UgY29udGFpbnMgdGhlIHNwZWNpYWwgbW9kdWxlIGZ1bmN0aW9ucyBmb3IgdGhlIGNvbmZpZyBpdGVtIG92ZXJ2aWV3IG5hdmJhci4KICovCklUU00uQWdlbnQuQ29uZmlnSXRlbS5PdmVydmlldyA9IChmdW5jdGlvbiAoVGFyZ2V0TlMpIHsKCiAgICAvKgogICAgKiBAbmFtZSBJbml0CiAgICAqIEBtZW1iZXJvZiBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uT3ZlcnZpZXcKICAgICogQGZ1bmN0aW9uCiAgICAqIEBkZXNjcmlwdGlvbgogICAgKiAgICAgIFRoaXMgZnVuY3Rpb24gaW5pdGlhbGl6ZXMgdGhlIG92ZXJ2aWV3IGJlaGF2aW91cnMuCiAgICAqLwogICAgVGFyZ2V0TlMuSW5pdCA9IGZ1bmN0aW9uICgpIHsKICAgICAgICB2YXIgSVRTTUNvbmZpZ0l0ZW1TZWFyY2ggICAgPSBDb3JlLkNvbmZpZy5HZXQoJ0lUU01Db25maWdJdGVtU2VhcmNoJyk7CiAgICAgICAgdmFyIElUU01Db25maWdJdGVtQWN0aW9uUm93ID0gQ29yZS5Db25maWcuR2V0KCdJVFNNQ29uZmlnSXRlbUFjdGlvblJvdycpIHx8IHt9OwoKICAgICAgICAkKCcjU2hvd0NvbnRleHRTZXR0aW5nc0RpYWxvZycpLm9uKCdjbGljaycsIGZ1bmN0aW9uIChFdmVudCkgewogICAgICAgICAgICBDb3JlLlVJLkRpYWxvZy5TaG93Q29udGVudERpYWxvZygkKCcjQ29udGV4dFNldHRpbmdzRGlhbG9nQ29udGFpbmVyJyksIENvcmUuTGFuZ3VhZ2UuVHJhbnNsYXRlKCJTZXR0aW5ncyIpLCAnMjAlJywgJ0NlbnRlcicsIHRydWUsICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbCiAgICAgICAgICAgICAgICAgICAgeyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBMYWJlbDogQ29yZS5MYW5ndWFnZS5UcmFuc2xhdGUoIlN1Ym1pdCIpLAogICAgICAgICAgICAgICAgICAgICAgICBUeXBlOiAnU3VibWl0JywgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQ2xhc3M6ICdQcmltYXJ5JwogICAgICAgICAgICAgICAgICAgIH0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIF0KICAgICAgICAgICAgKTsgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEV2ZW50LnByZXZlbnREZWZhdWx0KCk7CiAgICAgICAgICAgIEV2ZW50LnN0b3BQcm9wYWdhdGlvbigpOyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7CiAgICAgICAgfSk7CgogICAgICAgIGlmIChJVFNNQ29uZmlnSXRlbVNlYXJjaCkgewogICAgICAgICAgICAkKCcjSVRTTUNvbmZpZ0l0ZW1TZWFyY2gnKS5vbignY2xpY2snLCBmdW5jdGlvbiAoKSB7CiAgICAgICAgICAgICAgICBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uU2VhcmNoLk9wZW5TZWFyY2hEaWFsb2coCiAgICAgICAgICAgICAgICAgICAgJ0FnZW50SVRTTUNvbmZpZ0l0ZW1TZWFyY2gnLAogICAgICAgICAgICAgICAgICAgIENvcmUuQXBwLkVzY2FwZVNlbGVjdG9yKElUU01Db25maWdJdGVtU2VhcmNoLlByb2ZpbGUpLAogICAgICAgICAgICAgICAgICAgIENvcmUuQXBwLkVzY2FwZVNlbGVjdG9yKElUU01Db25maWdJdGVtU2VhcmNoLkNsYXNzSUQpCiAgICAgICAgICAgICAgICApOwogICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwogICAgICAgICAgICB9KTsKICAgICAgICB9CgogICAgICAgICQuZWFjaChJVFNNQ29uZmlnSXRlbUFjdGlvblJvdywgZnVuY3Rpb24oS2V5LCBEYXRhKSB7CiAgICAgICAgICAgIElUU00uVUkuQ29uZmlnSXRlbUFjdGlvblJvdy5BZGRBY3Rpb25zKCQoJyNDb25maWdJdGVtSURfJyArIEtleSksIERhdGEpOwogICAgICAgIH0pOwoKICAgICAgICBJVFNNLlVJLkNvbmZpZ0l0ZW1BY3Rpb25Sb3cuSW5pdCgpOwoKICAgICAgICBDb3JlLlVJLkluaXRDaGVja2JveFNlbGVjdGlvbigkKCd0YWJsZSB0ZC5DaGVja2JveCcpKTsKCiAgICAgICAgJCgnLk1hc3RlckFjdGlvbicpLm9uKCdjbGljaycsIGZ1bmN0aW9uIChFdmVudCkgewogICAgICAgICAgICB2YXIgJE1hc3RlckFjdGlvbkxpbmsgPSAkKHRoaXMpLmZpbmQoJy5NYXN0ZXJBY3Rpb25MaW5rJyk7CiAgICAgICAgICAgIC8vIG9ubHkgYWN0IGlmIHRoZSBsaW5rIHdhcyBub3QgY2xpY2tlZCBkaXJlY3RseQogICAgICAgICAgICBpZiAoRXZlbnQudGFyZ2V0ICE9PSAkTWFzdGVyQWN0aW9uTGluay5nZXQoMCkpIHsKICAgICAgICAgICAgICAgIHdpbmRvdy5sb2NhdGlvbiA9ICRNYXN0ZXJBY3Rpb25MaW5rLmF0dHIoJ2hyZWYnKTsKICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKICAgICAgICAgICAgfQogICAgICAgIH0pOwogICAgfTsKCiAgICBDb3JlLkluaXQuUmVnaXN0ZXJOYW1lc3BhY2UoVGFyZ2V0TlMsICdBUFBfTU9EVUxFJyk7CgogICAgcmV0dXJuIFRhcmdldE5TOwp9KElUU00uQWdlbnQuQ29uZmlnSXRlbS5PdmVydmlldyB8fCB7fSkpOwo=
// --
// Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
// Copyright (C) 2021 Znuny GmbH, https://znuny.org/
// --
// This software comes with ABSOLUTELY NO WARRANTY. For details, see
// the enclosed file COPYING for license information (GPL). If you
// did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
// --


// TODO:
//Remove this line and fix JSDoc
// nofilter(TidyAll::Plugin::Znuny::JavaScript::ESLint)


"use strict";

var ITSM = ITSM || {};
ITSM.Agent = ITSM.Agent || {};
ITSM.Agent.ConfigItem = ITSM.Agent.ConfigItem || {};

/**
 * @namespace
 * @exports TargetNS as ITSM.Agent.ConfigItem.Search
 * @description
 *      This namespace contains the special module functions for the search.
 */
ITSM.Agent.ConfigItem.Search = (function (TargetNS) {

    /**
     * @function
     * @return nothing
     *      This function initializes the search screen.
     */
    TargetNS.Init = function () {
        var ITSMConfigItemOpenSearchDialog = Core.Config.Get('ITSMConfigItemOpenSearchDialog');
        if (ITSMConfigItemOpenSearchDialog) {
            ITSM.Agent.ConfigItem.Search.OpenSearchDialog(
                ITSMConfigItemOpenSearchDialog.Action,
                Core.App.EscapeSelector(ITSMConfigItemOpenSearchDialog.Profile),
                Core.App.EscapeSelector(ITSMConfigItemOpenSearchDialog.ClassID)
            );
        }
    };

    /**
     * @function
     * @return nothing
     *      This function rebuild attribute selection, only show available attributes.
     */
    TargetNS.AdditionalAttributeSelectionRebuild = function () {

        // get original selection with all possible fields and clone it
        var $AttributeClone = $('#AttributeOrig option').clone(),
            $AttributeSelection = $('#Attribute').empty(),
            Value;

        // strip all already used attributes
        $AttributeClone.each(function () {
            Value = Core.App.EscapeSelector($(this).attr('value'));
            if (!$('#SearchInsert label#Label' + Value).length) {
                $AttributeSelection.append($(this));
            }
        });

        $AttributeSelection.trigger('redraw.InputField');

        return true;
    };

    /**
     * @function
     * @param {String} of attribute to add.
     * @return nothing
     *      This function adds one attribute for search
     */
    TargetNS.SearchAttributeAdd = function (Attribute) {

        // escape possible colons (:) in element id because jQuery can not handle it in id attribute selectors
        Attribute = Core.App.EscapeSelector(Attribute);

        var $Label = $('#SearchAttributesHidden label#Label' + Attribute);

        if ($Label.length) {
            $Label.prev().clone().appendTo('#SearchInsert');
            $Label.clone().appendTo('#SearchInsert');
            $Label.next().clone().appendTo('#SearchInsert')

                // bind click function to remove button now
                .find('.RemoveButton').on('click', function () {
                    var $Element = $(this).parent();
                    TargetNS.SearchAttributeRemove($Element);

                    // rebuild selection
                    TargetNS.AdditionalAttributeSelectionRebuild();

                    return false;
                });

                // set autocomplete to customer type fields
                $('#SearchInsert').find('.ITSMCustomerSearch').each(function() {
                    var InputID = $(this).attr('id') + 'Autocomplete';
                    $(this).removeClass('ITSMCustomerSearch');
                    $(this).attr('id', InputID);
                    $(this).prev().attr('id', InputID + 'Selected');

                    // escape possible colons (:) in element id because jQuery can not handle it in id attribute selectors
                    ITSM.Agent.CustomerSearch.Init($('#' + Core.App.EscapeSelector(InputID)));

                    // prevent dialog closure when select a customer from the list
                    $('ul.ui-autocomplete').on('click', function(Event) {
                        Event.stopPropagation();
                        return false;
                    });
                });

                // set autocomplete to user type fields
                $('#SearchInsert').find('.ITSMUserSearch').each(function() {
                    var InputID = $(this).attr('id') + 'Autocomplete';
                    $(this).removeClass('ITSMUserSearch');
                    $(this).attr('id', InputID);
                    $(this).prev().attr('id', InputID + 'Selected');

                    // escape possible colons (:) in element id because jQuery can not handle it in id attribute selectors
                    ITSM.Agent.UserSearch.Init( $('#' + Core.App.EscapeSelector(InputID) ) );

                    // prevent dialog closure when select a customer from the list
                    $('ul.ui-autocomplete').bind('click', function(Event) {
                        Event.stopPropagation();
                        return false;
                    });
                });
            }

        Core.UI.InputFields.Activate($('#SearchInsert'));

        return false;
    };

    /**
     * @function
     * @param {jQueryObject} $Element The jQuery object of the form  or any element
     *      within this form check.
     * @return nothing
     *      This function remove attributes from an element.
     */

    TargetNS.SearchAttributeRemove = function ($Element) {
        $Element.prev().prev().remove();
        $Element.prev().remove();
        $Element.remove();
    };

    /**
     * @function
     * @private
     * @param {String} Profile The profile name that will be delete.
     * @return nothing
     * @description Delete a profile via an ajax requests.
     */
    function SearchProfileDelete(Profile) {
        var Data = {
            Action: 'AgentITSMConfigItemSearch',
            Subaction: 'AJAXProfileDelete',
            Profile: Profile,
            ClassID: $('#SearchClassID').val()
        };
        Core.AJAX.FunctionCall(
            Core.Config.Get('CGIHandle'),
            Data,
            function () {}
        );
    }

    /**
     * @function
     * @private
     * @return nothing
     * @description Shows waiting dialog until search screen is ready.
     */
    function ShowWaitingDialog() {
        var Content = Core.Template.Render(
            'Agent/ITSMCore/LoadingDialog',
            {
                SpanTitle: Core.Config.Get('LoadingMsg')
            }
        );

        Core.UI.Dialog.ShowContentDialog(
            Content,
            Core.Config.Get('LoadingMsg'),
            '10px',
            'Center',
            true
        );
    }

    /**
     * @function
     * @param none
     * @return nothing
     *      This function sets all search dialog settings
     */

    TargetNS.SetSearchDialog = function() {

        // hide add template block
        $('#SearchProfileAddBlock').hide();

        // hide save changes in template block
        $('#SaveProfile').parent().hide().prev().hide().prev().hide();

        // search profile is selected
        if ($('#SearchProfile').val() && $('#SearchProfile').val() !== 'last-search') {

            // show delete button
            $('#SearchProfileDelete').show();

            // show profile link
            $('#SearchProfileAsLink').show();

            // show save changes in template block
            $('#SaveProfile').parent().show().prev().show().prev().show();

            // set SaveProfile to 0
            $('#SaveProfile').prop('checked', false);
        }

        Core.UI.InputFields.Activate($('.Dialog:visible'));

        // register add of attribute
        $('#Attribute').on('change', function () {
            var Attribute = $('#Attribute').val();
            TargetNS.SearchAttributeAdd(Attribute);
            TargetNS.AdditionalAttributeSelectionRebuild();

            // Register event for tree selection dialog
            $('.ShowTreeSelection').off('click').on('click', function () {
                Core.UI.TreeSelection.ShowTreeSelection($(this));
                return false;
            });

            return false;
        });

        // register return key
        $('#SearchForm').off('keypress.FilterInput').on('keypress.FilterInput', function (Event) {
            if ((Event.charCode || Event.keyCode) === 13) {
                if (!CheckForSearchedValues()) {
                    return false;
                }
                else {
                    $('#SearchFormSubmit').trigger('click');
                }
                return false;
            }
        });

        // register submit
        $('#SearchFormSubmit').off('click.DoSearch').on('click.DoSearch', function () {

            var ShownAttributes = [];

            if ($('#SearchProfileAddAction, #SearchProfileAddName').is(':visible') && $('#SearchProfileAddName').val()) {
                $('#SearchProfileAddAction').trigger('click');
            }

            // remember shown attributes
            $('#SearchInsert label').each(function () {
                if ($(this).attr('id')) {
                    ShownAttributes.push($(this).attr('id'));
                }
            });
            $('#SearchForm #ShownAttributes').val(ShownAttributes.join(';'));

            // Normal results mode will return HTML in the same window
            if ($('#SearchForm #ResultForm').val() === 'Normal') {

                if (!CheckForSearchedValues()) {
                    return false;
                }
                else {
                    CheckSearchStringsForStopWords(function () {
                        $('#SearchForm').submit();
                        return false;
                   });
                }
            }
            else { // Print and CSV should open in a new window, no waiting dialog
                $('#SearchForm').attr('target', 'SearchResultPage');
                if (!CheckForSearchedValues()) {
                    return false;
                }
                else {
                    CheckSearchStringsForStopWords(function () {
                        $('#SearchForm').submit();
                        $('#SearchForm').attr('target', '');
                    });
                }
            }
            return false;
        });

        Core.Form.Validate.Init();
        Core.Form.Validate.SetSubmitFunction($('#SearchForm'), function (Form) {
            Form.submit();

            // Show only a waiting dialog for Normal results mode, because this result
            //  will return the HTML in the same window.
            if ($('#SearchForm #ResultForm').val() === 'Normal') {
                ShowWaitingDialog();
            }
        });

        // load profile
        $('#SearchProfile').bind('change', function () {
            var SearchProfile = $('#SearchProfile').val(),
                SearchProfileClassID = $('#SearchClassID').val(),
                SearchProfileAction = $('#SearchAction').val();

            TargetNS.OpenSearchDialog(SearchProfileAction, SearchProfile, SearchProfileClassID);
            return false;
        });

        // show add profile block or not
        $('#SearchProfileNew').on('click', function (Event) {
            $('#SearchProfileAddBlock').toggle();
            $('#SearchProfileAddName').focus();
            Event.preventDefault();
            return false;
        });

        // add new profile
        $('#SearchProfileAddAction').bind('click', function () {
            var ProfileName, $Element1;

            // get name
            ProfileName = $('#SearchProfileAddName').val();

            // check name
            if (!ProfileName.length || ProfileName.length < 2) {
                return;
            }

            // add name to profile selection
            $Element1 = $('#SearchProfile').children().first().clone();
            $Element1.text(ProfileName);
            $Element1.attr('value', ProfileName);
            $Element1.prop('selected', true);
            $('#SearchProfile').append($Element1).trigger('redraw.InputField');

            // set input box to empty
            $('#SearchProfileAddName').val('');

            // hide add template block
            $('#SearchProfileAddBlock').hide();

            // hide save changes in template block
            $('#SaveProfile').parent().hide().prev().hide().prev().hide();

            // set SaveProfile to 1
            $('#SaveProfile').prop('checked', true);

            // show delete button
            $('#SearchProfileDelete').show();

            // show profile link
            $('#SearchProfileAsLink').show();
        });

        // direct link to profile
        $('#SearchProfileAsLink').bind('click', function () {
            var SearchProfile = $('#SearchProfile').val(),
                SearchProfileAction = $('#SearchAction').val(),
                ClassID = $('#SearchClassID').val();

            window.location.href = Core.Config.Get('Baselink') + 'Action=' + SearchProfileAction +
            ';Subaction=Search;TakeLastSearch=1;SaveProfile=1;Profile=' + encodeURIComponent(SearchProfile)
            + ';ClassID=' + ClassID;
            return false;
        });

        // delete profile
        $('#SearchProfileDelete').on('click', function (Event) {

            // strip all already used attributes
            $('#SearchProfile').find('option:selected').each(function () {
                if ($(this).attr('value') !== 'last-search') {

                    // rebuild attributes
                    $('#SearchInsert').text('');

                    // remove remote
                    SearchProfileDelete($(this).val());

                    // remove local
                    $(this).remove();

                    // show fulltext
                    TargetNS.SearchAttributeAdd('Fulltext');

                    // rebuild selection
                    TargetNS.AdditionalAttributeSelectionRebuild();
                }
            });
            $('#SearchProfile').trigger('change');

            if ($('#SearchProfile').val() && $('#SearchProfile').val() === 'last-search') {

                // hide delete link
                $('#SearchProfileDelete').hide();

                // show profile link
                $('#SearchProfileAsLink').hide();

            }

            Event.preventDefault();
            return false;
        });

    };

    /**
     * @function
     * @param {Profile} The profile that is set to the search dialog
     * @return nothing
     *      This function refresh the search dialog with the selected profile
     */

    TargetNS.LoadProfile = function (Profile) {
        var BaseLink = Core.Config.Get('Baselink'),
            Action = 'Action=AgentITSMConfigItemSearch;',
            SubAction = 'Subaction=AJAXUpdate;',
            ClassID = 'ClassID=' + $('#SearchClassID').val() + ';',
            SearchProfile = 'Profile=' + Profile,
            URL =  BaseLink + Action + SubAction + ClassID + SearchProfile;

        $('#DivClassID').addClass('ui-autocomplete-loading');
        Core.AJAX.ContentUpdate($('#AJAXUpdate'), URL, function() {
            var ITSMSearchProfileAttributes = Core.Config.Get('ITSMSearchProfileAttributes') || [];
            $.each(ITSMSearchProfileAttributes, function(Idx, Attribute) {
                ITSM.Agent.ConfigItem.Search.SearchAttributeAdd(Core.App.EscapeSelector(Attribute));
                ITSM.Agent.ConfigItem.Search.AdditionalAttributeSelectionRebuild();
            });

            TargetNS.SetSearchDialog('$Env{"Action"}');
            $('#ITSMSearchProfile').removeClass('Hidden');
            $('#ITSMSearchFields').removeClass('Hidden');
            $('.Dialog:visible #SearchFormSubmit').appendTo($('.Dialog:visible > .Content > .ContentFooter'));
            $('#SearchFormSubmit').removeClass('Hidden');
            $('#DivClassID').removeClass('ui-autocomplete-loading');
            Core.UI.InputFields.Activate($('#SearchForm'));
        });
    };

    /**
     * @private
     * @name CheckForSearchedValues
     * @memberof Core.Agent.Search
     * @function
     * @returns {Boolean} False if no values were found, true if values where there.
     * @description
     *      Checks if any values were entered in the search.
     *      If nothing at all exists, it alerts with translated:
     *      "Please enter at least one search value or * to find anything"
     */
    function CheckForSearchedValues() {
        // loop through the SerachForm labels
        var SearchValueFlag = false;
        $('#SearchForm label').each(function () {
            var ElementName,
                $Element,
                $LabelElement = $(this),
                $FieldElement = $LabelElement.next('.Field');
            // those with ID's are used for searching
            if ($(this).attr('id')) {

                // substring "Label" (e.g. first five characters ) from the
                // label id, use the remaining name as name string for accessing
                // the form input's value
                ElementName = $(this).attr('id').substring(5);
                $Element = $('#SearchForm input[name=' + Core.App.EscapeSelector(ElementName) + ']');

                // If there's no input element with the selected name
                // find the next "select" element and use that one for checking
                if (!$Element.length) {
                    $Element = $(this).next().find('select');
                }

                // Fix for bug#10845: make sure time slot fields with TimeInputFormat
                // 'Input' set are being considered correctly, too. As this is only
                // relevant for search type 'TimeSlot', we check for the first
                // input type=text elment in the corresponding field element.
                // All time field elements have to be filled in, but if only one
                // is missing, we treat the whole field as invalid.
                if ($FieldElement.find('input[name$="SearchType"]').val() === 'TimeSlot' && !$FieldElement.find('select').length) {
                    $Element = $FieldElement.find('input[type=text]').first();
                }

                if ($Element.length) {
                    if ($Element.val() && $Element.val() !== '') {
                        SearchValueFlag = true;
                    }
                }
            }
        });
        if (!SearchValueFlag) {
           alert(Core.Language.Translate('Please enter at least one search value or * to find anything.'));
        }
        return SearchValueFlag;
    }

        /**
     * @private
     * @name CheckSearchStringsForStopWords
     * @memberof Core.Agent.Search
     * @function
     * @param {Function} Callback - function to execute, if no stop words were found.
     * @description Checks if specific values of the search form contain stop words.
     *              If stop words are present, a warning will be displayed.
     *              If stop words are not present, the given callback will be executed.
     */
    function CheckSearchStringsForStopWords(Callback) {
        var SearchStrings = {},
            SearchStringsFound = 0,
            RelevantElementNames = {
                'From': 1,
                'To': 1,
                'Cc': 1,
                'Subject': 1,
                'Body': 1,
                'Fulltext': 1
            };

        if (!Core.Config.Get('CheckSearchStringsForStopWords')) {
            Callback();
            return;
        }

        $('#SearchForm label').each(function () {
            var ElementName,
                $Element;

            // those with ID's are used for searching
            if ($(this).attr('id')) {

                // substring "Label" (e.g. first five characters ) from the
                // label id, use the remaining name as name string for accessing
                // the form input's value
                ElementName = $(this).attr('id').substring(5);
                if (!RelevantElementNames[ElementName]) {
                    return;
                }

                $Element = $('#SearchForm input[name=' + ElementName + ']');

                if ($Element.length) {
                    if ($Element.val() && $Element.val() !== '') {
                        SearchStrings[ElementName] = $Element.val();
                        SearchStringsFound = 1;
                    }
                }
            }
        });

        // Check if stop words are present.
        if (!SearchStringsFound) {
            Callback();
            return;
        }

        AJAXStopWordCheck(
            SearchStrings,
            function (FoundStopWords) {
                alert(Core.Config.Get('SearchStringsContainStopWordsMsg') + "\n" + FoundStopWords);
            },
            Callback
        );
    }

    /**
     * @function
     * @param {Event} Action
     * @return nothing
     *      This function open the search dialog
     */

    TargetNS.OpenSearchDialog = function (Action, Profile, Class) {

        var Referrer = Core.Config.Get('Action'),
            Data;

        if (!Action) {
            Action ='AgentITSMConfigItemSearch';
        }
        Data = {
            Action: Action,
            Referrer: Referrer,
            Profile: Profile,
            Subaction: 'AJAX',
            ClassID: Class
        };

        ShowWaitingDialog();
        Core.AJAX.FunctionCall(
            Core.Config.Get('CGIHandle'),
            Data,
            function (HTML) {

                // if the waiting dialog was canceled, do not show the search
                //  dialog as well
                if (!$('.Dialog:visible').length) {
                    return;
                }

                Core.UI.Dialog.ShowContentDialog(HTML, Core.Config.Get('SearchMsg'), '10px', 'Center', true, [], true);

                $('#SearchClassID').off('change.SearchProfile').on('change.SearchProfile', function() {
                    if ($('#SearchClassID').val() !== "") {
                        ITSM.Agent.ConfigItem.Search.LoadProfile($('#SearchProfile').val());
                    }
                    else {
                        $('#SearchProfile').attr("id", "SearchProfileOld");
                    }
                });

                var ClassSelected = $('#SearchClassID').val();
                if (ClassSelected) {
                    ITSM.Agent.ConfigItem.Search.LoadProfile($('#SearchProfile').val());
                }

                TargetNS.SetSearchDialog();
            }, 'html'
        );
    };

    Core.Init.RegisterNamespace(TargetNS, 'APP_MODULE');

    return TargetNS;
}(ITSM.Agent.ConfigItem.Search || {}));

// --
// Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
// Copyright (C) 2021 Znuny GmbH, https://znuny.org/
// --
// This software comes with ABSOLUTELY NO WARRANTY. For details, see
// the enclosed file COPYING for license information (GPL). If you
// did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
// --


// TODO:
//Remove this line and fix JSDoc
// nofilter(TidyAll::Plugin::Znuny::JavaScript::ESLint)


"use strict";

var ITSM = ITSM || {};
ITSM.Agent = ITSM.Agent || {};
ITSM.Agent.ConfigItem = ITSM.Agent.ConfigItem || {};

/**
 * @namespace
 * @exports TargetNS as ITSM.Agent.ConfigItem.Search
 * @description
 *      This namespace contains the special module functions for the search.
 */
ITSM.Agent.ConfigItem.Search = (function (TargetNS) {

    /**
     * @function
     * @return nothing
     *      This function initializes the search screen.
     */
    TargetNS.Init = function () {
        var ITSMConfigItemOpenSearchDialog = Core.Config.Get('ITSMConfigItemOpenSearchDialog');
        if (ITSMConfigItemOpenSearchDialog) {
            ITSM.Agent.ConfigItem.Search.OpenSearchDialog(
                ITSMConfigItemOpenSearchDialog.Action,
                Core.App.EscapeSelector(ITSMConfigItemOpenSearchDialog.Profile),
                Core.App.EscapeSelector(ITSMConfigItemOpenSearchDialog.ClassID)
            );
        }
    };

    /**
     * @function
     * @return nothing
     *      This function rebuild attribute selection, only show available attributes.
     */
    TargetNS.AdditionalAttributeSelectionRebuild = function () {

        // get original selection with all possible fields and clone it
        var $AttributeClone = $('#AttributeOrig option').clone(),
            $AttributeSelection = $('#Attribute').empty(),
            Value;

        // strip all already used attributes
        $AttributeClone.each(function () {
            Value = Core.App.EscapeSelector($(this).attr('value'));
            if (!$('#SearchInsert label#Label' + Value).length) {
                $AttributeSelection.append($(this));
            }
        });

        $AttributeSelection.trigger('redraw.InputField');

        return true;
    };

    /**
     * @function
     * @param {String} of attribute to add.
     * @return nothing
     *      This function adds one attribute for search
     */
    TargetNS.SearchAttributeAdd = function (Attribute) {

        // escape possible colons (:) in element id because jQuery can not handle it in id attribute selectors
        Attribute = Core.App.EscapeSelector(Attribute);

        var $Label = $('#SearchAttributesHidden label#Label' + Attribute);

        if ($Label.length) {
            $Label.prev().clone().appendTo('#SearchInsert');
            $Label.clone().appendTo('#SearchInsert');
            $Label.next().clone().appendTo('#SearchInsert')

                // bind click function to remove button now
                .find('.RemoveButton').on('click', function () {
                    var $Element = $(this).parent();
                    TargetNS.SearchAttributeRemove($Element);

                    // rebuild selection
                    TargetNS.AdditionalAttributeSelectionRebuild();

                    return false;
                });

                // set autocomplete to customer type fields
                $('#SearchInsert').find('.ITSMCustomerSearch').each(function() {
                    var InputID = $(this).attr('id') + 'Autocomplete';
                    $(this).removeClass('ITSMCustomerSearch');
                    $(this).attr('id', InputID);
                    $(this).prev().attr('id', InputID + 'Selected');

                    // escape possible colons (:) in element id because jQuery can not handle it in id attribute selectors
                    ITSM.Agent.CustomerSearch.Init($('#' + Core.App.EscapeSelector(InputID)));

                    // prevent dialog closure when select a customer from the list
                    $('ul.ui-autocomplete').on('click', function(Event) {
                        Event.stopPropagation();
                        return false;
                    });
                });

                // set autocomplete to user type fields
                $('#SearchInsert').find('.ITSMUserSearch').each(function() {
                    var InputID = $(this).attr('id') + 'Autocomplete';
                    $(this).removeClass('ITSMUserSearch');
                    $(this).attr('id', InputID);
                    $(this).prev().attr('id', InputID + 'Selected');

                    // escape possible colons (:) in element id because jQuery can not handle it in id attribute selectors
                    ITSM.Agent.UserSearch.Init( $('#' + Core.App.EscapeSelector(InputID) ) );

                    // prevent dialog closure when select a customer from the list
                    $('ul.ui-autocomplete').bind('click', function(Event) {
                        Event.stopPropagation();
                        return false;
                    });
                });
            }

        Core.UI.InputFields.Activate($('#SearchInsert'));

        return false;
    };

    /**
     * @function
     * @param {jQueryObject} $Element The jQuery object of the form  or any element
     *      within this form check.
     * @return nothing
     *      This function remove attributes from an element.
     */

    TargetNS.SearchAttributeRemove = function ($Element) {
        $Element.prev().prev().remove();
        $Element.prev().remove();
        $Element.remove();
    };

    /**
     * @function
     * @private
     * @param {String} Profile The profile name that will be delete.
     * @return nothing
     * @description Delete a profile via an ajax requests.
     */
    function SearchProfileDelete(Profile) {
        var Data = {
            Action: 'AgentITSMConfigItemSearch',
            Subaction: 'AJAXProfileDelete',
            Profile: Profile,
            ClassID: $('#SearchClassID').val()
        };
        Core.AJAX.FunctionCall(
            Core.Config.Get('CGIHandle'),
            Data,
            function () {}
        );
    }

    /**
     * @function
     * @private
     * @return nothing
     * @description Shows waiting dialog until search screen is ready.
     */
    function ShowWaitingDialog() {
        var Content = Core.Template.Render(
            'Agent/ITSMCore/LoadingDialog',
            {
                SpanTitle: Core.Config.Get('LoadingMsg')
            }
        );

        Core.UI.Dialog.ShowContentDialog(
            Content,
            Core.Config.Get('LoadingMsg'),
            '10px',
            'Center',
            true
        );
    }

    /**
     * @function
     * @param none
     * @return nothing
     *      This function sets all search dialog settings
     */

    TargetNS.SetSearchDialog = function() {

        // hide add template block
        $('#SearchProfileAddBlock').hide();

        // hide save changes in template block
        $('#SaveProfile').parent().hide().prev().hide().prev().hide();

        // search profile is selected
        if ($('#SearchProfile').val() && $('#SearchProfile').val() !== 'last-search') {

            // show delete button
            $('#SearchProfileDelete').show();

            // show profile link
            $('#SearchProfileAsLink').show();

            // show save changes in template block
            $('#SaveProfile').parent().show().prev().show().prev().show();

            // set SaveProfile to 0
            $('#SaveProfile').prop('checked', false);
        }

        Core.UI.InputFields.Activate($('.Dialog:visible'));

        // register add of attribute
        $('#Attribute').on('change', function () {
            var Attribute = $('#Attribute').val();
            TargetNS.SearchAttributeAdd(Attribute);
            TargetNS.AdditionalAttributeSelectionRebuild();

            // Register event for tree selection dialog
            $('.ShowTreeSelection').off('click').on('click', function () {
                Core.UI.TreeSelection.ShowTreeSelection($(this));
                return false;
            });

            return false;
        });

        // register return key
        $('#SearchForm').off('keypress.FilterInput').on('keypress.FilterInput', function (Event) {
            if ((Event.charCode || Event.keyCode) === 13) {
                if (!CheckForSearchedValues()) {
                    return false;
                }
                else {
                    $('#SearchFormSubmit').trigger('click');
                }
                return false;
            }
        });

        // register submit
        $('#SearchFormSubmit').off('click.DoSearch').on('click.DoSearch', function () {

            var ShownAttributes = [];

            if ($('#SearchProfileAddAction, #SearchProfileAddName').is(':visible') && $('#SearchProfileAddName').val()) {
                $('#SearchProfileAddAction').trigger('click');
            }

            // remember shown attributes
            $('#SearchInsert label').each(function () {
                if ($(this).attr('id')) {
                    ShownAttributes.push($(this).attr('id'));
                }
            });
            $('#SearchForm #ShownAttributes').val(ShownAttributes.join(';'));

            // Normal results mode will return HTML in the same window
            if ($('#SearchForm #ResultForm').val() === 'Normal') {

                if (!CheckForSearchedValues()) {
                    return false;
                }
                else {
                    CheckSearchStringsForStopWords(function () {
                        $('#SearchForm').submit();
                        return false;
                   });
                }
            }
            else { // Print and CSV should open in a new window, no waiting dialog
                $('#SearchForm').attr('target', 'SearchResultPage');
                if (!CheckForSearchedValues()) {
                    return false;
                }
                else {
                    CheckSearchStringsForStopWords(function () {
                        $('#SearchForm').submit();
                        $('#SearchForm').attr('target', '');
                    });
                }
            }
            return false;
        });

        Core.Form.Validate.Init();
        Core.Form.Validate.SetSubmitFunction($('#SearchForm'), function (Form) {
            Form.submit();

            // Show only a waiting dialog for Normal results mode, because this result
            //  will return the HTML in the same window.
            if ($('#SearchForm #ResultForm').val() === 'Normal') {
                ShowWaitingDialog();
            }
        });

        // load profile
        $('#SearchProfile').bind('change', function () {
            var SearchProfile = $('#SearchProfile').val(),
                SearchProfileClassID = $('#SearchClassID').val(),
                SearchProfileAction = $('#SearchAction').val();

            TargetNS.OpenSearchDialog(SearchProfileAction, SearchProfile, SearchProfileClassID);
            return false;
        });

        // show add profile block or not
        $('#SearchProfileNew').on('click', function (Event) {
            $('#SearchProfileAddBlock').toggle();
            $('#SearchProfileAddName').focus();
            Event.preventDefault();
            return false;
        });

        // add new profile
        $('#SearchProfileAddAction').bind('click', function () {
            var ProfileName, $Element1;

            // get name
            ProfileName = $('#SearchProfileAddName').val();

            // check name
            if (!ProfileName.length || ProfileName.length < 2) {
                return;
            }

            // add name to profile selection
            $Element1 = $('#SearchProfile').children().first().clone();
            $Element1.text(ProfileName);
            $Element1.attr('value', ProfileName);
            $Element1.prop('selected', true);
            $('#SearchProfile').append($Element1).trigger('redraw.InputField');

            // set input box to empty
            $('#SearchProfileAddName').val('');

            // hide add template block
            $('#SearchProfileAddBlock').hide();

            // hide save changes in template block
            $('#SaveProfile').parent().hide().prev().hide().prev().hide();

            // set SaveProfile to 1
            $('#SaveProfile').prop('checked', true);

            // show delete button
            $('#SearchProfileDelete').show();

            // show profile link
            $('#SearchProfileAsLink').show();
        });

        // direct link to profile
        $('#SearchProfileAsLink').bind('click', function () {
            var SearchProfile = $('#SearchProfile').val(),
                SearchProfileAction = $('#SearchAction').val(),
                ClassID = $('#SearchClassID').val();

            window.location.href = Core.Config.Get('Baselink') + 'Action=' + SearchProfileAction +
            ';Subaction=Search;TakeLastSearch=1;SaveProfile=1;Profile=' + encodeURIComponent(SearchProfile)
            + ';ClassID=' + ClassID;
            return false;
        });

        // delete profile
        $('#SearchProfileDelete').on('click', function (Event) {

            // strip all already used attributes
            $('#SearchProfile').find('option:selected').each(function () {
                if ($(this).attr('value') !== 'last-search') {

                    // rebuild attributes
                    $('#SearchInsert').text('');

                    // remove remote
                    SearchProfileDelete($(this).val());

                    // remove local
                    $(this).remove();

                    // show fulltext
                    TargetNS.SearchAttributeAdd('Fulltext');

                    // rebuild selection
                    TargetNS.AdditionalAttributeSelectionRebuild();
                }
            });
            $('#SearchProfile').trigger('change');

            if ($('#SearchProfile').val() && $('#SearchProfile').val() === 'last-search') {

                // hide delete link
                $('#SearchProfileDelete').hide();

                // show profile link
                $('#SearchProfileAsLink').hide();

            }

            Event.preventDefault();
            return false;
        });

    };

    /**
     * @function
     * @param {Profile} The profile that is set to the search dialog
     * @return nothing
     *      This function refresh the search dialog with the selected profile
     */

    TargetNS.LoadProfile = function (Profile) {
        var BaseLink = Core.Config.Get('Baselink'),
            Action = 'Action=AgentITSMConfigItemSearch;',
            SubAction = 'Subaction=AJAXUpdate;',
            ClassID = 'ClassID=' + $('#SearchClassID').val() + ';',
            SearchProfile = 'Profile=' + Profile,
            URL =  BaseLink + Action + SubAction + ClassID + SearchProfile;

        $('#DivClassID').addClass('ui-autocomplete-loading');
        Core.AJAX.ContentUpdate($('#AJAXUpdate'), URL, function() {
            var ITSMSearchProfileAttributes = Core.Config.Get('ITSMSearchProfileAttributes') || [];
            $.each(ITSMSearchProfileAttributes, function(Idx, Attribute) {
                ITSM.Agent.ConfigItem.Search.SearchAttributeAdd(Core.App.EscapeSelector(Attribute));
                ITSM.Agent.ConfigItem.Search.AdditionalAttributeSelectionRebuild();
            });

            TargetNS.SetSearchDialog('$Env{"Action"}');
            $('#ITSMSearchProfile').removeClass('Hidden');
            $('#ITSMSearchFields').removeClass('Hidden');
            $('.Dialog:visible #SearchFormSubmit').appendTo($('.Dialog:visible > .Content > .ContentFooter'));
            $('#SearchFormSubmit').removeClass('Hidden');
            $('#DivClassID').removeClass('ui-autocomplete-loading');
            Core.UI.InputFields.Activate($('#SearchForm'));
        });
    };

    /**
     * @private
     * @name CheckForSearchedValues
     * @memberof Core.Agent.Search
     * @function
     * @returns {Boolean} False if no values were found, true if values where there.
     * @description
     *      Checks if any values were entered in the search.
     *      If nothing at all exists, it alerts with translated:
     *      "Please enter at least one search value or * to find anything"
     */
    function CheckForSearchedValues() {
        // loop through the SerachForm labels
        var SearchValueFlag = false;
        $('#SearchForm label').each(function () {
            var ElementName,
                $Element,
                $LabelElement = $(this),
                $FieldElement = $LabelElement.next('.Field');
            // those with ID's are used for searching
            if ($(this).attr('id')) {

                // substring "Label" (e.g. first five characters ) from the
                // label id, use the remaining name as name string for accessing
                // the form input's value
                ElementName = $(this).attr('id').substring(5);
                $Element = $('#SearchForm input[name=' + Core.App.EscapeSelector(ElementName) + ']');

                // If there's no input element with the selected name
                // find the next "select" element and use that one for checking
                if (!$Element.length) {
                    $Element = $(this).next().find('select');
                }

                // Fix for bug#10845: make sure time slot fields with TimeInputFormat
                // 'Input' set are being considered correctly, too. As this is only
                // relevant for search type 'TimeSlot', we check for the first
                // input type=text elment in the corresponding field element.
                // All time field elements have to be filled in, but if only one
                // is missing, we treat the whole field as invalid.
                if ($FieldElement.find('input[name$="SearchType"]').val() === 'TimeSlot' && !$FieldElement.find('select').length) {
                    $Element = $FieldElement.find('input[type=text]').first();
                }

                if ($Element.length) {
                    if ($Element.val() && $Element.val() !== '') {
                        SearchValueFlag = true;
                    }
                }
            }
        });
        if (!SearchValueFlag) {
           alert(Core.Language.Translate('Please enter at least one search value or * to find anything.'));
        }
        return SearchValueFlag;
    }

        /**
     * @private
     * @name CheckSearchStringsForStopWords
     * @memberof Core.Agent.Search
     * @function
     * @param {Function} Callback - function to execute, if no stop words were found.
     * @description Checks if specific values of the search form contain stop words.
     *              If stop words are present, a warning will be displayed.
     *              If stop words are not present, the given callback will be executed.
     */
    function CheckSearchStringsForStopWords(Callback) {
        var SearchStrings = {},
            SearchStringsFound = 0,
            RelevantElementNames = {
                'From': 1,
                'To': 1,
                'Cc': 1,
                'Subject': 1,
                'Body': 1,
                'Fulltext': 1
            };

        if (!Core.Config.Get('CheckSearchStringsForStopWords')) {
            Callback();
            return;
        }

        $('#SearchForm label').each(function () {
            var ElementName,
                $Element;

            // those with ID's are used for searching
            if ($(this).attr('id')) {

                // substring "Label" (e.g. first five characters ) from the
                // label id, use the remaining name as name string for accessing
                // the form input's value
                ElementName = $(this).attr('id').substring(5);
                if (!RelevantElementNames[ElementName]) {
                    return;
                }

                $Element = $('#SearchForm input[name=' + ElementName + ']');

                if ($Element.length) {
                    if ($Element.val() && $Element.val() !== '') {
                        SearchStrings[ElementName] = $Element.val();
                        SearchStringsFound = 1;
                    }
                }
            }
        });

        // Check if stop words are present.
        if (!SearchStringsFound) {
            Callback();
            return;
        }

        AJAXStopWordCheck(
            SearchStrings,
            function (FoundStopWords) {
                alert(Core.Config.Get('SearchStringsContainStopWordsMsg') + "\n" + FoundStopWords);
            },
            Callback
        );
    }

    /**
     * @function
     * @param {Event} Action
     * @return nothing
     *      This function open the search dialog
     */

    TargetNS.OpenSearchDialog = function (Action, Profile, Class) {

        var Referrer = Core.Config.Get('Action'),
            Data;

        if (!Action) {
            Action ='AgentITSMConfigItemSearch';
        }
        Data = {
            Action: Action,
            Referrer: Referrer,
            Profile: Profile,
            Subaction: 'AJAX',
            ClassID: Class
        };

        ShowWaitingDialog();
        Core.AJAX.FunctionCall(
            Core.Config.Get('CGIHandle'),
            Data,
            function (HTML) {

                // if the waiting dialog was canceled, do not show the search
                //  dialog as well
                if (!$('.Dialog:visible').length) {
                    return;
                }

                Core.UI.Dialog.ShowContentDialog(HTML, Core.Config.Get('SearchMsg'), '10px', 'Center', true, [], true);

                $('#SearchClassID').off('change.SearchProfile').on('change.SearchProfile', function() {
                    if ($('#SearchClassID').val() !== "") {
                        ITSM.Agent.ConfigItem.Search.LoadProfile($('#SearchProfile').val());
                    }
                    else {
                        $('#SearchProfile').attr("id", "SearchProfileOld");
                    }
                });

                var ClassSelected = $('#SearchClassID').val();
                if (ClassSelected) {
                    ITSM.Agent.ConfigItem.Search.LoadProfile($('#SearchProfile').val());
                }

                TargetNS.SetSearchDialog();
            }, 'html'
        );
    };

    Core.Init.RegisterNamespace(TargetNS, 'APP_MODULE');

    return TargetNS;
}(ITSM.Agent.ConfigItem.Search || {}));

Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMjEgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKLy8gZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQuCi8vIC0tCgoKInVzZSBzdHJpY3QiOwoKdmFyIElUU00gPSBJVFNNIHx8IHt9OwpJVFNNLkFnZW50ID0gSVRTTS5BZ2VudCB8fCB7fTsKSVRTTS5BZ2VudC5Db25maWdJdGVtID0gSVRTTS5BZ2VudC5Db25maWdJdGVtIHx8IHt9OwoKLyoqCiAqIEBuYW1lc3BhY2UKICogQGV4cG9ydHMgVGFyZ2V0TlMgYXMgSVRTTS5BZ2VudC5Db25maWdJdGVtLlNlYXJjaAogKiBAZGVzY3JpcHRpb24KICogICAgICBUaGlzIG5hbWVzcGFjZSBjb250YWlucyB0aGUgc3BlY2lhbCBtb2R1bGUgZnVuY3Rpb25zIGZvciB0aGUgc2VhcmNoLgogKi8KSVRTTS5BZ2VudC5Db25maWdJdGVtLlVzZXJTZWFyY2ggPSAoZnVuY3Rpb24gKFRhcmdldE5TKSB7CgogICAgLyoqCiAgICAgKiBAbmFtZSBJbml0CiAgICAgKiBAbWVtYmVyb2YgSVRTTS5BZ2VudC5Vc2VyU2VhcmNoCiAgICAgKiBAZnVuY3Rpb24KICAgICAqIEBwYXJhbSB7alF1ZXJ5T2JqZWN0fSAkRWxlbWVudCAtIFRoZSBqUXVlcnkgb2JqZWN0IG9mIHRoZSBpbnB1dCBmaWVsZCB3aXRoIGF1dG9jb21wbGV0ZS4KICAgICAqIEBkZXNjcmlwdGlvbgogICAgICogICAgICBJbml0aWFsaXplcyB0aGUgc3BlY2lhbCBtb2R1bGUgZnVuY3Rpb25zLgogICAgICovCiAgICBUYXJnZXROUy5Jbml0ID0gZnVuY3Rpb24gKCkgewoKICAgICAgICB2YXIgVXNlclNlYXJjaEl0ZW1JRHMgPSBDb3JlLkNvbmZpZy5HZXQoJ1VzZXJTZWFyY2hJdGVtSURzJyksCiAgICAgICAgICAgIGk7CgogICAgICAgIGlmICh0eXBlb2YgVXNlclNlYXJjaEl0ZW1JRHMgIT09ICd1bmRlZmluZWQnICYmIEFycmF5LmlzQXJyYXkoVXNlclNlYXJjaEl0ZW1JRHMpICYmIFVzZXJTZWFyY2hJdGVtSURzLmxlbmd0aCkgewoKICAgICAgICAgICAgZm9yIChpID0gMDsgaSA8IFVzZXJTZWFyY2hJdGVtSURzLmxlbmd0aDsgaSsrKSB7CiAgICAgICAgICAgICAgICAvLyBlc2NhcGUgcG9zc2libGUgY29sb25zICg6KSBpbiBlbGVtZW50IGlkIGJlY2F1c2UgalF1ZXJ5IGNhbiBub3QgaGFuZGxlIGl0IGluIGlkIGF0dHJpYnV0ZSBzZWxlY3RvcnMKICAgICAgICAgICAgICAgIElUU00uQWdlbnQuVXNlclNlYXJjaC5Jbml0KCQoIiMiICsgQ29yZS5BcHAuRXNjYXBlU2VsZWN0b3IoVXNlclNlYXJjaEl0ZW1JRHNbaV0pKSk7CiAgICAgICAgICAgIH0KICAgICAgICB9CiAgICB9OwoKICAgIENvcmUuSW5pdC5SZWdpc3Rlck5hbWVzcGFjZShUYXJnZXROUywgJ0FQUF9NT0RVTEUnKTsKCiAgICByZXR1cm4gVGFyZ2V0TlM7Cn0oSVRTTS5BZ2VudC5Db25maWdJdGVtLlVzZXJTZWFyY2ggfHwge30pKTsK
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMjEgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKLy8gZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQuCi8vIC0tCgoidXNlIHN0cmljdCI7Cgp2YXIgSVRTTSA9IElUU00gfHwge307CklUU00uQWdlbnQgPSBJVFNNLkFnZW50IHx8IHt9OwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gPSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gfHwge307CgovKioKICogQG5hbWVzcGFjZSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uWm9vbQogKiBAbWVtYmVyb2YgSVRTTS5BZ2VudC5Db25maWdJdGVtCiAqIEBhdXRob3IgT1RSUyBBRwogKiBAZGVzY3JpcHRpb24KICogICAgICBUaGlzIG5hbWVzcGFjZSBjb250YWlucyB0aGUgc3BlY2lhbCBtb2R1bGUgZnVuY3Rpb25zIGZvciB0aGUgY29uZmlnIGl0ZW0gYWRkLgogKi8KSVRTTS5BZ2VudC5Db25maWdJdGVtLlpvb20gPSAoZnVuY3Rpb24gKFRhcmdldE5TKSB7CgogICAgLyoKICAgICogQG5hbWUgSW5pdAogICAgKiBAbWVtYmVyb2YgSVRTTS5BZ2VudC5Db25maWdJdGVtLlpvb20KICAgICogQGZ1bmN0aW9uCiAgICAqIEBkZXNjcmlwdGlvbgogICAgKiAgICAgIFRoaXMgZnVuY3Rpb24gaW5pdGlhbGl6ZXMgY29uZmlnIGl0ZW0gem9vbSBzZWN0aW9uLgogICAgKi8KICAgIFRhcmdldE5TLkluaXQgPSBmdW5jdGlvbiAoKSB7CiAgICAgICAgdmFyIElUU01TaG93Q29uZmlybURpYWxvZyA9IENvcmUuQ29uZmlnLkdldCgnSVRTTVNob3dDb25maXJtRGlhbG9nJyk7CgogICAgICAgIElUU00uQWdlbnQuWm9vbS5Jbml0KENvcmUuQ29uZmlnLkdldCgnVXNlckNvbmZpZ0l0ZW1ab29tVGFibGVIZWlnaHQnKSk7CgogICAgICAgICQoJ3VsLkFjdGlvbnMgYS5Bc1BvcHVwJykub24oJ2NsaWNrJywgZnVuY3Rpb24gKCkgewogICAgICAgICAgICBDb3JlLlVJLlBvcHVwLk9wZW5Qb3B1cCgkKHRoaXMpLmF0dHIoJ2hyZWYnKSwgJ0FjdGlvbicpOwogICAgICAgICAgICByZXR1cm4gZmFsc2U7CiAgICAgICAgfSk7CgogICAgICAgICQoJy5NYXN0ZXJBY3Rpb24nKS5vbignY2xpY2snLCBmdW5jdGlvbiAoRXZlbnQpIHsKICAgICAgICAgICAgdmFyICRNYXN0ZXJBY3Rpb25MaW5rID0gJCh0aGlzKS5maW5kKCcuTWFzdGVyQWN0aW9uTGluaycpOwogICAgICAgICAgICAvLyBvbmx5IGFjdCBpZiB0aGUgbGluayB3YXMgbm90IGNsaWNrZWQgZGlyZWN0bHkKICAgICAgICAgICAgaWYgKEV2ZW50LnRhcmdldCAhPT0gJE1hc3RlckFjdGlvbkxpbmsuZ2V0KDApKSB7CiAgICAgICAgICAgICAgICB3aW5kb3cubG9jYXRpb24gPSAkTWFzdGVyQWN0aW9uTGluay5hdHRyKCdocmVmJyk7CiAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7CiAgICAgICAgICAgIH0KICAgICAgICB9KTsKCiAgICAgICAgLy8gSW5pdGlhbGl6ZSBhbGxvY2F0aW9uIGxpc3QgZm9yIGxpbmsgb2JqZWN0IHRhYmxlLgogICAgICAgIENvcmUuQWdlbnQuVGFibGVGaWx0ZXJzLlNldEFsbG9jYXRpb25MaXN0KCk7CgogICAgICAgIGlmIChJVFNNU2hvd0NvbmZpcm1EaWFsb2cpIHsKICAgICAgICAgICAgJC5lYWNoKElUU01TaG93Q29uZmlybURpYWxvZywgZnVuY3Rpb24oS2V5LCBEYXRhKSB7CiAgICAgICAgICAgICAgICBJVFNNLkFnZW50LkNvbmZpcm1EaWFsb2cuQmluZENvbmZpcm1EaWFsb2coewogICAgICAgICAgICAgICAgICAgIEVsZW1lbnRJRDogICAgICAgICAgICAgICAgICBEYXRhLk1lbnVJRCwKICAgICAgICAgICAgICAgICAgICBFbGVtZW50U2VsZWN0b3I6ICAgICAgICAgICAgRGF0YS5FbGVtZW50U2VsZWN0b3IsCiAgICAgICAgICAgICAgICAgICAgRGlhbG9nQ29udGVudFF1ZXJ5U3RyaW5nOiAgIERhdGEuRGlhbG9nQ29udGVudFF1ZXJ5U3RyaW5nLAogICAgICAgICAgICAgICAgICAgIENvbmZpcm1lZEFjdGlvblF1ZXJ5U3RyaW5nOiBEYXRhLkNvbmZpcm1lZEFjdGlvblF1ZXJ5U3RyaW5nLAogICAgICAgICAgICAgICAgICAgIERpYWxvZ1RpdGxlOiAgICAgICAgICAgICAgICBEYXRhLkRpYWxvZ1RpdGxlLAogICAgICAgICAgICAgICAgICAgIFRyYW5zbGF0ZWRUZXh0OiAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgICAgIFllczogQ29yZS5MYW5ndWFnZS5UcmFuc2xhdGUoIlllcyIpLAogICAgICAgICAgICAgICAgICAgICAgICBObzogIENvcmUuTGFuZ3VhZ2UuVHJhbnNsYXRlKCJObyIpLAogICAgICAgICAgICAgICAgICAgICAgICBPazogIENvcmUuTGFuZ3VhZ2UuVHJhbnNsYXRlKCJPayIpCiAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgfSk7CiAgICAgICAgICAgIH0pOwogICAgICAgIH0KICAgIH07CgogICAgQ29yZS5Jbml0LlJlZ2lzdGVyTmFtZXNwYWNlKFRhcmdldE5TLCAnQVBQX01PRFVMRScpOwoKICAgIHJldHVybiBUYXJnZXROUzsKfShJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uWm9vbSB8fCB7fSkpOwo=
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMjEgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKLy8gZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQuCi8vIC0tCgoidXNlIHN0cmljdCI7Cgp2YXIgSVRTTSA9IElUU00gfHwge307CklUU00uQWdlbnQgPSBJVFNNLkFnZW50IHx8IHt9OwpJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gPSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0gfHwge307CgovKioKICogQG5hbWVzcGFjZSBJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uWm9vbQogKiBAbWVtYmVyb2YgSVRTTS5BZ2VudC5Db25maWdJdGVtCiAqIEBhdXRob3IgT1RSUyBBRwogKiBAZGVzY3JpcHRpb24KICogICAgICBUaGlzIG5hbWVzcGFjZSBjb250YWlucyB0aGUgc3BlY2lhbCBtb2R1bGUgZnVuY3Rpb25zIGZvciB0aGUgY29uZmlnIGl0ZW0gYWRkLgogKi8KSVRTTS5BZ2VudC5Db25maWdJdGVtLlpvb20gPSAoZnVuY3Rpb24gKFRhcmdldE5TKSB7CgogICAgLyoKICAgICogQG5hbWUgSW5pdAogICAgKiBAbWVtYmVyb2YgSVRTTS5BZ2VudC5Db25maWdJdGVtLlpvb20KICAgICogQGZ1bmN0aW9uCiAgICAqIEBkZXNjcmlwdGlvbgogICAgKiAgICAgIFRoaXMgZnVuY3Rpb24gaW5pdGlhbGl6ZXMgY29uZmlnIGl0ZW0gem9vbSBzZWN0aW9uLgogICAgKi8KICAgIFRhcmdldE5TLkluaXQgPSBmdW5jdGlvbiAoKSB7CiAgICAgICAgdmFyIElUU01TaG93Q29uZmlybURpYWxvZyA9IENvcmUuQ29uZmlnLkdldCgnSVRTTVNob3dDb25maXJtRGlhbG9nJyk7CgogICAgICAgIElUU00uQWdlbnQuWm9vbS5Jbml0KENvcmUuQ29uZmlnLkdldCgnVXNlckNvbmZpZ0l0ZW1ab29tVGFibGVIZWlnaHQnKSk7CgogICAgICAgICQoJ3VsLkFjdGlvbnMgYS5Bc1BvcHVwJykub24oJ2NsaWNrJywgZnVuY3Rpb24gKCkgewogICAgICAgICAgICBDb3JlLlVJLlBvcHVwLk9wZW5Qb3B1cCgkKHRoaXMpLmF0dHIoJ2hyZWYnKSwgJ0FjdGlvbicpOwogICAgICAgICAgICByZXR1cm4gZmFsc2U7CiAgICAgICAgfSk7CgogICAgICAgICQoJy5NYXN0ZXJBY3Rpb24nKS5vbignY2xpY2snLCBmdW5jdGlvbiAoRXZlbnQpIHsKICAgICAgICAgICAgdmFyICRNYXN0ZXJBY3Rpb25MaW5rID0gJCh0aGlzKS5maW5kKCcuTWFzdGVyQWN0aW9uTGluaycpOwogICAgICAgICAgICAvLyBvbmx5IGFjdCBpZiB0aGUgbGluayB3YXMgbm90IGNsaWNrZWQgZGlyZWN0bHkKICAgICAgICAgICAgaWYgKEV2ZW50LnRhcmdldCAhPT0gJE1hc3RlckFjdGlvbkxpbmsuZ2V0KDApKSB7CiAgICAgICAgICAgICAgICB3aW5kb3cubG9jYXRpb24gPSAkTWFzdGVyQWN0aW9uTGluay5hdHRyKCdocmVmJyk7CiAgICAgICAgICAgICAgICByZXR1cm4gZmFsc2U7CiAgICAgICAgICAgIH0KICAgICAgICB9KTsKCiAgICAgICAgLy8gSW5pdGlhbGl6ZSBhbGxvY2F0aW9uIGxpc3QgZm9yIGxpbmsgb2JqZWN0IHRhYmxlLgogICAgICAgIENvcmUuQWdlbnQuVGFibGVGaWx0ZXJzLlNldEFsbG9jYXRpb25MaXN0KCk7CgogICAgICAgIGlmIChJVFNNU2hvd0NvbmZpcm1EaWFsb2cpIHsKICAgICAgICAgICAgJC5lYWNoKElUU01TaG93Q29uZmlybURpYWxvZywgZnVuY3Rpb24oS2V5LCBEYXRhKSB7CiAgICAgICAgICAgICAgICBJVFNNLkFnZW50LkNvbmZpcm1EaWFsb2cuQmluZENvbmZpcm1EaWFsb2coewogICAgICAgICAgICAgICAgICAgIEVsZW1lbnRJRDogICAgICAgICAgICAgICAgICBEYXRhLk1lbnVJRCwKICAgICAgICAgICAgICAgICAgICBFbGVtZW50U2VsZWN0b3I6ICAgICAgICAgICAgRGF0YS5FbGVtZW50U2VsZWN0b3IsCiAgICAgICAgICAgICAgICAgICAgRGlhbG9nQ29udGVudFF1ZXJ5U3RyaW5nOiAgIERhdGEuRGlhbG9nQ29udGVudFF1ZXJ5U3RyaW5nLAogICAgICAgICAgICAgICAgICAgIENvbmZpcm1lZEFjdGlvblF1ZXJ5U3RyaW5nOiBEYXRhLkNvbmZpcm1lZEFjdGlvblF1ZXJ5U3RyaW5nLAogICAgICAgICAgICAgICAgICAgIERpYWxvZ1RpdGxlOiAgICAgICAgICAgICAgICBEYXRhLkRpYWxvZ1RpdGxlLAogICAgICAgICAgICAgICAgICAgIFRyYW5zbGF0ZWRUZXh0OiAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgICAgIFllczogQ29yZS5MYW5ndWFnZS5UcmFuc2xhdGUoIlllcyIpLAogICAgICAgICAgICAgICAgICAgICAgICBObzogIENvcmUuTGFuZ3VhZ2UuVHJhbnNsYXRlKCJObyIpLAogICAgICAgICAgICAgICAgICAgICAgICBPazogIENvcmUuTGFuZ3VhZ2UuVHJhbnNsYXRlKCJPayIpCiAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgfSk7CiAgICAgICAgICAgIH0pOwogICAgICAgIH0KICAgIH07CgogICAgQ29yZS5Jbml0LlJlZ2lzdGVyTmFtZXNwYWNlKFRhcmdldE5TLCAnQVBQX01PRFVMRScpOwoKICAgIHJldHVybiBUYXJnZXROUzsKfShJVFNNLkFnZW50LkNvbmZpZ0l0ZW0uWm9vbSB8fCB7fSkpOwo=
// --
// Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
// Copyright (C) 2021 Znuny GmbH, https://znuny.org/
// --
// This software comes with ABSOLUTELY NO WARRANTY. For details, see
// the enclosed file COPYING for license information (GPL). If you
// did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
// --


// TODO:
//Remove this line and fix JSDoc
// nofilter(TidyAll::Plugin::Znuny::JavaScript::ESLint)


"use strict";

var ITSM = ITSM || {};
ITSM.UI = ITSM.UI || {};

/**
 * @namespace
 * @exports TargetNS as ITSM.UI.ConfigItemActionRow
 * @description
 *      Action row functionality
 * @requires
 *      Core.JSON
 *      Core.Data
 */
ITSM.UI.ConfigItemActionRow = (function (TargetNS) {

    if (!Core.Debug.CheckDependency('ITSM.UI.ConfigItemActionRow', 'Core.JSON', 'JSON API')) {
        return;
    }
    if (!Core.Debug.CheckDependency('ITSM.UI.ConfigItemActionRow', 'Core.Data', 'Data API')) {
        return;
    }

    var ConfigItemElementSelectors = {
            'Small': 'div.Overview table td input:checkbox[name=ConfigItemID]',
            'Medium': 'ul.Overview input:checkbox[name=ConfigItemID]',
            'Large': 'ul.Overview input:checkbox[name=ConfigItemID]'
        },
        ConfigItemView;

    /**
     * @function
     * @private
     * @param {Object} Data The data that should be converted
     * @return {string} query string of the data
     * @description Converts a given hash into a query string
     */
    function SerializeData(Data) {
        var QueryString = '';
        $.each(Data, function (Key, Value) {
            QueryString += ';' + encodeURIComponent(Key) + '=' + encodeURIComponent(Value);
        });
        return QueryString;
    }

    /**
     * @function
     * @description
     *      This functions adds information about the valid action of an element to the element.
     *      These information are used to generate the action row individually for this element.
     * @param {jQueryObject} $Element
     *      The element for which the data is stored
     * @param {String} JSONString
     *      The JSON string which contains the information about the valid actions of the element (generated by Perl module)
     *      Could also be an javascript object directly
     * @return nothing
     */
    TargetNS.AddActions = function ($Element, JSONString) {
        var Actions;
        // The element of the given ID must exist, JSONString must not be empty
        if (isJQueryObject($Element)) {
            if (typeof JSONString === 'String') {
                Actions = Core.JSON.Parse(JSONString);
            }
            else {
                Actions = JSONString;
            }

            // save action data to the given element
            Core.Data.Set($Element, 'Actions', Actions);
        }
        else {
            Core.Debug.Log('Element does not exist or no valid data structure passed.');
        }
    };

    /**
     * @function
     * @description
     *      This function is called on click on the checkbox of an ConfigItem element and updates the action row for this element.
     * @param {jQueryObject} $ClickedElement
     *      jQueryObject of the clicked element (normally $(this))
     * @param {jQueryObject} Checkboxes
     *      jQueryObject of the checkboxes of the different ConfigItems
     * @param {jQueryObject} $ActionRow
     *      The jQueryObject of the ActionRow wrapper (normally the <ul>-Element)
     * @return nothing
     */
    TargetNS.UpdateActionRow = function ($ClickedElement, $Checkboxes, $ActionRow) {
        var ConfigItemActionData,
            ActionRowElement;

        // Check, if one or more items are selected
        $Checkboxes = $Checkboxes.filter(':checked');
        // No checkbox is selected
        if (!$Checkboxes.length) {
            // Remove actions and deactivate bulk action
            $ActionRow
                .find('li').filter(':not(.Bulk)').remove()
                .end().end()
                .find('#ConfigItemBulkAction').addClass('Inactive')
                .end()
                .find('li.Last').removeClass('Last')
                .end()
                .find('li:last').addClass('Last');
        }
        // Exactly one checkbox is selected
        else if ($Checkboxes.length === 1 && !$('#SelectAllConfigItems').is(':checked')) {
            // Update actions and activate bulk action
            $ActionRow.find('#ConfigItemBulkAction').removeClass('Inactive');

            // Find the element which is active (it must not be the clicked element!)
            // and get the data
            ConfigItemActionData = Core.Data.Get($Checkboxes.closest('li, tr'), 'Actions');
            if (typeof ConfigItemActionData !== 'undefined') {
                $.each(ConfigItemActionData, function (Index, Value) {
                    if (Value.HTML) {
                        $(Value.HTML).attr('id', Value.ID).appendTo($ActionRow);
                        ActionRowElement = $ActionRow.find('#' + Value.ID).find('a');
                        if (typeof Value.Target === 'undefined' || Value.Target === "") {
                            ActionRowElement.attr('href', Value.Link);
                        }
                        if (Value.PopupType) {
                            ActionRowElement.on('click.Popup', function () {
                                Core.UI.Popup.OpenPopup(Value.Link, Value.PopupType);
                                return false;
                            });
                        }
                    }
                });
            }

            // Apply the Last class to the right element
            $ActionRow
                .find('li.Last').removeClass('Last')
                .end()
                .find('li:last').addClass('Last');
        }
        // Two ore more checkboxes selected
        else {
            // Remove actions and activate bulk action
            $ActionRow
                .find('li').filter(':not(.Bulk)').remove()
                .end().end()
                .find('#ConfigItemBulkAction').removeClass('Inactive')
                .end()
                .find('li.Last').removeClass('Last')
                .end()
                .find('li:last').addClass('Last');
        }
    };

    /**
     * @function
     * @description
     *      This function initializes the complete ActionRow functionality and binds all click events.
     * @return nothing
     */
    TargetNS.Init = function () {
        // Get used ConfigItem view mode
        if ($('#ConfigItemOverviewMedium').length) {
            ConfigItemView = 'Medium';
        }
        else if ($('#ConfigItemOverviewLarge').length) {
            ConfigItemView = 'Large';
        }
        else {
            ConfigItemView = 'Small';
        }

        $('#SelectAllConfigItems').on('click', function () {
            var Status = $(this).prop('checked');
            $(ConfigItemElementSelectors[ConfigItemView]).prop('checked', Status).triggerHandler('click');
        });

        $(ConfigItemElementSelectors[ConfigItemView]).on('click', function (Event) {
            Event.stopPropagation();
            ITSM.UI.ConfigItemActionRow.UpdateActionRow($(this), $(ConfigItemElementSelectors[ConfigItemView]), $('div.OverviewActions ul.Actions'));
        });

        $('#ConfigItemBulkAction a').on('click', function () {
            var $Element = $(this),
                $SelectedConfigItems,
                ConfigItemIDParameter = "ConfigItemID=",
                ConfigItemIDs = "",
                URL;
            if ($Element.parent('li').hasClass('Inactive')) {
                return false;
            }
            else {
                $SelectedConfigItems = $(ConfigItemElementSelectors[ConfigItemView] + ':checked');
                $SelectedConfigItems.each(function () {
                    ConfigItemIDs += ConfigItemIDParameter + $(this).val() + ";";
                });
                URL = Core.Config.Get('Baselink') + "Action=AgentITSMConfigItemBulk;" + ConfigItemIDs;
                URL += SerializeData(Core.App.GetSessionInformation());
                Core.UI.Popup.OpenPopup(URL, 'ConfigItemAction');
            }
            return false;
        });
    };

    return TargetNS;
}(ITSM.UI.ConfigItemActionRow || {}));

// --
// Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
// Copyright (C) 2021 Znuny GmbH, https://znuny.org/
// --
// This software comes with ABSOLUTELY NO WARRANTY. For details, see
// the enclosed file COPYING for license information (GPL). If you
// did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
// --


// TODO:
//Remove this line and fix JSDoc
// nofilter(TidyAll::Plugin::Znuny::JavaScript::ESLint)


"use strict";

var ITSM = ITSM || {};
ITSM.UI = ITSM.UI || {};

/**
 * @namespace
 * @exports TargetNS as ITSM.UI.ConfigItemActionRow
 * @description
 *      Action row functionality
 * @requires
 *      Core.JSON
 *      Core.Data
 */
ITSM.UI.ConfigItemActionRow = (function (TargetNS) {

    if (!Core.Debug.CheckDependency('ITSM.UI.ConfigItemActionRow', 'Core.JSON', 'JSON API')) {
        return;
    }
    if (!Core.Debug.CheckDependency('ITSM.UI.ConfigItemActionRow', 'Core.Data', 'Data API')) {
        return;
    }

    var ConfigItemElementSelectors = {
            'Small': 'div.Overview table td input:checkbox[name=ConfigItemID]',
            'Medium': 'ul.Overview input:checkbox[name=ConfigItemID]',
            'Large': 'ul.Overview input:checkbox[name=ConfigItemID]'
        },
        ConfigItemView;

    /**
     * @function
     * @private
     * @param {Object} Data The data that should be converted
     * @return {string} query string of the data
     * @description Converts a given hash into a query string
     */
    function SerializeData(Data) {
        var QueryString = '';
        $.each(Data, function (Key, Value) {
            QueryString += ';' + encodeURIComponent(Key) + '=' + encodeURIComponent(Value);
        });
        return QueryString;
    }

    /**
     * @function
     * @description
     *      This functions adds information about the valid action of an element to the element.
     *      These information are used to generate the action row individually for this element.
     * @param {jQueryObject} $Element
     *      The element for which the data is stored
     * @param {String} JSONString
     *      The JSON string which contains the information about the valid actions of the element (generated by Perl module)
     *      Could also be an javascript object directly
     * @return nothing
     */
    TargetNS.AddActions = function ($Element, JSONString) {
        var Actions;
        // The element of the given ID must exist, JSONString must not be empty
        if (isJQueryObject($Element)) {
            if (typeof JSONString === 'String') {
                Actions = Core.JSON.Parse(JSONString);
            }
            else {
                Actions = JSONString;
            }

            // save action data to the given element
            Core.Data.Set($Element, 'Actions', Actions);
        }
        else {
            Core.Debug.Log('Element does not exist or no valid data structure passed.');
        }
    };

    /**
     * @function
     * @description
     *      This function is called on click on the checkbox of an ConfigItem element and updates the action row for this element.
     * @param {jQueryObject} $ClickedElement
     *      jQueryObject of the clicked element (normally $(this))
     * @param {jQueryObject} Checkboxes
     *      jQueryObject of the checkboxes of the different ConfigItems
     * @param {jQueryObject} $ActionRow
     *      The jQueryObject of the ActionRow wrapper (normally the <ul>-Element)
     * @return nothing
     */
    TargetNS.UpdateActionRow = function ($ClickedElement, $Checkboxes, $ActionRow) {
        var ConfigItemActionData,
            ActionRowElement;

        // Check, if one or more items are selected
        $Checkboxes = $Checkboxes.filter(':checked');
        // No checkbox is selected
        if (!$Checkboxes.length) {
            // Remove actions and deactivate bulk action
            $ActionRow
                .find('li').filter(':not(.Bulk)').remove()
                .end().end()
                .find('#ConfigItemBulkAction').addClass('Inactive')
                .end()
                .find('li.Last').removeClass('Last')
                .end()
                .find('li:last').addClass('Last');
        }
        // Exactly one checkbox is selected
        else if ($Checkboxes.length === 1 && !$('#SelectAllConfigItems').is(':checked')) {
            // Update actions and activate bulk action
            $ActionRow.find('#ConfigItemBulkAction').removeClass('Inactive');

            // Find the element which is active (it must not be the clicked element!)
            // and get the data
            ConfigItemActionData = Core.Data.Get($Checkboxes.closest('li, tr'), 'Actions');
            if (typeof ConfigItemActionData !== 'undefined') {
                $.each(ConfigItemActionData, function (Index, Value) {
                    if (Value.HTML) {
                        $(Value.HTML).attr('id', Value.ID).appendTo($ActionRow);
                        ActionRowElement = $ActionRow.find('#' + Value.ID).find('a');
                        if (typeof Value.Target === 'undefined' || Value.Target === "") {
                            ActionRowElement.attr('href', Value.Link);
                        }
                        if (Value.PopupType) {
                            ActionRowElement.on('click.Popup', function () {
                                Core.UI.Popup.OpenPopup(Value.Link, Value.PopupType);
                                return false;
                            });
                        }
                    }
                });
            }

            // Apply the Last class to the right element
            $ActionRow
                .find('li.Last').removeClass('Last')
                .end()
                .find('li:last').addClass('Last');
        }
        // Two ore more checkboxes selected
        else {
            // Remove actions and activate bulk action
            $ActionRow
                .find('li').filter(':not(.Bulk)').remove()
                .end().end()
                .find('#ConfigItemBulkAction').removeClass('Inactive')
                .end()
                .find('li.Last').removeClass('Last')
                .end()
                .find('li:last').addClass('Last');
        }
    };

    /**
     * @function
     * @description
     *      This function initializes the complete ActionRow functionality and binds all click events.
     * @return nothing
     */
    TargetNS.Init = function () {
        // Get used ConfigItem view mode
        if ($('#ConfigItemOverviewMedium').length) {
            ConfigItemView = 'Medium';
        }
        else if ($('#ConfigItemOverviewLarge').length) {
            ConfigItemView = 'Large';
        }
        else {
            ConfigItemView = 'Small';
        }

        $('#SelectAllConfigItems').on('click', function () {
            var Status = $(this).prop('checked');
            $(ConfigItemElementSelectors[ConfigItemView]).prop('checked', Status).triggerHandler('click');
        });

        $(ConfigItemElementSelectors[ConfigItemView]).on('click', function (Event) {
            Event.stopPropagation();
            ITSM.UI.ConfigItemActionRow.UpdateActionRow($(this), $(ConfigItemElementSelectors[ConfigItemView]), $('div.OverviewActions ul.Actions'));
        });

        $('#ConfigItemBulkAction a').on('click', function () {
            var $Element = $(this),
                $SelectedConfigItems,
                ConfigItemIDParameter = "ConfigItemID=",
                ConfigItemIDs = "",
                URL;
            if ($Element.parent('li').hasClass('Inactive')) {
                return false;
            }
            else {
                $SelectedConfigItems = $(ConfigItemElementSelectors[ConfigItemView] + ':checked');
                $SelectedConfigItems.each(function () {
                    ConfigItemIDs += ConfigItemIDParameter + $(this).val() + ";";
                });
                URL = Core.Config.Get('Baselink') + "Action=AgentITSMConfigItemBulk;" + ConfigItemIDs;
                URL += SerializeData(Core.App.GetSessionInformation());
                Core.UI.Popup.OpenPopup(URL, 'ConfigItemAction');
            }
            return false;
        });
    };

    return TargetNS;
}(ITSM.UI.ConfigItemActionRow || {}));

Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwovLyAtLQovLyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQovLyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91Ci8vIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCi8vIC0tCgoidXNlIHN0cmljdCI7Cgp2YXIgWm51bnkgPSBabnVueSB8fCB7fTsKdmFyIENvcmUgPSBDb3JlIHx8IHt9OwovKioKICogQG5hbWVzcGFjZQogKiBAZXhwb3J0cyBUYXJnZXROUyBhcyBabnVueS5EeW5hbWljRmllbGQuQ29uZmlnSXRlbQogKiBAZGVzY3JpcHRpb24KICogICAgICBUaGlzIG5hbWVzcGFjZSBjb250YWlucyB0aGUgc3BlY2lhbCBmdW5jdGlvbnMgZm9yIER5bmFtaWNGaWVsZENvbmZpZ0l0ZW0uCiAqLwpabnVueS5EeW5hbWljRmllbGQuQ29uZmlnSXRlbSA9IChmdW5jdGlvbiAoVGFyZ2V0TlMpIHsKCiAgICAvKioKICAgICAqIEBuYW1lIEluaXRBZGRpdGlvbmFsREZTdG9yYWdlCiAgICAgKiBAbWVtYmVyb2YgWm51bnkuRHluYW1pY0ZpZWxkLkNvbmZpZ0l0ZW0uSW5pdEFkZGl0aW9uYWxERlN0b3JhZ2UKICAgICAqIEBmdW5jdGlvbgogICAgICogQHBhcmFtIHtTdHJpbmd9IER5bmFtaWNGaWVsZE5hbWUgLSBOYW1lIG9mIGR5bmFtaWMgZmllbGQgd2hpY2ggY29udGFpbnMgdGhlIGNvbmZpZyBpdGVtIHNlbGVjdGlvbi4KICAgICAqIEBkZXNjcmlwdGlvbgogICAgICogICAgICBJbml0aWFsaXplcyBhZGRpdGlvbmFsIGR5bmFtaWMgZmllbGQgc3RvcmFnZSBmb3Igc2VsZWN0aW9uIG9mIGNvbmZpZyBpdGVtcy4KICAgICAqLwoKICAgIFRhcmdldE5TLkluaXRBZGRpdGlvbmFsREZTdG9yYWdlID0gZnVuY3Rpb24gKER5bmFtaWNGaWVsZE5hbWUpIHsKICAgICAgICAkKCcjRHluYW1pY0ZpZWxkXycgKyBEeW5hbWljRmllbGROYW1lKS5vbignY2hhbmdlJywgZnVuY3Rpb24oKSB7CiAgICAgICAgICAgIEZpbGxBZGRpdGlvbmFsRHluYW1pY0ZpZWxkcyhEeW5hbWljRmllbGROYW1lKTsKICAgICAgICB9KTsKICAgIH0KCiAgICAvLyBUT0RPOiB3ZSBzaG91bGQgdXNlIHRoZSBzYW1lIG5hbWluZyBsaWtlIERGLUxEQVAgKEF1dG9GaWxsKSBvciBzdGFuZGFyZGl6ZSB0aGVzZQogICAgZnVuY3Rpb24gRmlsbEFkZGl0aW9uYWxEeW5hbWljRmllbGRzKFNvdXJjZUR5bmFtaWNGaWVsZE5hbWUpIHsKICAgICAgICB2YXIgVVJMICA9IENvcmUuQ29uZmlnLkdldCgnQmFzZWxpbmsnKSwKICAgICAgICAgICAgU2VsZWN0ZWRDb25maWdJdGVtSURzLAogICAgICAgICAgICBEYXRhID0gewogICAgICAgICAgICAgICAgQWN0aW9uOiAgICAgICAgICAgICAgICAgJ0FKQVhEeW5hbWljRmllbGRDb25maWdJdGVtJywKICAgICAgICAgICAgICAgIFN1YmFjdGlvbjogICAgICAgICAgICAgICdHZXRBZGRpdGlvbmFsREZTdG9yYWdlRGF0YScsCiAgICAgICAgICAgICAgICBTb3VyY2VEeW5hbWljRmllbGROYW1lOiBTb3VyY2VEeW5hbWljRmllbGROYW1lLAogICAgICAgICAgICB9OwoKICAgICAgICAvLyBTdXBwb3J0IHNpbmdsZSBhbmQgbXVsdGkgc2VsZWN0LgogICAgICAgIFNlbGVjdGVkQ29uZmlnSXRlbUlEcyA9IFpudW55LkZvcm0uSW5wdXQuR2V0KCdEeW5hbWljRmllbGRfJyArIFNvdXJjZUR5bmFtaWNGaWVsZE5hbWUpOwogICAgICAgIGlmICh0eXBlb2YoU2VsZWN0ZWRDb25maWdJdGVtSURzKSA9PT0gJ3N0cmluZycpIHsKICAgICAgICAgICAgU2VsZWN0ZWRDb25maWdJdGVtSURzID0gW1NlbGVjdGVkQ29uZmlnSXRlbUlEc107CiAgICAgICAgfQogICAgICAgIERhdGFbJ1NlbGVjdGVkQ29uZmlnSXRlbUlEcyddID0gU2VsZWN0ZWRDb25maWdJdGVtSURzOwoKICAgICAgICBDb3JlLkFKQVguRnVuY3Rpb25DYWxsKAogICAgICAgICAgICBVUkwsCiAgICAgICAgICAgIERhdGEsCiAgICAgICAgICAgIGZ1bmN0aW9uIChSZXNwb25zZSkgewogICAgICAgICAgICAgICAgaWYgKCFSZXNwb25zZSkgewogICAgICAgICAgICAgICAgICAgIHJldHVybjsKICAgICAgICAgICAgICAgIH0KCiAgICAgICAgICAgICAgICAkLmVhY2goUmVzcG9uc2UsIGZ1bmN0aW9uKERlc3RpbmF0aW9uRHluYW1pY0ZpZWxkTmFtZSwgVmFsdWUpIHsKICAgICAgICAgICAgICAgICAgICB2YXIgRmllbGRJRCAgID0gWm51bnkuRm9ybS5JbnB1dC5GaWVsZElEKCdEeW5hbWljRmllbGRfJyArIERlc3RpbmF0aW9uRHluYW1pY0ZpZWxkTmFtZSksCiAgICAgICAgICAgICAgICAgICAgICAgIEZpZWxkVHlwZSA9IFpudW55LkZvcm0uSW5wdXQuVHlwZShGaWVsZElEKSwKICAgICAgICAgICAgICAgICAgICAgICAgRGF0ZVRpbWVQYXJ0czsKCiAgICAgICAgICAgICAgICAgICAgaWYgKFZhbHVlICYmIChGaWVsZFR5cGUgPT0gJ0R5bmFtaWNGaWVsZF9EYXRlJyB8fCBGaWVsZFR5cGUgPT0gJ0R5bmFtaWNGaWVsZF9EYXRlVGltZScpKSB7CiAgICAgICAgICAgICAgICAgICAgICAgIERhdGVUaW1lUGFydHMgPSBWYWx1ZS5tYXRjaCgvXihcZHs0fSktKFxkezJ9KS0oXGR7Mn0pKCAoXGR7Mn0pOihcZHsyfSk6KFxkezJ9KSk/JC8pOwogICAgICAgICAgICAgICAgICAgICAgICBpZiAoRGF0ZVRpbWVQYXJ0cy5sZW5ndGggIT0gOCkgewogICAgICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuOwogICAgICAgICAgICAgICAgICAgICAgICB9CgogICAgICAgICAgICAgICAgICAgICAgICBWYWx1ZSA9IHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgIFllYXI6ICAgcGFyc2VJbnQoRGF0ZVRpbWVQYXJ0c1sxXSwgMTApLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgTW9udGg6ICBwYXJzZUludChEYXRlVGltZVBhcnRzWzJdLCAxMCksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBEYXk6ICAgIHBhcnNlSW50KERhdGVUaW1lUGFydHNbM10sIDEwKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIEhvdXI6ICAgcGFyc2VJbnQoRGF0ZVRpbWVQYXJ0c1s1XSwgMTApLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgTWludXRlOiBwYXJzZUludChEYXRlVGltZVBhcnRzWzZdLCAxMCksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBVc2VkOiAgIHRydWUsCiAgICAgICAgICAgICAgICAgICAgICAgIH07CiAgICAgICAgICAgICAgICAgICAgfQoKICAgICAgICAgICAgICAgICAgICBabnVueS5Gb3JtLklucHV0LlNldCgKICAgICAgICAgICAgICAgICAgICAgICAgJ0R5bmFtaWNGaWVsZF8nICsgRGVzdGluYXRpb25EeW5hbWljRmllbGROYW1lLAogICAgICAgICAgICAgICAgICAgICAgICBWYWx1ZQogICAgICAgICAgICAgICAgICAgICk7CiAgICAgICAgICAgICAgICB9KTsKICAgICAgICAgICAgfQogICAgICAgICk7CiAgICB9CgogICAgcmV0dXJuIFRhcmdldE5TOwp9KFpudW55LkR5bmFtaWNGaWVsZC5Db25maWdJdGVtIHx8IHt9KSk7Cg==
LyoKQ29weXJpZ2h0IChDKSAyMDAxLTIwMjEgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KQ29weXJpZ2h0IChDKSAyMDIxIFpudW55IEdtYkgsIGh0dHBzOi8vem51bnkub3JnLwoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgoqLwoKLyoqCiAqIEBwYWNrYWdlICAgICBTa2luICJEZWZhdWx0IgogKiBAc2VjdGlvbiAgICAgSVRTTSBDb25maWdJdGVuCiAqLwoKQG1lZGlhIHNjcmVlbixwcm9qZWN0aW9uLHR2LGhhbmRoZWxkIHsKCiAuRmllbGQgYnV0dG9uLkRpc2FibGVWYWxpZGF0aW9uICB7Cgl6LWluZGV4OiAwOwp9Cgp9IC8qIGVuZCBAbWVkaWEgKi8K
# --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package var::packagesetup::ITSMConfigurationManagement;    ## no critic

use strict;
use warnings;

use Kernel::Language qw(Translatable);
use Kernel::Output::Template::Provider;
use Kernel::System::VariableCheck qw(:all);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::Console::Command::Maint::ITSM::Configitem::DefinitionPerl2YAML',
    'Kernel::System::DB',
    'Kernel::System::GeneralCatalog',
    'Kernel::System::Group',
    'Kernel::System::ITSMConfigItem',
    'Kernel::System::LinkObject',
    'Kernel::System::Log',
    'Kernel::System::Service',
    'Kernel::System::Stats',
    'Kernel::System::SysConfig',
    'Kernel::System::Valid',
);

=head1 NAME

var::packagesetup::ITSMConfigurationManagement - code to execute during package installation

=head1 DESCRIPTION

Functions for installing the ITSMConfigurationManagement package.

=head1 PUBLIC INTERFACE

=head2 new()

create an object

    use Kernel::System::ObjectManager;
    local $Kernel::OM = Kernel::System::ObjectManager->new();
    my $CodeObject = $Kernel::OM->Get('var::packagesetup::ITSMConfigurationManagement');

=cut

sub new {
    my ( $Type, %Param ) = @_;

    # allocate new hash for object
    my $Self = {};
    bless( $Self, $Type );

    # Force a reload of ZZZAuto.pm and ZZZAAuto.pm to get the fresh configuration values.
    for my $Module ( sort keys %INC ) {
        if ( $Module =~ m/ZZZAA?uto\.pm$/ ) {
            delete $INC{$Module};
        }
    }

    # Create common objects with fresh default config.
    $Kernel::OM->ObjectsDiscard();

    # define UserID parameter for the constructor of the stats object
    $Kernel::OM->ObjectParamAdd(
        'Kernel::System::Stats' => {
            UserID => 1,
        },
    );

    # define file prefix for stats
    $Self->{FilePrefix} = 'ITSMStats';

    return $Self;
}

=head2 CodeInstall()

run the code install part

    my $Result = $CodeObject->CodeInstall();

=cut

sub CodeInstall {
    my ( $Self, %Param ) = @_;

    # add the group itsm-configitem
    $Self->_GroupAdd(
        Name        => 'itsm-configitem',
        Description => 'Group for ITSM ConfigItem mask access in the agent interface.',
    );

    # install configuration item definitions
    $Self->_AddConfigItemDefinitions();

    # fill up empty last_version_id rows in C<configitem> table
    $Self->_FillupEmptyLastVersionID();

    # fill up empty inci_state_id rows in C<configitem_version> table
    $Self->_FillupEmptyVersionIncidentStateID();

    # fill up empty C<cur_depl_state_id> or C<cur_inci_state_id> rows in C<configitem> table
    $Self->_FillupEmptyIncidentAndDeploymentStateID();

    # update DynamicFieldConfig
    $Self->_UpdateDynamicFieldConfigItemLinkType();
    $Self->_UpdateDynamicFieldConfigDeplStateIDs();

    # set preferences for some configuration items
    $Self->_SetPreferences();

    # set default permission group
    $Self->_SetDefaultPermission();

   # set migrate webservice configs - InvokerType from Znuny4OTRSITSMConfigItemInvoker to ITSMConfigItemInvoker::Generic
    $Self->_MigrateWebserviceConfigs();

    # install stats
    $Kernel::OM->Get('Kernel::System::Stats')->StatsInstall(
        FilePrefix => $Self->{FilePrefix},
        UserID     => 1,
    );

    return 1;
}

=head2 CodeReinstall()

run the code reinstall part

    my $Result = $CodeObject->CodeReinstall();

=cut

sub CodeReinstall {
    my ( $Self, %Param ) = @_;

    # add the group itsm-configitem
    $Self->_GroupAdd(
        Name        => 'itsm-configitem',
        Description => 'Group for ITSM ConfigItem mask access in the agent interface.',
    );

    # install configuration item definitions
    $Self->_AddConfigItemDefinitions();

    # fill up empty last_version_id rows in C<configitem> table
    $Self->_FillupEmptyLastVersionID();

    # fill up empty inci_state_id rows in C<configitem_version> table
    $Self->_FillupEmptyVersionIncidentStateID();

    # fill up empty cur_depl_state_id or cur_inci_state_id rows in C<configitem> table
    $Self->_FillupEmptyIncidentAndDeploymentStateID();

    # update DynamicFieldConfig
    $Self->_UpdateDynamicFieldConfigItemLinkType();
    $Self->_UpdateDynamicFieldConfigDeplStateIDs();

    # set preferences for some configuration items
    $Self->_SetPreferences();

    # set default permission group
    $Self->_SetDefaultPermission();

   # set migrate webservice configs - InvokerType from Znuny4OTRSITSMConfigItemInvoker to ITSMConfigItemInvoker::Generic
    $Self->_MigrateWebserviceConfigs();

    # install stats
    $Kernel::OM->Get('Kernel::System::Stats')->StatsInstall(
        FilePrefix => $Self->{FilePrefix},
        UserID     => 1,
    );

    return 1;
}

=head2 CodeUpgradeFromLowerThan_4_0_2()

This function is only executed if the installed module version is smaller than 4.0.2.

    my $Result = $CodeObject->CodeUpgradeFromLowerThan_4_0_2();

=cut

sub CodeUpgradeFromLowerThan_4_0_2 {    ## no critic
    my ( $Self, %Param ) = @_;

    # migrate the template Content in the SysConfig
    $Self->_MigrateDTLInSysConfig();

    return 1;
}

=head2 CodeUpgradeFromLowerThan_4_0_8()

This function is only executed if the installed module version is smaller than 4.0.8.

    my $Result = $CodeObject->CodeUpgradeFromLowerThan_4_0_8();

=cut

sub CodeUpgradeFromLowerThan_4_0_8 {    ## no critic
    my ( $Self, %Param ) = @_;

    # fix a typo in the general catalog
    $Self->_FixTypo();

    return 1;
}

=head2 CodeUpgradeFromLowerThan_4_0_91()

This function is only executed if the installed module version is smaller than 4.0.91.

    my $Result = $CodeObject->CodeUpgradeFromLowerThan_4_0_91();

=cut

sub CodeUpgradeFromLowerThan_4_0_91 {    ## no critic
    my ( $Self, %Param ) = @_;

    # change configurations to match the new module location.
    $Self->_MigrateConfigs();

    return 1;
}

sub _UpdateDynamicFieldConfigItemLinkType {
    my ( $Self, %Param ) = @_;

    my $ConfigObject       = $Kernel::OM->Get('Kernel::Config');
    my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');
    my $LogObject          = $Kernel::OM->Get('Kernel::System::Log');

    my $UserID = 1;

    my $DeprecatedConfig = $ConfigObject->Get('ZnunyDynamicFieldConfigItem') // {};
    return 1 if !IsHashRefWithData($DeprecatedConfig);

    # Don't migrate deprecated SysConfig settings to dynamic field config
    # if config item linking is not enabled.
    return 1 if !$DeprecatedConfig->{TicketLink};

    # Fetch all dynamic fields of type ConfigItemDropdown and ConfigItemMultiselect
    # and object type 'Ticket', because linking is only supported for those.
    my $DynamicFields = $DynamicFieldObject->DynamicFieldListGet();
    my @DynamicFields = grep {
        $_->{FieldType} =~ m{\AConfigItem(Dropdown|Multiselect)\z}
            && $_->{ObjectType} eq 'Ticket'
    } @{$DynamicFields};

    return 1 if !@DynamicFields;

    # Migrate settings to existing dynamic field configs.
    my $ConfigItemLinkType    = $DeprecatedConfig->{LinkType} // 'RelevantTo';
    my $ConfigItemLinkSource  = 'ITSMConfigItem';
    my $ConfigItemLinkRemoval = 1;

    DYNAMICFIELD:
    for my $DynamicField (@DynamicFields) {

        # Ignore dynamic fields that somehow may have already set the link type.
        # This normally cannot happen.
        next DYNAMICFIELD if $DynamicField->{Config}->{ConfigItemLinkType};

        $DynamicField->{Config}->{ConfigItemLinkType}    = $ConfigItemLinkType;
        $DynamicField->{Config}->{ConfigItemLinkSource}  = $ConfigItemLinkSource;
        $DynamicField->{Config}->{ConfigItemLinkRemoval} = $ConfigItemLinkRemoval;

        my $DynamicFieldUpdated = $DynamicFieldObject->DynamicFieldUpdate(
            ID         => $DynamicField->{ID},
            Name       => $DynamicField->{Name},
            Label      => $DynamicField->{Label},
            FieldOrder => $DynamicField->{FieldOrder},
            FieldType  => $DynamicField->{FieldType},
            ObjectType => $DynamicField->{ObjectType},
            Config     => $DynamicField->{Config},
            ValidID    => $DynamicField->{ValidID},
            UserID     => $UserID,
        );
        next DYNAMICFIELD if $DynamicFieldUpdated;

        $LogObject->Log(
            Priority => 'error',
            Message  => "Dynamic field '$DynamicField->{Name}' (ID $DynamicField->{ID}) could not be updated.",
        );
    }

    return 1;
}

sub _UpdateDynamicFieldConfigDeplStateIDs {
    my ( $Self, %Param ) = @_;

    my $ConfigObject         = $Kernel::OM->Get('Kernel::Config');
    my $DynamicFieldObject   = $Kernel::OM->Get('Kernel::System::DynamicField');
    my $LogObject            = $Kernel::OM->Get('Kernel::System::Log');
    my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

    my $UserID = 1;

    my $DeplStateNameByID = $GeneralCatalogObject->ItemList(
        Class => 'ITSM::ConfigItem::DeploymentState',
    );

    # Fetch all dynamic fields of type ConfigItemDropdown and ConfigItemMultiselect
    # and object type 'Ticket', because linking is only supported for those.
    my $DynamicFieldConfigs = $DynamicFieldObject->DynamicFieldListGet();
    my @DynamicFieldConfigs = grep {
        $_->{FieldType} =~ m{\AConfigItem(Dropdown|Multiselect)\z}
    } @{$DynamicFieldConfigs};

    return 1 if !@DynamicFieldConfigs;

    DYNAMICFIELDCONFIG:
    for my $DynamicFieldConfig (@DynamicFieldConfigs) {
        next DYNAMICFIELDCONFIG if !exists $DynamicFieldConfig->{Config}->{DeplStateIDs};

        my @DeplStates = grep { defined $_ }
            map { $DeplStateNameByID->{$_} }
            @{ $DynamicFieldConfig->{Config}->{DeplStateIDs} // [] };

        delete $DynamicFieldConfig->{Config}->{DeplStateIDs};
        $DynamicFieldConfig->{Config}->{DeplStates} = \@DeplStates;

        my $DynamicFieldConfigUpdated = $DynamicFieldObject->DynamicFieldUpdate(
            ID         => $DynamicFieldConfig->{ID},
            Name       => $DynamicFieldConfig->{Name},
            Label      => $DynamicFieldConfig->{Label},
            FieldOrder => $DynamicFieldConfig->{FieldOrder},
            FieldType  => $DynamicFieldConfig->{FieldType},
            ObjectType => $DynamicFieldConfig->{ObjectType},
            Config     => $DynamicFieldConfig->{Config},
            ValidID    => $DynamicFieldConfig->{ValidID},
            UserID     => $UserID,
        );
        next DYNAMICFIELDCONFIG if $DynamicFieldConfigUpdated;

        $LogObject->Log(
            Priority => 'error',
            Message =>
                "Config of dynamic field '$DynamicFieldConfig->{Name}' (ID $DynamicFieldConfig->{ID}) could not be updated.",
        );
    }

    return 1;
}

=head2 CodeUpgrade()

run the code upgrade part

    my $Result = $CodeObject->CodeUpgrade();

=cut

sub CodeUpgrade {
    my ( $Self, %Param ) = @_;

    # install configuration item definitions
    $Self->_AddConfigItemDefinitions();

    # fill up empty last_version_id rows in C<configitem> table
    $Self->_FillupEmptyLastVersionID();

    # fill up empty inci_state_id rows in C<configitem_version> table
    $Self->_FillupEmptyVersionIncidentStateID();

    # fill up empty cur_depl_state_id or cur_inci_state_id rows in C<configitem> table
    $Self->_FillupEmptyIncidentAndDeploymentStateID();

    # update DynamicFieldConfig
    $Self->_UpdateDynamicFieldConfigItemLinkType();
    $Self->_UpdateDynamicFieldConfigDeplStateIDs();

    # set preferences for some configuration items
    $Self->_SetPreferences();

    # set default permission group
    $Self->_SetDefaultPermission();

   # set migrate webservice configs - InvokerType from Znuny4OTRSITSMConfigItemInvoker to ITSMConfigItemInvoker::Generic
    $Self->_MigrateWebserviceConfigs();

    # install stats
    $Kernel::OM->Get('Kernel::System::Stats')->StatsInstall(
        FilePrefix => $Self->{FilePrefix},
        UserID     => 1,
    );

    # Convert any Perl definition to YAML
    # This could be moved to a version specific
    $Self->_ConvertPerlDefinitions2YAML();

    return 1;
}

=head2 CodeUninstall()

run the code uninstall part

    my $Result = $CodeObject->CodeUninstall();

=cut

sub CodeUninstall {
    my ( $Self, %Param ) = @_;

    # delete all links with configuration items
    $Self->_LinkDelete();

    # deactivate the group C<itsm-configitem>
    $Self->_GroupDeactivate(
        Name => 'itsm-configitem',
    );

    # delete 'CurInciStateTypeFromCIs' service preferences
    $Self->_DeleteServicePreferences();

    return 1;
}

=head2 _SetPreferences()

    my $Result = $CodeObject->_SetPreferences()

=cut

sub _SetPreferences {
    my ( $Self, %Param ) = @_;

    my %Map = (
        Expired     => 'productive',
        Inactive    => 'postproductive',
        Maintenance => 'productive',
        Pilot       => 'productive',
        Planned     => 'preproductive',
        Production  => 'productive',
        Repair      => 'productive',
        Retired     => 'postproductive',
        Review      => 'productive',
        'Test/QA'   => 'preproductive',
    );

    NAME:
    for my $Name ( sort keys %Map ) {
        my $Item = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemGet(
            Name  => $Name,
            Class => 'ITSM::ConfigItem::DeploymentState',
        );

        next NAME if !$Item;

        $Kernel::OM->Get('Kernel::System::GeneralCatalog')->GeneralCatalogPreferencesSet(
            ItemID => $Item->{ItemID},
            Key    => 'Functionality',
            Value  => $Map{$Name},
        );
    }

    return 1;
}

=head2 _SetDefaultPermission()

set the default group that has access rights

=cut

sub _SetDefaultPermission {
    my ( $Self, %Param ) = @_;

    # get class list
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );

    # check if group already exists
    my $GroupID = $Kernel::OM->Get('Kernel::System::Group')->GroupLookup(
        Group  => 'itsm-configitem',
        UserID => 1,
    );

    # check if a permission group is already set. If not, set default permission group
    for my $ClassID ( sort keys %{$ClassList} ) {
        my $Class = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemGet(
            ItemID => $ClassID,
        );

        if ( !$Class->{Permission} ) {
            $Kernel::OM->Get('Kernel::System::GeneralCatalog')->GeneralCatalogPreferencesSet(
                ItemID => $Class->{ItemID},
                Key    => 'Permission',
                Value  => $GroupID,
            );
        }
    }

    return 1;
}

=head2 _GroupAdd()

add a group

    my $Result = $CodeObject->_GroupAdd(
        Name        => 'the-group-name',
        Description => 'The group description.',
    );

=cut

sub _GroupAdd {
    my ( $Self, %Param ) = @_;

    NEEDED:
    for my $Needed (qw(Name Description)) {

        next NEEDED if defined $Param{$Needed};

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Need $Needed!",
        );
        return;
    }

    # get valid list
    my %ValidList = $Kernel::OM->Get('Kernel::System::Valid')->ValidList(
        UserID => 1,
    );
    my %ValidListReverse = reverse %ValidList;

    # get list of all groups
    my %GroupList = $Kernel::OM->Get('Kernel::System::Group')->GroupList();

    # reverse the group list for easier lookup
    my %GroupListReverse = reverse %GroupList;

    # check if group already exists
    my $GroupID = $GroupListReverse{ $Param{Name} };

    # reactivate the group
    if ($GroupID) {

        # get current group data
        my %GroupData = $Kernel::OM->Get('Kernel::System::Group')->GroupGet(
            ID     => $GroupID,
            UserID => 1,
        );

        # reactivate group
        $Kernel::OM->Get('Kernel::System::Group')->GroupUpdate(
            %GroupData,
            ValidID => $ValidListReverse{valid},
            UserID  => 1,
        );

        return 1;
    }

    # add the group
    else {
        return if !$Kernel::OM->Get('Kernel::System::Group')->GroupAdd(
            Name    => $Param{Name},
            Comment => $Param{Description},
            ValidID => $ValidListReverse{valid},
            UserID  => 1,
        );
    }

    # lookup the new group id
    my $NewGroupID = $Kernel::OM->Get('Kernel::System::Group')->GroupLookup(
        Group  => $Param{Name},
        UserID => 1,
    );

    # add user root to the group
    $Kernel::OM->Get('Kernel::System::Group')->GroupMemberAdd(
        GID        => $NewGroupID,
        UID        => 1,
        Permission => {
            ro        => 1,
            move_into => 1,
            create    => 1,
            owner     => 1,
            priority  => 1,
            rw        => 1,
        },
        UserID => 1,
    );

    return 1;
}

=head2 _GroupDeactivate()

deactivate a group

    my $Result = $CodeObject->_GroupDeactivate(
        Name => 'the-group-name',
    );

=cut

sub _GroupDeactivate {
    my ( $Self, %Param ) = @_;

    # check needed stuff
    if ( !$Param{Name} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need Name!',
        );
        return;
    }

    # lookup group id
    my $GroupID = $Kernel::OM->Get('Kernel::System::Group')->GroupLookup(
        Group => $Param{Name},
    );

    return if !$GroupID;

    # get valid list
    my %ValidList = $Kernel::OM->Get('Kernel::System::Valid')->ValidList(
        UserID => 1,
    );
    my %ValidListReverse = reverse %ValidList;

    # get current group data
    my %GroupData = $Kernel::OM->Get('Kernel::System::Group')->GroupGet(
        ID     => $GroupID,
        UserID => 1,
    );

    # deactivate group
    $Kernel::OM->Get('Kernel::System::Group')->GroupUpdate(
        %GroupData,
        ValidID => $ValidListReverse{invalid},
        UserID  => 1,
    );

    return 1;
}

=head2 _AddConfigItemDefinitions()

installs configuration item definitions

    my $Result = $CodeObject->_AddConfigItemDefinitions();

=cut

sub _AddConfigItemDefinitions {
    my ( $Self, %Param ) = @_;

    # This is needed to extract translatable words from the config item definitions.
    my %Translations = (
        Vendor                 => Translatable('Vendor'),
        Model                  => Translatable('Model'),
        Description            => Translatable('Description'),
        Type                   => Translatable('Type'),
        CustomerCompany        => Translatable('Customer Company'),
        Owner                  => Translatable('Owner'),
        SerialNumber           => Translatable('Serial Number'),
        OperatingSystem        => Translatable('Operating System'),
        CPU                    => Translatable('CPU'),
        Ram                    => Translatable('Ram'),
        HardDisk               => Translatable('Hard Disk'),
        Capacity               => Translatable('Capacity'),
        FQDN                   => Translatable('FQDN'),
        NetworkAdapter         => Translatable('Network Adapter'),
        IPoverDHCP             => Translatable('IP over DHCP'),
        IPAddress              => Translatable('IP Address'),
        GraphicAdapter         => Translatable('Graphic Adapter'),
        OtherEquipment         => Translatable('Other Equipment'),
        WarrantyExpirationDate => Translatable('Warranty Expiration Date'),
        InstallDate            => Translatable('Install Date'),
        Note                   => Translatable('Note'),
        Phone1                 => Translatable('Phone 1'),
        Phone2                 => Translatable('Phone 2'),
        Fax                    => Translatable('Fax'),
        EMail                  => Translatable('E-Mail'),
        Address                => Translatable('Address'),
        NetworkAddress         => Translatable('Network Address'),
        SubnetMask             => Translatable('Subnet Mask'),
        Gateway                => Translatable('Gateway'),
        Version                => Translatable('Version'),
        LicenceType            => Translatable('Licence Type'),
        LicenceKey             => Translatable('Licence Key'),
        Quantity               => Translatable('Quantity'),
        ExpirationDate         => Translatable('Expiration Date'),
        Media                  => Translatable('Media'),
    );

    # Config item definitions.
    my %Definition = (
        Computer => << "EOF",
---
- Key: Vendor
  Name: Vendor
  Searchable: 1
  Input:
    Type: Text
    Size: 50
    MaxLength: 50
    # Example for CI attribute syntax check for text and textarea fields
    #RegEx: ^ABC.*
    #RegExErrorMessage: Value must start with ABC!

- Key: Model
  Name: Model
  Searchable: 1
  Input:
    Type: Text
    Size: 50
    MaxLength: 50

- Key: Description
  Name: Description
  Searchable: 1
  Input:
    Type: TextArea

- Key: Type
  Name: Type
  Searchable: 1
  Input:
    Type: GeneralCatalog
    Class: ITSM::ConfigItem::Computer::Type
    Translation: 1

- Key: CustomerID
  Name: Customer Company
  Searchable: 1
  Input:
    Type: CustomerCompany

- Key: Owner
  Name: Owner
  Searchable: 1
  Input:
    Type: Customer

- Key: SerialNumber
  Name: Serial Number
  Searchable: 1
  Input:
    Type: Text
    Size: 50
    MaxLength: 100

- Key: OperatingSystem
  Name: Operating System
  Input:
    Type: Text
    Size: 50
    MaxLength: 100

- Key: CPU
  Name: CPU
  Input:
    Type: Text
    Size: 50
    MaxLength: 100
  CountMax: 16

- Key: Ram
  Name: Ram
  Input:
    Type: Text
    Size: 50
    MaxLength: 100
  CountMax: 10

- Key: HardDisk
  Name: Hard Disk
  Input:
    Type: Text
    Size: 50
    MaxLength: 100
  CountMax: 10
  Sub:
  - Key: Capacity
    Name: Capacity
    Input:
      Type: Text
      Size: 20
      MaxLength: 10

- Key: FQDN
  Name: FQDN
  Searchable: 1
  Input:
    Type: Text
    Size: 50
    MaxLength: 100

- Key: NIC
  Name: Network Adapter
  Input:
    Type: Text
    Size: 50
    MaxLength: 100
    Required: 1
  CountMin: 0
  CountMax: 10
  CountDefault: 1
  Sub:
  - Key: IPoverDHCP
    Name: IP over DHCP
    Input:
      Type: GeneralCatalog
      Class: ITSM::ConfigItem::YesNo
      Translation: 1
      Required: 1
  - Key: IPAddress
    Name: IP Address
    Searchable: 1
    Input:
      Type: Text
      Size: 40
      MaxLength: 40
      Required: 1
    CountMin: 0
    CountMax: 20
    CountDefault: 0

- Key: GraphicAdapter
  Name: Graphic Adapter
  Input:
    Type: Text
    Size: 50
    MaxLength: 100

- Key: OtherEquipment
  Name: Other Equipment
  Input:
    Type: TextArea
    Required: 1
  CountMin: 0
  CountDefault: 0

- Key: WarrantyExpirationDate
  Name: Warranty Expiration Date
  Searchable: 1
  Input:
    Type: Date
    YearPeriodPast: 20
    YearPeriodFuture: 10

- Key: InstallDate
  Name: Install Date
  Searchable: 1
  Input:
    Type: Date
    Required: 1
    YearPeriodPast: 20
    YearPeriodFuture: 10
  CountMin: 0
  CountDefault: 0

- Key: Note
  Name: Note
  Searchable: 1
  Input:
    Type: TextArea
    Required: 1
  CountMin: 0
  CountDefault: 0
EOF
        Hardware => << "EOF",
---
- Key: Vendor
  Name: Vendor
  Searchable: 1
  Input:
    Type: Text
    Size: 50
    MaxLength: 50

- Key: Model
  Name: Model
  Searchable: 1
  Input:
    Type: Text
    Size: 50
    MaxLength: 50

- Key: Description
  Name: Description
  Searchable: 1
  Input:
    Type: TextArea

- Key: Type
  Name: Type
  Searchable: 1
  Input:
    Type: GeneralCatalog
    Class: ITSM::ConfigItem::Hardware::Type
    Translation: 1

- Key: CustomerID
  Name: Customer Company
  Searchable: 1
  Input:
    Type: CustomerCompany

- Key: Owner
  Name: Owner
  Searchable: 1
  Input:
    Type: Customer

- Key: SerialNumber
  Name: Serial Number
  Searchable: 1
  Input:
    Type: Text
    Size: 50
    MaxLength: 100

- Key: WarrantyExpirationDate
  Name: Warranty Expiration Date
  Searchable: 1
  Input:
    Type: Date
    YearPeriodPast: 20
    YearPeriodFuture: 10

- Key: InstallDate
  Name: Install Date
  Searchable: 1
  Input:
    Type: Date
    Required: 1
    YearPeriodPast: 20
    YearPeriodFuture: 10
  CountMin: 0
  CountMax: 1
  CountDefault: 0

- Key: Note
  Name: Note
  Searchable: 1
  Input:
    Type: TextArea
    Required: 1
  CountMin: 0
  CountMax: 1
  CountDefault: 0
EOF
        Location => << "EOF",
---
- Key: Type
  Name: Type
  Searchable: 1
  Input:
    Type: GeneralCatalog
    Class: ITSM::ConfigItem::Location::Type
    Translation: 1

- Key: CustomerID
  Name: Customer Company
  Searchable: 1
  Input:
    Type: CustomerCompany

- Key: Owner
  Name: Owner
  Searchable: 1
  Input:
    Type: Customer

- Key: Phone1
  Name: Phone 1
  Searchable: 1
  Input:
    Type: Text
    Size: 50
    MaxLength: 100

- Key: Phone2
  Name: Phone 2
  Searchable: 1
  Input:
    Type: Text
    Size: 50
    MaxLength: 100

- Key: Fax
  Name: Fax
  Searchable: 1
  Input:
    Type: Text
    Size: 50
    MaxLength: 100

- Key: E-Mail
  Name: E-Mail
  Searchable: 1
  Input:
    Type: Text
    Size: 50
    MaxLength: 100

- Key: Address
  Name: Address
  Searchable: 1
  Input:
    Type: TextArea

- Key: Note
  Name: Note
  Searchable: 1
  Input:
    Type: TextArea
    Required: 1
  CountMin: 0
  CountDefault: 0
EOF
        Network => << "EOF",
---
- Key: Description
  Name: Description
  Searchable: 1
  Input:
    Type: TextArea

- Key: Type
  Name: Type
  Searchable: 1
  Input:
    Type: GeneralCatalog
    Class: ITSM::ConfigItem::Network::Type
    Translation: 1

- Key: CustomerID
  Name: Customer Company
  Searchable: 1
  Input:
    Type: CustomerCompany

- Key: Owner
  Name: Owner
  Searchable: 1
  Input:
    Type: Customer

- Key: NetworkAddress
  Name: Network Address
  Searchable: 1
  Input:
    Type: Text
    Size: 30
    MaxLength: 20
    Required: 1
  CountMin: 0
  CountMax: 100
  CountDefault: 1
  Sub:
  - Key: SubnetMask
    Name: Subnet Mask
    Input:
      Type: Text
      Size: 30
      MaxLength: 20
      ValueDefault: 255.255.255.0
      Required: 1
    CountMin: 0
    CountMax: 1
    CountDefault: 0
  - Key: Gateway
    Name: Gateway
    Input:
      Type: Text
      Size: 30
      MaxLength: 20
      Required: 1
    CountMin: 0
    CountMax: 10
    CountDefault: 0

- Key: Note
  Name: Note
  Searchable: 1
  Input:
    Required: 1
    Type: TextArea
  CountMin: 0
  CountMax: 1
  CountDefault: 0
EOF
        Software => << "EOF",
---
- Key: Vendor
  Name: Vendor
  Searchable: 1
  Input:
    Type: Text
    Size: 50
    MaxLength: 50

- Key: Version
  Name: Version
  Searchable: 1
  Input:
    Type: Text
    Size: 50
    MaxLength: 50

- Key: Description
  Name: Description
  Searchable: 1
  Input:
    Type: TextArea

- Key: Type
  Name: Type
  Searchable: 1
  Input:
    Type: GeneralCatalog
    Class: ITSM::ConfigItem::Software::Type
    Translation: 1

- Key: CustomerID
  Name: Customer Company
  Searchable: 1
  Input:
    Type: CustomerCompany

- Key: Owner
  Name: Owner
  Searchable: 1
  Input:
    Type: Customer

- Key: SerialNumber
  Name: Serial Number
  Searchable: 1
  Input:
    Type: Text
    Size: 50
    MaxLength: 50

- Key: LicenceType
  Name: Licence Type
  Searchable: 1
  Input:
    Type: GeneralCatalog
    Class: ITSM::ConfigItem::Software::LicenceType
    Translation: 1

- Key: LicenceKey
  Name: Licence Key
  Searchable: 1
  Input:
    Type: Text
    Size: 50
    MaxLength: 50
    Required: 1
  CountMin: 0
  CountMax: 100
  CountDefault: 0
  Sub:
  - Key: Quantity
    Name: Quantity
    Input:
      Type: Integer
      ValueMin: 1
      ValueMax: 1000
      ValueDefault: 1
      Required: 1
    CountMin: 0
    CountMax: 1
    CountDefault: 0
  - Key: ExpirationDate
    Name: Expiration Date
    Input:
      Type: Date
      Required: 1
      YearPeriodPast: 20
      YearPeriodFuture: 10
    CountMin: 0
    CountMax: 1
    CountDefault: 0

- Key: Media
  Name: Media
  Input:
    Type: Text
    Size: 40
    MaxLength: 20

- Key: Note
  Name: Note
  Searchable: 1
  Input:
    Type: TextArea
    Required: 1
  CountMin: 0
  CountMax: 1
  CountDefault: 0
EOF
    );

    # get list of installed configuration item classes
    my $ClassList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class => 'ITSM::ConfigItem::Class',
    );
    my %ReverseClassList = reverse %{$ClassList};

    CLASSNAME:
    for my $ClassName ( sort { lc $a cmp lc $b } keys %Definition ) {

        # check if class exists
        my $ClassID = $ReverseClassList{$ClassName};

        next CLASSNAME if !$ClassID;

        # check if definition already exists
        my $DefinitionList = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->DefinitionList(
            ClassID => $ClassID,
        );

        next CLASSNAME if !defined $DefinitionList;
        next CLASSNAME if $DefinitionList && ref $DefinitionList eq 'ARRAY' && @{$DefinitionList};

        # add the new definition
        $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->DefinitionAdd(
            ClassID    => $ClassID,
            Definition => $Definition{$ClassName},
            UserID     => 1,
        );
    }

    return 1;
}

=head2 _LinkDelete()

delete all existing links to configuration items

    my $Result = $CodeObject->_LinkDelete();

=cut

sub _LinkDelete {
    my ( $Self, %Param ) = @_;

    # get all configuration items
    my $ConfigItemIDs = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->ConfigItemSearch();

    return if !$ConfigItemIDs;
    return if ref $ConfigItemIDs ne 'ARRAY';

    # delete the configuration item links
    for my $ConfigItemID ( @{$ConfigItemIDs} ) {
        $Kernel::OM->Get('Kernel::System::LinkObject')->LinkDeleteAll(
            Object => 'ITSMConfigItem',
            Key    => $ConfigItemID,
            UserID => 1,
        );
    }

    return 1;
}

=head2 _FillupEmptyLastVersionID()

fill up empty entries in the last_version_id column of the C<configitem> table

    my $Result = $CodeObject->_FillupEmptyLastVersionID();

=cut

sub _FillupEmptyLastVersionID {
    my ( $Self, %Param ) = @_;

    # get configuration items with empty last_version_id
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT id FROM configitem WHERE '
            . 'last_version_id = 0 OR last_version_id IS NULL',
    );

    # fetch the result
    my @ConfigItemIDs;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        push @ConfigItemIDs, $Row[0];
    }

    CONFIGITEMID:
    for my $ConfigItemID (@ConfigItemIDs) {

        # get the last version of this configuration item
        $Kernel::OM->Get('Kernel::System::DB')->Prepare(
            SQL => 'SELECT id FROM configitem_version '
                . 'WHERE configitem_id = ? ORDER BY id DESC',
            Bind  => [ \$ConfigItemID ],
            Limit => 1,
        );

        # fetch the result
        my $VersionID;
        while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
            $VersionID = $Row[0];
        }

        next CONFIGITEMID if !$VersionID;

        # update C<inci_state_id>
        $Kernel::OM->Get('Kernel::System::DB')->Do(
            SQL => 'UPDATE configitem '
                . 'SET last_version_id = ? '
                . 'WHERE id = ?',
            Bind => [ \$VersionID, \$ConfigItemID ],
        );
    }

    return 1;
}

=head2 _FillupEmptyVersionIncidentStateID()

fill up empty entries in the C<inci_state_id column> of the C<configitem_version> table

    my $Result = $CodeObject->_FillupEmptyVersionIncidentStateID();

=cut

sub _FillupEmptyVersionIncidentStateID {
    my ( $Self, %Param ) = @_;

    # get operational incident state list
    my $InciStateList = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemList(
        Class       => 'ITSM::Core::IncidentState',
        Preferences => {
            Functionality => 'operational',
        },
    );

    # error handling
    if ( !$InciStateList || ref $InciStateList ne 'HASH' || !%{$InciStateList} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Can't find any item in general catalog class ITSM::Core::IncidentState!",
        );
        return;
    }

    # sort ids
    my @InciStateKeyList = sort keys %{$InciStateList};

    # update inci_state_id
    return $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL => 'UPDATE configitem_version '
            . 'SET inci_state_id = ? '
            . 'WHERE inci_state_id = 0 OR inci_state_id IS NULL',
        Bind => [ \$InciStateKeyList[0] ],
    );
}

=head2 _FillupEmptyIncidentAndDeploymentStateID()

fill up empty entries in the cur_depl_state_id or cur_inci_state_id column of the C<configitem> table

    my $Result = $CodeObject->_FillupEmptyIncidentAndDeploymentStateID();

=cut

sub _FillupEmptyIncidentAndDeploymentStateID {
    my ( $Self, %Param ) = @_;

    # get configuration items with empty cur_depl_state_id or cur_inci_state_id
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL => 'SELECT id FROM configitem WHERE '
            . 'cur_depl_state_id = 0 OR cur_depl_state_id IS NULL OR '
            . 'cur_inci_state_id = 0 OR cur_inci_state_id IS NULL',
    );

    # fetch the result
    my @ConfigItemIDs;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        push @ConfigItemIDs, $Row[0];
    }

    CONFIGITEMID:
    for my $ConfigItemID (@ConfigItemIDs) {

        # get last version
        my $LastVersion = $Kernel::OM->Get('Kernel::System::ITSMConfigItem')->VersionGet(
            ConfigItemID => $ConfigItemID,
        );

        next CONFIGITEMID if !$LastVersion;
        next CONFIGITEMID if ref $LastVersion ne 'HASH';
        next CONFIGITEMID if !$LastVersion->{DeplStateID};
        next CONFIGITEMID if !$LastVersion->{InciStateID};

        # complete configuration item
        $Kernel::OM->Get('Kernel::System::DB')->Do(
            SQL => 'UPDATE configitem SET '
                . 'cur_depl_state_id = ?, '
                . 'cur_inci_state_id = ? '
                . 'WHERE id = ?',
            Bind => [
                \$LastVersion->{DeplStateID},
                \$LastVersion->{InciStateID},
                \$ConfigItemID,
            ],
        );
    }

    return 1;
}

=head2 _DeleteServicePreferences()

Deletes the service preferences for the key 'CurInciStateTypeFromCIs'.

    my $Result = $CodeObject->_DeleteServicePreferences();

=cut

sub _DeleteServicePreferences {
    my ( $Self, %Param ) = @_;

    # get service list
    my %ServiceList = $Kernel::OM->Get('Kernel::System::Service')->ServiceList(
        Valid  => 0,
        UserID => 1,
    );

    SERVICEID:
    for my $ServiceID ( sort keys %ServiceList ) {

        # delete the current incident state type from CIs of the service
        $Kernel::OM->Get('Kernel::System::Service')->ServicePreferencesSet(
            ServiceID => $ServiceID,
            Key       => 'CurInciStateTypeFromCIs',
            Value     => '',
            UserID    => 1,
        );
    }

    return 1;
}

=head2 _MigrateDTLInSysConfig()

Converts template settings in sysconfig to template toolkit.

    my $Result = $CodeObject->_MigrateDTLInSysConfig();

=cut

sub _MigrateDTLInSysConfig {

    # create needed objects
    my $ConfigObject    = $Kernel::OM->Get('Kernel::Config');
    my $SysConfigObject = $Kernel::OM->Get('Kernel::System::SysConfig');
    my $ProviderObject  = Kernel::Output::Template::Provider->new();

    my @NewSettings;

    # handle hash settings
    NAME:
    for my $Name (
        qw(
        ITSMConfigItem::Frontend::MenuModule
        ITSMConfigItem::Frontend::PreMenuModule
        )
        )
    {

        # get setting's content
        my $Setting = $ConfigObject->Get($Name);
        next NAME if !$Setting;

        MENUMODULE:
        for my $MenuModule ( sort keys %{$Setting} ) {

            # Setting is a hash.
            SETTINGITEM:
            for my $SettingItem ( sort keys %{ $Setting->{$MenuModule} } ) {

                my $SettingContent = $Setting->{$MenuModule}->{$SettingItem};

                # Do nothing if there is no value for migrating.
                next SETTINGITEM if !$SettingContent;

                my $TTContent;
                eval {
                    $TTContent = $ProviderObject->MigrateDTLtoTT( Content => $SettingContent );
                };
                if ($@) {
                    $Kernel::OM->Get('Kernel::System::Log')->Log(
                        Priority => 'error',
                        Message  => "$MenuModule->$SettingItem : $@!",
                    );
                }
                else {
                    $Setting->{$MenuModule}->{$SettingItem} = $TTContent;
                }
            }

            # Build new setting.
            push @NewSettings, {
                Name           => $Name . '###' . $MenuModule,
                EffectiveValue => $Setting->{$MenuModule},
            };
        }
    }

    return 1 if !@NewSettings;

    # Write new setting.
    $SysConfigObject->SettingsSet(
        UserID   => 1,
        Comments => 'ITSMConfigurationManagement - package setup function: _MigrateDTLInSysConfig',
        Settings => \@NewSettings,
    );

    return 1;
}

=head2 _FixTypo()

Fixes a typo in the general catalog

    my $Result = $CodeObject->_FixTypo();

=cut

sub _FixTypo {
    my ( $Self, %Param ) = @_;

    # get the item data with the wrong name "Keybord"
    my $ItemDataRef = $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemGet(
        Class => 'ITSM::ConfigItem::Hardware::Type',
        Name  => 'Keybord',
    );

    # check if general catalog entry exists
    if ( $ItemDataRef && ref $ItemDataRef eq 'HASH' && %{$ItemDataRef} ) {

        # update the item data with the correct name "Keyboard"
        $Kernel::OM->Get('Kernel::System::GeneralCatalog')->ItemUpdate(
            %{$ItemDataRef},
            Name   => 'Keyboard',
            UserID => 1,
        );
    }

    return 1;
}

=head2 _MigrateConfigs()

change configurations to match the new module location.

    my $Result = $CodeObject->_MigrateConfigs();

=cut

sub _MigrateConfigs {
    my ( $Self, %Param ) = @_;

    my $SysConfigObject = $Kernel::OM->Get('Kernel::System::SysConfig');

    my @NewSettings;

    for my $Type (qw(MenuModule Overview)) {

        # migrate ITSMConfiguration Preferences
        # get setting content for ITSMConfiguration Preferences
        my $Setting = $Kernel::OM->Get('Kernel::Config')->Get( 'ITSMConfigItem::Frontend::' . $Type );

        CONFIGITEM:
        for my $MenuModule ( sort keys %{$Setting} ) {
            my $MenuModuleSettings = $Setting->{$MenuModule};

            # update module location
            my $Module = $MenuModuleSettings->{'Module'};
            if ( $Module !~ m{Kernel::Output::HTML::ITSMConfigItem(\w+)} ) {
                next CONFIGITEM;
            }

            $Module =~ s{Kernel::Output::HTML::ITSMConfigItem(\w+)}{Kernel::Output::HTML::ITSMConfigItem::$1}xmsg;
            $MenuModuleSettings->{Module} = $Module;

            # Build new setting.
            push @NewSettings, {
                Name           => 'ITSMConfigItem::Frontend::' . $Type . '###' . $MenuModule,
                EffectiveValue => $MenuModuleSettings,
            };
        }
    }

    return 1 if !@NewSettings;

    # Write new setting.
    $SysConfigObject->SettingsSet(
        UserID   => 1,
        Comments => 'ITSMConfigurationManagement - package setup function: _MigrateConfigs',
        Settings => \@NewSettings,
    );

    return 1;
}

=head2 _ConvertPerlDefinitions2YAML()

Converts Perl definitions to YAML.

    my $Result = $CodeObject->_ConvertPerlDefinitions2YAML();

=cut

sub _ConvertPerlDefinitions2YAML {

    my $CommandObject
        = $Kernel::OM->Get('Kernel::System::Console::Command::Maint::ITSM::Configitem::DefinitionPerl2YAML');

    my ( $Result, $ExitCode );

    {
        local *STDOUT;
        open STDOUT, '>:utf8', \$Result;    ## no critic
        $ExitCode = $CommandObject->Execute();
    }

    return 1 if !$ExitCode;

    $Kernel::OM->Get('Kernel::System::Log')->Log(
        Priority => 'error',
        Message  => "$Result",
    );

    return;
}

=head2 _MigrateZnunyCustomerCIsSysConfig()

Migrate SysConfig of package Znuny-CustomerCIs.

=cut

sub _MigrateZnunyCustomerCIsSysConfig {
    my ( $Self, %Param ) = @_;

    my $SysConfigMigrationObject = $Kernel::OM->Get('Kernel::System::SysConfig::Migration');
    my $ConfigObject             = $Kernel::OM->Get('Kernel::Config');

    my $Home      = $ConfigObject->Get('Home');
    my $FileClass = 'Kernel::Config::Files::ZZZAAuto';
    my $FilePath  = "$Home/Kernel/Config/Backups/ZZZAAuto.pm";

    if ( !-f $FilePath ) {
        print "\n\n Error: ZZZAAuto backup file not found.\n";
        return;
    }

    my %MigrateSysConfigSettings = (
        "ITSMConfigItem::Frontend::AgentZnunyCustomerCIs###SearchLimit" => {
            UpdateName => 'ITSMConfigItem::Frontend::AgentITSMConfigItemCustomerCIs###SearchLimit',
        },
        "ITSMConfigItem::Frontend::AgentZnunyCustomerCIs###CustomerUser" => {
            UpdateName => 'ITSMConfigItem::Frontend::AgentITSMConfigItemCustomerCIs###CustomerUser',
        },
        "ITSMConfigItem::Frontend::AgentZnunyCustomerCIs###CustomerCompany" => {
            UpdateName => 'ITSMConfigItem::Frontend::AgentITSMConfigItemCustomerCIs###CustomerCompany',
        },
        "ITSMConfigItem::Frontend::AgentZnunyCustomerCIs###Mapping" => {
            UpdateName => 'ITSMConfigItem::Frontend::AgentITSMConfigItemCustomerCIs###Mapping',
        },
        "ITSMConfigItem::Frontend::AgentZnunyCustomerCIs###Permission" => {
            UpdateName => 'ITSMConfigItem::Frontend::AgentITSMConfigItemCustomerCIs###Permission',
        },
        "Frontend::Output::FilterElementPost###ZnunyCustomerCIs" => {
            UpdateName => 'Frontend::Output::FilterElementPost###AgentITSMConfigItemCustomerCIs',
        },
        "ZnunyCustomerCIsWidget###Group" => {
            UpdateName => 'AgentITSMConfigItemCustomerCIsWidget###Group',
        },
        "ZnunyCustomerCIsWidget###LinkType" => {
            UpdateName => 'AgentITSMConfigItemCustomerCIsWidget###LinkType',
        },
        "Frontend::Module###AgentZnunyCustomerCIsWidgetAJAX" => {
            UpdateName => 'Frontend::Module###AgentITSMConfigItemCustomerCIsWidget',
        },

    );

    my $Success = $SysConfigMigrationObject->MigrateSysConfigSettings(
        FileClass => $FileClass,
        FilePath  => $FilePath,
        Data      => \%MigrateSysConfigSettings,
    );

    return 1;
}

sub _MigrateWebserviceConfigs {
    my ( $Self, %Param ) = @_;

    my $WebserviceObject = $Kernel::OM->Get('Kernel::System::GenericInterface::Webservice');

    my $Webservices = $WebserviceObject->WebserviceList(
        Valid => 0,
    );
    return 1 if !IsHashRefWithData($Webservices);

    my %InvokerTypeMapping = (
        'Znuny4OTRSITSMConfigItemInvoker::Generic' => 'ITSMConfigItem::Generic',
        'ITSMConfigItemInvoker::Generic'           => 'ITSMConfigItem::Generic',
    );

    WEBSERVICEID:
    for my $WebserviceID ( sort keys %{$Webservices} ) {
        my $WebserviceData = $WebserviceObject->WebserviceGet(
            ID => $WebserviceID,
        );
        next WEBSERVICEID if !IsHashRefWithData($WebserviceData);

        my $InvokerConfigs   = $WebserviceData->{Config}->{Requester}->{Invoker};
        my $OperationConfigs = $WebserviceData->{Config}->{Provider}->{Operation};

        # Migrate web service invoker types.
        if ( IsHashRefWithData($InvokerConfigs) ) {
            INVOKER:
            for my $Invoker ( sort keys %{$InvokerConfigs} ) {
                my $InvokerConfig = $InvokerConfigs->{$Invoker};
                next INVOKER if !defined $InvokerConfig->{Type};
                next INVOKER if !exists $InvokerTypeMapping{ $InvokerConfig->{Type} };

                $InvokerConfig->{Type} = $InvokerTypeMapping{ $InvokerConfig->{Type} };
            }
        }

        $WebserviceObject->WebserviceUpdate(
            ID      => $WebserviceID,
            Name    => $WebserviceData->{Name},
            Config  => $WebserviceData->{Config},
            ValidID => $WebserviceData->{ValidID},
            UserID  => 1,
        );
    }

    return 1;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (L<https://otrs.org/>).

This software comes with ABSOLUTELY NO WARRANTY. For details, see
the enclosed file COPYING for license information (GPL). If you
did not receive this file, see L<https://www.gnu.org/licenses/gpl-3.0.txt>.

=cut

JVBERi0xLjUKJeTw7fgKNCAwIG9iago8PC9UeXBlL1hPYmplY3QvU3VidHlwZS9JbWFnZS9XaWR0aCA2NTAvSGVpZ2h0IDIwMC9Db2xvclNwYWNlL0RldmljZUdyYXkvQml0c1BlckNvbXBvbmVudAo4L0RlY29kZVBhcm1zPDwvQml0c1BlckNvbXBvbmVudCA4L0NvbG9ycyAxL0NvbHVtbnMgNjUwL1ByZWRpY3RvciAyPj4vRmlsdGVyL0ZsYXRlRGVjb2RlL0xlbmd0aAo0MTg4Pj4Kc3RyZWFtCnja7V1/rBxVFZ7XmkKD8W1pahqIdKo1NUp5SyQlGqEjJCIV6LYpMWkjfbUiSkQfKkQhylZJIBrjFk0NQtItCQQF8ZU0aVMQ96WmTbWGfRRUSNVdCCmkCm+1SizWZ/v62u7ec+7Mndm5v3a/75/2zczuzpnzzf3uOffce4MAAAAAAAAAAADAbVSKOr99pI4nDKihtkznt3+8hicMgIkAmAgmAmAiACaCiQCYCICJYCIAJgJgIpgIgIkAmAgmAmAiACZmRqvYwBMGHGDixjIeMKCI4TCnLyoNkUPjRTxfwDSKz9JjF6M6ETCO+hC0GXAA5bugzf2B+Ztvfg3aDFjH7NrSl685IBwcGFp66QcWFM4J/nm4+cLe/S9BmwHtePTTQfDWDY93HFu+5vL3tP35x6ce3QttBvRi47en/vnmvWcODd2+vCBcdegX974KbQY0Ys3D0/+p3nR0Wpg3fHc+c+GBW39l4/6Y7Di0uRextDb71H/Hrj984p+zy7fNYC9tff1B8/c38kNoc1/g/P1tzd9U3HLW92+RXXz0tvtM319YH4Q290XYvG9J+59Hrt951rfulF9+9MaHoM2ABsx84rrOA8dGXtoxI+YDr680G0JDm/sE93yDHHr4gsviPvH06ha0Gcgbw1uYg3tfWxn3mc8/YFebN43Abz2HZbtmcYf/MvaZd8g/9NwnXrepzc3iBBzXa1i0Zx5/4o2t686Vf2ydsaCF02bMGOg9DO75oDRE3nJVKP3czlVvQZuBHMPm7Z+MOfuz910iO3XkI89Dm4H88KMvxZ4e+9dy2akbzYy0GNHmsFgMw3BBm91BfaLWaPS2752y+oubEy74w/61M/kzlVt7Q5vDKIoW8KdatVqtR5NFrll9xa6ZSZe88cgN72JPbL/WfW2ebP9j6zC9oFAaGYr/iuZoNYtbJoW/M48IiW/iWOSO1aRitFXIaOXifYPJFx396arzuOO/W2qAiIVGV9o8Ge/D4sg6lW9plqv+MlGn1TQRvXI0k5Hz9ixSum7LJUuYo88vMcDE0RVdaXMnHwY6T0Zl5WnYrXLFVybqtJq2E9tKWWycteMKtQvffvFC5qiJgd/SL7uLmzt90jFAGFZWpLmT5nDNUybqtLpKWteFWeKdBzcoXvj8hWxYHWknYpfaLPqkXTpGyoMpb2bryISXTNRpdfhX8citlfQmfvUHihc+dxH/Pqx3XZtFn5xhQ1jNsD7K+HDdRyZqtZpkNpph6l+47omZahf+afEAe/yOe1zXZtEnpzsx0ehglvtplWoeMlGr1TRmSZ3rXbJb8bZemXc2b+5VT7muzaJPTr2vbO2REtZX/WOiXqsnRB9xWaPYsHn/BWoXThyby5946WOHzWtz2tBM4MNAly5J8Sa4w0S9Vle+IjahYSrVmrVbMRn49ivvlZz53h3HjGtzSiuJT6aeaDcuCVpR3TsmarWaxizqunECD69RvPCFD8lu7srfm9fm1HnTSRrYdeWS4x34aMI3Juq1msQsqbJ7d96tegfSQaEHbpp0XpuJT44H3twE/lRQDN4dYqJeqynHU6QUVz+meOGLi2VnDn3qWb1EzEObiU/GIq6lDVr1WqNxpgQlDIvF4lCXXUWHmKjZahKzqCfaPrx7ttqFL58nnT6gO4WTizZTPgzQlna8yhaehKVhiVvU1MchJmq2msYsqmUQ5+87X+3Cv82QTh54fP0RD7SZ+mST8NSa1apcSorlFdlTOS4xUa/VNGZRbTJmP7Ra6br/HAplp55Zo3k6VT7aTH0ieCSp2iSqcC1EM/SMiZqtJjGLepvBTG9m8LUvvF9yZtdnNS8XVqgvyEObE3yysZLI7UJlXcaeorNMzN/qbmKW4ftnJV5z13eG7rucO/Hfn3/573qJSLseWcuNYnzSLCllBplKXaVhBEeZqMVqErOkKINYum1+whWPrA2Cc29f/25y4s+bNmtOaQfRr/PR5jifqOYFuQVEgzkTnjJRj9Wk4VDqv0zjgu3xda6/jabmkC773LWdfH9l9P4XNPMwP22O8UmK0VEmeFKIWZxkoiaracySpjzgnY/FzS599eLpQeWZS6++8uJTSZ9/7Nvx9IFAO3LTZrlP0tRWMi+Gwu24yERtVpOYJVUZxMyKfH7pW5e2EW7ueRctmnvO/468dvDA6yaWZcpPm6U+URYpSSCvkDFzkIn6rCYxS0qHbdgsiVuOrXoysAZOm9MNqif6RLmMQfbKKyxU5h4TdVpNYpaUHls2yrO8fdF342C0Ofs8Bd4naSvcaSud/A3uMVGn1cRpaSc53c2uG2tgWkA6bc6+bS7rkzSR3Uk0FqTuKDrHRK1W05gl5cyqBrcKwG+uPOqWNmeZpBPnk/RrmZD0WvIr7xwT9VpNpDzdfCO2VujgRw/bI2K+2sz7JMP30Qc14BsTNVtNY5ZCl24PWpcdsEjEfLWZ90mW8IeIR2LI4hoTdVtNYpZUGeAJWnl17JqdFomYszazPsm0dAuZYZ6odY4xUbvVpFVLkwJmCl6CW35skYh5azPrk0wLjZEuU+L74RgTtVvdVcxCl5IIfnKzTSLmrc2sTzLtWUDuLJFYjjFRv9UkZlEXM6Ys+pmrj9pkIrNtblfazPkkfTJj6lm96TcTDVhNYhb136RlZQcvadkkIlP/0e3aO9QnaSeGS74osRfkFhNNWE2iDuV2mFRb/HvooE0iMjmlLrWZ80nGgUMxjEzkg1tMNGE16eSr0p92Mdc+YpOIGrSZ88nChhk+uMVEE1YTQqmWQZDA6MkVVomoQZsZn2TrMPnORDNWk5hFsSUmTVDGYlSHtZnxSVZy+81EM1aTwEPtZ2lbWrBJRE6bc9g2dzKv7/SbiYasJjGLUqeA9C/tbvvEaHMeqydP5tR195yJhqzOxqn0I6mGtTmXGyI+ybqnkN9MNGQ10VmV/ilxvd2Nu/VoM+OTORP9yERTVpPEoELsQUb6uk6YOKjNjE8Ggn5koimrSSWDQhkE6VzOsbgnoy5tpj7Jms7wm4nmrCZ9vkRaZSGvf9pMfZI5Q+k1E81ZTcQtUWqzCLp/2gwmmrY6dcxCiixsJhPpsGN+gTyYaNZq2sQleJJkw20mE5ltc3PSZmYlwZF+ZKI5q2m3L6EMgjjfYjKRWZYqv4ySPT64xESTvy3GLPFlEEQPLSYTuS3t83svwETTv006/bHjO6QZyrzah9vaDCaa/23SysV+hmRN7CUTtWozmGj+t2nMElMGQRLJGWvLNWlz1kFSMNEJJpKYJSZeIiUT9pKJjDbnGsaDicaZSGKWZpjHpea1OeW2uWCic0wkMYu0oSOzWK0lE3VrM5hog4kkZpGOJJMynIzTbdzXZjDRBhNpzCILiMUyHGvJRO3aDCZaYSKJWSRlEOQ6W8lE/doMJlphomogotx2+q/NYKIdJpKYhR00I2U4tpKJzN7XeWszmGiHiSRmYTlGCGApmchtm5u3NvcEE8UBMQ+YSHSXLYMQL5JouHYweyBpyCb1ABMzFBtaZ6JKLEIazhzLDdKAWUU0f20GEy0xkcQsTEqRZE7sJBPNaDOYaIuJJGahNMvQ6/BXm3uBiemqrFxhIrlrklIkV9hJJhrS5l5gIhmb9YKJpKkhwydiq5l5a8bctVlPCO8/E8lL6wcTyW2LKUWxJ2knmchos6b51v4zkfS4/GAiYZrQ+SI1sjqihCzarKtt9p+JpJjUEyaKb1AzjDVLON1r2twLTCRjop4wkUQknfIsNplWkonmtLkXmEgWMPKEicTNHR1BEofZSCYa1OYeYCJdIMMXJop+7lhmRKyRtZFMNKnNLjExY7NP31tfmEgEuN3LYktvI5loUptdYmLGt57uZOgNE8WYpU2eY9tLQ+C25tOY0/SeiXRBP2+YKHYs2ugmirOFZCK3ba7OsjR3mJjttWfWUfOGiUT+zjhaFGcLyzIx2+ZqXUPUnk9IY5ZphWGmnNgfJor6d7rlE8W5Gfa6Ntv0CckDZhpEYGZY+MNEMWY5LQuiOJtf4920Ntv0CYnMsjxubpFTj5goliCe8rUozuaTiaa12aZPyHhxFkuZdZ99YqI4Z2p67FkUZ/NrvBvXZps+IT28DCELl3v1iYmiDE/3B0VxNp5M5LRZdw/Bnk/oa5e+H8I1iV4xUXwIJ2NkQZzNJxMZbdY+yGPRJ2R7ntRJM7ZJ9IqJYswydSPSkNqmNne/ba67PiFpnNQ9EebV9Y2JQswyVbktmmU6mWhDm636hCyElfbHmVfXOyaKMcuJ9UaEdnK689jb2mzVJzQpna5R5LXZMyaK7+Px4ESs1jadTLSizVZ9QpZ9SdcjKtSGgh5gouD3409ADMMML8tkR5vt+oTZjTBF+MzULPnIREGLj0ux8FhMJxPtaLNdnzDLQ7Yixd65tEX0jonCU1g48WbmlzMPMNvmmtBmuz5h5FmVisWqlIi+MVF4CuuDzu6z6WQiI1RmOqpWfUKj5+NPvlRTaEfKg0GvMFF4CpsKnQ/F8BrvzFiBoYkLVn3Cp2ESbyGqDMWd9o2JkmTUNMwmE7kt7fOFtOLKrk+Ymq4TvfZyNeYzw8PCh8aH/GYimc/SYZzZNd7rQ33KRFlz0KyM8p3kUqkk6vK28rOeM5GJ3Az30aQPpm+YyPYUpxuDWq3ewcZiWIyYJrRZnJj0nIlc5HYKhpOJ/ctE2TjJKZqd5mJxUN6PEiTFOybGvI+mk4n9y0RumYFUOFG6J+S4/WOiPGYxvcZ7HzNRUk+jiqnbFTIP/jFRGrMYr0zsZybGSFMyTo5TC+2qh0yUxSzGN4zsaybKB5ATMR1YCuLmIRNlMYvxmVT9zcSsrWJrpMoa4SETJY/A/IaRfc5EfjpKEsaH63w3y0cm8jGL+WWZ+p2JQVRdkPb3NpUnJL/tIxP5mMX8hpF9z8SgUE4XQo+N1KVtqpdM5GIWC8sygYlBEJbVe4vj5c40W+c8BC+ZyMUsFjaMBBNPcrE0qHLdtkottpflJROZmKUZBmCiJZ8USqWkjM54lauNmPSfiTRmMZ5MBDrIGEWRrDKpVatJSnR6AWTl+oUN0ME2ojAsFgpthGw2Go16vac9Q2pUzScTAYDrJ9rZMBLoe4jibGfDSKDvQRbEsLNhJND3yGetZwDoEmSV5maIhwJYAKkXtrJhJAAgmQg4ATKbx8aGkQBAy9aRTASciFeQTASsgNSsI5kIWAEp2EYyEXAiXmmGeCiABZDiByQTARugEweQTASciFfMbxgJAFy8YmEmFQDQejDEK4AbTSLGVwAbIFPuMb4CWAmcyaK6SOEANkBqH9AkAjZAF3dGkwhYQLE2iCYRcKCTSLfBRJMIOEFENImAE0Q0vr89ALAbA2MtHMA42I2BUSELGEZU5nZxxYqJgAs8PLEZJp4NoAnDUaPWaLTHKVFUkmy0YHZTcaC/cLIMtnWaY0X5UuKImwHtTFQBZpYCTjBxPEInEXCAiSAi4AQTQUTACSaCiIATTBwrgYiAA0zE0ArgAhObwzU8JsA+EzdWoMyAfSZuLTfwjADbTGxWq+AhYAjRSMSPNDdHa1j+BjCKQrFYiNpLH8aCWo9vzAoAAAAAAAD0EP4PudtXiAplbmRzdHJlYW0KZW5kb2JqCjUgMCBvYmoKPDwvQ29sb3JTcGFjZS9EZXZpY2VSR0IvU01hc2sgNCAwIFIvVHlwZS9YT2JqZWN0L1N1YnR5cGUvSW1hZ2UvV2lkdGggNjUwL0hlaWdodCAyMDAvQml0c1BlckNvbXBvbmVudAo4L0RlY29kZVBhcm1zPDwvQml0c1BlckNvbXBvbmVudCA4L0NvbG9ycyAzL0NvbHVtbnMgNjUwL1ByZWRpY3RvciAxNT4+L0ZpbHRlci9GbGF0ZURlY29kZS9MZW5ndGgKMjI1Nj4+CnN0cmVhbQp42u3dfYwcZR3A8eeZ2b2X3h3tcdIetQWBaEtEsRoUIzEhsURM0ASNojFKiP+JgRheFUMM4Q9DoonBGIwvxJBARBsVocSiBoIRARUpFQoUWvr+em2vey+7OzPuXIEgGoHdhTuOzyebSy+9bDrP/fHtb+aZnVgURQAA5pLEEgCAPAMA8gwA8gwAdKjy0m9Wn3u+FQHgzWXd2jWmZwBAngFAngEAeQYA5BkA5BkAkGcAkGcAQJ4BQJ4BAHkGAHm2BAAgzwCAPAOAPAMA8gwA803FEgDQLevWrrEIpmcAMD0DwJtHduC2ML6vCEVRO1hM7C+masX0kaI5HUKM1d7YN5T0DcXBkVjpC9W+YvJwGBwJk4di30Cs9heNqZkZthLSSuwZeH6iXfQZeQaADtq895Z837P53k35vs15K9LTR0JWb5U6hGLm72NI0lDpib2DycLR5LhT0pF3mJ4B4PWS7/15tntTtmNDtmtjMXEgZFmZ5Ji0Zuby9fxV3dZQnYf6ZDFdax7eHXc+kR1zXDJ6auWEVfIMAF2dmLffFLJG87lHsy0P5Yd3hzwPSRLStJyVXy6+8CWNrZE6b+Rj2/NDe7I9T1dPPrNy4uxHWp4BmBdD89gvi/pU44l7sm2PhcZ0md701TWuHKnT8ueLrNi/pV47kNcOVN/5EXkGgE7bnB94rr7+d/nOJ1vflUNzG8oT4CFMjTefvC9M16qnnTOLR+TGKgDe3IrxdcWh7Y1Wm3c8XrY5dpa2JA1Zvbn5wcaGdfIMAO3mebpZX39HtnPjzOavbnSt9SZ5s7nl4ebTf5ZnAGirZCOri2KwaNReuGmqG2IaGlPNTQ9ku56UZwBoI6WVBef9Kl364ZBnIa//r33abRa6qB3InnmgqO0vN4G3XvIMAK8ppQs+fVf1g5fHtDdk090p9Mx90tneZ7LykrbpGQDaqmnvGZf3fvQ7sdIbmpPdKXSShvpEc9v6MHEw5A15BoB2VE79Qv9n74mDS0Oz1q0Zuhjbmo1tNT0DQAdhG14x8MW/JsMrQ/1wKLKO85wU9Yls37Mzn9ctzwDQ/hDdv+CCP1VO/NjMZrGsozE6xtY75Id2FVOH5RkAOpP29n3y9p4PXFLu5S43i7Xdu5m0T4wVU0fkGQA6FpOeM7/Ze/b3YnWgo0vRMRSNqXzioDwDQHdU3/2l/s/9IfaPzGznbrfPWSM4uQ0A3UzdwpMXXHBfHDx+5jp0W4q8LLQ8A0AXxYHR9LjTywdmtD1Ad+vDyOQZAI5qbLi58cydIam2W8sk9A7IMwB0Tbbnkal7Lo6V/jYn4KKI1b5kwbA8A0B3FONbJ25fHatD5UOo2nykVRH6hmLfoDwDQDc0p2q3nxNjUn56dnttLj8sLEkG3xb7jpFnAOh8cM4n15wXJve3f8m5lIdqTzqyPPT2yzMAdGry7ouy3Q+HtLejd8nzZGAkOfakUNi5DQCdqf/zpuZTa0J1oN3rzc/P3yFJk9EVcWjxG/zvl2cA5pts6731ey8rt4N1eLNynsWhJZVlp4eevpBW5BkA2p14j+yY+PWnQrlVO3Y2Omeh0lM54X3JoqVv/FHIMwDzqM2NWu3Ws8pbnGNngSuKVtnTJe+qnHTGrBxIxe8SgHkizyZuPSvUj4S0p8PKhyKPw8uqK86OA8fOvHPT9AwA7TR1+v6r8oObO21zOTcXcXhp9bSPJ4tPma2DkWcA5kecd4WeWrJoWcjzcsd1m++StSblOLy8+p5PVN5+2iwejpPbAMwHsff46kkfShYMNzfem+3fXD7/MUlf9RXo8mx2+Ur70qWnVleeHfsXzu7hyDMA80Sy5MuhcmsytLi5+cHmtkeLI/vKSLcKHePMLu7/usmq/MDOF8PcUw7NJ6yqLF8VevqKxpQ8A0CXCj3y+TASYv9P09EVzR3/yvduysf3heZUyLKZETv+Z5tDOWH3DKQLR9PjVyajK+PQSKjX58KByDMA8y7SoxeFeHPPomX5xMH84LZ8bHvemqQnx4tWp/PsaJVj70DsH4qDi5PhZenwsljtbf3w3DkEeQZgPhZ6yYVhZv9zc+sPKstPL+qTeW1sZmgud43FJC1iTAdHivpU0clHl8gzALTTueVfPfqH9BWLPqf+e+E3B8AbqSgKiyDPAMwtV19z3Y4dO62DPAMwh/zt749cfOmVj67fYCnkGYA5YfW557e+jo8fufIb377jzrstiDwDMIc0m83v3/ijG3/44zzPrYY8AzCH/Oa3d11z7fWTk1OWQp4BmB1Hz2y/zEMP/+OSr1+9Z88+6yPPAMwhz27e8rVLr3hsw+OWQp4BmP3R+UUHxg5eftW1v1/3RwslzwDMIc1m84bv3viTn91iKeQZgLnltl+sue76G6ZnHhtVrzfkGQBeL///zPbL3Hf/Xy674lu7du1+K1+NlmcA5pwnNj514Vcufv+q98ozAMwhWZa9lQ/fAyUB6JrXdBK7W++2bu0a0zMAIM8AIM8AgDwDAPIMAPIMAMgzAMgzACDPACDPAIA8A4A8WwIAkGcAQJ4BQJ4BAHkGAHkGAOQZAOQZAJBnAECeAUCeAQB5BgB5BgDkGQDkGQCQZwCQZwBAngEAeQYAeQYA5BkA5BkAkGcAkGcAQJ4BAHkGAHkGAOQZAOQZAJBnAJBnAECeAUCeAQB5BgDkGQDkGQCQZwCQZwBAngFAngEAeQYA5BkA5BkAkGcAkGcAQJ4BQJ4BAHkGAHkGAOQZAJBnAJBnAECeAUCeAQB5BgB5BgDkGQCQZwCQZwBAngFAngEAeQYAeQYA5BkA5BkAkGcAQJ4BQJ4BAHkGAHkGAOQZAOQZAJBnAECeAUCeAQB5BgB5BgDkGQDkGQCQZwCQZwBAngEAeQYAeQYA5BkA5BkAkGcAkGcAQJ4BAHkGAHkGAOQZAOQZAJBnAJBnAECeAUCeAQB5BgDkGQDkGQCQZwCQZwBAngFAngEAeQYA5BkA5BkAeGWVl36zbu0aKwIApmcAQJ4BQJ4BgNfq343Ab1gKZW5kc3RyZWFtCmVuZG9iagoxNSAwIG9iago8PC9GaWx0ZXIvRmxhdGVEZWNvZGUvTGVuZ3RoIDMyMT4+CnN0cmVhbQp42r1SQU7DMBC884r9QNz1rr32SlUPCCrBrSI3xKFNxYkIceL7jNOkqmi5omizznomnpmEvigS44pUhIp5qE7DSBibBuUSfdrOHCRVLVh2VoLH2FCrp5Hp4ZN2dN/TahupBhaA+ndS5ZDrGdwfX9esbsxaUK0zs6BnQUXQEnPaY11RGWs8Z2DTcMZMnGkeT73hU8Pz3GPjbDrJvsbowBwdhWmscweLh81b/0yPPVwGd1P6braKhFQSjZTMIF3Pkw96WRwKicBiag5zCVWNOqkcWNP/WmwGVlslDyZ1khOTQS30KL6hyEkO9Mw08LNdHrWkdJ3QzXTManC89iKeZXSRj+IPKDV7E9QFyRmoKsHNZz0t/yjLCVcc/Cx+m+N/chJLsKy/WXZYHOpxjm2Jb8COMXZkPweLmxyWE3Z3P5S9pwsKZW5kc3RyZWFtCmVuZG9iagozNSAwIG9iago8PC9GaWx0ZXIvRmxhdGVEZWNvZGUvTGVuZ3RoIDkzMT4+CnN0cmVhbQp42u1ZyW4cIRC95yv4gSkXUMUiWXOIkkjJzcrcohzGPTOnWFFO+f0UNNCbZ8nFWdyy2s3QFPWA18XrQv1QWqH8aeWN8i5CiKp7UlLtLFj0OubHjGAoWC/FjfMQtU6t7j4+oXr3XT2otzt190GrAGik0e6krEXg0BrvDl/u0UaHaL1c6Y6IRu5s5NJiRoi0l3KQi6Usv1naUtfaZJtcr/t7ak+pPZa7TjbbjeF4L1WPiDrKJbU6lLtYYbf9uvuk3u9klBCjs+pnGpY3QJ7UkyLnBLptNd/U5zpCo7QGa5xLQxQbTRY8+tHwMrTHAu/UD4FjgSc4+dCX07DI9W2zTRqmy8CSmwBOm5i8EDGYSGpjCKxgza7yDNZu8iyFOqaCc+ggwYwIhFxsNW03Mc9Q6oWz49pLP5cJcJ7zug6htCle86AOpT6B35/1Tk6Dk/maY3ATCysWPnC20Ba0jzJetBCZGui6gNtNiCg1Jo4AVgLsR8CPZ12EAES0cKG3G2ewlfSNpd+30Ku3F/AmlLE3Uk17YLRzqnGlmgSz/oU+DvyfvXHjbh0DOr/o9sUmZaXIn/cmJDI3ks9qQF6Qz7U4x6Ogy2XnSKGfh6C8DOLn3SLJJr30u67+a+AaETg0c675Fuh8US9d4VoSU24cArOGSUzsqsyyRcRRLE/2Q8sLQVIbL3rALzCtUfIv9vb8ljpTmyKPIVpfF5SL2mzyuAWqIpHztVD71xXlzI+/zv4gEjS4BmyhKBMgX9Rk14NNXx0NuB90fGJ7HoC7zPIWbae+V87+w9HW6hv5FmXxLc/51mSlORayY+Fc/bT1pX78rXi6kWdTn+vKvwKeWZTvjLDgmRvHtRy/cBS/jpPP97LLp5QJcWVd2+Ft2fdNeXpapl2GpMoSJgdg7xY4V7b8AW/Ps2q2g1sTIZohVVPSRcEN8anlsxqHSnroVHNZVxNTbSOfuruYybJswDAPe37NZI3DZ1e255q18jdAmPYbL2LwBInZFW6B0KK5Hsmb42X37MDzvEdaAHj2pQ8BtB1gLMRMKDnWkALA1VzFrLtVhb8ib8Hi7bwjlFcmhDnveLzZVN5Zvsq7WXcr71beneGdZjDD116Yi5x8IlIPVPaj05HD2a69gVSad72u8X/vLfQiaMK86WEgOjDRT04DS9XoOHBMpnyCiGCJxjrcnD+DEr2xMLn8KhAacGznVu6xCfVDkR/1LLSrwt7si1yXf+axenh48wvDFDMtCmVuZHN0cmVhbQplbmRvYmoKNDAgMCBvYmoKPDwvRmlsdGVyL0ZsYXRlRGVjb2RlL0xlbmd0aCA2Mzc+PgpzdHJlYW0KeNqtVcmO2zAMvfcr9AOjUJSoBQhyKNoB2tuguRU9ONupQdFTf7+UTMmyE8+GQeDIprg/Pkn9VUYB/4wKqIJPOiZ1vCoWe6stBJPKNoFGF23g1wcfdDIma22+XUF9+aOe1Oe92jwaFTUgK+0vylrQFJvy/vRzCzZ5ABv4ySsAIK+E/Bg2cwBu4PfID/E7fxPrumPTKTZFbsY167usD7KabLN7QEpbFh0ATOKHpSbKylZw3P3af1df91ylTslb9S+XFVC74NRVOe85ddskv9WPWiEqY7RF73OJbGMg6OikPHa+YwnAdqyQcuZeqpAsKUk1tcIoOl50WJdOIs9VDjVXiZ+0szHW8DZpG2ILX2tsaWDqQtUmDV0K5+J+82gZuhApiWMM7NeLX5Sc3aXreM6t1FW7bVGwpVw9S/0rExBNm2SXpBsd7mPEqknDpOlCt1vyoDYHc3/5+3DuOsyrOZXyTdRkiKsGxtaNVSNcdg8ec7zlDNHhOUwQrYZolpjQNBpWhvc8Qc5tXMchvA2H++5f6l3ZdTR5PcSJf4WXK/2yy36ts5z6TF5m573uWvC5P8vu+mniqSMSSfpxLK0S7ZaYH8aC94WfIVRnv/huyLgxMk3McoIX93Hw9bSY8Kr4F1/tTBzPliXexbsffbWszGv5gX0k04Z5DcEE2sWbMytM/AhjIsUdSRNmo10onRt8bIeHjJhLsjNMmh/JrefD9Bzrkx+l9VSHhuDqrNR4jaEN9fvs7jXew+G3YDq/M5mOmMLs0hRRd2suGh9yTOfqBHBPDc4mptePGunGJA9NWjVxgNqTXVr5Q4PxJFdEvSraHOEgdOQ/bMf806f/S9/xGgplbmRzdHJlYW0KZW5kb2JqCjQ3IDAgb2JqCjw8L0ZpbHRlci9GbGF0ZURlY29kZS9MZW5ndGggMTMxOT4+CnN0cmVhbQp42qVYy47rNgzd9yv8A6PR+wEEsyjaAdrdRWdXdGE7yeoOiru6v19SpCTacYKZDgLDNs23qEMq04/JTBp+Zkp2SrGoXKb1fQJydMrpZEr9HLSyPrsEj08xqWIMcj3/8a6n3/6dvk2/vk3Pr2bKSltgertOzmkVcmd+O/990q5ErV2CC+9aawv3YOEyIOa19jM8Z7gCPMN7AF6/dp4qU+mG7sjvkV/z3aDMy5MN5QSkRWtT4AKqyXwHKb2+/PP25/T7G0SpSolu+olhJat88tP75GME112nfJ/+ahHayRjlbIwYIsgYnVT2HJ424QUoWnOk1ZvIUaC3V75uIqv+PL86yF/KobByZ5RxmXVbFvAZbKQAFF9IvKqcRwJ8as/MeeMMphfT7C+3Kd5I6nAmSrU833yd9z6FvkDdSw4+xEaByqB0XFhf6rzrg6StzB3omwELzmTMTJbaffMpbDTPI+qWI7Qyt4ychRWg+UVmIMwtCz0GrK7M1jZyI64aTRrrsvPYsOWmu3GGWgxYW04FX3j5lwuFxCxc4TK5sJdQtefg9EgnbpxeAmmkEQuopjjSEpKGWlJJWKrUB/ZXKW0vXEpaFgP7ZHcFdaVnY9p+5D1WlHc5ty2WswrO9i3W9nHbal8J+XDb2aRcimyv7OI2Ke/iZsrDzdi4HFAdaLPXLncZOaJNk282Tb67afLBytzT8JXKYJ0f3p5dIowNROVfTqPktzHj25IHIPm5a/EMV31jNmoDsPrVD0/Qw2OP9tUtsoWVqA833+cCdzLEsMdjx42pLhkjROhFUmW5lKrsIl2tXx19qcte9ppreJXXn/aB4ncbROqzwL4N123CXc9Cf9vo9vcLTFuJvY6uEW1bLvREr61M0olib9lBGQN+GksXZr7GGsVqLJw31oZ89b11vMiy665PLgQp1S/QYy3HEm+7o9XHcfbtGceWbjBwd2gJj2DPxqB83KNeGKj3aaB9gHZJzGeUPliEUBsnN1d6qwZEqyO+yI3/uCF2TQKRiPIVRGKriC25W3hU3Jg2QF2XygHqkrZbD29QN2xys9smpGWPxyL6tfke/t8CNj8/Dke45E/KebdDtGN86q34+dVPRquii8EyIQVWw+TrWUEEDx3Yc5a9xEI/vzwFW2PDvOEXiCHyakZYQXiskg0KDHhrF8ogPruFNc70jvKSjvJV1lM+yBqO9Gbtb82uJZtmbR5K/uDbW/XPEx/yY1RIw+fqN64u3kFHNORbhTRDdvA8gzbRx3gmH5subwffvZh8u59ZBnPEcInzQo3XjtG1xr48gg4XrUrG7bEjbiamqkqLKriI6XiWR6hWzDj90pDnuAdRtxmDnDyG8SHrId7Uc2Cfmq6jETpnJPbUt+5cq+6Z90wQSLGRf+xm12sEsizdtmhvlfLZE91eT5RebSc5+bU1B4llFbd8OQ3vYe/QeTnziHBlXDrKW9yh8dav7qkTlqtMy3PZZWKswsgh4fDR7CTTtVkqNGxjq7FVUlwRdRm2HVvWKHH39oEJsIkpNOh1Hic0Xhv1pg0QtaWPdMlpbA/ZmzhG+2jysZ31gjxDbhvbiGevr+pxwrofbaQuUFtQ8vOoNZNfzfJV6NhaujOt0td7Q3ES3Sb2E+E6JggCi81x2HQIsfxPUEAU+Aiw8HB4+M+NjsqWtPnrhknivxsJRE/KhoB/8EBP862nmTo0bpBVyiSweCuDnaPclfHaqhjcXiouLV/uLOYVvHeQrT1GUx5sx/tvv/wHkR8keQplbmRzdHJlYW0KZW5kb2JqCjUyIDAgb2JqCjw8L0ZpbHRlci9GbGF0ZURlY29kZS9MZW5ndGggNTY1Pj4Kc3RyZWFtCnjazVWxjtswDN37FfqBU0hRoiTgkKFoD2i3Q70VHRynmRoUN93vl1Io147tSw7oUAQKKYqU9B5Jy7wYNCA/NNGZyNmmbIazETOTJYiY63IA63yiKOoDR5sRi9fuyxnMp9/m2XzszO4JTbLgxKk7GSKwIY3O3fH7I1BmAIoyigQAJzI4GShhHsD3oicZQXSZB/H1w+hTY6odL7L4++IPKrHE7B9cyI9iOgBgliFWTColCob9j+6r+dwJSpszk3ktsKKzPnpzNp5Zrk6j5Zf51hA6g2jJMReIEoMQbfIKD5D3YgEQNbHeMurNgiJDRSnr/qToWJGmeq/dEwmPMYWshxBapNTOWABqcNGJxuVwlMOQVMrm5U7jHNr66O11t6OMqCcc1T40L/ipe3iVpOfWpFZYw9+E1eScWmxNZ1LASVPIFbB6+KxJDs1CR01nS+sw4pwxUCkTkhxIXvw1SVEl3yCpEeN1NMJG+JeZEhMmerlEf5MkoAKDtB5aZRcq4khSnNQ9TipkpQ/K2r3Qw53Qy7yf1Mdhsz76CQ1pAT38N/UxMnCbLH5nnZBuv04WTRqIVWfd/W2ygP4VFZfbLPAtqKAtKvw7qOBJ+8ypmH1LVKdVKpYtQw3oZru4daBv55omrrdAzr4Fk3wPq82RtCpao/g78r2ZO3dqb+X6awVsXY6z50pNk/fq6iWJhRE/MpIqbt194Z+sC4uQmtPNEA/OcqDrKD7cAbfXZ1z+3KGd8PzhD6c8ytwKZW5kc3RyZWFtCmVuZG9iago1NyAwIG9iago8PC9GaWx0ZXIvRmxhdGVEZWNvZGUvTGVuZ3RoIDg4Mz4+CnN0cmVhbQp42uWawW7cOAxA7/sV+oFhSEqiJCDIodgtsL0VndtiDzOTzqlB0VN/fymZst1s03SCwBEwCBR5aFISH2lLsu2+OXKof+QSuyQFcnGnB6di8eAxUWmnIwKH7JMe7iRBIapaN38/oPvzq/vo3u3dzXtyGZBVaX923iPEPCvv7/+5RV8E0ScttUZE1jqyFlKzgBgOepy1RD3W31F1w2nWaTZNTlNd9UPVR6up2tztOJZbFR0RqWhRKWWr1QpPd//uP7i/9uollCLefa9uJYaQgntwQUSH7mfJF/epe8iOCDyLVBfVhjBBDuYeknatfeDto5GqB6HYCKWNsPZ/894rr5RjscY8AflsbbG53o2murv2TPM7BXs7SY+ftT5OSFdnKryzlTxLK2qc+/Cm99naXqxjk7SQtREka6mORKb+pnB0i+ZND2i2niyIS/C6dpOc1qH00eys9d7OYtNaypPPRD3ALWhrzJQhUtTAxaLZ7SfYjOe7XeCwzplafMuVnTBVFzqGaEPPk9PEP8/jJi8G5byEp8r90fJ/sv+d66LpzG7jlM3xaLDOzzucCgin13L4sFxwXceiPw2UVoONlp/B4petjXqurC9cg8OWYfdLzOu5CijWrPBzGy8H51umPwutFCiRxoG26JXJ0bDq78VAfj22S4Ax6zyht8xxsqxmU8+Yp6aVJy7hn4Pqemilt3EBpKBTY8nXAOkXek2n39ZXU8MlIAWhMF87yLT4Mc3Tl4PMpMs7uXaQhwXgBfA8EkQpVw7Py+pSjstK7RKQTLr78NcOMluf6cX3RR8YmF5trXde7TVsg7befFWdOeo8ZQKHdrzRhu8yOMIQA20JJ9rvQYFkhixxSyBisjGBBPRAOW8MpG3yBwXCXg94SyBp7AwJHrKXrYHkcYFIANqUR1/wDsojB9CyMZB23pYZA8/AEQNkTG8FR+Ucx4XDEcjT1nBkyZz2/G1QOCFCiPFt4AwJRCKkVN4oW2rD9VH2qHCyAL3eM/7fhZOGBSIoEHjTaboMvY4TFkgb3k4mPwdlERJgChsnR39oMPiSRSSBL/lt4AwJJCdIvOmW8DDCiv/HLwb0fsol/fDJgIlW3ww8esufGMGH0D8ZqJz4SdwZOP7PJFSzJ00CMkj0j63kOL9LvzcHu6On+S39wRDoPz72Hj7+8R8f/BxyCmVuZHN0cmVhbQplbmRvYmoKNjIgMCBvYmoKPDwvRmlsdGVyL0ZsYXRlRGVjb2RlL0xlbmd0aCA2NjE+PgpzdHJlYW0KeNrNms2OEzEMx+88RV6gXju28yGtekCABLcVvSEO/dg5USFOvD7ONC27oF69VhUl47GV/H9tZpyk6VeihPahVHOqpUPr6XhOZi4MjJX6elsRsjSu1tyUCp1oeD18PmP68DM9pfe79PCJUgPM5rRbEjOCtpvz7vTtEbkXRK5WRo2I2WrNVsjCBFH21m5W1Np2reYrx5vPGrPa6VIPfxn+OGsaMdtN1v5opgMidStmpTZri8Lj9vvuS/q4M5XQe+H0e8iqGaRKOicpxYbON8uP9PWqkE1hbdqHQmqgpGlDWKHJlJhx2W4ky8vuR+G1203JZHfy8xyuTinWlsW8TGZeXkicOF5KHT5ymDHZC+kFnvKwX+HdB5IbUFFvIMe4QKSB2LzyBbIKDAqkNGjE3kACT5nWgbh4AhnTpYYFkrGDePPQwDxyN5t4AjlNW1AglpEQNk8gzxdBUYFUBGHXNGSJPWU6QlX1BtLDAmEiwOqZhqyDfY4LhAkE2RmIlrhAlKDm6glk3LskqshDVNwHLNtyENWbzRKXR8/A1TMlGQMLDEQoQ+nNGch8A6MlRMhxl3zCDJjz28AJCUQZ2HWJs8bFfb5KZSilewOJuykgnaF31/Sk3PK1FUSOm7spCTBVbzhx3z7KAsV1G3oAmZuMFwDHeR0Ykgp0163pV5CWv0LXmKCQqkJuzRtS3IWzdoVC2RvIPiyQQgrddbt6FXWYbyZr50NcOFwge7MJfNhTdBynuq4TW+htp1ILdHRNXfrtfNSIXHZa3hbO65N3tBnT66uj92m6c/a+gaw6DujR1lDzp4W0H2rv4q/W4/8xMnjfjRG0Jb3yv1HlcP2PAZ+myqta+86M4vjO9pODjqfVtYend38ArLb86wplbmRzdHJlYW0KZW5kb2JqCjY3IDAgb2JqCjw8L0ZpbHRlci9GbGF0ZURlY29kZS9MZW5ndGggODc5Pj4Kc3RyZWFtCnja5VnNbhs5DL7vU+gFzFL80Q8Q5FC0BXZvRX1b7CFx6tMGRU99/VIajj3pJotV6sYZLwxBMkWN9H0jkuIofA0xoP1iyBRyqlBq2N0HEycGxhxr71YEksLZmpuUocbYtN78fo/h3ZfwMbzdhjcfYiiAZErbfWBG0HJQ3t79eYVcEyJnK61GRLJayUq0YYIoN9YuVtTa9l9NV3YHnT6my+NUN31p+uh1bGOuN6T1ykS3iLFaMWksXtso3F3/tf0jvN8aSqg1cfjWYGUCyRLug6RkS+eD5O/waUbIhjAXrQ1hLKBRwyZihiIOkXB/vRGS5fStcJ92kyhaD3325apDsbbsTctg0n4B0elYQm06emdte7rQRCHH3n4heicilZt8JvJpcqhATHomcjoZN6+XHCkgZm+nIacptsVKdmL2rpucPFs03U012wzU9HZHQB2I/RfTp+rt+Tn5UcI6qZ0oJ2IEfCpQIr8g+N7nINv4cxNQKkRO5yPgv5jEzwAkrCAviu/mgSn+cnxUTSYrAzhvBBnzVWQngIhlhWDb/y6TybRbcPgJp21eA4Tj2ojQxfM/jwGuCFl1bYCTh3OXd308rsv6pzHJjwg7N4t57rFdwTEC5ro2kvIiwpVj3whwjiDIawNe/G3vxt+0RsiU1wb41sEuzUGn888IeEuJcG2ugNlP9btnOXyuBJzXFuabrnh61MF75BsALpEg1bWFfL5zHRo+0AkzINGpAJNHoOwLmKMKTebYgeGU1HaZn1HU35SoEza/wWns88x/Qey83jnVGSFIGfh0R/pRguacbL8Ap+7M949n49326ej3znXEGCE5M6RUz0Py83Km5aeZ+ZOJz0O62NVl4Zf8hUlanM71KB8hrDLUypdCmO+aAQI0CnDMF0bAL8jdlAXSyT7VvhaiukntJv22jmeYkKpAPdln2ldAzL865hFisgKV8j8h5vD8eSf59+7BLEGrQop0KaTlxYXAmFmlqFA5XSgRnI9+Z5QYTkCXxIu6Xi9j1pK0XTfKpZDxWKYh0/rasw8byGU9w7kZIywnqHghx51JNx71D5nJo4Hq4Y01mhXV/ODK2kVP3Fn3a260HM63G8bGGT1JeQHSfwyRRvGTQwQJkvKPo9LtfC3fE/ToBPTgYj0Jr6a9051JC0W38wwff/sOxzPQ/gplbmRzdHJlYW0KZW5kb2JqCjcyIDAgb2JqCjw8L0ZpbHRlci9GbGF0ZURlY29kZS9MZW5ndGggOTE4Pj4Kc3RyZWFtCnja7ZrNbtswDIDvewq9QFiK1C9Q9DBsA7ZbsdyGHZK2Oa0Ydtrrj5Kp2Oma1iqKZTWCwJAsURL1RT8UZfPLWIPysyaSiSFDyubm3khyYGCMNtdsj0AucZToKkTI1hapi8/3aD78NNfm/dpcfLImAZIIrXeGGcGnvfD69tslcg6IHOUpISKShJ7ksVLMIbqNxJM8XuLy7kXW3exlapmaboewyLsijxraUuZqRT5fStIW0WZ5JNUmDaUU3lx9X38xH9fSS8g5sPlduhUJXHTm3rgQRHXep/wwX1sPWXoYk8+lhzaBt96sLEZITrtIuLtaOXLT5svDtdlVICs5dKfqeu2KxN1OpKSbTIpAuu55jJc82iq+gm6Qn4O0ylSUiqrWtVOUDa+2w6z4W3qLx0G2QKx6YYN4HAwlsMEvBYwb4lVWwx4YLoGTubUQGGHII9cHISRIlhcEYa9bHqaJC/0jI2WwHBYEpcq6sUwHDMIMbjEsmtxW2+3gQFnS3EJBlGWjbixpkO8BI5aIxbRUMDvdbYued31gIoJjuyAwNY37l5CMEP1izI4STjaaDhBsLWB8ayYHT215CWkzvjfd2k7bA4MtOHxrpkdtJz9iZpT3je62eigqulX9OqB4C5HiaaAcHfIvBtbRcTnaoT9lvyeLfCnvyqi+1XrDv+OQCTieyMwYdz/UhX53aDeWda/KTW3K0maL+yFe63dDvM4GhUttE1F3QcsrgGkyk2q7sQNurG6G5+A6SxByOsOdD5ce9wn1QGcGJFom9JcvBzPheQY+1fHr9eFtpl7BYcRWuWJObYc1t8CblH0R4GctlJnwI0MIeZnwXwZWZ4CLfSAzQ858BjmC9Oqz3E2OlfNgeuuAbTzDHGEG1VvTqzyOfZb8oUypc6NLQZy03TeaPTsIp7pe+D//gDixLNIDN+NMqN5BPtXVxP8JNWn9N/0jNHqglM4wR5hbBTldIvzkAm0m2OwhWDqD3evxrD+kHUbysOk1PTqgB+shczhDnw/9mBNqJnAOQGfeE95hvBmp9et5pYepL58wuDPUEeqtylD3qTjEABlfzQQO6jd52vcx5NNo5Rzk4zjjGrhjfprHvIjH/Cxt8z/Q6Slr68DLePg1DcqszvHgcxpNOvI9Tf0EB4GdDlu0Zbeko39PAvJ/FSmXifloEYcEwfPDUmHbPhmqo0QvHmoof6mAuhwO+RVrsfC3rYXrd38ANca98QplbmRzdHJlYW0KZW5kb2JqCjc3IDAgb2JqCjw8L0ZpbHRlci9GbGF0ZURlY29kZS9MZW5ndGggODk4Pj4Kc3RyZWFtCnja5VnLbhs7DN33K/QDZvkQ9QCCLIrbAu2uqHdFF7FTrxoUd3V//1Iajj1OY6CTJsFkCmMgWSJHOkcSh6TCv4EC2o9C5pBThVLD/i5YcxIQzFR7tyJwLJKtukkZKlGTevvxDsM/P8Pn8G4b3n6gUADZhLaHIIKg5Si8vf16hVITomR7WomIbKWyPWRqETHeWL3Yo1a3/2qycX+U6Tq9nYayyccmj15S07nesNYra9ohUrXHWql4aVq4v/62/RTebw0l1Jok/NdgZYaYY7gLMSWbuhxbfoQvI0IxhLlobQipgJKGDWGGEh0i4+F6EzlOh2+P9GE3icl6+LtPVx2K1ePBpHigRhvk5PAdXuub0DX086DXZM760SkZqcreH4d679sPfdb+y7LcG+u4NF3n/pwO3u7z5exLKD4364tpspxlkG2L0WQUx8W4TDAXoKTrIri3dd3kJOXpHvZ55nPi+vzV8TipfX5pIL3Xp/IzSI4Fop395ZM8GoGRYM1PsIuH/12mj7V/wPjICW/HaaXsh0Xvxuwwj/BUoJC8BOHZ+9xqjruz7RquXh93T75I3LBjp++eAbZUIElLBZt9VZtePY37DEQwVohr5WHU73LiprDVb9wUzuCJq7XFlRLVJj7a66OZSZNx3L530zaDNPPSCMvzk/Y7jtwfnZKMEIVWAKQiZNWlAhm2a29ndxtGl6O7F7v7rvUs8EIEmOuSwf/GZ38OYCGIKH8RYCXInP8iwBYS4qKPc5l8kU7O8bMe88ogOb5GUtw/4d15DNvfqcM8ZhARiSHVsmQingioCCDzgoEKeRaibWudOGgzQKqAvISr/vQgPRPR+4ZVPss2qD6UNZtHThZIacmf9mm0fkzHJHezH2f8YxWoVdYA2uOUY77IZfhmEsB5Eo/9C3HMOX2fRZpSBKG8DtJ+jXhfiESJkCItmcRJFPwHR0w1Qk1LdrBk96hsn2YFLuW1AOPhHX0MmaRm9bTVW5tcyFl0z1Inad0ZRFWFRPzaiXJ3ewbwRApV0kqAP0OckSQBr4UfGW0kTq7K5p2UpO2aMr5+Qoav5DQkwUd57CknqJhXTsjZ9ZicrscejuHOb7jRTlDNZ1fc3nThjnsDrNouwtFiId9q/Q698Uh8eUksQKTyoGK9qBTR4meV+1ppN17oy61DHQ2ILaTRceU+l4dBvBtH+Pzmfy+j2gIKZW5kc3RyZWFtCmVuZG9iago4MiAwIG9iago8PC9GaWx0ZXIvRmxhdGVEZWNvZGUvTGVuZ3RoIDgyMz4+CnN0cmVhbQp42u1ZvW4bMQze+xR6ATOkKOoHCDIUbYF2C+qt6FA78dSg6NTXL6WTfOekRiMHOPicIBCkUORR/CxRpGh+GzKof2SCNcEniMlsH4ySPQNjoFSmBcG6yEGHKx8gEWWuq88PaD78Mrfm/dpcfSITAa0yrXeGGUHinnl99+0aOXlEDtpyj4hWe7HaSMUcovuh46hNdKz/i/K67Z6nyBQ6DX3md5kfa09Z5mZlJV0raYNISZtSKdZepXB78339xXxcq5WQkmfzJ5sVLLjgzINx3uvSeU/5ab42C1ktDFFStpAiCIlZEQaIrppocXezctZN1efGRe3KW9IZe1+XK9UUHbudctkBGskm+2p+NU/nngNf4Xkkh7yp/E1HqDp44Ms0GyaQV7DyvGAD6zgANgJ5uTAAuO097AfERXB6jpYPiFSZ0rZ9IPgIkXjpIPwYPYubeKa8rqxnv2MqLevNXqkHqJiA2C8bqEGORtm9a5YuMCwmcAvHApnrRmlXVxuH7lNkbVKaWzogbnAlRS52+1Or0QdhXDoIfpi3rs/4gOCYLsD4/ZrScByc798JCSGIXAAYcnq8xUSAYenhBTeZzSDfAwATOOQLAyD7hXJTxGFn9AAiBMGGSwNkV6/NvL77PkA0dUO5ADwOaB32JwscFh82xEHnCXGkIws+nWvIUHji+L3Cj6N+nR9kfJ/RzIDWnqfRY4aQapp0f9ovKww8W4pQEl8ew/jil5qPekQv38uGZSd+V/XU79qWN/qJ4Xkt4QXHqAO0wOB9egOtC7TEkNIMYcaTm+4YIDMYLeSAKbwyo9mBn+MJ90noXzx8e2zbji/b5TaoOdIcAIiDNMcT7tkCEARsjHMAECbZb/tGdYTFSaZpLeP/4NiWTVf92Tk+KwZxA68L4+PUtNayf+UMgw6exipyWH/pAToJeLKvCOhj63hBeuNJIM3xZHwKiOcUEXv2YM8Up6MyT1xkxeUlG0ZySdO9AeGDh4Th1QBxWOFGPQ0pHJS4K+lIjXsFViQXwlFzrrp9Sg29NHscbk1EKf5TMB0Vcqg5u/BjKb9pBX2+q3A12PRHUsdwPZTpirvIrmLTNNy++wtUud9+CmVuZHN0cmVhbQplbmRvYmoKODcgMCBvYmoKPDwvRmlsdGVyL0ZsYXRlRGVjb2RlL0xlbmd0aCA4MjM+PgpzdHJlYW0KeNrtmktv2zAMgO/7FfoDYUmRegFFD8M2YLsVy23YocmW04php/39US8n6ZzCyWqgXY3ClUyRsfhJph6W+WXIoP6RCdYEnyAms703KvYMjIFSKXYIViIHza58gESUta4+3qN599Pcmrdrc/WBTAS0qrTeGWYEFwfl9bcv18jJI3LQK6eIaDV1Vi9SM0GUO81HvZzm9d6prmwHnWJT5FTTrC9ZH1tK2eZmZV26VtEGkZJeKqXYUrXC7c3X9Sfzfq1eQkqeze/sVrAgQcy9Ee+16jxIfpjP3UNWD0N0KXtIERw5syIMEKW5aHF3sxIrh4/PF5fHrrwlLXHqtvjmZnaPq6tZTra5Het9zmfZAaoq17zsqt5ROTYcHVO3l5ovZdtapvK/muTBs4ZmKTZhj7votftej15OtRka5NPgbATybgF3NjiJIPqeLuDOBecjROLXCG7X5K2eGWSJtNzqtq1+DlG3+TQVbExA7BewTw3WYgJZuD49V5tUJv8P2CIrtr7BCYdzoqpb6ngALD+DXQupDWapn6+wS17ODrNWZ4uEcYE7C9yAIEwL3FngJoTgnvuUtC+DOtj2u08xsyo65VnbkeUXNxtX60ua8rY2dlnO7SaDZiLAkBbQs4NmAkFeQM8O2hEEG+YGncu3+x2QHl/z72jwqvke/8JJYDXmHgKb6GSwgLOHx7t9/B8ap/o4jBWDXS5LrfP4ocPt9VLrjN/3+mMdaxKjx+s2mWGywEGeP0Rk2+bbj2zJTe5gE/Vs9zNeNIALWfApLnDH4f7b9okwA1q7wB2He7eHOhWoY+D5l/cvFCj7gzDgWn020+EGBu/TAnccbt9LCZfF2cSQ0szTystG6WkOOBJgCi/YARbwc392mtUBJ5Dm/vxz+USvvb+2vRt9A6IHoeNPjtOdDg5sjM/V6QlLpql6fQdIzoCTHHiyC5wxOJ4cpLk/sbxYOOzBLmzG2bh8nEFeA5zzZ+A+eEgYXjWc41MwqG9SCkfHYJroxDmYFVjn8mEZ1LVM62blnA2VvZbTTaCLSYqjhumkkaCu7x0/tPKbfuiHv7UxuY/N2nCK+7oCKKO2/rOb/oTbN38AnLil2gplbmRzdHJlYW0KZW5kb2JqCjkyIDAgb2JqCjw8L0ZpbHRlci9GbGF0ZURlY29kZS9MZW5ndGggODA2Pj4Kc3RyZWFtCnja7VpNb9swDL3vV+gPhCVF6gsoehi2AdutWG7DDouznFYMO+3vj/qw6zQ14HRJETRGYUimSVnvvUiiVZk/hgzqH5lgTfAJYjLdg1GzZ2AMlMpjh2AlctDqygdIRNnr5vMDmg+/zb15vzY3n8hEQKtO651hRnBxcF5vv90iJ4/IQa9cIqLV0lm9SMMEUX5oPerltK73Tn2lG3xKTLFTLbO/ZH9sJeWYu5V16VZNG0RKeqmVYis1Cru77+sv5uNaUUJKns3fDCtYkCDmwYj32nUeLL/M1x4hK8IQXcoIKYIjZ1aEAaI0iBZ3dyuxMn59vri8duUt6ROnsMU3mBkeV6jZTrbBjvU+19U2h7riM8RibcvW++JTaOsOaZ3rZ3u6QyVTqW5ETpNjI5B3CznPkiMRRMfbFZCza/f+OIJ8hEi8EDRJUExA7BeCpgiymEAWfqb5sUltcqkE5Xoar+3N7qu/bTlCiZPaL9k8zQfmk6FZDmG8QDIGdbP6P2v8XrszAQYEYXrDABNCcO7tAmQiwHCJKQPH0eQkNY/Pz/qxyi1n78fmXMBMIMhXBNgRBBuuCLB+ZuFrDNncr+Jra2x+VlaYzaGdtS65/W1tS2pbL1vJZvKQLHCQhQghCz7FhQhhBrT2vETUlSkP2z6bjP+ZuM0E5xj43Kl56XhRa/eYEZbYcZY53kkaZ5nZHtp81o3S8UP7WYkKDN6nyyXqlbb1JvtwVvITQ0q8kH9S8vsRJPMTX0cCTGER4rRCuJbG7epXyVwxWMCfe8v7+sQIrU/cMHdtR0Me/eYK5ATSubfdr0+g2Kat7qhvdhcc2BgXMU4rxmZ/w3PI1bv5wiQHnuwizEmFYR7xc9wGrCcHif0iyGkF6fFK61fbJZ8rCnuwiyYn1mTbfOxRnyPe5aMJsogx2YeDffbnt21mEx48JFw+O0b/iJSGcVOJze0MM8tL1/IRLomP3GUe94XaP7GDOjOlsHdkp5kmzuyswDqXD/YgsLRhVM4EUVFn+ufAgBSfDUyTQYIWvOOnUX7TH1AqkwA1eXKpPyKl67YSUgRwOdHp33D/7h+3Hc0WCmVuZHN0cmVhbQplbmRvYmoKOTcgMCBvYmoKPDwvRmlsdGVyL0ZsYXRlRGVjb2RlL0xlbmd0aCA5MjU+PgpzdHJlYW0KeNrtmktv2zgQgO/7K/gHPJ0Hn0CQQ7G7QHsL6luxh9hZnzYoeurf3yE5ejiJUwlIm9YxDEHUcChyPpPDGUnuqyOH+iOX2KVYIBe3v3cqjgKCiUqrDgjssyQtbmKCQlS13n24R/fnF3fj3m/du7/JZUBWpe3BiSCEPCpv7z5foZSIKEmPekZE1nNgPUibeUR/q+WsR9CyXgfV9ftRp7Vpcurnqu+rPtqZapvrDYdypaIdIhU9VErZztoK99f/bD+6v7ZqJZQSxX2rZiUGn7y7dz5GHbqMkv/cp8FCUQtTDqVaSBkCBbchTJC9mch4uN549vPu6yGt201k0pqgZvtoZlbzpJta5cRmdu7XtVxlFVnYT7LBbK1bgrXpzNs1nVkfTX/AGFp9bxM7OMVq0E6D4AwUw+8Fol9XA0vX9f9OEJYa7jN4XTfnY7iEXm6LLNm5yg5dLsVkhz6LfF4OK2bIJK8Hq4Gq96n3HI22dpw7kMmb/FpLLBcgiT8a3npAyMmAm2tuhto9uFg52wxKCwEOs2/YBh5sB8MMrH3XPmQOOhxvEQsBMxbwb4XvqXHwwWyhLl/KjovK/EvCsz+vup6maysnlNnA86TPtxO4QXeEtHTF2uQa7uEfTKrmAbhPTg1yOpy9/SGH70PSSIowXyA9CykheKELpGchFYQUVoVbtmu30Kbu+DsbUOrAfDqWexusjxMYfzeDFGe+1nwsz8PywX/R5MsHv1XrmvwH+LV63z7O70IUIsBUXg/io8jiWYgLjZ9vnGtgCIFHucBoMAJB4nSB0WBoKozh3FisiM/l1lxRmvW5kmFhkOTfGESJUybZ9onYx7sCnCeGWPIF3GpwIoDM5wbOnkcEA9JCOp6BWrcsfRCQdSnf7wApH6fNa4AkgRjPLiTaGYwHQXiNPdfAKQKlnFuIVLOQ8aHDPCvKPXNaASiQB6FzC5tewBkH8RDXvSZ4I2CChxLPLbiUfPyg4SVApQCczy0OkjvT49XJeygBIvEFyAAkUoAiv1oso0uhyZdCaADjE29/LdBbuSNFicAXJsdMQn3F/Yq5aJ/lZCkQTXHY2ueBa1bWUjgpQsGfHcKE4SWMGT8+ZH3KnbBF9xXE0LdBGevt3j89yxj7fforC9TVWNLRZxYmOvGdxQY4hPoxBmqeZlO2fcdBzdLTf6cmv5SfbFhONvLIEIM8bBV3w0clj+et1kS86q8M2mytj452Qw83f/wPktJaRAplbmRzdHJlYW0KZW5kb2JqCjEwMiAwIG9iago8PC9GaWx0ZXIvRmxhdGVEZWNvZGUvTGVuZ3RoIDMyNT4+CnN0cmVhbQp42r1Su04DQQzs+Qr/wDl+7UuKrkCABF3EdYgiD1IRISp+H+9mL4KQtOjks8+ekXdmDz6BgfxhSAIpFswFtgfwdlRUSlzaOBCKZU1eDjFhYa6oxeOB4O4DVnA7weKBISOJg6Y9qBKGfAJPu5claYlEmjxqJiLxHMSDnWZEtvY6ewSv/Ts41rYnTOO0Ph9zxVvFU89cOeMgoSy9tSHi4uFdzj07i7bj6/QE95OrxFKiwleVlQQtGRzAYvSj66nzDs+zQnWFKYdSFXLGwAEGpoTZukSh/TiY2M/1NbStHaKwT0LpMvzcXKVpl+N22NuZtNhz6tZo4/yTlUfT6s5Al02jiFLSL9d664ptA0oI1VtCNTu61q6lXZHMW/7areiQi8RylWQkGIOes+Jm/kd016XOkv2S3EqfyLqb4S/ZzBtWN99VJqwDCmVuZHN0cmVhbQplbmRvYmoKMTEzIDAgb2JqCjw8L0ZpbHRlci9GbGF0ZURlY29kZS9MZW5ndGggODM5Pj4Kc3RyZWFtCnjatVbJjtswDL33K/QDUShS1AIM5lC0A7S3QXMrenCcyamDoqf+fqnVdhK70xRFoMiWSepxeaLUT2UUyM8oj8q7qENU46uSZUeawJuYPzNotIG8PO6c19GYJLX/9Arqww/1rN4f1P7JqKABRehwVkSgOXThw+nrA1B0AORlpBkAUGZGGUbULIAd5DnIYHmWdxZZO3aZrJPXTZmTvE3yUGeTdB53yPFBlo4AJsqQVRPqLFowPn47fFYfD+KljtGR+pXc8qitt+pVWecEOvWV7+pL8xCVMZrQueSi6BjwOtjqnmzyKCsA8ogvE+rsRUL7ktHVvau9qC2F0MxR1ORDN9cwT2ZDDZzMaLOp/RNJ2H3gWI2gFxuu2qBTxdCiM7bo4FDjJn94zKYyAs02NuUWbi5p6CF2xamWpiGlIzTDKcSmGNxpZC5GuTmVvlKd0xg6IK47pPlcE2+ynclMx4bDJM2zsuDxtiu0iPs8ZF6zlP4OEXTklkqQXBlx0YyLaqJcPTup2+SKSyv9LS7exhyEFpKUSd+/5QDJgKyxhiulkqxugcPYkcyxZYtvxlbQNGzl2zq221WBYSOUxkWBbeUc6PIljzLHQgkUoHxVlqXUbTsQYIsmGKwGpkua8EQTO9GE+F9oko8srsiMuazq7mQp5cLVWs6uxTSh4TijTyjPcG4S6TzMq2M9LULXdasEaKHi6aTJp4wv79n6lG+eUe44We/HCCxpShXrhChZbhbJTada+8puhXiy4Wm1YqJ0B7Ujx9qhm3PPLtAvqj2/uSpTVriWwZjlNhhFPmrjJ05lcHNe2QWD37Kvr5kf7mQLRdQe7X9ji0XZEv0lW1xnS8pmMmWF91yP1QzjtMEc/8cGk4tkqKZcLpBWdn/XevotoLaepJbXQ70BDNctqWy4aEsuN+hbm+VUt54UL4tcWD/DnSl2riP0DnVJn7nvvmRw0dvaJcbf2a1EWW5nV93KbnSrYVG5oZ7zUx+gUhSzaG0yiZF02GhOdqM5rUMpm9Pie77G3UkuJinWeG8rssfbN0VwGqNfXBXr0uyuOEdTW4aXOwZZO+8Z6QTCVS/EW4nPbcVVJQuoHdOlluvxvcnZNWbKDs/vfgPeap2kCmVuZHN0cmVhbQplbmRvYmoKMTc0IDAgb2JqCjw8L0ZpbHRlci9GbGF0ZURlY29kZS9MZW5ndGggMzcyPj4Kc3RyZWFtCnjaXZJta4MwEMff+ynysoM543MFEfqwgi+2jtl+gDSenVBjiOmg+/TTO1dGBcVf7nL/fy7nbcptqVrLvA/Tywosa1pVGxj6q5HATnBuleMHrG6lnQm/shPa8TZvQr+LDpjnfgvjimEAO7g/6qpubtOr8X+vQVVCDW5pxaWVL9Y2z5xz93jY+QmroaEih5sGFsxcbqvbYKErVdOzPHcY8z5HzcGaG1us6v4ET9Pa3tRgWnVmi+OmwpXqqvUFOlCWcacosJxPrmVfw6CFBCPUGZx89MALlu/Gp3BA1Q9xP6Vtp0Z+CYPp6zGd82BZIG2IMiTfJ3olipBCn2hJFBNlRAlSkCBFIVLIibZEFIvnWEoUIUWkkJBCRAoJKUQropSIfCbkM46R0oCIaqZUM6YqaYztmM/t/3Xh3rWYTMYB1d3O2RQPHpoWhJPR3Zqj0TCbj5v9l5h6P83SfZTk1ZjxBnHgcDama2wV3GdS93rahe8vv9DC/gplbmRzdHJlYW0KZW5kb2JqCjE3NSAwIG9iago8PC9GaWx0ZXIvRmxhdGVEZWNvZGUvTGVuZ3RoIDM4Nj4+CnN0cmVhbQp42l2S3WqDQBCF732KvUyh1vU3EUSIJgEv2pSaPMBGx1RI1mVdC+nTV2dMKBEUv/HMzBl3nLzYFLI1zPnUXVWCYU0raw19N+gK2AnOrbRcj9VtZWbCZ3UVynLyd6E+xBWYY/8IbYu+B9Pbv3KQN7vp5Pi+VyBLIXs76y71mzHNK+fcPh52bsRqaKjE4aaAeTMXm/LWG7gWsulYkliMOV9jx97oG1us6+4EL1Nsr2vQrTyzxTEvMVIOSl3gCtIwbqUplnPJc9XV0CtRgRbyDFYyeuApS3bjlVog66fv47yYdmqqb6FRno1yzr1VipQTxUguJ9oQuURbpDFhoiAkiokioi1RhuRTlYCq+D5SyJFC6rekfuGGaI20ipDyAAeZHUd3/4953QBlLjnwXXJAQY9KBBT0Q5IsyYGHwYCUARmJSBls7xIM0hwhpYdrMunNtsjI83/1/KnuLuNY0I9nI/H/WabjmZbtsWvVoPV4yLiRuD7TSbcSHkurOjVl4f0H/kTJbwplbmRzdHJlYW0KZW5kb2JqCjE3NiAwIG9iago8PC9GaWx0ZXIvRmxhdGVEZWNvZGUvTGVuZ3RoIDM3ND4+CnN0cmVhbQp42l2S326CMBTG73mKXrpkjFIZkcSQIEjixeYy9AEqHByJlqaUJe7pR89Rt0jCn1/5vtOv7QnyTbFRnWXBh+nrCixrO9UYGPrR1MAOcOyUFwrWdLW9Ej7rs9RekL9J/S7PwAL/WxpfDgPYwf9Ro7r4ba+m760GVUk1+J9wHE/SvFjbPnPO/f2uDGPWQEtVdhcNTFx5U1SXwcJ5o9qeLZceY8Fk7wZrLmyWNf0BntzY1jRgOnVks31e4Ug1an2CMyjLuJemWC6k2HXfwKBlDUaqI3jLKQNP2bKcrtQD1Tz8DznZDm39JQ3KV5Occ7FIkXKiBElwpHmBFAmk1xIpJl9GymSNVORIWUhEyixGWkdIqwVSGRORr8wx7DVVdMv4tyaShQWlW1E6cSMMSZOJ4hbZvaKMkkeUnAbj8DoZlZ8/7Ijg5SQTrryjufOWKx46mie0lij5n9dts+ube9vUozHTYWFzYRu4E+sU3PtP99q58P4FrxbA2gplbmRzdHJlYW0KZW5kb2JqCjE3NyAwIG9iago8PC9GaWx0ZXIvRmxhdGVEZWNvZGUvTGVuZ3RoIDQxMD4+CnN0cmVhbQp42l2TwW6cMBCG7zyFj6lUBwOLSaUVUsJ2pT0kjbrJA7Bm2CIttmWM1H37Gv9WDkWA9Hk888+Mx3l3Opz05Fn+7ow6k2fjpAdHi1mdInah66SzomTDpHyi+Fdzb7O8e+3tWz8Ty/m6OL786R1xT3/nkY9G+4UbS9rfLXG7Xm6T4rc5vLPRphDc0XW99e7R+PG7EIJ/fhwLyQYaEfcjuLEy8elwvi+e5pMeDdvvM8by3yGNxbs7e3gezIW+bWu/3EBu0lf28Nmd48p5tfZGc8iCiaxtY7gChSgz0GJ7Ra7XV8r2IQfRsv0xPG1GevjPXiW3y6hCmdv2ogvbhZBFu1FZgUrQc6RdBXqBDVSVoBrUgCToKVKVbMmvAR1g+wk6wvYUaScilQdQAfUfoKSXSIJeQNCTHShlnaIk9URQl1CvoSCPIChU6ESNTtQCtIvUJIJeiSgSWTfomUTMGl2SiNkkqqGALknUUEJdImYQylBm9EuErJs6Hmk6u+1wt/n9Gl+1OhdGJA55HL5tTiZNX/fAGrt5xe8fX+nVNwplbmRzdHJlYW0KZW5kb2JqCjE4MCAwIG9iago8PC9MZW5ndGgxIDExODAxL0ZpbHRlci9GbGF0ZURlY29kZS9MZW5ndGggNDk3Mz4+CnN0cmVhbQp42u06a3QT15nfnRk9bMuWLMvyQ4BHHvygflt+ACFY2JYtY2P8JBpDQbIlWwa/sEWM2RBsCC2RSyE4IU02JSFhuzSlyRgIdVI2hTTpts1Jttlu222abtJN2rQNeZyUsClgeb87GhuTx5625+yPnuN7NTPf/b7vfu975545AgIAUTAKLOSvb84rPL7u6K2IOY+Xu+P2AB9zISEJgCzC8SOdA129d3rOxeIY6ZqzXT0jnX+e+eEagIgBgEi/3+fxflz95w6AuP9E/hKKMPD6IQCTAcdL/b2BnZvvVqFGUxEqfbCnv8PzwHNfrwRI+QnKy+/17BxQ5ZNHADKQHfg+T6+vOZD6Mo55ANV7A4O+gZaxoxMAOSiPRADLbSOHQQVa1YMqG2JSwk/2FehkUAATpVazKo5huDeAed8O/EZEZlLRBWuaK4AAPzOtTgmZ4C7tQcbFA3mY0jin6hzVDqzaBAlQj6ME2KE6p3oJbmqcE0xwD8DMJTq6cQ+Z6J3Oun7vzDuh/aHNoZMz74A5dB7+pqYNP8ZhDH4NX4ND8FV4AA7AGNHbqze2ia7Wluamxob19evqatfWOKurHJUV5WvsZatvXXXLyhXLS0uKC/LzcnOyMzPS05YKqdaURFOsQR8THRUZodWoVRzLEMjmJeJ2SGwaH1vlERyCx5mTzTsS/ZU52Q6hyi3xHl7CB5cuOJ0ySvBIvJuX0vHhmYd2S3bk7PwEpz3MaZ/jJAZ+FayiKgReeqlS4KdIW6ML4YOVgshL78rwOhnm0uVBNA6sVpwhW0Wt5R1S1e3+oMONNpLJqMgKocIXmZMNk5FRCEYhJGUKA5MkczWRASbTsXKSAW00VYueOjxeqaHR5ai0WK1iTnaNFCNUyiSokEVK6gpJI4vku6npMM5PZl8IfmXKAO3uLJ1X8Ho2uSTWg3ODrCMY/LIUmyUtEyqlZbveSkTPfVK2UOmQsqjU2qY5PbU3VBJJlWYQ+OBHgO4I7166GeNRMOo0w0dAQYmpkEiTy0qbpQpjHQxWCXxV0B30TM2Mtgu8QQhO6nTBAQeGGxpcKGJq5plxi1T1FVEyuP1kpai4XtVUK8U1bnRJTFoV7/cgBn9lgnW5xRo7x9PweWTAsGBwMMJWKw3D+JQd2nEgjTa6wmMe2i2nwZ6XJUqMm1IuzFLiWylldJYyN90tYG5rm11BiUur8QoOjPi4Rxptx+raShMjGKSYKxarEDTG8ivyRJmXR6tqvN28pErHIOGs+ROwbuiUoEEexFwJP961oIL0WCO/QkAxVI5DcLiV3+3+RBTAY6CdWeFCaHFJ9koE7B4lY47J/Dyc4XFjwror5WRKecKAZBLK57JLzXJ0N7vkKco0yVQh4W6qzJLyHPK64h1Bd2XYBCpLaHQ9DbaZNyaLeMsZGxSBWEmZzRVYZemOoMvbKaW4LV5cd528y2KV7CJmWBRcPpGWHUZo2RsWuThEuVZaXLXNQm1jm2u5YkiYQMVxaY5PiBFclrAYLEBJm6blXYyFFZHRgAi+CgGhfBXeJU2aFi8DBlzG0sItX8W7iAVmudEMaRnv8FUqfHR8k1AVLacK56w0NR2inAqnxSpawy0nm0EyryjGGVoaVOcsCbcpJGixPiucMorGMpEWPe8SfIIo+HnJ3uCivtHwyFFWgiHHXMlVy02jecHCMIEVybMDGkypKssyP7hStTyeGzo/Qa6ZJfNBrVDbHKTCBUUgoOU1EtASti+Ptch7AV3QAu69vAGXtLygg5N2O13M/pVUiFDjDQrNrlUyN+4nuy27qC4j1JLalvKcbNzayicFcqBx0k4ONLe5nsYXJH+gxXWaIUyFu1ycXIo019P4GrXLWIZiKZIOeDqgkppwoJX5LU/bAUZlKicj5HHHFAEZp53FEeiYYsI4Q1hRuqzIDgxSuDDFPsvNIU4bxo3KOLlNAg2ZPVJl19oj7DommrFMEoo6jZhn8LQQQeCMjkQTyyTOapLRU2R0MsJuCXOMIoc9bOGB1huqW9tcZ3SA0+Q7KiqnDcsl0Y/JxteKg/fSQrlD9AfdIl1sYMbU4I9IRFiNaRJWoyFqnRQp+MqlKKGc4ssoviyMV1O8BkuUmAlOH8XcN0iEVsBGlxWXJJ/8I0vQ8C7NlIibStDw2xw07iKeHSLxfMCCBnRnOFYDkFeYZyvIj7PGWkvxusjWXP+OnxmZ/rLq3NW1fu539P1PZiwhE7tPnYInky/YzbrkiMUGc5VogOTF1WKyHlRVIligLAsSw1cs2GJRJpuaUWy2FZaUxgllTEmJrdCcoEkXUtUajU0jsGq137Hcnl/XlPTmxeiXE3IfeyQzwGfV3XrbFxcnvZf07u8zHg2Z4g+a7xjiVlaUe3/4TwkxY4s9Yz17rp37lyeoTdAUOs+8jb5EQeQZlo3MgLIy9MNgJDaGiTXECUzTB0T79SvXBvdNEANpZuwkLvTh9MnQc6Fvh6ZJpOyXO3SeXEYZkShDy3JhGcTAEAFiDazNSC7/KXTtIfLB9n1HWHfoW9PfC33IZDC7SSshoT+hDVkzbyLrh2iD7qkIdr0YYcYwFOSXFlFv401qITU9q7E6q7ihMrdk7bqNRcXVW/D4SUgenhT3y3oN5zgNiVCTRFRtXLGiIN8qxBlLMUYZ6EfdqedvWVwonm4gF+9j7yGq9uKVfXvkAzHqbpi5JJ8XF0GOPcGCmTSuE/Eom5CQXCcmqDW6WlGTpCQl1ggrEvPkB7onW8YJSnaKi+ScZMRZNVa2hNlw7hdHQlcnL9594vHv/XxiX35729oXmIes01f23MbufuX08csj6090Xjh150R04grm1AOhmgQ87MJ+tKVN9SIk4xkXrYlOVMVa14mxsYkqVWStqNIkWmrFxM+0Jq7IaFxqK5QrA1hbSThwDI2cqrAErRNSY5h4k3n/yMtkyVR/jVDZ+R+h1yu1tzy59cGz+9+aOPev5NDmicZfl9/hzh8hJ18hDxFhIuEec+jnoVfXb37h1PHQRwd/J/Y8u/3jFVsG2+8+Fo7d0ZlLqki0NxES7DoT1IgmLlpTI0bHKSZilFI5qtZWWBqvRmOwHBQz2SeeCL17/PvfGv/Ohof6Pgxd+inx/uaF+x1Ble250N2//u7rK+5LWUSyCENU5OKlhoMOWV8LgHoKc2WCJLsukoljNTotC3qNEcpstjKbMYEmnhXUag0rEFucjViJYDSWCoQ58Q2tJsvw7KOk+PJ9EcwEiQy98sSdX7Kop20qg/Pa2uu7Veeu1XFnr65l6pnhdVffC/uXgfkQ0L904O2GFF08G7+oRozn2IgakY2blwZaDiYshfS5WshITy8uWhpeq7QwMAbmhCUMe0/o7Zmtx6sfPf3os92H7t9BtL/o+NaW8vITji89lrq2e6KrmJz86R8Wpz7ecv9d9cONrY8HvvtOeu7R9Ozx4ZJ1lbe6vCOyXc/g/nMW46ABsz2SGDUsx9aIHAvynpFYRquf4D6ksZK3T5H40FYr+8slIb8q5f77qV8MtKNfbm4TGHEnyrCbIjkdqPTaeEjGHcigi6sSdeYb+1BZrLygsN4TbKU2VigVaFUVFxlL2MISY7juaY6ZPZsOZ0ZV7PnD/mTtmR+MH/vq1WWeCzsHNvrPdG7mNnV88dS1qVd++4NTPyHeH1+3PE2W3de161To1ccGt8/ZxJ5FmxIhFdLtpkUxVeKiRaDRJFWJGgMY0TLzzYVfkK8KmyGXOmcVNCZ5m8RCi8OVGM4DuTT17/ddufeBB0NnQy9auX8YK9vfsflwUddT+bq3/lRjevXieSJe2D989qWGfRNL849kC+xvzv/z8PbiKrSJnRkLmbg6tCkTiqECnPZMi82GdmlWVIscH8Fq0m24B8dnpturxXRDJlMtZurjs6vF+Jt2cVofiXl5sVidWTQ/GEqNJryrZdC6SEgwy/ZTZ0pKwzd6x2VCTLPllKFWhwfKRqPm6pJ+uuwP3y9dPF5etuOpx0Kvhf7nudcOH6/v3FjW8P4z9XXTa/eOnnupZ+/DO0eJua1h/e1lrqb6HWs457bE1e7vvaktrl9m0caY7xqd+vHEl7ofLFnW4OjI4nc3P/ex6epH69r2BFY7YwxJd7LGVrFlYkPThsZ7MEe3YKJSVK/jey8GYp7SaRg9iccN9/nCgnybEGdjqdHh7frNk388OWK3uSqyM4s4J0lcXt1WWldAaxdlqL+MMc3Gyosz63Vx0Xqw6uIz1HHqKjE5zoBLOVzFuKLDlWeNm3Ual3Uchjz8Epzd5CjyBgMzda9xdW3LwyNW38k/jpsjb/v2pu27orQbnuvYM24+/Pi9JsfaxmPDKduYs5uXVdXvfP3roTTO+eKwf29g98iBvZV9027mrP8Lq2sDPzsS0in2csvQ3iRYbI8Jm6uLp7ZSU2mGZTM/z0pqHvPLe03V9eI3xxST3GfceyYSDqGezLK6nf91TLZgl29893rftFtZCx9wi7kNEI3nAdxZIZauS00iVp75xs46t8HTNUDmliHWiJHZ9+xrh6/844VXj3xM9u3e1vvtLXuG/d/4IrfhNy+cD50+99/PP0vWnwltHzp0b+jPx/cdOkKiw/sd+sr65X3BYo+OwbexPipePpYoroY9JeHXzJyjqxnmA4dtaXHRxoeHF22TXXRu7uI2xRyN3vmzvaFI9G5Lcw6V3zZzifmVqhLfb5h7g9Gui9cbjfHVolGvhipRPbfA6cEnOdHwUkF+mvLiQCfl849NQ9+vdNmQ8VIx1btk937jnlMxT+9uLyXG5ND7Y8aVO5LHYkzDt5e3MG019W9d3zN94rac5atk/7bjvreZc+Ieg/7pjZGJ+uh4I/pnVDTjK4T6R+XTpVYaF97uZvWT7x81Trge35vq/cbb42Zd++kto0fMRzjndI1YeGjnG8fIr65Nvbi7fXxsdR9zAmZm6HmPexTPe+lAv9FpoI7ZJ9uxAZNcLJ8fI88QFsNNz0s2YiMbSOnJ0I/omVHmwyvpQZqaLfpVH0FE+DvSjzqcF+Tn72tfvs5N/yril9p2rJkIYJTPTThPe3D6JJ66R65zoUVIB3Jm/vcopoR7CS4yYzOWv+TrFeOEJsY54/4UXg1Zc1/S/PQsNu/Lmh8a8Nr/134p456Ao38Jn/o0PQt8xnwOMtjr9P34V+rdD+3zx+yH4THHzIzJvmrp2pin/yrcwh3Da/eNeWxWmIf5GrQhbTsstIW20BbaQltoC+3/tXEHZyxM3kIcFtrfedsg3xdB52f0A3Byrj8v99+FO8kmd31uP/8Z/fKNztgZ9+f2x5V+mbnM3oH9mNKvcPlcj9wfwv6ayrTQF/pC/7vr/EJf6H/nvYh+72LvhyCY4TZQAQMGyINNACQGDgIrf0VLJqvlJ20rlC9r9B6BozDMgBrKFZiFNHAqMIfSmhVYBTroVWA16GFEgTXQhO9movzVywSvKnAUvsffU2Ad5JJIBY5GeLkCx8AXSCe1kovA0SA5rMAEjAyjwAzoGJMCs7CW4RWYA54pUWAVJDJbFVgNS5i9CqyBE8wjCqyFTLZSgaNgOdujwDrYxJ5Q4GjYxIECx0ALt6myu6s70L3L5+W9noCH7+gfGBns7vIH+G/yhfkF+Tl4K8jmq/v7u3p8fEX/4ED/oCfQ3d+Xu37A18c3e/qGagKenu6ONUMdvj6vb5DP4ecofJjEl+/o7vHyBfn5G3yDQziXL8gtyKdclCknzHRjUvcQ7+EDgx6vr9czuI3v75zV7unz8r2eEb7dxw/6urqHAr5BNLu7j+/wDQY8+Ny6Y7B7yNvdQe0byp2zaJ7V/kBgYGVe3vDwcK5HIXcgNbejvzfv/6IFRgZ8Xt9Qd1cfepDrD/T21HUjyxDq3yHrCPh9/JoBTwc+FEo2P+ttYW7+fNkyV27/YFdeT5hzKK+upsJR3+zIQU6ohG7owiuA1y7wgRd4vDw49iDUAf0wgJU5KHP5EcvDN/EqhHwowCtHgQogG7HVyN2PfD0oh4cKhAdxNr17ZPn90Ae5sB5xPoR4XAkefA5BjaytBzk6YA2OO2S6F++DyJWD16fn8DfN4nG17UCoR7afWpaPZ186f0jRS7G5MmVW1qyknJskfZambvlO4xGQfaGW9eJzELYhrh/P0p/03SPbz8tcI/hsl7GDeO+SpQVk28LR7pa1dcgYakd4vBX9GZR5vbJds/EbQi8+HaPPjjXNVwCxK3EHy4Nhueci/ebZHcrcXBnqRc6/dV4AfR2QvfLJce9C3nAOcmWZvRidOtkbn+xJ2P8d8/wIIB+N1BqU40G+8OjmObTSPpnbQtSQ/7l235CVK9vchdSem2QOIaYO66ACHFCPmXfIlU1lgv2BCKuTb+XJktYUJ9uasoTol5QteXIJu642PaWu1pZSW5WWkl5kaE2zLW1NiptJ0XAzKWp2JmVtjS2lBmlxNmOrirCtnA1ns0TPlrFPsmx1VVLKO1VESLWlLLJZWs22+NZYom812PStev16PZOi/zc9o9fP6Bk1Q6CV2KC1H/bAk/A+cAYgo2aiIlPk8GRLc1ZW7ZRmpqlWimjYKJEDUlozvdsb2yT1AQla2za6Jgn5qrj/4EEoX1wrFTa7JH6xWCt5ETAsnjRDuTg0lJW1eSiwI4u2QNZQIGt+k4eJm/8X7YViKwplbmRzdHJlYW0KZW5kb2JqCjE4MSAwIG9iago8PC9GaWx0ZXIvRmxhdGVEZWNvZGUvTGVuZ3RoIDI3Pj4Kc3RyZWFtCnjaY5DwOMDE0MzAseJpAwPZgIMEtUwAk94DxQplbmRzdHJlYW0KZW5kb2JqCjE4NCAwIG9iago8PC9MZW5ndGgxIDE1Mzk2L0ZpbHRlci9GbGF0ZURlY29kZS9MZW5ndGggNzQ1NT4+CnN0cmVhbQp42u16C3hU5bXo+vdznpn3TJIhyZ7sPIh5Z0hiEMgQyAMSQh4kzgDNyZBJSJCQkARpoJ6ClKLBqBW1lvb2oCJNOR6dWA5GD6cttfiqSq32erweK72t1xdi67GUAtmc9e89kwRt79d7vnu/7/b72H/23ut/rbX+tda/1vp3BggAGGEXsFC0urWw5KFtDyZiywm8O7tuHpEc+cn3ApB5WD/aM7ix/1+GD2/D+vMA4pmNm0d73vnXpDsA9BMAiZ293eHIH3/9XAJAug7Hl/ViQ0KO7l2sB7Ce0ds/8uW3vun4GtYjSPTg5oGucLMUTAUowG6xqD/85UE+nUkDKC3CBmlLuL/7RM/OdKw3AfD/PjjUPfi8/vJNADfkYf8FYHkLcwJ40PEHeT9ylaa92Vehh1zWMYxRJ7A8xzDcGWA+CYC0DmfNxxuKl7YuAwmkK9NCmuKEr+nGmaAE5B9oH9fBH6fUgRWc4EG6x/G5jT/OvwxXXVwHtu8EuHKW1mafipM+cZbr8r1XPlLuunJGOYEtdqUd/kuXTnu9C+cJkFx4DZ4gDMJ4w8fwDjyB614DEKhdtzYUbFvT2tLctLpxVUP9yhV1tTXVy5dVLQ1ULlm86IaFFdeXl5UWFxUW5OfNz87KzJDTfWmJTpvVkmA2GvQ6UeA5liGQJ0VJZ3WUzZRsNWG5Wg7X5edJ1Ym9y/PzquWazqgUlqL44rLkujq1SQ5HpU4pmoWv8JzmzmgAR/Z8bmRAGxmYGUms0iJYREnIUvTl5bI0RdY2BxEeXy6HpOjHKrxKhbkstWLGis+HM1SuKLdSdbTm5t6x6k7kkUwaDcvkZd2G/DyYNBgRNCIUnS8PTpL5S4gKMPOrF04yoDNTsrjS6nAk2tQcrF7u9flC+XkrognycrULlqkoo8KyqKiilPoo67Bfmsz78dgdU1bY0JlrisiR8PpglA3j3DG2emxsX9SWG82Rl0dzdvw2EVfeHc2Tl1dHcynW+pYZOvWzJEmUz7TK0tgfAJcjf3z26pZwrEXItP4BKBhllkVJS9BHL28NynpsrEaWasY6x8JTV3ZtkCWrPDZpMo0NVqO4oSmIKKauPL3fG625IxS1dvaShaHY0mta6qOO5nXBKJNZI/WGsQX/KmXf9V6fbWZM01/qBhQLCgcl7PNRMeyfCsAGrER3NQe1ugQbvE9AoDA3FGU6ac+P4z2uNtqzK94zM71TRt3WtwbHolzmiohcjRLfH47u2oDWtYkqRrZGE857ffKY3SZVFIbUsRJytSLSJ0X5LBQSzpo7Ae2GThmzqpWE89rrYy8SyLLZpQoZ0VA81XJ1Z+zv5t5ERCChoOtyNUNYE4wGliMQCMc0Vj1ZVIgzwp2osL7lqjKjhfJg1ClXzWiXslXd1xpUp8SmRZ3LouhZY7OihdXqvpKqx6il/bWq3IWq3H1HqHchkpGbg0+B/8qZyQWS9wd+WACh5RSxexlaZFb1WDDSE03r9EZwj/ZIQa8vGgghipAc7A5RE0Vp5pzxqoYUUu1qTbC+Va5vXhu8Psa01kHRcZnVn0MjB70aGjTWqC5TJwUZLxvCgVZskGoQkKsW4TMqZurwtqJy1FZq5FWLpCDxQnw0shHNkaq7l8fG0fpVSHlqesvq4tgEWkU8y+q8vpBPu/LzGOyWYoRxho4qoC7ehS4NO3Roy8vq1CYq90QqVSkod8shuVeKBpqCdG1UPKpGYsJQ9RPT65qranOEhWICH3bHK1SY0Zpc71zhRmvV+ky17nPdK+Ld0phOrm8do8jlGEJAzldEgZp74HqbV/Ub1GJk9NOSFW1GtZixyUCAWgs1DmlMXhEZk1uDi9TR6Htu8e6gtOxQT+rXVOXnoRusmpTJbc2TAXJb69rgU1aMfLetCT7BEGZZZ1VoMgP7gk9JGGDUVoa20kZakWiFYmrBik4d730KI/0utZdTG9R61xQBtU0XbyPQNcVobVaNUJZKKAAM9nBaTyA+msM2nda2S21Tr0mgIgsY+IAuoA+YGDPjnSS06QlseRrzGj2BH5iImXgncVaL2jxFdk3qA15txC4cEdA4vK1tlnTb2uAPTIDT1CcSqqIXmktiLyobQ1C1FKGG8pVQ71hniG42cKNq8I9EibwE1SQvQUYEU9Qgd1dFjXIVba+k7ZVau0DbRTRR4iY4fRfqvilKqAWsC/pwS0rJL3jHrB9TTYXQAY1Z381H5k5izmHAvIIFEVIDJo7lGZbR6TlWBCgsKfTb7KSiwua3+YuLHD6brxzvk+yKy0/2MqPT+/jjF1f2cv8LgCEe7gJ7SuAw6/OAFDDbTSBg9pZkYVmXHgr9iIBisvmtL5UUF5EFZf4St8spyOlZmXNg4qkPVNU3LA3UkwPNi5auagwsauZvWVqzcvGiFTWVNUvrKhfXLaV5C4EW5QQzinybkVqC0WBgOQ5E0WLMYFmorLQRJKj9Id9We7lfYFxOu0fOYlq+fc/F8bsP3HbhwEGmmOjJK4+dVEo++4NS/tRR8qyGezHijsRxI2JEj7gTkEbG53ETKyPKZfbSBUy2321nIt++58K+e+8ev0iRK39SFk5MkRfOf0ZeOfm4UqzhJhNMO/MQytv2zyyPRkhQOipKZLTU51rMJJGJixfVsduuvEteIxUoU3fAgOI06dk1Ib0DKnNjE8rnyG7b2vbWdetb1qy/a017sGX1upC2llHMXBtxLQa4LuDQc5j5EWI06USRcJxeIIm4INRMRVw9fntFRXGRz2crLxdEMZv42UblZyy/KnXiPvKGwjbuf2BV4r6nyXYVdyPaThP/M0iBxYHUeQlEBDdxp6a5RZEx2A2NIbvIkHlkXmOIksm12aEiUXvZ4mZFxUjJlZIlTOmCLDldELOXMLgmuihkwOXjmi4zZODhSF64ue3QjROb75zovf0XI7X3nTjB3PIGGf7e7i03BNsb655b35ATOTbSPfn0o5MJKm9XznJVyFsWrAsUpnvMDofIzsMDh2gWs+f7UjNSV4fYRE+ipylkSiRmNjHRkCGarQbr6pAhiTIL/sRKe0Xh33V8SeWYNsxsBbWogiJOQXRRZjlferbN7faXlNFlyKUqoC6nzFfCcZ4j908eV15Rfntuqu2lyLfumZga2Hr0v/287r51dz1PXL8hIjdw+48zBPc/fuO1j1YTMbesd3hj+8ehzUeKbvjl3VO4v0BGWW9V9eiAmkC6RWT1eBRh0YTA5eB5p+AgvINvQvNwsBZRYFXF4gL8lHPUL4lzrQpdY1624W4uJT6b31e6oKxcFtAotkaVX0zfzewmKVEl3cDqMhcon5FC5VVS+AYbvdx3YclZR0ujcpOq/wjKuBR58kJ7IDfR5fKwSU5ngi6B1bHzUpwOvcNkBRRoEoDgciU2hlyCYFodEqh8IWYJiZRBfM+ahMqinULIITVuiIlWKl3gowKlbogtL2N6Xr+k/EJ577P7V/+ig3iVt0p3zd9ZziZNn/fKi9kj507/QbmwmhivK33/bZepivlAuaj8WkTTQFnWId9N/HOQCBmwNOAT7CZTejrMs4M9M8uS1hSyWFysK2l1yJXE6tBMxKv4zZ0xA2TUXkHdC/hKPC5qvILo1zYk4IZkeM0Y0gWX011HBsiWW2pDN/7DCxG9uet//PRXyoUPH/r9Hsbatakr0rH3FqafPEEmLH90dp74p6Pn3/xQ+eR+Iv1o786bdu5o3n4YVL7x4vfyJ9BTm2FZwMeYWR0BgTcaWaC+Dz1VgplpCJnNrJFngbVDpV+1gdy49mP6R+lq25zFQnyEMFnZAsuVb5r+aJOCMZbsZIRp5UCmzvcdsl45zJ+4uJzpJG9+JWOXglGBBTRI7jPUuwUlKKHu80Snk+OSU43GdGtyssXi01mTrEmrQiYr7iyr1QKW5IaQxQ6JDSGcPVeY8e01y6BfUz+NNhLnmtleso/E95bNpkK+KfIcKSdJO3v271TeuTD9Nik79vdbd+59+MVdO5TL/PHJk3smbIa0o+PPn2GHG9euWTF9Stmzofs42u121P+r6BvcUBxIFB2ORNZs9jjBBg0hGyfyBr4hZHDEnYC68+PsoUmmMy6rr6TcJsgS2BbYM/wlHjEri818/Mdk4OBbx36lnFQeO0LKX/3lG5GWI9wLyp8+UjyrlMv1ZA/Z9Fty45MbL2cvRB6oDBmUoRGqAxk69MtmwWAgxCToG0ImAeUmCIRhkBOGJYaGELF/UW6zMlN9qM3nit0cc/k4u3L6LPlEsTFO/rii/EhR9mnxAOmyCtLVw4JAso5ljRhXCDFQsnMoIrVZe5+lM5cM+UA5zdZMf0R+p1gpiX3Tcfx8E+I3QVUgzcQwBqMRk6YEHcvzZqyyRqPA6nS8wBJetU97xYxdqoRKbFpordCoEZH6KFs5WinfpBxSdu9RXiRlO8gdJKgwTG/H9Hn++HTXo+T306Mz6xNeQ/pOWBiY5wS9UY+kExLMZjcvuOyMnhGMIpsAAiWu7g6NcInGhadCEyaR7S5KmPgJJb+gzCETVlYO6R3KR+QIWSvgdj6ivMOa9Mo+bqfy0LbpIH/80kNcx8WVzLHiXcR5aR/uV6rjo8iLDn02SpvT612QkGCxOM1Ad6nFatU1hKys5S9Ke8bmeQwwqreWOHKu5647t391nDmhvKn8bi+K4zViIC5W2Lq5v/fFsxenL/DH39X8BdVFnUrfA42BLIfHwxk4q9XgdiWBARWT6NCbQL8KMwWjyYSsmFi3UVXKrKub2ZrWxJ+W2OLueQ5nNo0rulPBg67kXOTAHXfd34FsHSXXkdQ7byUrI8qE8jCbv6GvNzg9On2aP/7m27tfqFAcdzPFms46cD96cD9m415I9yYn+8DHO02owvk5Th+YfKZUvSe1IeThWLRS1jFnJ8SMJ+aQZ5hDfzEnHGdlF2BygRvVTXeq5o/dHreb8yi/UT48du/LbT39RV+66+tfbyLi+0Ont0a2fWtlsCOr9dsvP0Duf/a3QSJVlTWuyl22ZHnlyMGNP/lVWfHvi7JaqnIWV6zccIryn4N6prmoCFkBO+4nlhCdXmQ5tiHExfcSxBOs+C6ihTmrvKgMc014HyOMoqDO2lEWN3Db0LOmw8pAlt2kMzHp6YmJGSk6nWxKqg+ZTLzTaakJOa1MGp9WE+Ld4IpRSPyiFdkrStQg5cuWBZol0jhfTjMTTExopLIRUUB5UEGxPTfoVj2858Gn/nTpJ499/Z+7T370698pr96899Z7N331gXD91NEnvq8Xio82v9L90+enPYzAccG1u3d0I8+HkOdjghPskIa73pfCsj691Zjg8RiNkgmMLr1VACG5NiS4wVEbAutcBSbGEupYzNeivt1mZVCDskht3olB1YOBH7Wm6ZNNePbMe8+dGnT5PyDFJtNNg1t7mZtGOgYGuWHlZ8p/KB8pP79rp+BU7q9+4Mhn+w/5jn3nnx555BHU0/orZ9m3uWGU13UBl8Msih4zw7h5Y22It4IFOXPHOZvJloqLeMmmBnjVdGxWZAF9v8D0TJ/H3Fb/k4Nr676y5Ny54P31K7/pZBaTFJLfcDYlU3lMmSosUS4VZqB8kC63NSafQMCnd1vRDSYkUEFJFpcJbHoQVPlYVfm4/7x8UEIeyk86Y7PaMer4s0pROjIqNht5msk0GObUr9994bUBg4eUfHiDuXfTtn5+4JaerUNOUkIsxEH8h2/ZQHr+dPYbj/zHbQ/HhaP6C8rnBpSPgBldTUBm7RyXZLOl2A2GeYkuO2tfGWJZ0WyGlSGzVXTVhET35zOjOZFckx5BxfkwPGKstIK/BFwun7otyzEXZAbCW3TkUWaoTPlU+TlJuvAx0U0X8nff2n1sQ9Nx9uDOrVt3Xm5BL2IjScSvfHru3lvvua7g7Pzs2J5jtwlpqMtFgVQnw4g8y7o9ZgAPT5wEeXNajNbakBFYdXcgl/7ZSK5mQZqt0X0ol6L4ytRATmXo8rEG5YJyemLixOnv7mxc37h0IdGxo5f3saMHWlp+FC08k7JqUQ1mMoLi5PpRXjlQBgHYFVhuTElZvJgrstmYHFHiZDxiVSVfd53HUy7LS5PNeO43C+YF4sL6kF7CA7aYtSCHWRnKycnKqqwPZVmd+StDTm9cpLidCxEqxPzNox4/MObH8vjZXMlToUGa23OJ6lkwO5vuaDUPxWOJuuWzZp/UDRLnzEklXXDEKnI61/940aV3yvMLJ344dVJ5Snnlwz/+/Y7CmpU1wY3n3i7cbVeyR/sPP71l+Ltrtg60trU3HZngOr6dX/+lYy+wfEZe1Xe/9dN/e/hA920pznX+QFtO1sTQk8/buEtcZd3axsqi1eyqdZs2rXsRdXcI/eUE7gcXlAaSMTvwWPQmnU6vd1s4u52rDdmtBiAYKzESVc41KnvFTBJCz1ZqhEb3EEu6uQnl9LFdQ8ppUiyK1p5fnXqJGf/0sWemP0VX8Ezm7WsPv/4s2vgBNJ53kLYOM5SMgA1jMoZCs9FicDHiihDm2Kr4Zw75mIcu8JeoJ2xqKgeOHGltPUKWUCL8V++7r3b1JQ/XAbE1nUe8CbAiIHOiqGNYVuB5a4LBbNbpDAaLEWirLiGBFQwu1qIdzzTVVc49ncXOZ6pCadpD4jmQSE7tJRWY9yzHk/gh5Yd7KBNWpvMoWaN4p/eRU33KYcE5vVKBGD/kVeSHBceTmOBxDBBX/OOFFoDIqxQBDomPF6vQlmVYFchyOhxqVjhvntebadbrDYYMh9lsk3gbXxvy2KxGyzxicKkZVGX8gKltrTnpnL0irqusmLL8Hn8Z3Wbq001btC7uduV0843bNimnP8ix5R/dcinkzX98yw+fUV5pvnFwgBkfHX301PSnXMddq2483Nj+zBvT2bTt0GNxuU8g306ay6MtuVV2XWbeRnmlrGqczrWkuXb0Od40M2rvoHaEzPh/MPTMi5TYPz6rMtDc+tpzs77yAaRrhCSYH3AKJpMXHI5kt7425LZa2NqQ5fPBRE1M4vsuXcCQg5uTKaXBxW0nB/p37rhp884dm1leOaNcefD810gqYfHFFB85+v3vHTnyyPeUT5SXxokuSuyk6A7l4oze2O3IhxUyAzbGrq7eRs3ZzPM00mLSFtN6zKZKPOxVa2atS3I8maX3PKycfi/fsmCSG9Yr/9Nw/77p57iOEx3DEIubjyKNTKgMpAlJmJpBdprTmQUmZ5opzZDiSakJeaysoSbEur+QnM1mItr3EkxCsktpDJjJzDTPi0xh9PKkMuyjn5zb8Wh92y+bj+RtXrt7tPy911/80fo19zTsu/HAnh0LScOjx3zS5fllnRn5FVll67ffeN/DwX/PKFiRs+iG0vVfpvwWIr/lfAP6l8WBVJPOYnE4DDrGprO5PQa7xV4TMpssFh5qae6khYaTV33fiYf+zBIMUzRpohGi3O+iJx8MsNubBoeOPPm9eyaCz5AK5bkV78i/8T/1FOPd3XP23LvT7y5dQnk4iPaB5wG0y5JAImOzOZ1uo07nsustDJ/AJ9TO5m3qZ7G5cpqTrVH/ppoMAuxUjrX42PCp50kxtU9SfFdj22vPMm9OD1PzZBIuPUTp0v8hjyNdI+QFnEbgeYZahJnBo6We4KKRZskcaloopCcen8ON5lDuQI9zVGknlf+WoRf4+W+QSqWd65jeM7q5cy9zi0YDXYZwCmmkYIaawXMGtPcU8HhYAyt6vfPmpWFKmeoxGFjeYvEKFuDV/acmBBW2uaQ9FfEPqZQDlpXZGTaQD0cqS+2TcsSVt4+ILylTyk9I0e40SeDSrGOkaQ9ns5qJoC8wfJ3kKj8k+8i5y/u5DsW298OGR9qYxOn33e3hL82ru5RHPkHGCf0vuiobNzQEMo1OpxXPAByhkSfRSsMQNtk5t8tqN9trQ2avDixUWFpmPevi5rhqbUvZZkVHtYbWTROcctL4mtcgGKXTpFbJXvr7B1c3lC9tHk23ozT3twxuXMdsu+R4/DHbp+auSLm2l/HFHkX+9NRuWZETkSmjDv2wyPE8h6m+hcGariZEXflV31Nn4kfM55YSX6nPRXzs0cvjjKAksVWKjjEcY05+8Ox0M8CVK9q3JLGdyYIMJC0i9XbSAoZjLCPzWZBbWopj6LnmAuZYWbCC/pQCGtik+BmbnFP/Z2A7DhxPWKBnHLp9iov8aKlTpIz+ZwA0eyRJBym1v7Ms+gPotd8dvFDlOaS+31/54CXT9IRhXDeOY/XaNy5tnm58egLAMH7JpIwaxjXLnr2YMu5lOMm9RTx/za8dmOPQgvfiq9oOwGKmDrbN/PLiGzCKd2PslvGO4F331/6igt+LdydM8UmwXcW3E6a4CJXV58Y1aW3i/tk+XtBgvg7n9EIH9xCV/f/Zxf0S2jGmHOJSYD2+13NPw3rUVw63FwS+Cg4xW2nuQ7MUFT4kvgWHaDv3GayP42AfgkPsRZw3CoXYd5Abx53+Jjg1r4L970E5/H9+UT1cu65d165r17Xr2vW3cvG3Qw5z5Jocrl1/45eW1WZgVvnFMoa9WnkFfkcLSSWbyetMKrOD+e9/rrDJ7A72BJYL3PyZ0ovlMbWc487xVfxD/PN/qQjpws3CE6Igrhe/q1uo+7LuMJazek7fqf+O/gwtBrthk+FRw/vGZOPNWJ41Xv6/XUwJ18q1cq38PynJ18q18jdeiujXLfabGB3dEAQeGLBCIf0qQ4AYgVW/fSWTJTPfwCpiX9ZA/WZWEYMZ4KAqBrMgQ00M5iABmmMwDyboi8ECmGFbDBahBW6NwTpwwusx2Ajz4P0YbIICwsVgM8IlMTgBriOdlEtOj7UhMhaDCdjJ5RjMgI5JiMEs1DDuGMzBPCYvBvOQyKyPwQIkMyMxWITDzJ0xWAfz2ZIYbITr2bUx2ATr2XtisBnhczE4AdZwK5f3bewb6dvRHZEi4ZGw1DUwODrUt7F3RPq+VFJUXJSPj+I8qXZgYOPmbmnZwNDgwFB4pG9gS8Hqwe4tUmt4y3DVwObI0uGu7i2R7iEpX5ppl2iHVLWtD5/FRUXt3UPDOE8qLiguomPokHw6ZHZC37AUlkaGwpHu/vDQTdJAT5xueEtE6g+PShu6paHujX3DI91DyHDfFqmre2gkjO9N24b6hiN9XZSz4YIZbubw2zsyMriwsHD79u0F4Vh3F/YWdA30F/7v+kZGB7sj3cN9G7cg/wW9I/2bG/pwyDDS36bSGOntlpYOhrvwFevJk+JrLSkomotbHVUwMLSxcLM2criwYcWy6sbW6nwcCcvR+jbiPYL3DuiGCEh4h7EeRqgLBmAQRmFIHdWLrRJ8H+8SKIJivPNjUDHkYWstjh7AcZsRjwTLEB7C2fQZVvEPwBYogNXY1o2QBK3YvgWGcY8M4JwILEW4S+2L4HMIR+Tj/cXx0swMCm1DzBpMOSqCdnXucIwebS1Qe+J44ljyZ7D8OQp96pPKYETln3LUj+8huAnbBqDnC+sNq3xL6qhRfG9QW4fwuVHFNqLypUm4T6XWpbZQSWv1TbiWIXVsBJ9dMzIbxhV8UTZ/Xr5URyPYuhD9VSFsV0sB9l89uys2t0CF+nHkf3XeCK51UF1VtyrzjThWk3+BirMfpdOgrqZbXYm2/m1z1jGC46ikliKeMI7TalfPodb1eb2WIIWiv8j3LK4CleeN2Lv5KpzD2NIAK1CO1dCImq9WrRlxBvbrfXVSm0TYtrQ6ti2VvZLW1OBOW70qktbYEEnLWmBty/RntCU5rqSJ3JU0AftXNaSmRRpIQ405zeG3t/E4lfPjdJZY2Er2cZYV6mt+VvNWDSv709vm+b1tbr+rzUYsbVa/pe1xy2kLU2QhxA9tA/BVeBw+Ac4KZJeb8GSK3D25pjU3t35KvNJSH9U3rYuS26KZrfQZaF4bFW6LQtvadcFJQu4M7R0fh6qU+mhJazAqpYTqoxEErCmTbqgKDQ/n4h+9hjuGR+hbfcxciR3/CR3qZrUKZW5kc3RyZWFtCmVuZG9iagoxODUgMCBvYmoKPDwvRmlsdGVyL0ZsYXRlRGVjb2RlL0xlbmd0aCAzMT4+CnN0cmVhbQp42mOQOH+A79B0Bv7fz98wAAETAzmAgwS1TACDWwX4CmVuZHN0cmVhbQplbmRvYmoKMTg4IDAgb2JqCjw8L0xlbmd0aDEgMTkxMTMvRmlsdGVyL0ZsYXRlRGVjb2RlL0xlbmd0aCAxMDE0Mz4+CnN0cmVhbQp42u17B1hUV9rwObfM3Ol9BhiBO1wHFJQ2IGJjlC4YEcVlNDGgoGAXUGOMsa0NjTUh9hjjskqMGYxiiTGWJGoSNaaYvn6JiVkTY8qmbWQu33vuzFBS9tln///5n+/7n9zLvXP6+563v2cGhBFCKrQI0Shx+MiE5McH7bwALSfgKZswp47XDQvZgxDuBvXdE2dOmvZgeYsB6qcQkh+aNHXexNaHppcipKhDKKZbVWV5xY9zPm5EKHE9jO9TBQ2aVzkZ1M9DvXvVtLr75lwSfFC/BUC3Tp0xoXxt8eYwhAashPUSp5XfN5NNpJQIZR6F8fz08mmVtd6Vy6D+DkLsuZk1lTMzjzQAPvm7AAcFopkWvB6xiGO3si5oifR/0lfQRPwNR1EqmYxmGYpi/gtRX7kRPxZW7QEPSho8MhNBQ5tPFima0Z+5h6hSHuHHSB+Tx7YQ6IiWmZEN3QU1G5rNtrAXUZeLyUNmtAGhtluk1vEWzeQNs0ytD7d9KS5r+1zcBy0h4gn0H12c/2M1Wow+RJvROrQWbUEr0WKsQ/UIuXPHjvGUlowaWTyiaPhdwwoLhubn5eZkZ2UOGezOGDRwQP9+6X3T+qQmJSbE9+7VIyba2V2IckSGmA16nVajUio4uYxlaAqjXrwXl2V7aSdvyCkXsoXyvN69+OyQqqzevbKFnDIvX8574YOJFvLypCah3MuX8d5o+Cjv1FzmdcPIib8Y6faPdLePxHp+ABpAQAi892KWwB/FY0aUQvmhLMHDe7+UysOkMhMtVTRQcThghoQVwZbP9ubMqarPLgMccbNKmSlkVip790LNShUUVVDy9hBmNuMeg7BUoHpk92umEKchYGGn2eUV3qIRpdlZdofD07tXvlcrZEldKFNa0ivL9MqlJflqgjpazTf3OlW/5qgejS+LU1cIFeV3l3rpcphbT2fX16/wGuK8PYUsb8/7PwmBnVd6ewlZ2d44smpBcTucgg6Q2Ms69QJf/z2C7Qhf3uraUh5okTn13yNS9FKZXlxc6iCXPQdoXV+fI/A59WX15UfbFo0XeL1Q36xW18/MBnKjolJY4mjb8dV2b84aj1dfVoX7eQJbzyku8JpGjC31Us4cvqocWuAvQ3D0tTsM7WOKfq8bAVmAOEBhh4OQYfVRNxoPFe+iEaX+Oo/G2w8id0Kcx0uVkZ5TwR5LCelZFOxpn14mAG8LRpbWexlnfoWQDRRfXe5dNB6kazJhjKD3an+wO4R6o4FPT/BIY3nAKr+imvey0UAkmNV5AsgNmVKvlyraH/wfX9oBQLTByKcLsAxZJ1vILgv8zakKgQV4IHRenF8QRpV63VlQcJcHOJbdnJgAM8rLgGHVWRIzvQnCTK9ZGNLOXYJWdvXIUmlKYJrXnOkFixqY5U3IlvSKz64vy/KjQNYSRpQeQ662/2pO4e3PuFAK8mSRwdZMkLLo7PrSioneyDJ7BejdRL7U7vC6PcBhj1Ba6SFiBxTq+V92STg8kqyMKi0YKRSMGFPaN4CIv4Msxzizf7GMUGr3LwMC6OWcHF9K2WkPDNRDA58DBWHIAHh75U4OHj0QXGolgjtkAF+K7Sg4GtDw9uSzK7MC40i9y6IsEafMvOBqMlKFdTLz7A6Pw3/17kVBNx8ADDM4QtS8YBeYKejgQD4z86QmQssQIvR8qVApeIQq3usuKiV7I+SRqBwghkTzAK9Gdal1IhaQCTmgO1ghxPTmxNk7E9ebK9Xbq3m/6M4PdvP1nFAwsp4sLgQWRIB5vhcREXb3NdglW0AUWgDby+tBpSWFrm92u4kyV/Ujiwj5FfXCyNIB0miwJwvs9xNYRlSAC0YN6d0LTNuQZgGvHNHsxitHjik9pgcvtnJU6UEKU5llQzzN3aGv9BgPTkNqpUgraSQVnlTISsVQ4aTx9mNuhBZJvYzUINUnHMVIauOCbRhNOEr52/R+QNESIDeioIfx97iDoxlo4/xti6Q26WpGhGRuJevm3Aq3mtJQ9mZMmg5Cy3GIURQYPaPGGmxvhlnFUvNRvKhZ4bb7RyyCEW4/hitLOkCXjCl9Ro1gmvQGQEPIBeISUgXMBreSzVcQQXnAU1Vf5iHKhqzAGvjDXiwMAjYJgwARmdqrFCqHeFXCENKeQdoz/O0y0i4HEcVWDNMXAe+LvJhIwNhSB6gkH3bBXq//knDKA0alXv9pb0DuNMQPSogRaCRHliMMzVIMLUcoITnBhRNcCa6kRJPD4EiD5zSd33qkiprnW8G2/Dy0irkBsQCFU1ktXcW+AZGbDUW41TKkNiJjSKiOtihQgutissGI09MNLgMshFP6uJKtFrNMiIp2dirj1FRndFpatDMVL0h1OtPSnM5UdlZKfHxKclJScuATgipULJ6gHICrBnVzq+VIySBGq1PF0DTKyOgEx6Q3prlklMVstAnRVPG2jbvXb3h49a5NW6kkrMCXDpwWk7/7WuzzXBN+icQzGA2EddXBdRklrIy0OiXNxHRdF+spudDHmJpCxbisRkq9beOu1Q9vWL+bLCz+U+y39wR+5evv8KXTT4tJ0rpV6DrTkzkLtOnhNiGGUbCsWkPLOXmRB+k4zKGEe1xxBuQiMAwuPwyHxWEQDI5Uh8FFPYS3ixWrxYl462raVi+Oxk31eD+smyH+hKeh2xCQGQ6zSKFkgV8XJUSTEp02mYRlGh6nCxufusA+WHPbViH+WFOFk++GuaPwu1QGNQv4bThMIZaBpgRXYK4p1WEZhb/A727eLOG/ue06XoW+BfxD3CoIntUaBT3co7CijDg/ykmJaZ34uHlger/BQ9JdmZMzs7MzB+dkkDXMICTvS/JlOkIhhqVpyhgkK2wXC5h633etkfD156FSvEuhiW23mN4Q5RKZ6uE2GxGIlSw0RGEp8CjktK7AQ4cCCijEj0YHd4QoyqA3upKNWHobpBam91c/3Prh9ne3f2z9qGFP4yOPNO5poD4Ul4n1eCGuwQ/iGvFBcaN4WvwQx+D+cDvFa4A3xMjUK4CPEnV36xUIc1ilZhRyOVbIcAhswJieYEwPgAfoDsGQkiaTyWOwi3plN2dJebMUL13DGJfUWeL31+A4iZ4VoGtO9hUUiuLcthAsNyGTFjNhdr2ywKOX45BCD1k6uDFbYGMORyoeRKWmRAtRsL6f3JQcWxyMszUDL/bO7Lv2wbFPVJS+fPvS59uviqeor9fhpQc3rx85e9WA4bP2vnFwtfj1a+I5zi/r44C2dsAhBmW6u9uRSS5HtDVKI+vRk7ZZbdYCj82mdDojCjxOudJQ4FF2UBqlhyRIH51ww2aZ3GIFhBhHVHRMqtXq4g16h5DqSu7TjjDBmL69Y723Sfyb+H3d6bH3vl2G54vj1m988vymB8uapo0a88WSt24x49YcjOCshzZe+UjotTMhCffEynWPLp9yf0rOzNwRZ4lcABWZyexJ4IkRDXBHGFkFZFZyGtEsbTJj1sgWehQ6udFIy2V0O4+CGkZQJjomoS1IOoah4gC+CTI5Zia/+5gvnWrxviuuVHKJsWIaLhK9uGgD/WFrT3xj3aGyDN8coN8yoF84yEUY6ueO1NGhZs5so5ludgTEQkhmsQAXLTKZusAj+xXpOsiWzFjMSIiSKAak0jui5DEmUH26D9X7M6wQr4k/Lsl5rdJ7Vlx172Oj06i3fUectfSCT89dF8Xhu3q7Gnfg5PA0av8WMd+GJJ2ZDXglAF+tqDvKcgthBpXDAfgY6GinWqUNH+ZRabVm2mwr8JhDaW6YB4x8Z+zSO1EpaEWRI9lmASZStCsgdUIUYoOclVnMVutsPBznzRo8bMIXP6nV02+/eP2fb14Xf8BfrN2xccOYBk/RJmoWfho/aVoXKr4vvrT/9qufiHdwyfln9m1oHLokZ9LBKr9MAl/jgKYyFOnWY5ZFFBhJmir00DJE+BgkHegCBstowTicGtd6nb7oa2LDtyz7+TJYlyrYf09p/1EoHg1z97Tpos296HCFgpYZzDpZQqLM0JPvyRd4evZUI3XYMI86FAnDPEj+awFP70QIP7uIhZNbA3YuNcUZDa8+tNnql/MYmQxHEXrY+pAqlXf+8+WbDjaIH37eipPr7/ti7l8ffaRx+5lHluN+C9bOeWzd3A3sK8f3TD2YX/Ls/JZ3L564s+auwzMfe+5O433L19xf/miuexs96b6Ku/88ZED93ZVz/fydDPsjtsOGnGiwO8puiFIqgWzA3xhtBHDXrNVSZnMIsFdOyYd5qK7CZ0yP68JcDBIHwgdG0iTI/HsCi2k1+rcBuyL7iGacvqq6rKKJX3+vUqe11Jz5pO21R67dJ5rXbV+/aeyW0uJNdE5ro3ldGKiqa+Sf/v7aJ5jbIr6PE4/uWf+XoYtyqg9O9OMu+RGmAuy5hdhzPagHQlab1jTco9W3u5SgPQ94ws6exZDiSv6Fj5kzv7OnoZ9dsYI4HArNxaeZUPq6FM/Eui1yioFGBcewRR4GoyKPDmPwePcAtITO3oO4PgzPXHp36zh6Nz1m1Spx0qpVfvztbdfpdKC9HUWjdHe4A7RdbjKhblompgdC3fXdI4Z7ulv1ynyPnulC9q40B5FJ69OnqyEHlwGaZKFTOqIhR974lblL5w9vKO//3KXn34wpfLBy8MH7+mW401Oy8KY+sx8dVVtXPHGGM2nFhBNP5s+YMH10zb0O8e2VOUPcuRkSvvPacmRH2UMoFWWgbHeUEN9H219utiPU06yNZ92DQ/r2ZQZyBhXccZHJdHQchAAZLmM6pKHG9GTJSAYRd7XbeYJbTFBYCPI22mKW5J/qLkQxlIXYizSLTOAROOHujmTGiKHfZJKMxdHSXVWj56i4Ho9MbNh361TW/uzQZWNrHha/ar4mthzAQ3DC6x+f+k58RJxxFa/B6G08/MidH85eNmrzSpZsot5de2tJ1Yg/jb/ofbUt1CrGWg++vf8w1m86Ju77SLwsHh29bBRehydiBjdcOyw+I+4RcTpmzYeAFnCxevYESIMWeBdBaVUcZol00Iyc0eu0VL5HxWq1EJggBgITl99pdIiGwWiDOMVBO2gBuzAGdZPB1LIWX8uh/dSQdVSGWNnkEKw99+MrYgJ74ucsajJ+cfSCslqxP4Cm0QVQ0pNg27SguTzKd0ebmHClLjRUq2TkWvCanC5EFzLUo9NpkTZ0qEdrRLahHpj3+143ILAOniFGycIYzIzgMDn8dlkwGAKlC3gHBFfMhtV4mPjjbZFqwiGHdnpP/YyTmp858izbcuDEkn2hynTx/Rc+oLNmLZ8/zbfB9+GqjSsX+eOEBWBvLkv2NNptprVmhZa2hRhRvsfIqGRAM1PXQAVEJAr4jlxBEUgxdncl2+TRdP9vxJtY888tL2y+Jj4nPr4PZ7xzY39eI+sSnxdvih+J59IeSccrcfXHeNTRURvvIvILNGNLgWYQ3xJ7oWUUkKwZTaxmqIelGe1QD+HUr+M/5IAshUc0FASDi4c1xLniOnEKPoVL8P2HANanP17GYM2pm2KDuJhtEZeLf8UROOrOTBKgYQKX/gngqohvlymVmEEcZtQamSLfAyaeoth8D0Vj0HNs/F3fTpCwBB76p9YrdIJvPjXOt5taxrY8KvZs8N1AnWEpULI7DGJrTHOYVqp+Acro936/giEEYeDapiAAWN5389EA79iBkr2CfZi5UIrS2jmmWziy50OMwqrVhnyPmmFt+R7W9PsxisMg68JJSbGJjsO+CvE3dV/vFA+Ja4/g0s8+f3nIS0fE78U3sQOHbN4gHqNEX7ozGq/GFZ/gPx0e3TBKPCPeEN8RLwn4jH/vbKREZ5c7VAH2mGMYFrEaNUfneziOVcpoiOaIMoJR6mSmOzYP4ZsL3i42sqlVbGqiqSbK6ytiW3wbqemS/LaViQtxlZTThbs1Ko7T6hQ0AqJCeCDtOJjA0H4/2CcVFjTL1OP65A6bMKnplLjQvsF83yxISkpPXgnwi7kp4QxeheDLUiolwZniWEVnfNO78AnLJVTBdDA3z/u+2NLURD16wXeIenGl7yygG0e95VvWRR5YiIG0NACDmBQiW0DY2IGwhDEhg8NyoYkI1M+fbPXPlc0i5/wgSyEcy8og1tPINGYLVhpoGUTIWiQj6BH8bOm/JCcWrBYSDgOOLgNJYwRMF4l6TrsDI3yZw2+ITVpOVLP6rW+2jmNb7gxlIFOjD+w++fO3AV5mAGw9indbFODOsE4HxtZgVNFyHUdjeWe6dGVjADCAjabSgJcZZ1o/4vQ7TtHhHKWlHmNeeedQ62kA2M+OnRl0rt8XE9k++Vu5Yb5HwdC6fA9t+m3b4JdeHnXODdmT4g7xBWKWcBnOgtxv/J1a3zf/+OnHb//hgxzxcXE6yHAVrsSrxJniLvFt8SJOxrGQJyaJF/12iqmSdNiIUtxhSqA8UBqZzEpmqEeplMnlxqEeOS37hRand4QEkDdJqQiPAxmTwFSJV8Qb+5vwfKq7T7Xl45dPXjjJqP/2lQ8o4bNvfHzD2gAdxD0SHXSg5SluSCGtgAytBC2nNXJNaL5Hw8hNoOCQZGa81K7infVbD0kkyOdv0+XjS/juH8Xrab9Lm8/ENUPEBjyU+hcUCthy8CMtSA2xX57bqcGYUisMcpVSKVdQjNWm0ICKDvVoNBRNAw9pWkVJEvO7rk/CXaIZY9GzsANnkHRgC/EGDBDE9/BDF8Sd4uWbh/Y++dyHVJlvF9ty6bL4t4m+GVTZxnXrNiySdI7E1BTYye6EghEmCEpDTCzjjFZH0FYr2EorQyu6CBRK75r8MgIfTOEgXI6J9kdGxFZKcTTk01aGEv/+tdi6ZsybVU37B67b+OrT4pX3jqQefnLF5r7LVt14Ci87/U7mnuhei2sLy4tT8s8/vu980cOFdZMKy0ckFZ/w+2Ij0HAM0FCOuruNCLOYpjkFRCFgHZigj2jP3vwGEkJZB7W6SUxi8sQkNvJRyS8cB90JhXUMyOk26LFSjuUmo4GjGQ0sFNilq7PaGMxEMiTT6LeRbKj4uvgT3Neazp85dp5tab3rZ/FjzLfSB1pzjrzw4lH6KMLSd5JnpbOTMLcKHKiKg+CKpgmirsChCSzuwiSigpwpjVEf8X2z1/fdIVzUL6p7P/9JUOtdT2zd/bi0f4hc5YWwXig5iwlV2sMsWi2rCDEoaazwmxfJvgSPscCyRJMwjaxvIusHwJhc7It7xXctKdiaKH68V5x/6LNEqz0Vyw5hU6LDlHrjEP3G4JfMf97R6gLw805uPfI0Pa914baza1+llxE8ONhXg+QDBLeRkbHgAGTEATDEATBdHYCEBmBAgkYw/g2nIRaZeQjfEPufwFPw9MNif2qpbwHVSh33PUdl+gpRgG7zpbiHd+sYlmLltAJ8DJbTbAftAtEWLApKgF3UQLz8qBiyXww9Tr1Pvd+62HeBSqBXSHQrhvUyJPsU57ZBeCFHcgYzShXLgGGgMdcljuly3iRl2Q4mo9VEcz49/XzrT3TEMmbzlmV3qsG3toov0lvb8qQzRECPwRQ5Q8SBsCHVQW9tHU/vFF9cCvo/ru0WfZEZQ3iHct3OKE4VEREaauRoiA8oVUSuh1JBWmjR5XhAm8NyPKwVDEXGvz6acATzEL0zTUqnAqdMFoMThDRVLp1N0MUOrviJ+584RpmOz7h/1VOuEafLX3hO1G5rbnzp6WnbJ+Xv3YYL9bKsxfNHLeyVfOCUzzy7acsEuXxa7ZhxgLcXbMNsmRl8aiTKckcpbTadTh1Oq2neoUFqi9GgNIDjAYRlVmTO8YD/62wkXCGdTo6Dhwh+4wpJlJyEBGa5q48tmG6TU4MD73z1zdUpzQPUwtxGjqt7talha9OWhgZmjPiu+C3cbw0vfkhmFpcvrNyz+uxnn527duXq6377UAs0Xs3c7Y/VdWBHFXSITaYB3PTICLhZfxmrs4FY3QqmCuI6Sbshbqcm3hbvYMX3w3f2dqUtSRYPPrFr1frpZuzEamzCvaJsD1nDxdEvv9N/U7pk2wEukwY0MgKNhrgd4TIb0moNMgPvMFp0gBhW0woFkEihp005Htr6+ySSKOQUZMETXJsrJgbaBYmfKQFzSm9V1H769ldfX702VyNnGleIO5u2bGvauG3rpr/iaKyDu9fu4cPwyX/emnvsknDz3PXLr19tx9MI9DGhMNTfHRmitKloGqxHN7tNleOx2ZBMZpaIpe1CrM4Rsasz2YwWi8PqJ5mMhlzeAdxb+A/xM8x+cOlrn4Y9trf5qdId2/+8Q0sNXGPGPbAcK3Bf8Zu/VZ8+P/ThaAf96f7NO/7q5104xH1aWSQyk5MFs1ptoihQd9piVYJKKMHYy9kcj0muo4lWSGeZHaENIZwxXQrVIYpITUvVO4LOB6z/TvFm4wsv4PI/zY4ryxo3Btvoc63p9LmCAQPxw8KyyAX1uSQ/jRHNTBLQJhalocFountQXGhfpzpyIJtkwiaWio3qFukMVQ7J7KZL1aXmeLj+uR5lFBer43ScNTaWyvXE6npk5Hp66K3xuR6rPUC8djaHJCQYwCTG/Y4ftwQP1WIkRZAOQchRMjkqie54g0DgoJ4IsD0TqaSmQIlJeqVbrw9OJcVNHjrmzMHnxQ/Ev799c1FdbLo7u2TKOy+NzhYNDWuuXJi++dysB8csqfvHD7MfZPKqQ4RZuY+f4vqW9I5rWNfy/K6NFRvDTEWpA8bECnunHjprvoM8dy+Y4smeSg+onXPrxweBT16wpVkg61aSs2jVHKdAVoXVFqJVGI1MjseoVyKksBAX0B5pBbWfWCyDX4gNQc33myt65/1LnnqssZFTJh2uu3CBenH5n09c9Z0FLe9Z0nf42Odf86US+d0NgjKRvQbc0oHnMZBzP4z1Bo08z6OhdJhIxqXOR2ck/aU7f8E2urHxUN/YHv369Yjty+Thnumpffr2TUuDtds2iGZpbTUKQb3cVpNKpeG40DCrPs9jdSt0CKQPBbga1gWIqVPipO8MLe7e/tmZd+V3QBTNoSvMJX9iWu/oxRPye4LA/TQNB5rqUR+IYLVKpUrFyRiWYwxGiMX0eo6j5SoLrfP7vgSAnOEXofQAWbHk23EwyzLL8ai4Ujz+RTELv39BXLBgzx6OSho4Ac8Ve/tWU7Kp4kSZufVcWq0fNh4NsGkE2wRTBe4WSZQMJoaEiIRwMAb5x8teAT0RUI5bsBgMRhAAQWEU6O7OblaLxRjKaELB4kaG6k06yAcsUrqVEfjyAdTA5frFAU4nkeiQDRuRDN4WlA+Zc+acbQ83zpy7fUPjCjuX8ORkjIdzScfnHj9GXVi69OAx33by+exbvtNMXkPRmOOjK55/nchMQF4BXzNKcocgMxFYs8JqUSv0ehBXvV6p+z1x7Sqtts6yun8XQcN1dNZL54msHr8qwR3hkYD67ex4gElkCXIzE9JAxBEWqrSCJdPT+i4uoHNuBkJDpRLDijq0HFR7vPjt7U2fPIDVt29gXetze594Yt++vzzRSDnF78Q36jH1FLilOPGSeOf1D95/48q7flvvBXs2W9q3A2W4+VAVI5dzkUbOGCUwKqTTWXI8Or1Cx9lRtw5jn9ER3LcrrWTvwVlbO5GBeHBi8Ds5beKw1Y0rbJy7afp7X35ze28DtbVp7eOPm4cXl40WB8pSGsYUiVfFfxAHTl8//orzs3M3Xr74vt8vAa5pEr38MYY+3EqHhYXqQ3lHmN2mC4+IsGpMJjnYf70G5Xg0/8qB+j1ocp+0dndpa/elaYG8Xkbd1biZ3b5v47YtC968/fXVj+YpQpY2qjS1cw++4bzx8vXLl99dDcmbCnKo+KaGf76K36rI+atflugYwFOPYt1mjUKhVFKQ3+s0SGmR7EMg9jGmd/m2POiKguSj9uT3s2WlLHuucZWJG7ifuVu9VffO475DTN4rU+qC+RhdC3C6gz3o5s/HWFN7PpbjsepltKJdihJ+KyOTdXyplhIdE0/9OiGjaz+59N79ww+PWrJmxuNbFme8d7J5X/+/LJ9zX++KtWdX4bgtjdlbe8SPLHGPHZReMrVg+ba8FVlDB/ca1Dc1dz3gGNl2i9rL5oDkkKzbbFaoFEaaCbEpTXpTrkfr1uvkwCp5gFVhF7sEr34OWci5moE47DSXhZzjma2UsVdxiL06VjyzY0dOGR4knhk3WyNfqDHg4dSaouzPxcW++RMmExrtAR1Ll37xmeIOxSa5Wq00KS1WtUaj58w6SbetqmAETTIGV9cTh2AmByIRjKINuBBU+7HGVSEK1+G68+eYPF86OKK3KPedo5tGjD55hbroj1VIfkIBbPJ7ACVWqjWsAuukcN0VzOocUsrl6mM0gV3eJZYdvFWk5VSzXz0olsGycz7JSsXDqMQ7R6X1bGBTBVivG3ges8IeHmGzajWQjTJcmBFcD2fpnNu5XO35HU2+iJCTA3mjkUgWZHYS0D4AlEm4YErhekRcPiW+dWDyDI5TJRkvHHqhr5ljhOf3i1eopf2vPH2vbyHkxRPEooL0w6nUbN/q/bO7N1AfSGgBXjGwT07aZ5TbIGdlEIWRnz5gTmVhYMNd07xAhoelbO+ouPQ57MD8s+JSvOGEeFF8+QSVRNnEu/Ee303fZXxCzIL1KdB7GaxvIfk8bJS22hgN0kCorrFz4OMzkjud1oCn7didzC/fEHqmUQe/Hm7iNCNvHBR75jy7snBoWtaT+QOByOuu3uv6kXrgDn9sm2GZ+tR2FMwF6WkA89e5YK6H1WEu10O83+/ngvS01k+p0b7L1Ge+g9Q9s+jRixe3HgfvWdF2S5bl/54A9QHfGBNjo3UJkZFmhc7GpvWVqZFRp45UU8ioN1Jqo9pojxNMLjsTpGJCwj33GDp/4+T/LsGfF0mqEtV+qoKlo2jGaDFTjBDVnUozGxlXcndj8CsoesGCFQuXjl6QWJlz5rUPn180p9/U1kfP43Evk+eMuPu1y+Lus9UHcO/9B3Ds0wfEt70HxKtPM8KBrXt3937A3O2bd678MGCOSzwqzRF3v/yi2PjaZTzmpafEN546gHt6A9NQW5v/LEXupKJRXyCvnPxaBhcj5SEKx7DRKC41ldhN0NXtENsnkm9se8ay4QKyqdUo3BDLJiUbjbHx8dE5nngUa5G8UtAX/fq7Q79p86e1xIRZOswcsXL+QNn/zSIvxRPEYzOh60r72nPvWjPxzDMnpmRsz3t75PSFZdm5Be6VC8VbjR/87dJHzLcranMyHXxsuuveHZW7nszeGpPQUjAlp3j+qIzJqeljUotKrt0pZA4efHYHInuWcnn2DdhzFuwZ1OJjBDumqX4G/47JGCkGkMYU+8fY/WPiO42RbJg0ptQ/xoEkysUGxxA7h7+jxtHXQQdNLYGfHgV+5CQFn53izT2zplbX1U6eUkPdmvfgA/PnLl5KYEBexXQD2kejfFhPjgqpluA5O66Vfo9kaEEMi2kU/DmS/2T9QhM5fgraOxy6lRTv1Q34Hin8v3+/kJq0Wvr8e2Hhna2+txS7ufFQVfi/a/TP4x7y7YWm/Xe2ivMVu/2Ws+Oi+jAX0WlmJ079d351T41DxfAMlKzvJlSFf0AZ1CY0inKgzdTXyAxtE+E5AU8FPOPgiYNnGTyzA/UqeCaT8e3/MbAJST8roGcguzwJzWP1CLEJ6AKrRgvYt9AFphYeB9TfgPpnhGaAh9BWxtyE9hh0QZ6OLsg4ePqjBcyVwOe30FeBJjPTkBHmHWfOAtmrkJ3ZiThmPsDcQCSi60U3tLVK+NQCnp8hL30Z1cJnLbMA1QK/wpnxKAZgeikZ2k3J2jYwLqnslc9CXtLOvCWN95I5dBbMvwL7fANFQt8eBvghS0c2JgnW4BBFn/w1/H/nku0lv9v6/+MifGmXgZ9JXPeL/ySpJb/F+T+/CF9+sx34gv64/rj+uP64/rj+o4um/ufZUOYECqf6/8GbP67/5ZcU6SIejf6Newk6APd76CesxYl4KF6DW/BP1GhqM/UJPYSuoN9kkpkq6d4I9+fsTNYrU8uGypbLTsg+l/eTT4X7YfkL8o85nivldnIfKBIVDypeV/ygjFUWKxuULylvq8La70S416g+Vkeoa9SnNEjzsOZ1LdIO0j6gfVh7Sxetq9Kd1Bv1OfrN+ncMVsMgQ73huOE7I/8v7u3/T+4D5DZZ2+8e/8bdr9Nd+Mf9x/0/+p76f+2+74/7j/t/+b2WnG7Rj6J6ZAUfySIK6VECuhshrEU/IVo6+wrDg9rPwNKD3yTAWwE1f5lCcjQkUKZRBMoPlJlOY1ikRpMCZRniUG2gLEfFaH6gzCEzuhgoq1A39FGgrEbxyBcoa1A8jguUtSgWk1/bY0YBtRq8NFDGyIi/D5QppKXkgTKNBlCWQJnpNIZFIdSIQFmG9FR1oCxHe6h5gTKHetDhgbIK9aWzA2U1upteEChroPxuoKxFo5jUrOpJ1XXV91dW8BXldeX8hBkz59VUT6qq4/fxyYlJib3hldSLz50xY9LUSj5zRs3MGTXlddUzpscPn1k5nR9ZPr22uHLS7KnlNYNrJ1ROr6is4Xvz7V38kNnVUyv4pMTE0ZU1tTCLT4pPSiTdpLdjWHUtX87X1ZRXVE4rr5nCz5gYBFg+vYKfVj6PH1/J11ROqq6tq6wBTKun8xMqa+rK4XPy7Jrq2orqCQSl2vh2HDohWlVXN7NfQsLcuXPjywPdE6A3fsKMaQn/qq9u3szKisra6knTAfX4qrppUwurYUgtwJ8twairquQHzyyfAB+Bnl58cJvJ8Ymd15ZGxc+omZQw1T+yNqEwPzP7rpHZvWEkykLVIHjVqA6e+1ElqoAAsQKVQ70cShPQDDQTzUM10qgqaOXRPniSUSJKgqd3oJSEekFrLoyeAeOmwjo8yoRyDcwm73Jp/RloOojqcGirhBKPRkL7dBD1YqhPQrNhXjmMHQwtE6QRFfCugXG94fn1LB6UajasOlXCmWCTCEpKZtQGYJHWeKknODs497dWq5beZNd1EsYE+jQJoynQNgNN/NUOyyUceWnUPPgcL7XWSLshq9VJ2PhpWi1BmyC1ENr665NhBzXS2Ap4T2inUi3g/Ws6/DZFCVfqoLUfmKYENFe646G/6+wJgbnxUmkajPxP59XBXmdKu6qUKD0JxvqpHi+tOQ2oUyjtplLaiX//szvtow7GEUoNhnXKYZy/1nVOLylR6crNZICQ+Lt4d6wVL+E8CXqndlmzFloKwfxmomx0F3A+W5JfsqZ7i8KRx5fwOKIkMo8uiYzAuoiMiKcj6GEF0ZGFBa7IghxnZHSKvsTp6l4SamqLlDNtkTK6LXJovisyH/pMLmMJi+kSxgWzaayjM+inaTo3JzTyixwsuKJKurnsJVaXpcSAdSV6l65EpxuuoyJ1l3WUTtemo2QURiXYhUpmoIXoafQVYvQIL7JiFh/F65tHjYyLKzgqbysu8CqKxnrxSq9zJHm7R4zxylZ6UcmYsaXNGK/1LHvoITQkvMCbPLLUy4d7CrwVUNCHN1vREE9tbVzcuNq62XHkqourrYvrfEnVkHH/Dcs9Y7UKZW5kc3RyZWFtCmVuZG9iagoxODkgMCBvYmoKPDwvRmlsdGVyL0ZsYXRlRGVjb2RlL0xlbmd0aCA0MT4+CnN0cmVhbQp42mOQ/v9n//f/f/T////DIMAAAkxODAxNDMQCRhDBwUA8YAIAG34J0gplbmRzdHJlYW0KZW5kb2JqCjE5MSAwIG9iago8PC9TdWJ0eXBlL0NJREZvbnRUeXBlMEMvRmlsdGVyL0ZsYXRlRGVjb2RlL0xlbmd0aCAzMDgxPj4Kc3RyZWFtCnjahVcLXFPXGb8BckMlxcI1/lyuOze6tra1IMXHLGoHqKiodOUhrFUQJBAgIZIECITwfggnylMgvBGUV2XYWu2QVlqrhNVfm1pc11VtO3FtbbVd1+673WG/324AhW7rdknuzbl85zv/7zv/73FElIsLJRKJlkSFREWEhK3ctXu3NlX7lI9XqDIxXR2rc/xvO7+U/zmWIh5RPCfiWSde4cwvc7FIRWSD1NkudZmKJPYftP+wiH9OUaL8hY77mAc4SZcLv9zvSh92vDgvfYSSOFEiyoNaTCHqF9S31D9E1L8vFxCvjVPuiFemGpIMWb7ePj5rNmsPZumSElUGha+Pz+onHfenFYHeiuDYAynaTH1KkiI2NV4R7K3Y7a0I0WYKb5MUj2lTFXFKVaw6QaFNUIQroxTpeqVOr0jUadMP6h/3VoSrkvSKTK0uRSE8dUq1MlavjFekp8YrdQqDSqnYFhEWrgjSphoUu5IOKFP1SoWXl0KhVyoVKoPhoN+qVYb0RG+tLnFVgiCjX6WeEdKvcszzCno2JNxr147NW0PCtnobjAZFglaniFcaYpPUeu97BododZpYNSVcSwVfPEGtpHypX1LrqQ3UJupXlD8VQAVSQdR2KpjaSe2mnqNCqTAqnIqi9lOxVBylpBIoFZVEaZwFhwqXglI4XOtClVG3RZGiXiexU7LTR86XXTxcylz6Xb4V/1r8Mb2M/lCSJ/m761rXUtfJB2IeyH3gswUVbh5uFrcR6WJpmvQmBLiX2MJtcMXm+ZodTHbGyDf3y/p0OCe2oOpIEQK15ErEx+Th/Y/gQE2Hqaupp27IgmotlvbDyGJtxHXsB+fVWziy9RBZ+JuVciY/6PfBfx0ar+kfRoy/uvNcxSm2pxP3c2TTtzIDzojPQkxjdmI0LmJ3x518aeTTIXikkhMgwFUbiG2ir+xQbXfmc0Atg4W+d5a9EJ2lTkIQLOkvOZqEAmmTDqs1x7GVe3Wqms7EGYMncXs3ij19pnCABdFXt4Hhghav3Bq8dZPq5jk0Ibl0cfT3Vy8GrkHu+Ta+1iZ62Q6XBP1G2Ch7UY/zCw04tRw9Xuynj2N91n4PT8Djf/5+cuzNfX5WLvvIYVMfO4g72jngJM24W5NVWlBcisrLcHmuzrUneV97FEtkxIXISTDZCcITFoPrp5O3pu35o000YQf9pDOfACJZTyouMRWUZxaj3BLj3gB23eb3/nKuByTAdryECzmVpc3cxXZ1tPRO/AJHE/9nydLVRPrFI+AJ3Mk7LQ6FeOyDMVhk82QO8vGzCrOKDhmLUVR6RKGW9fd/7VMTB6/THW14COidd8hi4rZyBVnIMUPE/ZtHQQZLhm91oy2wXEba6AnrpTffY8+P6x6u49yxDcZt4GETwS77V3925nfwb8jGNmDThmf2rUF7Vkimnd6Nm7hvyPijdE4a1qiPCyPoopuO477+NJzDPQrj4E43dQtDPTZxxH2Y/vAd3IfK6uIH24819zj8D/02WGsT3bRDhbAFagiVffPMHbJAFVOcHocgFIolc/M76eTiwhSOtEmysGGgtbLqaC0aOHG69SJ7/dXA4A1bnv2ldmftm1quqqq2pl3ea2jOyNEUxq//eCu4gMfdW+CM3EnWGAS89eYY/GpMM+bZe9FoD7XDdnvaxTL7cxeZhyi+ml8iO2bsj8h/viJpH2IWUPUmVds+1muL/7r9LclNRo4RUfmmwnSNXGXV1eUhpvihgGi8LV++9rsQWASLvhv/BDEBiqGs16JOoJrc2M6n2K10US4+XMTllGBrHscsz8+3VnRieeVha+VRVH7klGGE/Wzi/UmO8fXvK+zN7kEdDV2VjRbXg9XHio+yLccaejmynTjLQmNiIiNih0cRc7nqrbNnz7/xyt49CChCZDH++1es8D9ztrVuYKAdNXfX4Ql2EsR4LefOq+CSrLGovqAGkU9BKa4zH7IUsiZjnsEovC3m4GWygwhfcX5xbpFRbmjLa647YqmzIrgOB8R11Y21bXIHF969F4nhwjbV9cn6ddiEYiCH/ibqzMbt0UlGI4KQ/xKQ5pi5EaymrQ5uCFO5GNhKf3311OujVu2ziGTGz/DphEPKa1qKc4eu90TT8AdVOA+RK3SeCqtUg7iBgyt0wyAedJCUNwsMNQqYQgVMAukQoXgzUHMkJKfmkZOaMhNq3vjUtNws2a9+7QlJXwfaGRv/wqyy5XCdnraJLJ83S0zGyYP3zBLG52a0MD5kOd8p60nv1GrT07XazvSens5OgeUzoeQmAPWDF8IuhMELi5byVbNLqGnyOOyBhSARD85D7U2b9LMBNkSTdeRJMXw3495gOkcnIOnCzdzIlPn5uRF4z8Xa89foj66CG1klTsycVQML7ht732mJAghBHJHFfPWsob7z9mux4C7ZvPH16QWm3fW0DZptnm9d2H/howuwVHgwx/hKQVsGTkG8WTLQIaR3Zq2jZKAwOq7MnFbIbSfReYmFqSWpP6ugC4ZXQ0N5tWtXgQ6b2bBpIzimV92BB9CUWZKSgdUcU+XAjWYXZd6Z0Z1mkdTve5uEV+bWGhqVVcWuRdUNxQ1so7XWWssdqf4EqhtGXS1TabRDx7ThTME83ONCXfMcvQBujs+ipcwI3ziNOhFBqaQdA+2ricuN16KTOdrWZDZFY07Xc8y1vvJMnMWmzHCgA/+WI+OSxIxZwt6c2TZBBwfbJMzI5Onh08es2cGImCXGe6z+w4+E/ofIHMrpYGNe/YloY+wDuOnAj4KNjP+PYGN6/0+4zZSnt+zQ7SCGo9Z6eN8VCpmL92PEgzC3nwQncLr9JXiioBqZ927/gIDdf/j8L5cvX3n/7WAfB8l5dxu4Z4qgVJgfMkssH76FdlRJY0leXilSrovMN5a7mioyyrVyXWtpU9vp6lNn0ZSfj+Q+26FybrN22eDQdNIpF1QaeH/Z5wG3iAQRw1ww9k2ZNfNCUTYXQhrYBR40uE7c+lwo/3v2B61Ey+DQ9/SXZ4L2cO5EZ4NVjrQ27eizdlDYmSX/3mIwd3vLrcofJ7Sen/RxHP3TLQfjOdd0YPAQwV7wcIYPx2VDmuMpSp1andCXNnCqu78f3d9+8LKDsx1/wvjwJt5N1p73yvqonVm7wtFJCfPHmVh1mgv+a2Tcb14qsM5LJc6CG5iHv7j0JTyI9JaWzG62s735BDfT/HievxxsH7YDZ1deZtRCAfaTvZ34Sl4jinv5YM2mVr/WJbENysFh+esj567AgiYvdTVKP1Jf0j7bAy2Z7oH0pdmaIjQYF9EQzRJn/43rtv9uxw09B5KDYqapp+RG5hp55P5fb9Q+X3niAIo9UfZKRatrS0VFawqrwhmZHFkiMWFdX2e1daAOaV62md5mwfX9m5P3OPm+DVptznwMeMj+RmQdKVVoW4M4tXJbw0vy4wP9t29YNWlHULblcHb/DCr7DKgyc3E56oiJwUaWbDA8nbjqu+zPzNxI0Ue5e+QJCck+e8PqP9Sh1tKy+rQZHE9Mw6hrtgw1oMSR0fIeFnwHbpxzeAqcxsFlXMh6nqNXtvwJyv4UaWeG+RSIkAG9epI4RUZmx0UjWGb94vpN9mRzfmkVV1paWpIl17dmHu9ub+kbSXzjCaEnlJMHSTQpAJfAu4i5BaI7H4NUYMraHaGbg+NHbJmooOtwl1U+PvbaB9dGtwcgdz7hhxWy1pyWzPTinNwSRG78M0hcbKrARfKspuz2zpqmxmoEN34IEtc0W3CtfDa5v2P/TPhbtJ4v5UtkEzN1Y4qlk4uKkrlAiRGnn3Q4Co3Cz6bMkZI52gTMVZBIoYwu2UPfl3XsxbtC2RBY6StEYwR4ylpxhyq5IiMb7a4ILTCwz+Dz3RwUwvp5Ldv663Qb7kxyuBc9vWauabxJDtEr302cuHrtzCSaOWyMC5Q/Dt5B9iDwFhJzDl8nHDnSsBlF3yvCP8p1STQz8pRfXspG/OIAgmP/2XqQOnqlXT9x+vTRniHE7E2xdOBWdoJu7hYqk7vjNPsQz0ofFU6klJ9wW+jkmV/Pb66HTfWV9TRa4BKulT6ApW62BXa3U83VFsdV1V5ZLZUOdhyurLG0WY6+8eJR6YN856K/y/4FkDsMtwplbmRzdHJlYW0KZW5kb2JqCjE5MyAwIG9iago8PC9GaWx0ZXIvRmxhdGVEZWNvZGUvTGVuZ3RoIDIyPj4Kc3RyZWFtCnjaa2Bg4BAwUJz6aEtFw7MjAB2RBbcKZW5kc3RyZWFtCmVuZG9iagoxMyAwIG9iago8PC9UeXBlL09ialN0bS9OIDE2Mi9GaXJzdCAxNDE4L0ZpbHRlci9GbGF0ZURlY29kZS9MZW5ndGggMzkzOT4+CnN0cmVhbQp42u1c23IbR5J936+otyXCwUbdLw4FYyRxNeZYkjWWPNYuoh9AsClhhwIYBDge7dfvOVUNoAE0POQE7NgLacN9qUtmnTyZld1dZaWFFMoKJaNQXmiLqyiMxZ0krFNCS+G8FloJH4zQWoRohTYiJofaaCdR6IRSCkcvlEYVHYQyBrXRp0WPOgnlbBAGnXtnhVFCBe+E0RAWvEBVlSLK0bNMKIciGp2ZCEloZBJEmSAs1HGeekEUtLGoF6ISFvWSMsKhqoRwqG0UOnMGIrwXDqJM4EAgIqAcI/QyCY/6Ae086keOEPUTGkE1K9EpmlqFxj4Kq6EfurCoKAJE2phEVgHtAuoHoIeh2AhQAuonCIsQKaEMVHTKJRGhIhQV0UG1iHKoZjHeGImxFwn1A/pDFRcBWkL9BOUAtZfoL3mo6q1IUE3jAuBDZ9RWEsrbBJgltPfEV6JNoEEkGkXHkyA8Uaapg8KwFSAImqZWEmMJmlbEYNAHrRk8rhS4EYLlnUTTozkHmBRMDg1DwpgIdeQ4lXEYKyoqgBGVZ4cBJxiXAqsAMQgBQgGjzBScBEgnm2A3nKA57Q3y4MRBluKdyMpAzZKnGFK0lpXRs4VFFFgWLSBRaBkdh2wIMeymQLDowFdlCDaHDCpFT96BljEQKJAqxnyCysmzToQFABu4K5KOLHIigRE4gRFcbm5gDdhFAZEUKQt8SRlesBCGYZnLluEQHZpIUlHZbCSOzdAUltqxuXTAlTZxdMFsL08kWSnkW9lyuJXbAR7cirQhUCEeMhE5l60JwypP0CW9g/aQCa087QnPE7mWSmiVbaRBaMdaJLkiBAAMtGEt8l3lQsvmnq7uYHDPW45j5ajQD8jA+h5YKU/ThUwUnlB9+BA4APU5YJXVz/1TfUUz5ggApwJBSECohDMITzS6pH1JUK3gvyS6VkQYHggeBaKViUQgqJm2IBX05BlGnrkFL2OfkQpSL51gH7JFUvFAAmm2zYx2JC5txBhGCEwCBRXxsVldRk3H/5Kd1lM/4mkTqEZ3cSR6Zj5c3gnygpZlJPQZdFoUHdP9XETzZ8+GH77eNsPns9l8OXx/f7nk1evp7K/DF/O7q+ZuJBGqZT38bngxfDlS5eI5mr0f/nH+YT48P7kenJ0Nf2wmy1FIVYLUaCtDgspKApnkK6NUfXb2L4+UJQncStZPP17wd/J5ubxdfDsc/vLLL9V/ze5nX6vJ/MtwrYENtvIwQKsC4l3lotnW4eMPl/+Juji7+ILQB5E/ovmr+Yy3XiEU8c7wFTQvJyRmW+nd3XzyvlmOhu/OXw0/NH9foovxp+ZlObwoh4t/Yqx9uMoNrhoR2lcMyuBQRf6BV5VP6kii1FpUdBVp6rypDMU5VUXOacFV4OmRxOk9cTZVUZHvpspTqFdVCOFI4syeOOMrzYDjY2UNA1OstI5HEmf3xGldBQacCNsh5iOQVT6kI4lzWzSRsrJMfCAiYD5DPKgQR48kyu+OzEZfxRw2K8PLFKuQ7JGkhT1pwVQMjJBGsljc1tYdSVrck+ZSxalaa1kFTLPIteBxx3KB1LUasszK2jyJV5Lpn6WTH0vUeEuUwujoaRpYYjq0Gha0x4ojl11RmLvoxJxjq0R7SVs5eaxRTXbtZXJkZKaIyMhcPKkK2dWRxF3tifOu8mSjhlhkxSaEyqVjiWv2xDldyZwthCpm8RazqzmSuM0UHmWFfNhiskEqhBlcJz64VMgvfucZvFVhNYNv6fC/ZQZ/gvU3hrVNODe4MuH8P4Pr8JVlov5E3Kd48ATr/7948MTSJzSf0DyIpqukims4XX5weoLzCc4nOJ/gfILzCc4nOP/Hwtk7xGVzgwF+Y5ORXqborIqym9g7vt7GAA0/BgZfectP7LEy5lgafBlPb5bzb6ez6/kf1iB3VeBHPqdSVT6phspHQK0qbdSRQVDJJmuUdmnznjiqKilhJd9C84sfnm2EVQZ2d78rAAZcc/mjYwbARF8554/NAhWUi1Eak3Y5oF2Au/FDrqq4HEJ7sDz+hhB83kVAa77Q5rd0aJSM0FZVMaqnuPKbviV4Pf0yXS5GJ3JwYgb18O34S5OvuFoD3Z2oAdd/8EwPuACEZ2bAFSA42+nDDk7Cpg874Fdz1ncDLvngmR9wpQjPwoBLRfb7iIOTy00fccB1JKyfBlwtwrPxgMtFeHY54HqR/T4mA5h73cdkwMUkrH814HdCnjUDLifh2fWAC1Q2fXw/vVqMuOwFt/LymHJsr41dVR2ZYoGP//4fAu7jLWyoxOz+5qbeLXM+VPZAmY26svZAmeSbllWZ1VvyELCS7y9z2hxsZ7Wr1kVcZ7Eu42IhfrxaF27pYsBRGfr1NPz4dUAXY2wVN2VpS08lKxzaspS28TQIQ72KotCpSukDhU7ayroDhYZfH1fDgLXPm8VykZc6ZdfJzj9d3jTPrpvraymtlzJoKb3C7wo/h+uA43V7//Ks54v/u/FdM1vmpUcU/RYemJco9YpAj6X7y7Z7y+NZz5f9nW7f3TV/y0ugOjLUgWHoTd9Z7VSuPc6DaX+pPdq27Oqs53N/rw5laCsd9IFxst9YZGcdcI3Devy8l/VL7X1TYPa+vR829fbwsYd1OwCJ6x0yzn3HuhTnNhCtIRyX+77BL5z1LC95Nb1bLNfGeT1erIEZvpzfQ8lTu1E3dNEzByyoCmKZhKmI1ji37flqCFlt3zLK9CDlu0gVaSvRh3zgsjXEVftT7chti9S4RY+qTHpEhh6RrXFsV77/laE3Hf/TxVk2FiiE0bEQeqXLily5/nVbNtlY2rftdTrrWbLQr+8hiFZWWJFJtaImLYlsCw/VaM56Fpq0hGkBaQnju4Qxu4Rp/X8rxoR+/Sa2Y7IWwi58mdyT1jfNgRiUtnysq8M2jeIBjLrUSS1LV85lz3rWVvTL2vKWdECWK/LyUZ31rHBYm7cbPp06zMBVd06f9axg2OmuqOq6YdHpw3EoO2psQ1zL3jA+61m60C9H/QPHbTqsvOxB+7LDwFbpwkCnH8DArsFXcS92aZw6nThmTkqXZKrkUmz3Y7OY399NmoUouWbJwN8hTV3LdLbtZrbENVK07It1SdMXedltUWNUUlPRpnAlBRUl/xQl+RQl8xRtdlcyTFHSS1FyS1ESS9EmfiWBFCV7FCV1FCVvFKZvHOah4zC741iFwVFBUZRJZKf7Mq09oPsyN3e6X4WIUQlmokSZ3e7jQ7sPO913vX/VetEafzPnuRIxcq5dQCzZpijMEmWuzGl2IaQoJN9Rs+2mR820rWYh8kZNv0rGRoU2wvWh4OJDu99Bwa88clQECd+nvX+o9n5Xe73uvmjv+7T3D9Xe72pvHm3D4uOisEMUAVxxvjJiKDCEPhjCIRj8DtfCLgxrjw8FhtAHQ4gP7X4XhrUjxqJ97NM+PlT7uKv92hFj0T72aR8fqn3c1f7xjuiL2UIxYihXcWPEVGBIfTCkgzDskC3twBDWnpgKDKkPhhQf2v0ODGHtidxekwON7NOfW20eJkHJ3RHojQjXivDtMbTHdkaS7VsE1apS3tjk/Tk9KpX58SEqKbOj0uM9OBbVUjF+ahWVtvNmZK8n5du3Ja3ft5FlRc6VovXwTXM1Hb+Y/z2/OXPJVXmXj6riagX1y7tmvJzfnbwef2g+il+my8/iM6Td3TXX4nY8+SuEDvYybj5RMn3KqZnZ5Do5s0xtZus6j3JbWfhORqpK/d2Hg5wr3S8/z+9W+dq4LVhVSPl929X9pLk7+dhQfVkl/MVBGdV0PjsfL5uT82+11EZqLZHle+1PpfxX/DsoAPxw28yeT1i5zIfIpJZ1BvvN/KoZ/rRofrhf3kxnwD6/1cqboFhvdXuTlNFAa9tmm70cL8c380+ZomqkEW/wq1UYaWdqE0cOVzaOogm1s5hwNVec1x4F3tVBI7qbOrBA1cGjka2jRMDh7onAPQV1dCOTfB3DyGhuakA/HvV9qpOKIx1MXa8JxDeY67eqLy/Oec0CPXwxXjS59Ps/v3795vk3xOT9eLY4vYD600lued4sJnfTW3Alb/LJbnBx/v7rYtl8uZhdz7MHfZoulndfT55fzS9Bmx/4znY6+3RycQXWT5dfBxB/e3vTfKETSKS+5z8L59Tw57yTqO3yw/yPF+dvxrfDVauOD2wrMny+mLCn4B1fJ+XzU23l8D10+ouIAfjfftdMP30udZ7/7dPP0ysw3Dmdu3pBzzjlprdTzY1VOu/4s7EeloE/n326acSpQu2b8SfMaOz6K1zhGYw9my+aZ5J/Wl5Kj38MnnotrrS0Z+0bY1Lm11B9Nb1puHtTrgFt4N1xEzvVyCANxa9WHjbWmUDGY8qQ3JiIVCbUVsLikntOaosSb8mpZE3tFDKYWDuDm5JIgyGJ9PIILz7vpmiPiVs3SSknvOMOU9AP1TR3gjpMAjrf9iqRcRZRh9vBLPevObZKIjpfJz3K20pjrJWxVDGz0Sed2WiMehwbz99+/+ZPrze4vZjfXO1xMeojcjFkLraZxTG5iJTpEBm96ZKRmxVPdeLeOW5bRNjcJqNsqai9xvPfHh1jpt8ljofpeBDWFRntNhldh4zcLotfrdxIW+485m5lj2euUOs0wmkNw3O/TsRERmZ57v+FWwXcc8j1Hff3YhIKJtYWnXiyyvIjpEjIBILjPlluKdbliEdK8BpNTW6GKUwkPC6W+Md9vqr2fmRtiZqOO3uNsKEcHXckG+7GilkKom7+OTx38sitnryf3SFvYEZbyZ0xcAFuYKZmyFoCAh73lTp+ofRgtwyI2RooROTGuPZ0NFxDFX46BIR0sVolOUIftdOIz8GX2Nx6g4YnPsobXrz9y/MPf16bbd8R/DEdQWdHCL9nUI6p4wfcEnuqy+5V2xOUV35gD7rAr0XkPixX7I/b7F+96MLsjYdZxF6XD1bzQCbVTuY7zm3upHwHrMgHsypY/epQmgRdDnbdMoR8J6q9JrGIT75bAiaua+Lcds6LCv+YYXLDsI9vP/709v03r9+8mc/mSp6COvc347sjUGqXq+WBhkzTUZJpSe/kzQdIFWKHVFq3pPKpSyrU2ZBKyu5Mjxnw1DAC5Yle+X5Ohd5ZXua32kl2/rY4dQi9Fbf4P2dQW9xKpmfQaxvtGOdAEvFvs8n8CtivHfP0uxahqzHkzvNrhvIs8GH+02yK2o1QwT5Ccu988RC5cV+ue4TcXSd9iEgl92X6R8g8YMTTjsiHqGHNvhqrtw7/DTxBGpIKZW5kc3RyZWFtCmVuZG9iagoxOTQgMCBvYmoKPDwvVHlwZS9YUmVmL1Jvb3QgMSAwIFIvSW5mbyAyIDAgUi9JRFs8MWQ1NmE3MjNhZDhmYzYxMmM5NzRlZTg5NmFkZTRlYzA+PDFkNTZhNzIzYWQ4ZmM2MTJjOTc0ZWU4OTZhZGU0ZWMwPl0vU2l6ZQoxOTUvV1sxIDIgMl0vRmlsdGVyL0ZsYXRlRGVjb2RlL0xlbmd0aCA0NDI+PgpzdHJlYW0KeNot0+kzVXEcx/Hf95I1a6S6KmvZEkolRLZrJ0u2UMjSTauyLyXLJaQQ0jZNU0/NeOxP6A/oYc/6K3Te8+3Ja37n8z4zZ86ZOcaYgwOb8TOr8AbuiwkwRoIqjOHyHXyCDfgMW/AFjOx/0/tE7L56ssEDcAN3OAQe4Ale4A0+4AuHwQ/8IQACIQiC4YiEr+ozQuARhMJRiRjWEAZPYAe+wjE4LtGZessJeAZ2CJe4nxpOwiCcgtOSGKUhAoYhEqIk+beGaBiFGIiV1D0NZ2ACzkKcpPdqiIcpSIBEycjXkAQv4RwkS7aPhvPwClIgVXJnNaTBHFyAi1Jg05AOLrgEl8XRr+EKLEIGXJWyKg2ZsARZkA3XIAdy4TrkQb6U/3/9AqiBUiiDciiECqiFSqiCIqiGOqiHm+CABmiEJrgBxVACzXAP2qEFbkErtEEn3IY70AFdcBf6oBt6oBdW4Ck4oR8ewmMYgwF4DkMwAjMwDpPwAqZhGWZhHhbgtVT3WH9KTaRF7V+L+u/6/dbgrTTMWJvTodt7WBfnD2tz2XXbhk1xTVrb7oBuH2X3j54+yH6xFX71GfMPZRVWsQplbmRzdHJlYW0KZW5kb2JqCnN0YXJ0eHJlZgo1Mjg0MQolJUVPRgo=
ITSM::ConfigItem::Class
Computer
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Class
Hardware
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Class
Location
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Class
Network
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Class
Software
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::DeploymentState
Expired
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::DeploymentState
Inactive
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::DeploymentState
Maintenance
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::DeploymentState
Pilot
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::DeploymentState
Planned
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::DeploymentState
Production
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::DeploymentState
Repair
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::DeploymentState
Retired
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::DeploymentState
Review
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::DeploymentState
Test/QA
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::YesNo
Yes
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::YesNo
No
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Computer::Type
Laptop
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Computer::Type
Desktop
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Computer::Type
Phone
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Computer::Type
PDA
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Computer::Type
Server
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Computer::Type
Other
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
Monitor
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
Printer
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
Switch
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
Router
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
WLAN Access Point
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
Security Device
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
Backup Device
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
Mouse
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
Keyboard
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
Camera
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
Beamer
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
Modem
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
PCMCIA Card
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
USB Device
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
Docking Station
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
Scanner
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Hardware::Type
Other
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Location::Type
Building
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Location::Type
Office
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Location::Type
Floor
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Location::Type
Room
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Location::Type
Rack
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Location::Type
Workplace
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Location::Type
Outlet
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Location::Type
IT Facility
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Location::Type
Other
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Network::Type
LAN
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Network::Type
WLAN
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Network::Type
Telco
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Network::Type
GSM
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Network::Type
Other
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::Type
Client Application
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::Type
Middleware
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::Type
Server Application
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::Type
Client OS
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::Type
Server OS
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::Type
Admin Tool
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::Type
User Tool
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::Type
Embedded
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::Type
Other
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::LicenceType
Single Licence
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::LicenceType
Per User
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::LicenceType
Concurrent Users
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::LicenceType
Per Processor
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::LicenceType
Per Server
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::LicenceType
Per Node
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::LicenceType
Volume Licence
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::LicenceType
Enterprise Licence
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::LicenceType
Developer Licence
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::LicenceType
Demo
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::LicenceType
Time Restricted
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::LicenceType
Freeware
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::LicenceType
Open Source
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Software::LicenceType
Unlimited
1
current_timestamp
1
current_timestamp
1
ConfigItemCreate
1
current_timestamp
1
current_timestamp
1
ConfigItemDelete
1
current_timestamp
1
current_timestamp
1
LinkAdd
1
current_timestamp
1
current_timestamp
1
LinkDelete
1
current_timestamp
1
current_timestamp
1
NameUpdate
1
current_timestamp
1
current_timestamp
1
VersionCreate
1
current_timestamp
1
current_timestamp
1
ValueUpdate
1
current_timestamp
1
current_timestamp
1
DefinitionUpdate
1
current_timestamp
1
current_timestamp
1
IncidentStateUpdate
1
current_timestamp
1
current_timestamp
1
DeploymentStateUpdate
1
current_timestamp
1
current_timestamp
1
VersionDelete
1
current_timestamp
1
current_timestamp
1
AttachmentAdd
1
current_timestamp
1
current_timestamp
1
AttachmentDelete
1
current_timestamp
1
current_timestamp
1
ITSM::ConfigItem::Class
Location
1
current_timestamp
1
current_timestamp
1
ConfigItemCreate
1
current_timestamp
1
current_timestamp
1
ConfigItemDelete
1
current_timestamp
1
current_timestamp
1
LinkAdd
1
current_timestamp
1
current_timestamp
1
LinkDelete
1
current_timestamp
1
current_timestamp
1
NameUpdate
1
current_timestamp
1
current_timestamp
1
VersionCreate
1
current_timestamp
1
current_timestamp
1
ValueUpdate
1
current_timestamp
1
current_timestamp
1
DefinitionUpdate
1
current_timestamp
1
current_timestamp
1
IncidentStateUpdate
1
current_timestamp
1
current_timestamp
1
DeploymentStateUpdate
1
current_timestamp
1
current_timestamp
1
VersionDelete
1
current_timestamp
1
current_timestamp
1
AttachmentAdd
1
current_timestamp
1
current_timestamp
1
AttachmentDelete
1
current_timestamp
1
current_timestamp
1
{Name}->{Content};
$Kernel::OM->Get($CodeModule)->CodeInstall();
]]>
{Name}->{Content};
$Kernel::OM->Get($CodeModule)->CodeReinstall();
]]>
{Name}->{Content};
# discard internally stored object, so that the next access to object creates them newly
$Kernel::OM->ObjectsDiscard(
Objects => [$CodeModule],
ForcePackageReload => 1,
);
$Kernel::OM->Get($CodeModule)->CodeUpgrade();
]]>
{Name}->{Content};
$Kernel::OM->Get($CodeModule)->CodeUninstall();
]]>
Get('Kernel::System::Stats')->StatsCleanUp(
ObjectNames => [ 'ITSMConfigItem' ],
UserID => 1,
);
}
]]>
WELCOME
You are about to install the Znuny package ITSMConfigurationManagement.
((enjoy))
]]>
WILLKOMMEN
Sie sind im Begriff das Znuny-Paket ITSMConfigurationManagement zu installieren.
((enjoy))
]]>
BIENVENIDO
Usted está punto de instalar el paquete ITSMConfigurationManagement de Znuny.
((enjoy))
]]>
ÜDVÖZÖLJÜK
Ön az ITSMConfigurationManagement Znuny csomag telepítésére készül.
((enjoy))
]]>
NOTICE
In order to grant users access to the config item menu, you need to add them as member to the group 'itsm-configitem'.
The menu items that were added by this package will be visible after you log-in to the system again.
((enjoy))
]]>
HINWEIS
Um Benutzern Zugriff auf das ConfigItem-Menü zu gewähren, müssen diese Mitglied der neuen Gruppe 'itsm-configitem' sein.
Die von diesem Paket hinzugefügten Menü-Punkte sind erst nach einem erneuten Anmeldevorgang im System sichtbar.
((enjoy))
]]>
ATENCIÓN
Para permitirles a los usuarios el acceso al menú de los elementos de configuración, es necesario que se den de
alta como miembros del grupo 'itsm-configitem'.
Los elementos del menú que se agregaron por este paquete, serpan visibles después reiniciar la sesión al sistema.
((enjoy))
]]>
FIGYELMEZTETÉS
Annak érdekében, hogy hozzáférést adjon a felhasználók számára a konfigurációelem menühöz, hozzá kell adnia őket az „itsm-configitem” csoporthoz tagként.
Azok a menüpontok, amelyeket ez a csomag adott hozzá, azután lesznek láthatóak, miután ismét bejelentkezik a rendszerbe.
((enjoy))
]]>
WELCOME
You are about to upgrade the Znuny package ITSMConfigurationManagement.
((enjoy))
]]>
WILLKOMMEN
Sie sind im Begriff das Znuny-Paket ITSMConfigurationManagement zu aktualisieren.
((enjoy))
]]>
BIENVENIDO
Usted está a punto de actualizar el paquete ITSMConfigurationManagement de Znuny.
((enjoy))
]]>
ÜDVÖZÖLJÜK
Ön az ITSMConfigurationManagement Znuny csomag frissítésére készül.
((enjoy))
]]>
ATTENTION
If you uninstall this package, all database tables that were created during installation will be deleted.
All data from these tables will be irrevocably lost!
The group 'itsm-configitem' that was created during package installation will be deactivated.
You can activate this group again in the admin area.
All links to config item objects will be irrevocably deleted!
The statistics that are associated with this package, will no longer be available if the package is uninstalled.
((enjoy))
]]>
ACHTUNG
Bei der Deinstallation werden die von diesem Paket angelegten Datenbank-Tabellen gelöscht.
Alle darin enthaltenen Daten gehen unwiderruflich verloren!
Die von diesem Paket angelegte Gruppe 'itsm-configitem' wird deaktiviert.
Sie kann jederzeit im Admin-Bereich wieder aktiviert werden.
Alle Verknüpfungen zu ConfigItem-Objekten werden unwiderruflich gelöscht!
Nach der Deinstallation sind die zu diesem Paket zugehörigen Statistiken nicht mehr verfügbar.
((enjoy))
]]>
ATENCIÓN
Si usted desinstala este paquete, todas las tablas de la base de datos que fueron creadas durante su instalación se eliminarán.
¡Todos los datos de dichas tablas se perderán irrevocablemente!
El grupo 'itsm-configitem' que se creó durante la instalación de este paquete se desactivará.
Es posible reactivar este grupo en el área de administración.
¡Todos los vínculos a objetos de elementos de configuración serán eliminados permanentemente!
Si usted desinstala este paquete, las estadísticas que están asociadas a él ya no estarán disponibles.
((enjoy))
]]>
FIGYELEM
Ha eltávolítja ezt a csomagot, akkor a telepítés során létrehozott összes adatbázistábla törlésre kerül.
Az ezekben a táblákban lévő összes adat visszavonhatatlanul el fog veszni!
A csomagtelepítés során létrehozott „itsm-configitem” csoport ki lesz kapcsolva.
Ezt a csoportot az adminisztrációs területen tudja ismét bekapcsolni.
A konfigurációelem objektumokra mutató összes hivatkozás visszavonhatatlanul törölve lesz!
Az ehhez a csomaghoz hozzárendelt statisztikák többé nem lesznek elérhetők, ha a csomag eltávolításra kerül.
((enjoy))
]]>