JIRAPackage
6.1.0
catworkx GmbH
http://www.catworkx.com
GNU AFFERO GENERAL PUBLIC LICENSE Version 3, November 2007
fix AMSOTRSTOJ-204: avoid double '/' in link to OTRS
Fix bug: ArticleGet is *not* a method of the ArticleObject (use the BackendObject instead)
Fix bug: Filling Multiselects resulted in a 400 response code as the values weren't build correctly. This is fixed now.
Fix bug: Jira user was marked as invalid as the list of assignable users was too short. Ask for 1000 users, now...
Fix bug: Server error occured on closing a ticket due to a typo
port Addon for OTRS6
fix ACL handling. Now it is possible to hide the button via ACLs
add queue specific settings (project and assignee)
do not search for description. search term should search for issue id as well
clean up summary - all vertical spaces are removed
new feature: set format for comments to be transferred to JIRA
new feature: setting the reporter can be disabled
fix bug: mapping created with the wizard is correct now (std:reporter:name instead of std:reporter)
fix bug: custom fields had a superfluous ':' in the mapping
fix bug: select fields weren't handled correctly.
Fix sending attachments with newer JIRA::REST releases
Improve wizard
Improve wizard
Add wizard for JIRA connector configuration
fix bug: utf-8 chars in attachment filenames led to internal server error
when no article is available, the ticket data is taken for the issue creation
Now article attachments for issue updates are sent to JIRA, too
When SysConfig option TransferTicketHistory is set to 'all articles', all articles of the ticket are sent to JIRA
Improve options for dialog: Now you can really hide the fields as well (before you could only make them editable or readonly)
add support for 'automatic-user'
In the dialog you can link a ticket to an existing JIRA issue.
add whitelist and blacklist options. That way you can define which projects should be shown in the dialog. Those options can be perl regular expression and the project *keys* are checked whether they match or not.
Fix bug: Do not translate Projects and Assignees in dialog
Support creating issues for specific articles
Minor UI improvements
Improve translation file
use dynamic field backend object to set JIRAIssueID field (instead of value object). Now the field update events are triggered.
add sysconfig option to suppress assignee setting in issue create request
Fix bug: When JIRA issue is closed in JIRA and this triggered an state update of the OTRS ticket, this triggered an event to close the JIRA issue. Now it is checked
if the agent who closes is the JIRA user, then the event is skipped.
Do not check field type for standard fields.
do not show subtasks in dialog (OTRSJIRA-108)
make assignee field in dialoag an autocomplete field (OTRSJIRA-109)
Followup for OTRSJIRA-98
Add dialog to create issue, fix bug OTRSJIRA-98
Fixed typo in package name
Moved TicketMenu module to correct folder.
Switched to OTRS::OPM::Maker::Command::sopm to generate .sopm file
Moved AgentTicketJIRA.tt to correct folder.
Added compatibility with OTRS 5.
Fixed a bug preventing automatic issue creation.
Do not update dynamic field.
Make debugging configurable, do not verify hostnames on SSL connections when configured.
Do not run ticket events when JIRA user triggers them.
Fix bug in typechecks.
Add typechecks, more debugging and fix update bug.
updates for OTRS4
Support IDs and names for custom fields and fix handling of some field types.
Switched to JIRA REST API.
Added article to mapFields call.
Cleanup in dynamic field code.
Removed authentication for license check. The API is available for anon access anyway.
Removed unnecessary returns.
Removed a strange if condition.
Removed obsolete and broken dependecy.
Attempt to debug JIRA 6.0 problems with JIRA::Client.
Bugfix.
Added licensing code.
Changed demo code for right hand side.
Finalized eval call again.
Bugfix.
Finalized eval call.
Added missing %Ticket.
Debugging.
Readded the $Self to the eval calls.
Changed eval call for external modules again.
Changed eval call for external modules.
Bugfix.
Bugfix.
Typo in config.
Added code to warn upon eval failure.
License info.
Skip file-2 attachment, fixes OTRSJIRA-3.
Bugfix for OTRSJIRA-5.
Bugfix.
Bugfix.
Bugfix for Perl 5.10 (SLES11SP2).
Lowered required module versions to work with Perl 5.10.
Rewrote date formatting completely.
Do not handle Checkboxes (for now).
Dynamic field data can be transferred.
Individial mapping for dynamic field types.
Changes to mapping code.
Bugfix.
Actually map field values if anything is present.
Avoid bounces and loops by checking the username of the user triggering an update.
Pass the correct variables down to the Jira Module.
More dependencies for Events. Better logging.
More dependencies for Events.
Readded original headers for actions. Corrected Actions dependencies. Fixed broken setup for ArticleCreateEvent. Added multiple events for TicketUpdate.
Removed broken dependency.
Fixed requirement check for events.
Fixed language typo.
Added more events.
Try to add the JIRAIssueID programatically.
Fix for the typo. 3.2.x Version.
Major rewrite of the OTRS2JIRA fieldmapping.
Added code to transform OTRS-dynamic fields/values to JIRA-custom fields/values and pass them to JIRA.
Added code to pass attachments to JIRA.
Added code to read configured Jira-project and Jira-issue from the 'SysConfig'.
Added ConfigItem to JIRA XML file and made field mapping from OTRS to JIRA configurable.
Added ConfigItem to JIRA XML file and made JIRA-date format configurable.
Added ConfigItem to JIRA XML file and made JIRA-Issue type configurable.
Added ConfigItem to JIRA XML file and made JIRA-project configurable.
Added code to check the OTRS Tickets state prior to any action. States are now configurable.
Added code to fetch the possible transitions of the given JIRA issue before attempting the transition.
Added template for showing an error in case a ticket is closed when one wants to open a JIRA issue.
Added code to pass the last article body to JIRA upon closing.
Added code to pass the article body to JIRA upon creation and comment the closing of a ticket in JIRA appropriately.
Added event code to close the JIRA issue when the ticket is closed.
1.1.10 - 2012-05-16 13:45:0
Added a hyperlink to JIRA.
Fixed the issue creation code.
Cleanup.
Added the options into the XML file.
Added error handling to the installer.
Added debugging to the installer.
Bugifxed the code for adding the JiraIssueID to the searchable and visible fields.
Added code for adding the JiraIssueID to the searchable and visible fields.
Bugfix for REST code for the remote issue links.
Bugfix for REST code for the remote issue links.
Added REST code for the remote issue links.
Uninstaller and Upgrader code done.
Added assignee and reporter. Nicer statuspage.
Updated MD5 dependecy.
New MD5 code in installer.
Updated DTL.
Updated SQL.
Debugging for dynamic field.
Added missing backslashes.
Added dynamic field.
New package.
Ein Modul, um Tickets in ein Jira zu eskalieren.
A module giving the option to escalate tickets into Jira.
6.0.x
Thank you for choosing the JIRAPackage module.<br />
Copyright (C) 2008-2016 catworkx GmbH http://www.catworkx.de<br />
This is commercial software.<br />
Please install the otrs2jira-licensing plugin into your Jira installation and obtain a valid license.<br />
Please contact catworkx GmbH for a valid license.
Vielen Dank für die Auswahl des JIRAPackage Modules.<br />
Copyright (C) 2008-2016 catworkx GmbH http://www.catworkx.de<br />
Dies ist kommerzielle Software.<br />
Bitte installieren Sie das Plugin otrs2jira-licensing in Ihre Jira Installation und hinterlegen dort eine gültige Lizenz.<br />
Bitte setzen Sie sich mit der catworkx GmbH bzgl. einer gültigen Lizenz in Verbindung.
JIRA::REST
REST::Client
Time::Piece
$Kernel::OM->Get('var::packagesetup::' . $Param{Structure}->{Name}->{Content} )->CodeInstall();
$Kernel::OM->Get('var::packagesetup::' . $Param{Structure}->{Name}->{Content} )->CodeUpgrade();
2020-01-27 06:50:02
debian.local
cGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6SklSQU1hcHBpbmc7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9IHF3KAogICAgS2VybmVsOjpTeXN0ZW06OlRpbWUKKTsKCnN1YiBUb2RheSB7CiAgICBteSAkVGltZU9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpUaW1lJyk7CiAgICBteSAoJFNlYywgJE1pbiwgJEhvdXIsICREYXksICRNb24sICRZZWFyKSAgPSAkVGltZU9iamVjdC0+U3lzdGVtVGltZTJEYXRlKAogICAgICAgIFN5c3RlbVRpbWUgPT4gJFRpbWVPYmplY3QtPlN5c3RlbVRpbWUsCiAgICApOwoKICAgIG15ICgkRGF0ZSkgPSBzcHJpbnRmICIlMDRkLSUwMmQtJTAyZFQwMDowMDowMC4wKzAwMDAiLCAkWWVhciwgJE1vbiwgJERheTsKCiAgICByZXR1cm4gJERhdGU7Cn0KCnN1YiBGaWxsU2VsZWN0IHsKICAgIG15ICgkU2VsZiwgJEtleSwgJUFydGljbGUpID0gQF87CgogICAgbXkgKCRNb2R1bGUsICRGdW5jdGlvbiwgJFZhbHVlKSA9IHNwbGl0IC8sLywgJEtleTsKCiAgICByZXR1cm4geyB2YWx1ZSA9PiAkVmFsdWUgfTsKfQoKc3ViIEZpbGxNdWx0aVNlbGVjdCB7CiAgICBteSAoJFNlbGYsICRLZXksICVBcnRpY2xlKSA9IEBfOwoKICAgIG15ICgkTW9kdWxlLCAkRnVuY3Rpb24sIEBWYWx1ZXMpID0gc3BsaXQgLywvLCAkS2V5OwoKICAgIHJldHVybiBbIG1hcCB7ICt7IHZhbHVlID0+ICRfIH0gfUBWYWx1ZXMgXTsKfQoKMTsK
<?xml version="1.0" encoding="UTF-8" ?>
<otrs_config version="2.0" init="Config">
    <Setting Name="Frontend::Module###AgentTicketJIRA" Required="1" Valid="1">
        <Description Lang="en">FrontendModuleRegistration for the TicketJira module.</Description>
        <Description Lang="de">FrontendModulRegistration für das TicketJira Modul.</Description>
        <Navigation>Jira::Frontend::Agent::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="Group">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="GroupRo">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="Title">TicketJira</Item>
                    <Item Key="Description">TicketJira</Item>
                    <Item Key="NavBarName">TicketJira</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Frontend::Navigation###AgentTicketJIRA###003-JIRA" Required="0" Valid="0">
        <Description Translatable="1">Main menu item registration.</Description>
        <Navigation>Jira::Frontend::Agent::ModuleRegistration::MainMenu</Navigation>
        <Value>
            <Array>
                <DefaultItem ValueType="FrontendNavigation">
                    <Hash>
                    </Hash>
                </DefaultItem>
            </Array>
        </Value>
    </Setting>
    <Setting Name="Ticket::Frontend::MenuModule###900-JIRA" Required="0" Valid="1">
        <Description Lang="en">Module to shows a Jira-link in menu of the ticket zoom.</Description>
        <Description Lang="de">Mit diesem Modul wird der Jira-Link in der Linkleiste der Ticketansicht angezeigt.</Description>
        <Navigation>Jira::Frontend::Agent::View::TicketZoom::MenuModule</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::TicketMenu::JIRA</Item>
                <Item Key="Name">Jira</Item>
                <Item Key="Action">AgentTicketJIRA</Item>
                <Item Key="Target"></Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###URL" Required="1" Valid="1">
        <Description Lang="en">The URL of the Jira server, include the protocol and the port if required a well as the base if any.</Description>
        <Description Lang="de">Die URL des Jira Servers, inklusive Protokoll sowie Port und Pfad, sofern zutreffend.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">http://cat-pc-050.catworkx.de:8080</Item>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###Path" Required="1" Valid="1">
        <Description Lang="en">The path of the REST API.</Description>
        <Description Lang="de">Der Pfad der REST API.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">/rest/api/latest/</Item>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###User" Required="1" Valid="1">
        <Description Lang="en">The username to use when logging in into Jira issues.</Description>
        <Description Lang="de">Der Benutzername mit dem sich beim Jira angemeldet wird.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">otrs</Item>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###Password" Required="1" Valid="1">
        <Description Lang="en">The password to use when logging in into Jira issues.</Description>
        <Description Lang="de">Das Passwort mit dem sich beim Jira angemeldet wird.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">otrs</Item>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###Reporter" Required="1" Valid="1">
        <Description Lang="en">The name of the reporter to use for the new Jira issues.</Description>
        <Description Lang="de">Der Name des Reporters für neue Jira Issues.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">otrs</Item>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###Assignee" Required="1" Valid="1">
        <Description Lang="en">The name of the assignee to use for the new Jira issues.</Description>
        <Description Lang="de">Der Name des Assignee für neue Jira Issues.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">admin</Item>
        </Value>
    </Setting>
    <Setting Name="JIRA::NoAssignee" Required="0" Valid="1">
        <Description Lang="en">If enabled, the assignee isn't sent to Jira.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###IncomingUser" Required="1" Valid="1">
        <Description Lang="en">The username that is used when Jira logs into OTRS to update tickets.</Description>
        <Description Lang="de">Der Benutzername mit dem sich Jira am OTRS anmeldet, um Tickets zu aktualisieren.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">jira</Item>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###WorkflowCloseActionName" Required="1" Valid="1">
        <Description Lang="en">The name of the close issue workflow action (default: "Close Issue", pay attention to the Jira language of user in TicketJIRA###User, English(UK) works).</Description>
        <Description Lang="de">Der Name der Aktion zum Schließen von Issues (Standard: "Close Issue", achten Sie auf die Jira Sprache des Benuters in TicketJIRA###User, English(UK) ist empfehlenswert).</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String">Close Issue</Item>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###OTRSTicketAllowCreationInAllQueues" Required="1" Valid="1">
        <Description Lang="en">Allow manual issue creation in all queues.</Description>
        <Description Lang="de">Manuelle Problemerstellung in allen Warteschlangen zulassen.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###OTRSTicketAllowedQueues4Create" Required="1" Valid="1">
        <Description Lang="en">The names of the OTRS Ticket queues that allow the creation of a Jira Issue.</Description>
        <Description Lang="de">Die Namen der OTRS Ticket Warteschlange die das Erzeugen eines Jira Issue zulassen.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Array>
                <DefaultItem ValueType="Entity" ValueEntityType="Queue" ValueRegex=""></DefaultItem>
            </Array>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###OTRSTicketStatesEnforcingCreate" Required="1" Valid="1">
        <Description Lang="en">The names of the states in OTRS for which a Jira issue is automatically created.</Description>
        <Description Lang="de">Die Namen der OTRS Ticket Zustände, für die automatisch ein Jira Issue erstellt wird.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String">new,open,merged,pending reminder</Item>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###OTRSTicketAllowedStates4Close" Required="1" Valid="1">
        <Description Lang="en">The names of the OTRS Ticket states that allow the closing of a Jira Issue.</Description>
        <Description Lang="de">Die Namen der OTRS Ticket Zustände die das Schließen eines Jira Issue zulassen.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String">closed successful,closed unsuccessful,removed</Item>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###OTRSTicketAllowedStates4Update" Required="1" Valid="1">
        <Description Lang="en">The names of the OTRS Ticket states that allow updating the Jira Issue.</Description>
        <Description Lang="de">Die Namen der OTRS Ticket Zustände die das Aktualisieren eines Jira Issue zulassen.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String">new,open,merged,pending reminder</Item>
        </Value>
    </Setting>
    <Setting Name="Ticket::EventModulePost###001-TicketCloseJIRA" Required="0" Valid="1">
        <Description Lang="en">Update Jira once the ticket is closed.</Description>
        <Description Lang="de">Das Jira nach dem Schließen des Tickets aktualisieren.</Description>
        <Navigation>Jira::Core::Ticket</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::Ticket::Event::CloseJIRAIssue</Item>
                <Item Key="Event">TicketStateUpdate</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Ticket::EventModulePost###001-TicketCreateJIRA" Required="0" Valid="1">
        <Description Lang="en">Create Jira Issue once the ticket is pushed into a specific state. See TicketJIRA###OTRSTicketStatesEnforcingCreate for details.</Description>
        <Description Lang="de">Erzeuge das Jira Issue sobald das OTRS Ticket in einem bestimmten Zustand angekommen ist. Für Details, siehe TicketJIRA###OTRSTicketStatesEnforcingCreate.</Description>
        <Navigation>Jira::Core::Ticket</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::Ticket::Event::CreateJIRAIssue</Item>
                <Item Key="Event">TicketStateUpdate</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Ticket::EventModulePost###001-TicketJIRAUpdate" Required="0" Valid="1">
        <Description Lang="en">Update Jira once the ticket updated the title field. Duplicate this entry for *all* fields you want to sync into Jira.</Description>
        <Description Lang="de">Das Jira beim aktualisieren des Feldes "title" ebenfalls aktualisieren. Dieser Eintrag muss für *jedes* zu übertragende Feld extra angelegt werden.</Description>
        <Navigation>Jira::Core::Ticket</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::Ticket::Event::UpdateJIRAIssue</Item>
                <Item Key="Event">TicketTitleUpdate|TicketCustomerUpdate|TicketFreeTextUpdate|TicketOwnerUpdate|TicketDynamicFieldUpdate_field001|TicketDynamicFieldUpdate_field002|TicketDynamicFieldUpdate_field003|TicketDynamicFieldUpdate_field004|TicketDynamicFieldUpdate_field005|TicketDynamicFieldUpdate_field010|TicketDynamicFieldUpdate_field011</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Ticket::EventModulePost###001-ArticleToJIRA" Required="0" Valid="1">
        <Description Lang="en">Create a Jira Comment once an article is created.</Description>
        <Description Lang="de">Erzeuge einen Jira kommentar sobald ein OTRS Artikel erzeugt wurde.</Description>
        <Navigation>Jira::Core::Ticket</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::Ticket::Event::AddArticleToJIRAIssue</Item>
                <Item Key="Event">ArticleCreate|ArticleUpdate</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###Project" Required="1" Valid="1">
        <Description Lang="en">The name of the project to use for the new Jira issues.</Description>
        <Description Lang="de">Der Name des Projektes für neue Jira Issues.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">OTRS</Item>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###IssueType" Required="1" Valid="1">
        <Description Lang="en">The name of the type to use for the new Jira issues.</Description>
        <Description Lang="de">Der Name des Typ für neue Jira Issues.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">Bug</Item>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###JIRA-DateTimeFormat" Required="1" Valid="1">
        <Description Lang="en">Date format used in Jira, e.g.: Jira: dd/MMM/yy h:mm a = OTRS %d/%b/%y %I:%M %p</Description>
        <Description Lang="de">Das von Jira akzeptierte Datumsformat, z.B.: Jira: dd/MMM/yy h:mm a = OTRS %d/%b/%y %I:%M %p</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">%d/%b/%y %I:%M %p</Item>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###JIRA-DateFormat" Required="1" Valid="1">
        <Description Lang="en">Date format used in Jira, e.g.: Jira: dd/MMM/yy = OTRS %d/%b/%y</Description>
        <Description Lang="de">Das von Jira akzeptierte Datumsformat, z.B.: Jira: dd/MMM/yy = OTRS %d/%b/%y</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">%d/%b/%y</Item>
        </Value>
    </Setting>
    <Setting Name="TicketJIRA###OTRS-JIRA-FieldMapping" Required="1" Valid="1">
        <Description Lang="en">Mapping from OTRS fields to Jira Fields. Key: OTRS; Content: Jira. Prefixes: "std:" = standard field; "dyn:" = OTRS dynamic field; "cust:" = Jira custom field; "call:" = OTRS/Jira mapping call with full Perl modulepath Pkg::Sub,func (e.g. MIME::Base64,encode_base64).</Description>
        <Description Lang="de">Abbildung der OTRS Feldern zu den Jira Feldern. Schlüssel: OTRS; Inhalt: Jira. Präfixe: "std:" = Standard Feld; "dyn:" = OTRS Dynamic Feld; "cust:" = Jira Custom Feld; "call:" = OTRS/Jira Mapping-Funktionsaufruf mit vollem Perl Modulepfad Pkg::Sub,func (z.B. MIME::Base64,encode_base64).</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Hash>
                <Item Key="dyn:dynamicField1">cust:customField1</Item>
                <Item Key="dyn:jira_prio">std:priority</Item>
                <Item Key="std:Title">std:summary</Item>
                <Item Key="std:Queue">cust:queue</Item>
                <Item Key="map:Mod::SubMod::func">cust:field1</Item>
                <Item Key="dyn:dynamicField2">map:Mod::SubMod,func2</Item>
                <Item Key="map:Mod::SubMod::func3">map:Mod::SubMod,func4</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="JIRA::OTRSTicketIDField" Required="1" Valid="1">
        <Description Lang="en">The ID of the custom field in Jira that stores the OTRS ticket ID.</Description>
        <Description Lang="de">Die ID des Feldes in Jira, das die OTRS-TicketID speichert.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="^\d+$">10000</Item>
        </Value>
    </Setting>
    <Setting Name="JIRA::CacheTTL" Required="1" Valid="1">
        <Description Lang="en">TimeToLive for Cache.</Description>
        <Description Lang="de">TimeToLive für Cache</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="^\d+$">20000</Item>
        </Value>
    </Setting>
    <Setting Name="JIRA::SSLVerify" Required="1" Valid="1">
        <Description Lang="en">Verify hostname for SSL connections.</Description>
        <Description Lang="de">Verifiziere den Hostnamen bei SSL-Verbindungen</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="Checkbox">1</Item>
        </Value>
    </Setting>
    <Setting Name="JIRA::Debug" Required="1" Valid="1">
        <Description Lang="en">Enables Debugging for Jira package.</Description>
        <Description Lang="de">Aktiviert das Debugging für das Jira-Paket</Description>
        <Navigation>Jira::Core</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="JIRA::ShowDialog" Required="0" Valid="1">
        <Description Translatable="1">Show dialog when issue in Jira is about to be created.</Description>
        <Navigation>Jira::Core</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="JIRA::TransferTicketHistory" Required="0" Valid="1">
        <Description Translatable="1">If set to 'only first article', only the first article is sent to Jira on issue creation. If set to 'all articles', all existing articles will be sent to Jira.</Description>
        <Navigation>Jira::Core</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="0">
                <Item ValueType="Option" Value="0" Translatable="1">only first article</Item>
                <Item ValueType="Option" Value="1" Translatable="1">all articles</Item>
            </Item>
        </Value>
    </Setting>
    <Setting Name="JIRA::SendReporter" Required="1" Valid="1">
        <Description Translatable="1">If disabled, no reporter is sent to Jira.</Description>
        <Navigation>Jira::CreateIssue</Navigation>
        <Value>
            <Item ValueType="Checkbox">1</Item>
        </Value>
    </Setting>
    <Setting Name="JIRA::ShowProject" Required="0" Valid="1">
        <Description Translatable="1">Show Project field in create dialog.</Description>
        <Navigation>Jira::CreateIssue</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="1">
                <Item ValueType="Option" Value="-1" Translatable="1">Hide</Item>
                <Item ValueType="Option" Value="0" Translatable="1">Readonly</Item>
                <Item ValueType="Option" Value="1" Translatable="1">Editable</Item>
            </Item>
        </Value>
    </Setting>
    <Setting Name="JIRA::ShowAssignee" Required="0" Valid="1">
        <Description Translatable="1">Show Assignee field in create dialog.</Description>
        <Navigation>Jira::CreateIssue</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="0">
                <Item ValueType="Option" Value="-1" Translatable="1">Hide</Item>
                <Item ValueType="Option" Value="0" Translatable="1">Readonly</Item>
                <Item ValueType="Option" Value="1" Translatable="1">Editable</Item>
            </Item>
        </Value>
    </Setting>
    <Setting Name="JIRA::ShowAutomaticUser" Required="0" Valid="1">
        <Description Translatable="1">Show automatic user in Assignee field in create dialog.</Description>
        <Navigation>Jira::CreateIssue</Navigation>
        <Value>
            <Item ValueType="Checkbox">0</Item>
        </Value>
    </Setting>
    <Setting Name="JIRA::ShowIssueType" Required="0" Valid="1">
        <Description Translatable="1">Show issue type field in create dialog.</Description>
        <Navigation>Jira::CreateIssue</Navigation>
        <Value>
            <Item ValueType="Select" SelectedID="0">
                <Item ValueType="Option" Value="-1" Translatable="1">Hide</Item>
                <Item ValueType="Option" Value="0" Translatable="1">Readonly</Item>
                <Item ValueType="Option" Value="1" Translatable="1">Editable</Item>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AgentTicketJIRADialog" Required="1" Valid="1">
        <Description Lang="en">FrontendModuleRegistration for the TicketJira dialog module.</Description>
        <Description Lang="de">FrontendModulRegistration für das TicketJira-Dialog Modul.</Description>
        <Navigation>Jira::Frontend::Agent::ModuleRegistration</Navigation>
        <Value>
            <Item ValueType="FrontendRegistration">
                <Hash>
                    <Item Key="Group">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="GroupRo">
                        <Array>
                        </Array>
                    </Item>
                    <Item Key="Title">TicketJira</Item>
                    <Item Key="Description">TicketJira</Item>
                    <Item Key="NavBarName">TicketJira</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Loader::Module::AgentTicketJIRADialog###003-JIRA" Required="1" Valid="1">
        <Description Translatable="1">Loader module registration for the agent interface.</Description>
        <Navigation>Jira::Frontend::Agent::ModuleRegistration::Loader</Navigation>
        <Value>
            <Hash>
                <Item Key="JavaScript">
                    <Array>
                        <Item>JIRA.Autocomplete.js</Item>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Frontend::Navigation###AgentTicketJIRADialog###003-JIRA" Required="0" Valid="0">
        <Description Translatable="1">Main menu item registration.</Description>
        <Navigation>Jira::Frontend::Agent::ModuleRegistration::MainMenu</Navigation>
        <Value>
            <Array>
                <DefaultItem ValueType="FrontendNavigation">
                    <Hash>
                    </Hash>
                </DefaultItem>
            </Array>
        </Value>
    </Setting>
    <Setting Name="Frontend::Output::FilterElementPost###OutputFilterArticleMenuOTRSToJIRALink" Required="0" Valid="1">
        <Description Translatable="1">Add a link in the article menu to create issue in Jira for the article.</Description>
        <Navigation>Jira::Frontend::Base::OutputFilter</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::Output::HTML::FilterElementPost::OTRSToJIRALink</Item>
                <Item Key="Templates">
                    <Hash>
                        <Item Key="AgentTicketZoom">1</Item>
                    </Hash>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="JIRA::ProjectWhitelist" Required="0" Valid="0">
        <Description Translatable="1">Here you can define a list of regular expressions. Only project keys that matches at least one of those regular expressions are shown in the project dropdown (Jira dialog).</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Array>
            </Array>
        </Value>
    </Setting>
    <Setting Name="JIRA::ProjectBlacklist" Required="0" Valid="0">
        <Description Translatable="1">Here you can define a list of regular expressions. Project keys that matches at least one of those regular expressions are excluded from the project dropdown (Jira dialog).</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Array>
            </Array>
        </Value>
    </Setting>
    <Setting Name="JIRA::IssueSearchMaxResults" Required="1" Valid="1">
        <Description Translatable="1">Maximum number of issues shown in the autocomplete dropdown.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="^\d+$">300</Item>
        </Value>
    </Setting>
    <Setting Name="JIRA::IssueSearchStates" Required="0" Valid="0">
        <Description Lang="en">Here you can define a list of states to restrict the issues that are searched for.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Array>
            </Array>
        </Value>
    </Setting>
    <Setting Name="Ticket::EventModulePost###101-SendAttachments" Required="0" Valid="1">
        <Description Translatable="1">Send attachments to Jira.</Description>
        <Navigation>Jira::Core::Ticket</Navigation>
        <Value>
            <Hash>
                <Item Key="Module">Kernel::System::Ticket::Event::SendAttachmentsToJIRAIssue</Item>
                <Item Key="Event">AttachmentWrite</Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Ticket::CustomModule###005-AttachmentEvents" Required="0" Valid="1">
        <Description Translatable="1">Overrides and some methods with regards to attachments.</Description>
        <Navigation>Jira::Core::Ticket</Navigation>
        <Value>
            <Item ValueType="String" ValueRegex="">Kernel::System::Ticket::Custom::AttachmentEvents</Item>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AdminJIRAWizard" Required="0" Valid="1">
        <Description Translatable="1">Wizard for basic Jira settings.</Description>
        <Navigation>Jira::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">Basic settings for Jira.</Item>
                    <Item Key="Title" Translatable="1">Jira</Item>
                    <Item Key="NavBarName">Admin</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Loader::Module::AdminJIRAWizard###003-JIRA" Required="0" Valid="1">
        <Description Translatable="1">Loader module registration for the agent interface.</Description>
        <Navigation>Jira::Frontend::Admin::ModuleRegistration::Loader</Navigation>
        <Value>
            <Hash>
                <Item Key="JavaScript">
                    <Array>
                        <Item>JIRA.AdminWizard.js</Item>
                    </Array>
                </Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="Frontend::Navigation###AdminJIRAWizard###003-JIRA" Required="0" Valid="0">
        <Description Translatable="1">Main menu item registration.</Description>
        <Navigation>Jira::Frontend::Admin::ModuleRegistration::MainMenu</Navigation>
        <Value>
            <Array>
                <DefaultItem ValueType="FrontendNavigation">
                    <Hash>
                    </Hash>
                </DefaultItem>
            </Array>
        </Value>
    </Setting>
    <Setting Name="Frontend::NavigationModule###AdminJIRAWizard" Required="0" Valid="1">
        <Description Translatable="1">Admin area navigation for the agent interface.</Description>
        <Navigation>Jira::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">Jira</Item>
                <Item Key="Block">System</Item>
                <Item Key="Description" Translatable="1">Basic settings for Jira.</Item>
                <Item Key="IconBig"></Item>
                <Item Key="IconSmall"></Item>
            </Hash>
        </Value>
    </Setting>
    <Setting Name="JIRA::ArticleFormat" Required="0" Valid="0">
        <Description Translatable="1">Format for Article content that is sent to Jira.</Description>
        <Navigation>Jira::Core::REST</Navigation>
        <Value>
            <Item ValueType="Textarea"><![CDATA[[% Data.From %] / [% Data.To %]
[% Data.Body %] ]]>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Frontend::Module###AdminJIRAQueueSettings" Required="0" Valid="1">
        <Description Translatable="1">Queue specific settings for dialog.</Description>
        <Navigation>Jira::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">Queue specific settings for Jira.</Item>
                    <Item Key="Title" Translatable="1">Jira</Item>
                    <Item Key="NavBarName">Admin</Item>
                </Hash>
            </Item>
        </Value>
    </Setting>
    <Setting Name="Frontend::Navigation###AdminJIRAQueueSettings###003-JIRA" Required="0" Valid="0">
        <Description Translatable="1">Main menu item registration.</Description>
        <Navigation>Jira::Frontend::Admin::ModuleRegistration::MainMenu</Navigation>
        <Value>
            <Array>
                <DefaultItem ValueType="FrontendNavigation">
                    <Hash>
                    </Hash>
                </DefaultItem>
            </Array>
        </Value>
    </Setting>
    <Setting Name="Frontend::NavigationModule###AdminJIRAQueueSettings" Required="0" Valid="1">
        <Description Translatable="1">Admin area navigation for the agent interface.</Description>
        <Navigation>Jira::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">Queue settings for Jira</Item>
                <Item Key="Block">System</Item>
                <Item Key="Description" Translatable="1">Queue specific settings for Jira.</Item>
                <Item Key="IconBig"></Item>
                <Item Key="IconSmall"></Item>
            </Hash>
        </Value>
    </Setting>

</otrs_config>

IyAgLS0KIyAgS2VybmVsL0xhbmd1YWdlL2RlX0FnZW50VGlja2V0SklSQS5wbSAtIGxhbmd1YWdlIG1vZHVsZQojICBDb3B5cmlnaHQgKEMpIDIwMDgtMjAxNSBjYXR3b3JreCBHbWJIIGh0dHA6Ly93d3cuY2F0d29ya3guZGUKIyAgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmRlX0FnZW50VGlja2V0SklSQTsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgIG15ICRMYW5nID0gJFNlbGYtPntUcmFuc2xhdGlvbn07CgogICAgJExhbmctPnsnQSBuZXcgSmlyYSBJc3N1ZSBoYXMgYmVlbiBjcmVhdGVkJ30gPSAnRWluIG5ldWVzIEppcmEgSXNzdWUgd3VyZGUgZXJzdGVsbHQnOwogICAgJExhbmctPnsnQSBKaXJhIElzc3VlIGNvdWxkIG5vdCBiZSBjcmVhdGVkJ30gPSAnRWluIEppcmEgSXNzdWUga29ubnRlIG5pY2h0IGVyc3RlbGx0IHdlcmRlbic7CiAgICAkTGFuZy0+eydUbyB0aGUgaXNzdWUgKEppcmEpJ30gICAgICAgICAgICAgICA9ICdadW0gSmlyYSBJc3N1ZSc7CiAgICAkTGFuZy0+eydCYWNrIHRvIHRoZSB0aWNrZXQgKE9UUlMpJ30gICAgICAgICA9ICdadXLDvGNrIHp1bSBPVFJTIFRpY2tldCc7CiAgICAkTGFuZy0+eydUaWNrZXQgbm90IG9wZW4uJ30gICAgICAgICAgICAgICAgICA9ICdUaWNrZXQgbmljaHQgaW0gb2ZmZW5lbiBadXN0YW5kLic7CiAgICAkTGFuZy0+eydKaXJhIHRvIE9UUlMnfSAgICAgICAgICAgICAgICAgICAgICA9ICdKaXJhIG5hY2ggT1RSUyc7CiAgICAkTGFuZy0+eydUYWtlIHRoaXMgdGlja2V0IGJhY2sgZnJvbSBKaXJhJ30gICA9ICdEaWVzZXMgVGlja2V0IGF1cyBKaXJhIHp1csO8Y2tob2xlbic7CiAgICAkTGFuZy0+eydPVFJTIHRvIEppcmEnfSAgICAgICAgICAgICAgICAgICAgICA9ICdPVFJTIG5hY2ggSmlyYSc7CiAgICAkTGFuZy0+eydFc2NhbGF0ZSB0aGlzIHRpY2tldCB0byBKaXJhJ30gICAgICA9ICdEaWVzZXMgVGlja2V0IG5hY2ggSmlyYSBlc2thbGllcmVuJzsKICAgICRMYW5nLT57J0NyZWF0ZSBpc3N1ZSBpbiBKaXJhJ30gICAgICAgICAgICAgID0gJ1ZvcmdhbmcgaW4gSmlyYSBlcnN0ZWxsZW4nOwogICAgJExhbmctPnsnUHJvamVjdCd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnUHJvamVrdCc7CiAgICAkTGFuZy0+eydBc3NpZ25lZSd9ICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdCZWFyYmVpdGVyJzsKICAgICRMYW5nLT57J0lzc3VlVHlwZSd9ICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1ZvcmdhbmdzdHlwJzsKICAgICRMYW5nLT57J0lzc3VlJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1ZvcmdhbmcnOwogICAgJExhbmctPnsnQ3JlYXRlIG5ldyBpc3N1ZSd9ICAgICAgICAgICAgICAgICAgPSAnTmV1ZW4gVm9yZ2FuZyBlcnN0ZWxsZW4nOwogICAgJExhbmctPnsnTGluayB3aXRoIEppcmEgaXNzdWUnfSAgICAgICAgICAgICAgPSAnTWl0IEppcmEtVm9yZ2FuZyB2ZXJrbsO8cGZlbic7CiAgICAkTGFuZy0+eydMaW5rIHdpdGggZXhpc3RpbmcgaXNzdWUnfSAgICAgICAgICA9ICdNaXQgYmVzdGVoZW5kZW0gVm9yZ2FuZyB2ZXJrbsO8cGZlbic7CiAgICAkTGFuZy0+e0VkaXRhYmxlfSAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICfDhG5kZXJiYXInOwogICAgJExhbmctPntIaWRlfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnQXVzYmxlbmRlbic7CiAgICAkTGFuZy0+e1JlYWRvbmx5fSAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdOdXIgQW56ZWlnZSc7CiAgICAKICAgICRMYW5nLT57J2FsbCBhcnRpY2xlcyd9ID0gJ0FsbGUgQXJ0aWtlbCc7CiAgICAkTGFuZy0+eydvbmx5IGZpcnN0IGFydGljbGUnfSA9ICdOdXIgZXJzdGVyIEFydGlrZWwnOwoKICAgICRMYW5nLT57J0dvIHRvIEFkbWluIGFyZWEnfSA9ICdHZWhlIHp1bSBBZG1pbmJlcmVpY2gnOwogICAgJExhbmctPnsnQ29uZmlndXJhdGlvbiBkb25lJ30gPSAnS29uZmlndXJhdGlvbiBlcmxlZGlndCc7CiAgICAkTGFuZy0+eydOZXh0J30gPSAnV2VpdGVyJzsKICAgICRMYW5nLT57J09UUlMgVGlja2V0IEZpZWxkJ30gPSAnT1RSUyBUaWNrZXRmZWxkJzsKICAgICRMYW5nLT57J0ppcmEgRmllbGQnfSA9ICdKaXJhIEZlbGQnOwogICAgJExhbmctPnsnSmlyYSBTZXR0aW5ncyd9ID0gJ0ppcmEgRWluc3RlbGx1bmdlbic7CiAgICAkTGFuZy0+eydKaXJhIGNvcmUgc2V0dGluZ3MnfSA9ICdKaXJhIEdydW5kZWluc3RlbGx1bmdlbic7CiAgICAkTGFuZy0+eydJc3N1ZSBTZXR0aW5ncyd9ID0gJ0VpbnN0ZWxsdW5nZW4gSmlyYSBWb3JmYWxsJzsKICAgICRMYW5nLT57J0lzc3VlIHNldHRpbmdzJ30gPSAnRWluc3RlbGx1bmdlbiBKaXJhIFZvcmZhbGwnOwogICAgJExhbmctPnsnRmllbGRtYXBwaW5nJ30gID0gJ0ZlbGR6dW9yZG51bmcnOwoKICAgICRMYW5nLT57J0ppcmEgVVJMJ30gPSAnSmlyYSBVUkwnOwogICAgJExhbmctPnsnQVBJIFBhdGgnfSA9ICdBUEkgUGZhZCc7CiAgICAkTGFuZy0+eydVc2VyJ30gICAgID0gJ0JlbnV0emVyJzsKICAgICRMYW5nLT57J0ppcmEgdXNlciBPVFJTIHVzZXMgZm9yIEFQSSBjYWxscyd9ID0gJ0ppcmEtQmVudXR6ZXIsIGRlbiBPVFJTIGbDvHIgZGllIEFQSS1BdWZydWZlIGJlbnV0enQnOwogICAgJExhbmctPntQYXNzd29yZH0gPSAnUGFzc3dvcnQnOwogICAgJExhbmctPnsnSW5jb21pbmcgVXNlcid9ID0gJ0VpbmdlaGVuZGVyIEJlbnV0emVyJzsKICAgICRMYW5nLT57J09UUlMgdXNlciBKaXJhIHVzZXMgdG8gY2FsbCBPVFJTIFdlYnNlcnZpY2UnfSA9ICdPVFJTLUJlbnV0emVyLCBkZW4gSmlyYSBmw7xyIGRpZSBPVFJTLVdlYnNlcnZpY2VzIGJlbnV0enQnOwogICAgJExhbmctPnsnVXNlIERpYWxvZyd9ID0gJ051dHplIGRlbiBEaWFsb2cnOwogICAgJExhbmctPnsnV2hlbiB1c2luZyB0aGUgZGlhbG9nLCB5b3UgY2FuIGVkaXQgc29tZSBzZXR0aW5ncyBmb3IgdGhlIGlzc3VlIChlLmcuIHByb2plY3QsIGFzc2lnbmVlLi4uKSd9ID0KICAgICAgICAnV2VubiBkZXIgRGlhbG9nIHZlcndlbmRldCB3aXJkLCBrw7ZubmVuIGVpbmlnZSBFaW5zdGVsbHVuZ2VuIGbDvHIgZGVuIFZvcmZhbGwgKHouQi4gUHJvamVrdCwgQmVzaXR6ZXIsIC4uLikgZ2XDpG5kZXJ0IHdlcmRlbi4nOwogICAgJExhbmctPnsnU2hvdyBBc3NpZ25lZSAoZGlhbG9nKSd9ID0gJ1plaWdlIEJlc2l0emVyIChpbSBEaWFsb2cpJzsKICAgICRMYW5nLT57J1Nob3cgIkF1dG9tYXRpYyIgdXNlciAoZGlhbG9nKSd9ID0gJ1N0ZWxsZSBkZW4gImF1dG9tYXRpc2NoZW4iIEJlbnV0emVyIHp1ciBBdXN3YWhsJzsKICAgICRMYW5nLT57J1Nob3cgSXNzdWVUeXBlIChkaWFsb2cpJ30gPSAnWmVpZ2UgVm9yZmFsbHN0eXAgKGltIERpYWxvZyknOwogICAgJExhbmctPnsnU2hvdyBQcm9qZWN0IChkaWFsb2cpJ30gPSAnWmVpZ2UgUHJvamVrdCAoaW0gRGlhbG9nKSc7CiAgICAkTGFuZy0+eydQcm9qZWN0J30gPSAnUHJvamVrdCc7CiAgICAkTGFuZy0+eydJc3N1ZSBUeXBlJ30gPSAnVm9yZmFsbHN0eXAnOwogICAgJExhbmctPnsnUmVwb3J0ZXInfSA9ICdSZXBvcnRlcic7CiAgICAkTGFuZy0+eydEbyBub3Qgc2V0IEFzc2lnbmVlJ30gPSAnU2V0emUgbmljaHQgZGVuIEJlc2l0emVyJzsKICAgICRMYW5nLT57J0Fzc2lnbmVlJ30gPSAnQmVzaXR6ZXInOwogICAgJExhbmctPnsnRHluYW1pYyBGaWVsZCd9ID0gJ0R5bmFtaXNjaGVzIEZlbGQnOwoKICAgICRMYW5nLT57J1RoZXNlIGluZm9ybWF0aW9uIGFyZSB0cmFuc2ZlcnJlZCBhdXRvbWF0aWNhbGx5Oid9ID0gJ0RpZXNlIEluZm9ybWF0aW9uZW4gd2VyZGVuIGF1dG9tYXRpc2NoIMO8YmVydHJhZ2VuJzsKICAgICRMYW5nLT57J1Byb2plY3QgKHN0YW5kYXJkIHZhbHVlIGZyb20gU3lzQ29uZmlnIG9yIHNlbGVjdGVkIHByb2plY3QpJ30gPSAnUHJvamVrdCAoU3RhbmRhcmR3ZXJ0IGF1cyBkZXIgU3lzQ29uZmlnIG9kZXIgYXVzZ2V3w6RobHRlcyBQcm9qZWt0KSc7CiAgICAkTGFuZy0+eydJc3N1ZXR5cGUgKHN0YW5kYXJkIHZhbHVlIGZyb20gU3lzQ29uZmlnIG9yIHNlbGVjdGVkIGlzc3VldHlwZSknfSA9ICdWb3JnYW5nc3R5cCAoU3RhbmRhcmR3ZXJ0IGF1cyBkZXIgU3lzQ29uZmlnIG9kZXIgYXVzZ2V3w6RobHRlciBWb3JnYW5nc3R5cCknOwogICAgJExhbmctPnsnU3VtbWFyeSAodGlja2V0IHRpdGxlKSd9ID0gJ1p1c2FtbWVuZmFzc3VuZyAoVGlja2V0LVRpdGxlKSc7CiAgICAkTGFuZy0+eydEZXNjcmlwdGlvbiAoYXJ0aWNsZSBib2R5KSd9ID0gJ0Jlc2NocmVpYnVuZyAoQXJ0aWtlbHRleHQpJzsKICAgICRMYW5nLT57J1JlcG9ydGVyIChzdGFuZGFyZCB2YWx1ZSBmcm9tIFN5c0NvbmZpZyBvciBzZWxlY3RlZCByZXBvcnRlciknfSA9ICdSZXBvcnRlciAoU3RhbmRhcmR3ZXJ0IGF1cyBkZXIgU3lzQ29uZmlnIG9kZXIgYXVzZ2V3w6RobHRlciBSZXBvcnRlciknOwogICAgJExhbmctPnsnQXNzaWduZWUgKHN0YW5kYXJkIHZhbHVlIGZyb20gU3lzQ29uZmlnIG9yIHNlbGVjdGVkIGFzc2lnbmVlKSd9ID0gJ1p1Z2V3aWVzZW5lIFBlcnNvbiAoU3RhbmRhcmR3ZXJ0IGF1cyBkZXIgU3lzQ29uZmlnIG9kZXIgYXVzZ2V3w6RobHRlIFBlcnNvbiknOwogICAgJExhbmctPnsnT1RSU1RpY2tldElEIGN1c3RvbWZpZWxkICh0aWNrZXQgaWQpJ30gPSAnQmVudXR6ZXJkZWZpbmllcnRlcyBGZWxkIE9UUlNUaWNrZXRJRCAoVGlja2V0LUlEKSc7CiAgICAkTGFuZy0+eydUaGlzIG1hcHBpbmcgZGVmaW5lcyB3aGljaCBpbmZvcm1hdGlvbiBhcmUgdHJhbnNmZXJyZWQgZnJvbSBPVFJTIHRvIEppcmEuIE9uIHRoZSBsZWZ0IHlvdSBzZWUgYWxsIHRoZSBhdmFpbGFibGUgaW5mb3JtYXRpb24gaW4gT1RSUyBhbmQgb24gdGhlIHJpZ2h0IHlvdSBzZWUgdGhlIGZpZWxkcyBpbiBKaXJhLiBQbGVhc2Uga2VlcCBpbiBtaW5kLCB0aGF0IHRoZSB0eXBlIG9mIHRoZSBmaWVsZHMgc2hvdWxkIG1hdGNoIChkYXRlL3RpbWUgaW5mb3JtYXRpb24gaW4gT1RSUyBuZWVkIHRvIGJlIHN0b3JlZCBpbiBhIGRhdGUvdGltZSBmaWVsZCBpbiBKaXJhKSd9ID0gJ0RpZSBGZWxkenVvcmRudW5nIGJlc3RpbW10LCB3ZWxjaGUgSW5mb3JtYXRpb25lbiBhdXMgZGVtIE9UUlMgbmFjaCBKaXJhIMO8YmVydHJhZ2VuIHdlcmRlbi4gQXVmIGRlciBsaW5rZW4gU2VpdGUgc2VoZW4gU2llIGFsbGUgSW5mb3JtYXRpb25lbiBkaWUgaW0gT1RSUyB6dXIgVmVyZsO8Z3VuZyBzdGVoZW4sIGF1ZiBkZXIgcmVjaHRlbiBTaWV0ZSBkaWUgRmVsZGVyIGluIEppcmEuIEJpdHRlIGJlYWNodGVuIFNpZSwgZGFzcyBkaWUgRmVsZHR5cGVuIHp1ZWluYW5kZXIgcGFzc2VuIHNvbGx0ZW4gKERhdHVtL1plaXQtSW5mb3JtYXRpb25lbiBpbiBPVFJTIGJlbsO2dGlnZW4gZWluIERhdHVtL1plaXQtRmVsZCBpbiBKaXJhKSc7CgogICAgJExhbmctPnsnU2VuZCBSZXBvcnRlcid9ID0gJ1NldHplIGRlbiBSZXBvcnRlcic7CgogICAgcmV0dXJuIDE7Cn0KMTsK
IyAgLS0KIyAgS2VybmVsL0xhbmd1YWdlL2VuX0FnZW50VGlja2V0SklSQS5wbSAtIGxhbmd1YWdlIG1vZHVsZQojICBDb3B5cmlnaHQgKEMpIDIwMDgtMjAxNSBjYXR3b3JreCBHbWJIIGh0dHA6Ly93d3cuY2F0d29ya3guZGUKIyAgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmVuX0FnZW50VGlja2V0SklSQTsKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwogICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBIG5ldyBKaXJhIElzc3VlIGhhcyBiZWVuIGNyZWF0ZWQnfSA9ICdBIG5ldyBKaXJhIElzc3VlIGhhcyBiZWVuIGNyZWF0ZWQnOwogICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBIEppcmEgSXNzdWUgY291bGQgbm90IGJlIGNyZWF0ZWQnfSA9ICdBIEppcmEgSXNzdWUgY291bGQgbm90IGJlIGNyZWF0ZWQnOwogICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUbyB0aGUgaXNzdWUgKEppcmEpJ30gPSAnVG8gdGhlIGlzc3VlIChKaXJhKSc7CiAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0JhY2sgdG8gdGhlIHRpY2tldCAoT1RSUyknfSA9ICdCYWNrIHRvIHRoZSB0aWNrZXQgKE9UUlMpJzsKICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGlja2V0IG5vdCBvcGVuLid9ID0gJ1RpY2tldCBub3Qgb3Blbi4nOwogICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydKaXJhIHRvIE9UUlMnfSA9ICdKaXJhIHRvIE9UUlMnOwogICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWtlIHRoaXMgdGlja2V0IGJhY2sgZnJvbSBKaXJhJ30gPSAnVGFrZSB0aGlzIHRpY2tldCBiYWNrIGZyb20gSmlyYSc7CiAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09UUlMgdG8gSmlyYSd9ID0gJ09UUlMgdG8gSmlyYSc7CiAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VzY2FsYXRlIHRoaXMgdGlja2V0IHRvIEppcmEnfSA9ICdFc2NhbGF0ZSB0aGlzIHRpY2tldCB0byBKaXJhJzsKICAgIyRTZWxmLT57VHJhbnNsYXRpb259LT57Jyd9ID0gJyc7CiAgIHJldHVybiAxOwp9CjE7Cg==
# --
# Copyright (C) 2017 Perl-Services.de, http://www.perl-services.de
# --
# 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::AdminJIRAQueueSettings;

use strict;
use warnings;

our @ObjectDependencies = qw(
    Kernel::Config
    Kernel::System::SysConfig
    Kernel::System::Queue
    Kernel::System::Valid
    Kernel::Output::HTML::Layout
    Kernel::System::Web::Request
    Kernel::System::DynamicField
    Kernel::System::JIRA::QueueSettings
);

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

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

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

    return $Self;
}

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

    my $LayoutObject     = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $ParamObject      = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $SysConfigObject  = $Kernel::OM->Get('Kernel::System::SysConfig');
    my $SettingsObject   = $Kernel::OM->Get('Kernel::System::JIRA::QueueSettings');
    my $ConfigObject     = $Kernel::OM->Get('Kernel::Config');
    my $JIRA             = $Kernel::OM->Get('Kernel::System::JIRA::Utils');

    my %GetParam;
    for my $Param ( qw/ProjectID Assignee QueueID ID ValidID/ ) {
        $GetParam{$Param} = $ParamObject->GetParam( Param => $Param );
    }

    my %AllSettings = $SettingsObject->SettingsList();
    my $PageOutput;

    $Self->{Subaction} //= '';

    # ------------------------------------------------------------ #
    # update action
    # ------------------------------------------------------------ #
    if ( $Self->{Subaction} eq 'Save' ) {

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

        # server side validation
        my %Errors;
        if ( !$Self->_CheckQueue( %GetParam ) ) {
            %Errors = ( QueueInvalid => 'ServerError' );
        }

        if ( %Errors ) {
            $Self->{Subaction} = 'Edit';

            my $Output = $LayoutObject->Header();
            $Output .= $LayoutObject->NavigationBar();
            $Output .= $Self->_Mask(
                %GetParam,
                %Errors,
            );
            $Output .= $LayoutObject->Footer();

            return $Output;
        }

        my $Method = 'SettingsAdd';
        if ( $AllSettings{ $GetParam{ID} } ) {
            $Method = 'SettingsUpdate';
        }

        $SettingsObject->$Method(
            %GetParam,
            UserID => $Self->{UserID},
        );

        return $LayoutObject->Redirect(
            OP => 'Action=AdminJIRAQueueSettings',
        );
    }

    elsif ( $Self->{Subaction} eq 'AJAXUpdate' ) {
        my $Users   = {};
        my $Project = $GetParam{ProjectID};

        if ( $Project ) {
            $Users = $JIRA->UsersGet( Project => $Project );
        }

        my $JSON = $LayoutObject->BuildSelectionJSON(
            [
                {
                    Name         => 'Assignee',
                    Data         => $Users,
                    SelectedID   => $GetParam{Assignee},
                    Translation  => 0,
                    PossibleNone => 1,
                },
            ],
        );

        return $LayoutObject->Attachment(
            ContentType => 'application/json; charset=' . $LayoutObject->{Charset},
            Content     => $JSON,
            Type        => 'inline',
            NoCache     => 1,
        );
    }

    elsif ( $Self->{Subaction} eq 'Delete' ) {

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

        $SettingsObject->SettingsDelete( QueueID => $GetParam{QueueID} );

        return $LayoutObject->Redirect(
            OP => 'Action=AdminJIRAQueueSettings',
        );
    }

    # ------------------------------------------------------------ #
    # Add/Edit mask
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'Edit' ) {
        $PageOutput = $Self->_Mask( %GetParam );
    }

    $PageOutput //= $Self->_Overview( AllSettings => \%AllSettings );

    my $Output = $LayoutObject->Header();
    $Output .= $LayoutObject->NavigationBar();
    $Output .= $PageOutput;
    $Output .= $LayoutObject->Footer();
    return $Output;
}

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

    my $SettingsObject = $Kernel::OM->Get('Kernel::System::JIRA::QueueSettings');

    return if !$Param{QueueID};
    return 1 if $Param{ID} && $Param{ID} == $Param{QueueID};

    my %Settings = $SettingsObject->SettingsGet( QueueID => $Param{QueueID} );
    return 1 if !%Settings;
}

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

    my $LayoutObject   = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $SettingsObject = $Kernel::OM->Get('Kernel::System::JIRA::QueueSettings');
    my $JIRA           = $Kernel::OM->Get('Kernel::System::JIRA::Utils');

    $LayoutObject->Block( Name => 'AddAction' );
    $LayoutObject->Block( Name => 'SettingsOverview' );

    my %AllSettings  = %{ $Param{AllSettings} || {} };
    my $JIRAProjects = $JIRA->ProjectsGet( Force => 1 );

    if ( !%AllSettings ) {
        $LayoutObject->Block(
            Name => 'NoSettingsFound',
        );
    }

    for my $SettingsID ( sort { $AllSettings{$a} cmp $AllSettings{$b} } keys %AllSettings ) {
        my %Data = $SettingsObject->SettingsGet( QueueID => $SettingsID );

        if ( $Data{ProjectID} ) {
            $Data{Project} = $JIRAProjects->{ $Data{ProjectID} } // '';
        }

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

    return $LayoutObject->Output(
        TemplateFile => 'AdminJIRAQueueSettings',
        Data         => {},
    );
}

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

    my $LayoutObject   = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $QueueObject    = $Kernel::OM->Get('Kernel::System::Queue');
    my $ValidObject    = $Kernel::OM->Get('Kernel::System::Valid');
    my $JIRA           = $Kernel::OM->Get('Kernel::System::JIRA::Utils');
    my $SettingsObject = $Kernel::OM->Get('Kernel::System::JIRA::QueueSettings');

    $LayoutObject->Block( Name => 'OverviewAction' );

    my %Setting;
    if ( $Param{QueueID} ) {
        %Setting   = $SettingsObject->SettingsGet( QueueID => $Param{QueueID} );
        $Param{ID} = $Param{QueueID};
    }

    my $Users   = {};
    my $Project = $Param{ProjectID} || $Setting{ProjectID};

    if ( $Project ) {
        $Users = $JIRA->UsersGet( Project => $Project );
    }

    my %Queues       = $QueueObject->QueueList( UserID => 1 );
    my $JIRAProjects = $JIRA->ProjectsGet( Force => 1 );

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

    $Param{QueueSelect} = $LayoutObject->BuildSelection(
        Name        => 'QueueID',
        Data        => \%Queues,
        Class       => 'Modernize ' . ( $Param{QueueInvalid} // '' ),
        Translation => 0,
        SelectedID  => $Param{QueueID} || $Setting{QueueID},
    );

    $Param{ProjectSelect} = $LayoutObject->BuildSelection(
        Name         => 'ProjectID',
        Data         => $JIRAProjects,
        Class        => 'Modernize',
        Translation  => 0,
        SelectedID   => $Param{ProjectID} || $Setting{ProjectID},
        PossibleNone => 1,
    );

    $Param{UsersSelect} = $LayoutObject->BuildSelection(
        Name         => 'Assignee',
        Data         => $Users,
        Class        => 'Modernize',
        Translation  => 0,
        SelectedID   => $Param{Assignee} || $Setting{Assignee},
        PossibleNone => 1,
    );

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

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

1;

# --
# Copyright (C) 2016 Perl-Services.de, http://www.perl-services.de
# --
# 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::AdminJIRAWizard;

use strict;
use warnings;

our @ObjectDependencies = qw(
    Kernel::Config
    Kernel::System::SysConfig
    Kernel::System::Valid
    Kernel::Output::HTML::Layout
    Kernel::System::Web::Request
    Kernel::System::DynamicField
);

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

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

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

    $Self->{Step1} = [
        {
            Label     => 'Jira URL',
            Field     => 'URL',
            SysConfig => 'URL',
            Extra     => {
                Type   => 'DynamicField',
                Name   => 'JIRAIssueID',
                Attr   => 'Link',
                Format => '%s/browse/[%% Data.JIRAIssueID | url %%]',
            },
        },
        {
            Label     => 'API Path',
            Field     => 'Path',
            SysConfig => 'Path',
        },
        {
            Label     => 'User',
            Hint      => 'Jira user OTRS uses for API calls',
            Field     => 'User',
            SysConfig => 'User',
        },
        {
            Label     => 'Password',
            Field     => 'Password',
            SysConfig => 'Password',
        },
        {
            Label     => 'Incoming User',
            Hint      => 'OTRS user Jira uses to call OTRS Webservice',
            Field     => 'IncomingUser',
            SysConfig => 'IncomingUser',
        },
    ];

    $Self->{Step2} = [
        {
            Label     => 'Project',
            Field     => 'Project',
            SysConfig => 'TicketJIRA###Project',
            Data      => 'JIRAProjects',
        },
        {
            Label     => 'Issue Type',
            Field     => 'IssueType',
            SysConfig => 'TicketJIRA###IssueType',
        },
        {
            Label        => 'Send Reporter',
            Field        => 'SendReporter',
            SysConfig    => 'JIRA::SendReporter',
            Translatable => 1,
            Data         => {
                0 => 'No',
                1 => 'Yes',
            },
            Sub => {
                1 => [
                    {
                        Label     => 'Reporter',
                        Field     => 'Reporter',
                        SysConfig => 'TicketJIRA###Reporter',
                    },
                ],
            },
        },
        {
            Label     => 'Custom field OTRSTicketID',
            Field     => 'OTRSTicketID',
            SysConfig => 'JIRA::OTRSTicketIDField',
            Data      => 'JIRACustomFields',
        },
        {
            Label        => 'Do not set Assignee',
            Field        => 'NoAssignee',
            SysConfig    => 'JIRA::NoAssignee',
            Translatable => 1,
            Data         => {
                0 => 'No',
                1 => 'Yes',
            },
            Sub => {
                0 => [
                    {
                        Label     => 'Assignee',
                        Field     => 'Assignee',
                        SysConfig => 'TicketJIRA###Assignee',
                    },
                ],
            },
        },
        {
            Label        => 'Use Dialog',
            Field        => 'Dialog',
            SysConfig    => 'JIRA::ShowDialog',
            Hint         => 'When using the dialog, you can edit some settings for the issue (e.g. project, assignee...)',
            Translatable => 1,
            Data         => {
                0 => 'No',
                1 => 'Yes',
            },
            Sub => {
                1 => [
                    {
                        Label        => 'Show Assignee (dialog)',
                        Field        => 'ShowAssignee',
                        SysConfig    => 'JIRA::ShowAssignee',
                        Translatable => 1,
                        Data         => {
                            -1 => 'Hide',
                            0 => 'Readonly',
                            1 => 'Editable',
                        },
                    },
                    {
                        Label        => 'Show "Automatic" user (dialog)',
                        Field        => 'ShowAutomaticUser',
                        SysConfig    => 'JIRA::ShowAutomaticUser',
                        Translatable => 1,
                        Data         => {
                            0 => 'No',
                            1 => 'Editable',
                        },
                    },
                    {
                        Label        => 'Show IssueType (dialog)',
                        Field        => 'ShowIssueType',
                        SysConfig    => 'JIRA::ShowIssueType',
                        Translatable => 1,
                        Data         => {
                            -1 => 'Hide',
                            0 => 'Readonly',
                            1 => 'Editable',
                        },
                    },
                    {
                        Label        => 'Show Project (dialog)',
                        Field        => 'ShowProject',
                        SysConfig    => 'JIRA::ShowProject',
                        Translatable => 1,
                        Data         => {
                            -1 => 'Hide',
                            0 => 'Readonly',
                            1 => 'Editable',
                        },
                    },
                ],
            },
        },
    ];

    return $Self;
}

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

    my $LayoutObject     = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $ParamObject      = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $SysConfigObject  = $Kernel::OM->Get('Kernel::System::SysConfig');
    my $ConfigObject     = $Kernel::OM->Get('Kernel::Config');

    $Self->{Subaction} //= 'Step1';

    my ($Index) = $Self->{Subaction} =~ m{(\d)}xms;
    $Index    //= 1;

    my @Steps = qw(JIRA Issue Mapping);

    $LayoutObject->Block( Name => 'Steps' );
    for my $StepIndex ( 0 .. $#Steps ) {
        my $Active = $StepIndex == $Index-1 ? 'Active' : '';

        $LayoutObject->Block(
            Name => 'Step' . $Steps[$StepIndex],
            Data => {
                Step   => $StepIndex + 1,
                Active => $Active,
            },
        );
    }

    # ------------------------------------------------------------ #
    # basic JIRA information
    # ------------------------------------------------------------ #
    if ( !$Self->{Subaction} || $Self->{Subaction} eq 'Step1' ) {
        my $Output = $LayoutObject->Header();
        $Output .= $LayoutObject->NavigationBar();
        $Output .= $Self->_Step1();
        $Output .= $LayoutObject->Footer();
        return $Output;
    }

    # ------------------------------------------------------------ #
    # update action
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'Step2' ) {

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

        # server side validation
        my %Errors = $Self->_CheckStep1();

        if ( %Errors ) {
            $Self->{Subaction} = 'Edit';

            my $Output = $LayoutObject->Header();
            $Output .= $LayoutObject->NavigationBar();
            $Output .= $Self->_Step1(
                %Errors,
                Submitted => 1,
            );
            $Output .= $LayoutObject->Footer();

            return $Output;
        }

        if ( $ParamObject->GetParam( Param => 'Step1Save' ) ) {
            $Self->_SaveStep1();
        }

        my $Output = $LayoutObject->Header();
        $Output .= $LayoutObject->NavigationBar();
        $Output .= $Self->_Step2();
        $Output .= $LayoutObject->Footer();
        return $Output;
    }

    # ------------------------------------------------------------ #
    # step3
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'Step3' ) {

        # server side validation
        my %Errors = $Self->_CheckStep2();

        if ( %Errors ) {
            $Self->{Subaction} = 'Edit';

            my $Output = $LayoutObject->Header();
            $Output .= $LayoutObject->NavigationBar();
            $Output .= $Self->_Step2(
                %Errors,
                Submitted => 1,
            );
            $Output .= $LayoutObject->Footer();

            return $Output;
        }

        if ( $ParamObject->GetParam( Param => 'Step2Save' ) ) {
            $Self->_SaveStep2();
        }

        my $Output = $LayoutObject->Header();
        $Output .= $LayoutObject->NavigationBar();
        $Output .= $Self->_Step3();
        $Output .= $LayoutObject->Footer();
        return $Output;
    }

    # ------------------------------------------------------------ #
    # step3
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'Step3Save' ) {

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

        # server side validation
        my %Errors = $Self->_CheckStep3();

        if ( %Errors ) {
            $Self->{Subaction} = 'Edit';

            my $Output = $LayoutObject->Header();
            $Output .= $LayoutObject->NavigationBar();
            $Output .= $Self->_Step3(
                %Errors,
            );
            $Output .= $LayoutObject->Footer();

            return $Output;
        }

        $Self->_SaveStep3();

        $LayoutObject->Block( Name => 'ThankYou' );

        my $Output = $LayoutObject->Header();
        $Output .= $LayoutObject->NavigationBar();
        $Output .= $LayoutObject->Output( TemplateFile => 'AdminJIRAWizard' );
        $Output .= $LayoutObject->Footer();
        return $Output;
    }


    # ------------------------------------------------------------ #
    # else ! print form
    # ------------------------------------------------------------ #
    else {
        return $LayoutObject->Redirect( OP => 'Action=AdminJIRAWizard' );
    }
}

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

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

    $LayoutObject->Block(
        Name => 'Step1',
        Data => {
            Item => 'JIRA core settings',
            Step => 1,
        },
    );

    if ( $Param{Message} ) {
        $LayoutObject->Block(
            Name => 'Step1HasError',
            Data => { ErrorMessage => $Param{Message} },
        );
    }

    my @Fields = @{ $Self->{Step1} || [] };
    my $Config = $ConfigObject->Get('TicketJIRA');

    FIELD:
    for my $Field ( @Fields ) {
        my $Subconfig = $Field->{SysConfig};
        my $Value = $Config->{$Subconfig};

        if ( $Param{Submitted} ) {
            $Value = $ParamObject->GetParam( Param => $Field->{Field} );
        }

        $LayoutObject->Block(
            Name => 'Field',
            Data => {
                %{ $Field || {} },
                Value => $Value,
            },
        );

        $LayoutObject->Block(
            Name => 'FieldInput',
            Data => {
                %{ $Field || {} },
                Value => $Value,
            },
        );

        if ( $Field->{Hint} ) {
            $LayoutObject->Block(
                Name => 'FieldHint',
                Data => $Field,
            );
        }
    }

    return $LayoutObject->Output(
        TemplateFile => 'AdminJIRAWizard',
        Data         => {},
    );
}

sub _CheckStep1 {
    my ($Self) = @_;

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

    return if !$ParamObject->GetParam( Param => 'Step1Save' );

    # all fields are mandatory
    my @Fields = @{ $Self->{Step1} || [] };

    my %Errors;
    my %GetParam;

    FIELD:
    for my $Field ( @Fields ) {
        my $Value = $ParamObject->GetParam( Param => $Field->{Field} );

        $GetParam{$Field->{Field}} = $Value;

        if ( !defined $Value || $Value eq '' ) {
            $Errors{Message} = 'All fields are mandatory';
            last FIELD;
        }
    }

    return %Errors if %Errors;

    # check if API is reachable
    {
        local $Kernel::OM = Kernel::System::ObjectManager->new(
            'Kernel::System::JIRA::Utils' => {
                JIRAURL  => $GetParam{URL} . $GetParam{Path},
                JIRAUSER => $GetParam{User},
                JIRAPWD  => $GetParam{Password},
             },
        );

        my $JIRA               = $Kernel::OM->Get('Kernel::System::JIRA::Utils');
        my ($Status, $Message) = $JIRA->Check();

        $Status //= 0;

        return if $Status == 200;

        if ( $Status == 404 && $Message =~ /parameter/ ) {
            return;
        }
        elsif ( $Status == 404 || $Status == 500 ) {
            $Errors{Message} = "The URL may not be correct, we can't connect to Jira";
        }
        elsif ( $Status == 401 ) {
            $Errors{Message} = "The credentials for the Jira user are not correct";
        }
        elsif ( $Status == 403 ) {
            $Errors{Message} = "Access denied, check API path";
        }
        else {
            $Errors{Message} = "Error $Status: $Message";
        }
    }

    return %Errors;
}

sub _SaveStep1 {
    my ($Self) = @_;

    my $SysConfigObject    = $Kernel::OM->Get('Kernel::System::SysConfig');
    my $ParamObject        = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');

    my $Base   = 'TicketJIRA';
    my @Fields = @{ $Self->{Step1} || [] };

    my @Settings;

    FIELD:
    for my $Field ( @Fields ) {
        my ($Value) = $ParamObject->GetParam( Param => $Field->{Field} );

        my $Fieldname = $Field->{SysConfig};
        push @Settings, {
            Name           => "$Base###$Fieldname",
            EffectiveValue => $Value,
        };

        if ( $Field->{Extra} ) {
            my $Extra = $Field->{Extra};

            if ( $Extra->{Type} eq 'DynamicField' ) {
                my $DynamicField = $DynamicFieldObject->DynamicFieldGet(
                    Name => $Extra->{Name},
                );

                my $Attr = $Extra->{Attr};
                $DynamicField->{Config}->{$Attr} = sprintf $Extra->{Format}, $Value;

                $DynamicFieldObject->DynamicFieldUpdate(
                    %{ $DynamicField || {} },
                    UserID => $Self->{UserID},
                );
            }
        }
    }

    my $Success = $SysConfigObject->SettingsSet(
        Settings => \@Settings,
        UserID   => 1,
    );
}

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

    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    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 $MainObject   = $Kernel::OM->Get('Kernel::System::Main');

    $LayoutObject->Block(
        Name => 'Step2',
        Data => {
            Item => 'Issue settings',
            Step => 2,
        },
    );

    if ( $Param{Message} ) {
        $LayoutObject->Block(
            Name => 'Step2HasError',
            Data => { ErrorMessage => $Param{Message} },
        );
    }

    my @Fields = @{ $Self->{Step2} || [] };
    my $Config = $ConfigObject->Get('TicketJIRA');

    my $FieldCount = 1;

    FIELD:
    for my $Field ( @Fields ) {
        my $ConfigKey          = $Field->{SysConfig};
        my (undef, $Subconfig) = split /\#{3}/, $ConfigKey;
        my $Value              = $ConfigKey =~ m{::} ? $ConfigObject->Get( $ConfigKey ) : $Config->{$Subconfig};

        if ( $Param{Submitted} ) {
            $Value = $ParamObject->GetParam( Param => $Field->{Field} );
        }

        my $Class  = $Field->{Class} // '';

        $LayoutObject->Block(
            Name => 'FieldStep2',
            Data => {
                %{ $Field || {} },
                FieldClass => $Class,
                Value      => $Value,
            },
        );

        if ( !$Field->{Data} ) {
            $LayoutObject->Block(
                Name => 'FieldInput2',
                Data => {
                    %{ $Field || {} },
                    Value => $Value,
                },
            );
        }
        else {

            if ( !ref $Field->{Data} ) {
                my $Sub = $Self->can( $Field->{Data} );
                ($Field->{Data}, $Field->{Value}) = $Self->$Sub( $Field->{Field} ) if $Sub;

                if ( !$Param{Submitted} ) {
                    $Value = $Field->{Value};
                }
            }

            my $Select = $LayoutObject->BuildSelection(
                Name        => $Field->{Field},
                Data        => $Field->{Data},
                Translation => $Field->{Translatable},
                SelectedID  => $Value,
                Class       => "Modernize",
            );

            $LayoutObject->Block(
                Name => 'FieldSelect2',
                Data => {
                    Field => $Select,
                },
            );
        }

        if ( $Field->{Hint} ) {
            $LayoutObject->Block(
                Name => 'FieldHintStep2',
                Data => $Field,
            );
        }

        if ( $Field->{Sub} ) {
            $LayoutObject->Block(
                Name => 'FieldChangeStep2',
                Data => $Field,
            );

            my @SubFields;

            KEY:
            for my $Key ( sort keys %{ $Field->{Sub} } ) {
                my $FieldClass = $Field->{Field} . '-' . $Key;
                push @SubFields, map{
                    $_->{Class} = $FieldClass;
                    $_;
                }@{ $Field->{Sub}->{$Key} };
            }

            splice @Fields, $FieldCount, 0, @SubFields;
        }
    }
    continue {
        $FieldCount++;
    }

    return $LayoutObject->Output(
        TemplateFile => 'AdminJIRAWizard',
        Data         => {},
    );
}

sub JIRACustomFields {
    my ($Self) = @_;

    my $JIRATicketObject   = $Kernel::OM->Get('Kernel::System::TicketJIRA');
    my $ConfigObject       = $Kernel::OM->Get('Kernel::Config');


    my $FieldTypes = $JIRATicketObject->FieldsGet( Return => 'types' );
    my $JIRAFields = $JIRATicketObject->FieldsGet();

    my %CustomFields = reverse %{ $JIRAFields || {} };

    my %JIRAFieldMap;

    FIELDNAME:
    for my $FieldName ( keys %{ $FieldTypes || {} } ) {
        next FIELDNAME if $FieldName !~ m{\A customfield_ }xms;

        $FieldName  =~ s{customfield_}{};
        my $FieldID = $FieldName;
        $FieldName  = $CustomFields{$FieldName};

        $JIRAFieldMap{$FieldID} = $FieldName;
    }

    my $Selected = $ConfigObject->Get("JIRA::OTRSTicketIDField");

    return ( \%JIRAFieldMap, $Selected );
}

sub _CheckStep2 {
    my ($Self) = @_;

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

    return if !$ParamObject->GetParam( Param => 'Step2Save' );

    my @Fields = @{ $Self->{Step2} || [] };

    my $Users;
    my $Projects;
    my $JIRA;

    # check if API is reachable
    {
        local $Kernel::OM = Kernel::System::ObjectManager->new();

        $JIRA     = $Kernel::OM->Get('Kernel::System::JIRA::Utils');
        $Projects = $JIRA->ProjectsGet( Force => 1 );
    }

    my %Values;
    my $FieldCount = 1;

    FIELD:
    for my $Field ( @Fields ) {
        my ($Value) = $ParamObject->GetParam( Param => $Field->{Field} );

         $Values{ $Field->{Field} } = $Value;

        if ( $Field->{Data} && ref $Field->{Data} ) {
            if ( ! $Field->{Data}->{$Value} ) {
                return ( Message => 'InvalidValue' );
            }
        }

        if ( $Field->{Field} eq 'Reporter' ) {

            # get available users
            $Users      //= $JIRA->UsersGet( Project => $Values{Project} );
            my $ValidUser = $Users->{$Value};

            if ( !$ValidUser ) {
                return ( Message => 'Invalid User (field ' . $Field->{Field} . '), please select an existing Jira user' );
            }
        }

        if ( !$Values{NoAssignee} && $Field->{Field} eq 'Assignee' ) {

            # get available users
            $Users      //= $JIRA->UsersGet( Project => $Values{Project} );
            my $ValidUser = $Users->{$Value};

            if ( !$ValidUser ) {
                return ( Message => 'Invalid User (field ' . $Field->{Field} . '), please select an existing Jira user' );
            }
        }

        if ( $Field->{Field} eq 'Project' ) {
            my $ValidProject = $Projects->{$Value};

            if ( !$ValidProject ) {
                return ( Message => 'Invalid Project, please select an existing Jira project' );
            }
        }

        if ( $Field->{Field} eq 'IssueType' ) {
            # get available issue types
            my $IssueTypes     = $JIRA->IssueTypesGet( Project => $Values{Project} );
            my $ValidIssueType = $IssueTypes->{$Value};

            if ( !$ValidIssueType ) {
                return ( Message => 'Invalid IssueType' );
            }
        }

        if ( $Field->{Sub} ) {
            my @SubFields;

            KEY:
            for my $Key ( sort keys %{ $Field->{Sub} } ) {
                push @SubFields, @{ $Field->{Sub}->{$Key} };
            }

            splice @Fields, $FieldCount, 0, @SubFields;
        }

        $FieldCount++;
    }

    return;
}

sub _SaveStep2 {
    my ($Self) = @_;

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

    my @Fields     = @{ $Self->{Step2} || [] };
    my $FieldCount = 1;

    my @Settings;

    FIELD:
    for my $Field ( @Fields ) {
        my ($Value) = $ParamObject->GetParam( Param => $Field->{Field} );

        my $Fieldname = $Field->{SysConfig};
        push @Settings, {
            Name           => $Fieldname,
            EffectiveValue => $Value,
        };

        if ( $Field->{Sub} ) {
            my @SubFields;

            KEY:
            for my $Key ( sort keys %{ $Field->{Sub} } ) {
                push @SubFields, @{ $Field->{Sub}->{$Key} };
            }

            splice @Fields, $FieldCount, 0, @SubFields;
        }

        $FieldCount++;
    }

    my $Success = $SysConfigObject->SettingsSet(
        Settings => \@Settings,
        UserID   => 1,
    );
}

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

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

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

    if ( $Param{Message} ) {
        $LayoutObject->Block(
            Name => 'Step3HasError',
            Data => { ErrorMessage => $Param{Message} },
        );
    }

    $Self->_DefineStep3();

    my $Fields = $Self->{Step3} || {};
    my $OTRSFields = $Fields->{OTRSFields};
    my $JIRAFields = $Fields->{JIRAFields};

    my %Defaults = (
        0 => {
            OTRS => '',
            JIRA => '',
        },
    );

    my $FieldMappings = $ConfigObject->Get('TicketJIRA')->{'OTRS-JIRA-FieldMapping'};
    my $FieldMappingsKeys = [keys %{$FieldMappings}];
    my $FieldMappingsValues = [values %{$FieldMappings}];
    my $OTRSKeys = [keys %{$OTRSFields}];
    my $JIRAKeys = [keys %{$JIRAFields}];
    foreach my $MappingKey (@{$FieldMappingsKeys}) {
        if (! grep { $_ eq $MappingKey } @{$OTRSKeys}) {
            $OTRSFields->{$MappingKey} = $MappingKey;
        }
    }
    foreach my $MappingValue (@{$FieldMappingsValues}) {
        if (! grep { $MappingValue =~ /$_(?::name|:key)?$/ } @{$JIRAKeys}) {
            $JIRAFields->{$MappingValue} = $MappingValue;
        }
    }
    my $RowMappingID = 0;

    EXISTINGROWID:
    foreach my $FieldKey (@{$FieldMappingsKeys}) {

        my $OTRSDropdown = $LayoutObject->BuildSelection(
            Name          => 'OTRS' . $RowMappingID . 'E',
            Data          => $OTRSFields,
            PossibleNone  => 1,
            Class         => 'Modernize',
            SelectedValue => $OTRSFields->{$FieldKey},
        );

        my $JIRAFieldKey = $FieldMappings->{$FieldKey} =~ s/(?::name|:key)$//r;
        my $JIRADropdown = $LayoutObject->BuildSelection(
            Name          => 'JIRA' . $RowMappingID,
            Data          => $JIRAFields,
            PossibleNone  => 1,
            Class         => 'Modernize',
            SelectedValue => $JIRAFields->{$JIRAFieldKey},
        );

        $LayoutObject->Block(
            Name => 'Row',
            Data => {
                OTRS => $OTRSDropdown,
                JIRA => $JIRADropdown
            },
        );

        $RowMappingID++;
    }

    my $RowsToDisplayCount = $RowMappingID + 10;

    ROWID:
    for my $RowID ( $RowMappingID .. $RowsToDisplayCount ) {

        my $OTRSDropdown = $LayoutObject->BuildSelection(
            Name          => 'OTRS' . $RowID,
            Data          => $OTRSFields,
            PossibleNone  => 1,
            Class         => 'Modernize',
            SelectedValue => $Defaults{$RowID}->{OTRS},
        );

        my $JIRADropdown = $LayoutObject->BuildSelection(
            Name          => 'JIRA' . $RowID,
            Data          => $JIRAFields,
            PossibleNone  => 1,
            Class         => 'Modernize',
            SelectedValue => $Defaults{$RowID}->{OTRS},
        );

        $LayoutObject->Block(
            Name => 'Row',
            Data => {
                OTRS => $OTRSDropdown,
                JIRA => $JIRADropdown
            },
        );
    }

    return $LayoutObject->Output(
        TemplateFile => 'AdminJIRAWizard',
        Data         => {},
    );
}

sub _CheckStep3 {
    return;
}

sub _SaveStep3 {
    my ($Self) = @_;

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

    my %Mapping;

    my @ParamNames = $ParamObject->GetParamNames();
    my @OTRSParamNames = grep { $_ =~ /^OTRS\d+E?$/ } @ParamNames;
    my $FieldMappings = $ConfigObject->Get('TicketJIRA')->{'OTRS-JIRA-FieldMapping'};

    ROWID:
    foreach my $OTRSParamName ( @OTRSParamNames ) {
        my ( $RowID ) = $OTRSParamName =~ m/(\d+)E?$/;

        my $OTRS = $ParamObject->GetParam( Param => $OTRSParamName );
        my $JIRA = $ParamObject->GetParam( Param => 'JIRA' . $RowID );

        next ROWID if !$OTRS || !$JIRA;

        if ($OTRSParamName =~ /E$/ && $FieldMappings->{$OTRS} && $FieldMappings->{$OTRS} =~ /^$JIRA/) {
            $JIRA = $FieldMappings->{$OTRS};
        } else {
            JIRAFIELDCHECK:
            for my $JIRAFieldCheck ( qw/reporter assignee issuetype priority project/ ) {
                next JIRAFIELDCHECK if $JIRA ne "std:$JIRAFieldCheck";

                my $Suffix = ( $OTRS ne 'std:CustomerUserID' && $OTRS =~ m{ID\z} ) || $JIRA eq "std:project" ? ':key' : ':name';
                $JIRA .= $Suffix;
            }
        }

        $Mapping{$OTRS} = $JIRA;
    }

    my $Fieldname = 'TicketJIRA###OTRS-JIRA-FieldMapping';
    my @Settings = ({
        Name           => $Fieldname,
        EffectiveValue => \%Mapping,
    });

    my $Success = $SysConfigObject->SettingsSet(
        Settings => \@Settings,
        UserID   => 1,
    );
}

sub _DefineStep3 {
    my ($Self) = @_;

    return 1 if $Self->{Step3};

    my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');
    my $JIRATicketObject   = $Kernel::OM->Get('Kernel::System::TicketJIRA');
    my $LayoutObject       = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

    my $DynamicFields = $DynamicFieldObject->DynamicFieldListGet(
        Valid       => 1,
        ObjectType  => [ 'Ticket' ],
    );

    my @TicketAttributes = qw(
       Queue Priority TicketID TicketNumber Title Owner Responsible State StateType
       QueueID StateID OwnerID ResponsibleID SLAID ServiceID PriorityID Lock LockID
       CustomerID CustomerUserID Type TypeID Age Created CreatedTimeUnits CreateBy
       ChangeBy Changed ArchiveFlag RealTillTimeNotUsed EscalationUpdateTime
       UnlockTimeout EscalationResponseTime EscalationSolutionTime EscalationTime
       From To Cc Subject ReplyTo  MessageID InReplyTo References Body ContentType
       SenderTypeID ArticleTypeID IncomingTime ArticleID SenderType ArticleType
    );

    my %OTRSFieldMap = map{ ("std:$_" => $_) } @TicketAttributes;
    for my $DynamicField ( @{ $DynamicFields || [] } ) {
       my $Name  = $DynamicField->{Name};
       my $Label = $LayoutObject->{LanguageObject}->Translate('Dynamic Field');
       $OTRSFieldMap{ "dyn:$Name" } = '(' . $Label . ') ' . $Name;
    }

    $Self->{Step3}->{OTRSFields} = \%OTRSFieldMap;

    my $FieldTypes = $JIRATicketObject->FieldsGet( Return => 'types', DoNotCache => 1 );
    my $JIRAFields = $JIRATicketObject->FieldsGet( DoNotCache => 1 );

    my %CustomFields = reverse %{ $JIRAFields || {} };

    my %JIRAFieldMap;
    for my $FieldName ( keys %{ $FieldTypes || {} } ) {
        my $Type = 'std';

        if ( $FieldName =~ m{\A customfield_ }xms ) {
            $Type = 'cust';
            $FieldName =~ s{customfield_}{};
            $FieldName = $CustomFields{$FieldName};
        }

        $JIRAFieldMap{"$Type:$FieldName"} = $FieldName;
    }

    $Self->{Step3}->{JIRAFields} = \%JIRAFieldMap;
}

sub JIRAProjects {
    my ($Self) = @_;

    my $UtilsObject  = $Kernel::OM->Get('Kernel::System::JIRA::Utils');
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    my $Projects = $UtilsObject->ProjectsGet( Force => 1 );
    my $Selected = $ConfigObject->Get('TicketJIRA')->{Project};

    return ( $Projects, $Selected );
}


1;

IyAgLS0KIyAgQ29weXJpZ2h0IChDKSAyMDA4LTIwMTggY2F0d29ya3ggR21iSCBodHRwOi8vd3d3LmNhdHdvcmt4LmRlCiMgIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok1vZHVsZXM6OkFnZW50VGlja2V0SklSQTsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSBMaXN0OjpVdGlsIHF3KGZpcnN0KTsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gcXcoCiAgICBLZXJuZWw6OkNvbmZpZwogICAgS2VybmVsOjpTeXN0ZW06OkxvZwogICAgS2VybmVsOjpTeXN0ZW06Ok1haW4KICAgIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQKICAgIEtlcm5lbDo6U3lzdGVtOjpUaWNrZXQKICAgIEtlcm5lbDo6U3lzdGVtOjpUaWNrZXRKSVJBCiAgICBLZXJuZWw6OlN5c3RlbTo6V2ViOjpSZXF1ZXN0Cik7CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHslUGFyYW19OwogICAgYmxlc3MgKCRTZWxmLCAkVHlwZSk7CgogICAgbXkgJENvbmZpZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyk7CiAgICAkU2VsZi0+e0RlYnVnfSAgID0gJENvbmZpZ09iamVjdC0+R2V0KCdKSVJBOjpEZWJ1ZycpOwoKICAgIGlmICggJFNlbGYtPntEZWJ1Z30gKSB7CiAgICAgICAgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpLT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdpbmZvJywKICAgICAgICAgICAgTWVzc2FnZSA9PiBfX1BBQ0tBR0VfXyAuICc6Om5ldygpOiBTdWJhY3Rpb249JyAuICRQYXJhbXtTdWJhY3Rpb259IC4gJyBUaWNrZXRJRD0nIC4gJFBhcmFte1RpY2tldElEfSwKICAgICAgICApOwogICAgfQoKICAgIHJldHVybiAkU2VsZjsKfQoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRUaWNrZXRPYmplY3QgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlRpY2tldCcpOwogICAgbXkgJENvbmZpZ09iamVjdCAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OkNvbmZpZycpOwogICAgbXkgJE1haW5PYmplY3QgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TWFpbicpOwogICAgbXkgJExvZ09iamVjdCAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyk7CiAgICBteSAkVGlja2V0SklSQU9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpUaWNrZXRKSVJBJyk7CiAgICBteSAkTGF5b3V0T2JqZWN0ICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnKTsKICAgIG15ICRQYXJhbU9iamVjdCAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OldlYjo6UmVxdWVzdCcpOwoKICAgIG15ICVEYXRhICA9ICgpOwogICAgbXkgJEJsb2NrID0gJ1N1Y2Nlc3MnOwoKICAgICMgU3ViYWN0aW9uID0gSklSQTJPVFJTIHx8IE9UUlMySklSQQogICAgbXkgJVRpY2tldCA9ICRUaWNrZXRPYmplY3QtPlRpY2tldEdldCgKICAgICAgICBUaWNrZXRJRCA9PiAkU2VsZi0+e1RpY2tldElEfSwKICAgICAgICBVc2VySUQgICA9PiAkU2VsZi0+e1VzZXJJRH0sCiAgICApOwoKICAgIGlmICggJFNlbGYtPntEZWJ1Z30gKSB7CiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnaW5mbycsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+IF9fUEFDS0FHRV9fIC4gIjo6UnVuKCk6IFRpY2tldElEOiAkVGlja2V0e1RpY2tldElEfSBUaXRsZTogJFRpY2tldHtUaXRsZX0gU3RhdGU6ICRUaWNrZXR7U3RhdGV9IgogICAgICAgICk7CiAgICB9CgogICAgbXkgJEpJUkFDb25maWcgPSAkQ29uZmlnT2JqZWN0LT5HZXQoJ1RpY2tldEpJUkEnKSB8fCB7fTsKCiAgICBteSAkQWxsb3dlZFF1ZXVlc0xpc3QgICAgICAgICA9ICRKSVJBQ29uZmlnLT57T1RSU1RpY2tldEFsbG93ZWRRdWV1ZXM0Q3JlYXRlfTsKICAgIG15ICRBbGxvd0NyZWF0aW9uSW5BbGxRdWV1ZXMgID0gJEpJUkFDb25maWctPntPVFJTVGlja2V0QWxsb3dDcmVhdGlvbkluQWxsUXVldWVzfTsKCiAgICBpZiAoICRBbGxvd0NyZWF0aW9uSW5BbGxRdWV1ZXMKICAgICAgICAgfHwgKEB7JEFsbG93ZWRRdWV1ZXNMaXN0fSAmJiBncmVwIHsgJFRpY2tldHtRdWV1ZX0gZXEgJF8gfSBAeyRBbGxvd2VkUXVldWVzTGlzdH0pICkgewogICAgICAgIG15ICRBcnRpY2xlSUQgPSAkUGFyYW1PYmplY3QtPkdldFBhcmFtKCBQYXJhbSA9PiAnQXJ0aWNsZUlEJyApOwoKICAgICAgICBteSAkSklSQUlzc3VlID0gJFRpY2tldEpJUkFPYmplY3QtPklzc3VlQ3JlYXRlKAogICAgICAgICAgICBUaWNrZXRJRCAgPT4gJFNlbGYtPntUaWNrZXRJRH0sCiAgICAgICAgICAgIEFydGljbGVJRCA9PiAkQXJ0aWNsZUlELAogICAgICAgICAgICBVc2VySUQgICAgPT4gJFNlbGYtPntVc2VySUR9LAogICAgICAgICAgICBRdWV1ZUlEICAgPT4gJFRpY2tldHtRdWV1ZUlEfSwKICAgICAgICApOwoKICAgICAgICBpZiAoICRTZWxmLT57RGVidWd9ICkgewogICAgICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgICAgICBQcmlvcml0eSA9PiAnaW5mbycsCiAgICAgICAgICAgICAgICBNZXNzYWdlICA9PiBfX1BBQ0tBR0VfXyAuICI6OlJ1bigpOiAiIC4gJE1haW5PYmplY3QtPkR1bXAoJEpJUkFJc3N1ZSksCiAgICAgICAgICAgICk7CiAgICAgICAgfQoKICAgICAgICBpZiAoICRKSVJBSXNzdWUtPntlcnJvcn0gKSB7CiAgICAgICAgICAgICRCbG9jayAgICAgICA9ICdFcnJvcic7CiAgICAgICAgICAgICREYXRhe2Vycm9yfSA9ICRKSVJBSXNzdWUtPntlcnJvcn07CiAgICAgICAgfQoKICAgICAgICBteSAkVVJMID0gJEpJUkFDb25maWctPntVUkx9OwoJICAgICAgCiAgICAgICAgJERhdGF7aWR9ICA9ICRKSVJBSXNzdWUtPntpZH07CiAgICAgICAgJERhdGF7a2V5fSA9ICRKSVJBSXNzdWUtPntrZXl9OwogICAgICAgICREYXRhe3VybH0gPSAiJFVSTC9icm93c2UvIiAuICRKSVJBSXNzdWUtPntrZXl9OwoKICAgICAgICBpZiAoICRTZWxmLT57RGVidWd9ICkgewogICAgICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgICAgICBQcmlvcml0eSA9PiAnaW5mbycsCiAgICAgICAgICAgICAgICBNZXNzYWdlICA9PiBfX1BBQ0tBR0VfXyAuICI6OlJ1bigpOiBUaWNrZXQtPlN0YXRlICRUaWNrZXR7U3RhdGV9IGFsbG93cyBmZXRjaGluZyAiCiAgICAgICAgICAgICAgICAgICAgLiAib2YgdGhlIEpJUkFJc3N1ZS0+a2V5OiAiIC4gJEpJUkFJc3N1ZS0+e2tleX0sCiAgICAgICAgICAgICk7CiAgICAgICAgfQogICAgfQogICAgZWxzZSB7CiAgICAgICAgbXkgJEFsbG93ZWRRdWV1ZXNOYW1lcyA9IGpvaW4oJywgJywgQHskQWxsb3dlZFF1ZXVlc0xpc3R9KTsKICAgICAgICAkQmxvY2sgICAgICAgID0gJ0Vycm9yJzsKICAgICAgICAkRGF0YXtlcnJvcn0gID0gJExheW91dE9iamVjdC0+e0xhbmd1YWdlT2JqZWN0fS0+VHJhbnNsYXRlKAogICAgICAgICAgICAiVGlja2V0IG5vdCBpbiAkQWxsb3dlZFF1ZXVlc05hbWVzOiAiIC4gJFRpY2tldHtRdWV1ZX0gLiAiLiIKICAgICAgICApOwoKICAgICAgICBpZiAoICRTZWxmLT57RGVidWd9ICkgewogICAgICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgICAgICBQcmlvcml0eSA9PiAnaW5mbycsCiAgICAgICAgICAgICAgICBNZXNzYWdlICA9PiBfX1BBQ0tBR0VfXyAuICI6OlJ1bigpOiBPVFJTIFRpY2tldCBub3QgaW4gJEFsbG93ZWRRdWV1ZXNOYW1lczogIiAuICRUaWNrZXR7UXVldWV9CiAgICAgICAgICAgICk7CiAgICAgICAgfQogICAgfQoKICAgICREYXRhe1RpY2tldElEfSAgICAgPSAkU2VsZi0+e1RpY2tldElEfTsKICAgICREYXRhe1RpY2tldE51bWJlcn0gPSAkVGlja2V0e1RpY2tldE51bWJlcn07CgogICAgJExheW91dE9iamVjdC0+QmxvY2soCiAgICAgICAgRGF0YSA9PiBcJURhdGEsCiAgICAgICAgTmFtZSA9PiAkQmxvY2ssCiAgICApOwoKICAgICMgYnVpbGQgb3V0cHV0CiAgICBteSAkT3V0cHV0ID0gJExheW91dE9iamVjdC0+SGVhZGVyKFRpdGxlID0+ICJUaWNrZXRKSVJBIik7CiAgICAkT3V0cHV0ICAgLj0gJExheW91dE9iamVjdC0+TmF2aWdhdGlvbkJhcigpOwogICAgJE91dHB1dCAgIC49ICRMYXlvdXRPYmplY3QtPk91dHB1dCgKICAgICAgICBEYXRhICAgICAgICAgPT4gXCVEYXRhLAogICAgICAgIFRlbXBsYXRlRmlsZSA9PiAnQWdlbnRUaWNrZXRKSVJBJywKICAgICk7CiAgICAkT3V0cHV0ICAgLj0gJExheW91dE9iamVjdC0+Rm9vdGVyKCk7CgogICAgcmV0dXJuICRPdXRwdXQ7Cn0KCjE7Cg==
#  --
#  Copyright (C) 2016 - 2018 catworkx GmbH http://www.catworkx.de
#  --
# 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::AgentTicketJIRADialog;

use strict;
use warnings;

use List::Util qw(first);

our @ObjectDependencies = qw(
    Kernel::Config
    Kernel::System::Log
    Kernel::System::JSON
    Kernel::System::Main
    Kernel::Output::HTML::Layout
    Kernel::System::Ticket
    Kernel::System::TicketJIRA
    Kernel::System::JIRA::Utils
    Kernel::System::Web::Request
    Kernel::System::JIRA::QueueSettings
);

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

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

    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');
    $Self->{Debug}   = $ConfigObject->Get('JIRA::Debug');

    if ( $Self->{Debug} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'info',
            Message => __PACKAGE__ . '::new(): TicketID=' . $Param{TicketID},
        );
    }

    return $Self;
}

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

    my $TicketObject     = $Kernel::OM->Get('Kernel::System::Ticket');
    my $ConfigObject     = $Kernel::OM->Get('Kernel::Config');
    my $LanguageObject   = $Kernel::OM->Get('Kernel::Language');
    my $MainObject       = $Kernel::OM->Get('Kernel::System::Main');
    my $LogObject        = $Kernel::OM->Get('Kernel::System::Log');
    my $TicketJIRAObject = $Kernel::OM->Get('Kernel::System::TicketJIRA');
    my $LayoutObject     = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    my $UtilsObject      = $Kernel::OM->Get('Kernel::System::JIRA::Utils');
    my $ParamObject      = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $SettingsObject   = $Kernel::OM->Get('Kernel::System::JIRA::QueueSettings');

    my %GetParam;
    for my $Param ( qw/TicketID Project Assignee IssueType State ArticleID IssueID/ ) {
        $GetParam{$Param} = $ParamObject->GetParam( Param => $Param ) || '';
    }

    my $TemplateFile = 'AgentTicketJIRADialog';
    my $Defaults     = $ConfigObject->Get('TicketJIRA') || {};
    my %Data         = %GetParam;

    if ( !$Self->{Subaction} ) {
        $LayoutObject->Block(
            Name => 'Form',
            Data => {
                %GetParam,
                %Param,
            },
        );

        my %Ticket = $TicketObject->TicketGet(
            TicketID => $Self->{TicketID},
            UserID   => $Self->{UserID},
        );

        my %Settings = $SettingsObject->ValidSettingsGet( QueueID => $Ticket{QueueID} );

        my $Projects       = $UtilsObject->ProjectsGet();
        my $DefaultProject = $Settings{ProjectID} || $Defaults->{Project};

        my $ShowProject = $ConfigObject->Get('JIRA::ShowProject');

        if ( !$ShowProject ) {
            $LayoutObject->Block(
                Name => 'Project',
                Data => {
                    %GetParam,
                    %Param,
                    Project => $DefaultProject,
                },
            );
        }
        elsif ( $ShowProject == 1 ) {
            $Param{ProjectSelect} = $LayoutObject->BuildSelection(
                Name         => 'Project',
                Data         => $Projects,
                PossibleNone => 1,
                Translation  => 0,
                Class        => 'Validate Modernize',
                SelectedID   => $GetParam{Project} || $Param{Project} || $DefaultProject,
            );

            $LayoutObject->Block(
                Name => 'ProjectSelect',
                Data => {
                    %GetParam,
                    %Param,
                },
            );
        }

        my $Users = $UtilsObject->UsersGet(
            Project => $DefaultProject,
        );

        my $DefaultAssignee = $Settings{Assignee} || $Defaults->{Assignee};
        my $NoAssignee      = $ConfigObject->Get('JIRA::NoAssignee');
        my $ShowAssignee    = $ConfigObject->Get('JIRA::ShowAssignee');

        if ( !$ShowAssignee && !$NoAssignee ) {
            $LayoutObject->Block(
                Name => 'Assignee',
                Data => {
                    %GetParam,
                    %Param,
                    Assignee => $DefaultAssignee,
                },
            );
        }
        elsif ( $ShowAssignee == 1 && !$NoAssignee ) {
            if ( $ConfigObject->Get('JIRA::ShowAutomaticUser') ) {
                $Users->{'-1'} = $LanguageObject->Translate('automatic-user');
            }

            $Param{AssigneeSelect} = $LayoutObject->BuildSelection(
                Name         => 'Assignee',
                Data         => $Users,
                PossibleNone => 1,
                Translation  => 0,
                Class        => 'Validate Modernize',
                SelectedID   => $GetParam{Assignee} || $Param{Assignee} || $DefaultAssignee,
            );

            $LayoutObject->Block(
                Name => 'AssigneeSelect',
                Data => {
                    %GetParam,
                    %Param,
                    Value => $GetParam{Assignee} || $Param{Assignee} || $DefaultAssignee,
                },
            );
        }

        my $IssueTypes = $UtilsObject->IssueTypesGet(
            Project => $DefaultProject,
        );

        my $DefaultIssueType = $Defaults->{IssueType};
        my $ShowIssueType    = $ConfigObject->Get('JIRA::ShowIssueType');

        if ( !$ShowIssueType ) {
            $LayoutObject->Block(
                Name => 'IssueType',
                Data => {
                    %GetParam,
                    %Param,
                    IssueType => $DefaultIssueType,
                },
            );
        }
        elsif ( $ShowIssueType == 1 ) {
            $Param{IssueTypeSelect} = $LayoutObject->BuildSelection(
                Name         => 'IssueType',
                Data         => $IssueTypes,
                PossibleNone => 1,
                Class        => 'Validate Modernize',
                SelectedID   => $GetParam{IssueType} || $Param{IssueType} || $DefaultIssueType,
            );

            $LayoutObject->Block(
                Name => 'IssueTypeSelect',
                Data => {
                    %GetParam,
                    %Param,
                },
            );
        }

# TODO: When we know how to get a list of states, then this should be reenabled...
#        my $DefaultState = $Defaults->{State};
#
#        if ( !$ConfigObject->Get('JIRA::ShowState') ) {
#            $LayoutObject->Block(
#                Name => 'State',
#                Data => {
#                    %GetParam,
#                    %Param,
#                    State => $DefaultState,
#                },
#            );
#        }
#        else {
#            $Param{StateSelect} = $LayoutObject->BuildSelection(
#                Name         => 'State',
#                Data         => {},
#                PossibleNone => 1,
#                Class        => 'Validate',
#            );
#
#            $LayoutObject->Block(
#                Name => 'StateSelect',
#                Data => {
#                    %GetParam,
#                    %Param,
#                },
#            )
#        }
    }
    elsif ( $Self->{Subaction} eq 'AssigneeSearch' ) {

        # build result list
        my @Result;

        my $Term = $ParamObject->GetParam( Param => 'Term' );
        $Term    =~ s{\*}{.*}g;

        my $RE;
        eval {
            $RE = qr/$Term/i;
        };

        if ( $GetParam{Project} && $RE ) {
            my $Users = $UtilsObject->UsersGet(
                Project => $GetParam{Project},
            );

            if ( $ConfigObject->Get('JIRA::ShowAutomaticUser') ) {
                $Users->{'-1'} = $LanguageObject->Translate('automatic-user');
            }

            ASSIGNEE:
            for my $User ( keys %{ $Users || {} } ) {
                if ( ( $User =~ $RE ) || ( $Users->{$User} =~ $RE ) ) {
                    push @Result, {
                        Label => "$Users->{$User} ($User)",
                        Value => $User,
                    };
                }
            }
        }

        my $JSON = $LayoutObject->JSONEncode(
            Data => \@Result,
        );

        return $LayoutObject->Attachment(
            ContentType => 'application/json; charset=' . $LayoutObject->{Charset},
            Content     => $JSON || '',
            Type        => 'inline',
            NoCache     => 1,
        );
    }
    elsif ( $Self->{Subaction} eq 'AJAXUpdate' ) {
        my $JSONData = [];

        if ( $GetParam{Project} ) {
            my $Users = $UtilsObject->UsersGet(
                Project => $GetParam{Project},
            );

            my $DefaultAssignee = $Defaults->{Assignee};

            if ( $ConfigObject->Get('JIRA::ShowAutomaticUser') ) {
                $Users->{'-1'} = $LanguageObject->Translate('automatic-user');
            }

            push @{ $JSONData }, {
                Name         => 'Assignee',
                Data         => $Users,
                PossibleNone => 1,
                Class        => 'Validate Modernize',
                SelectedID   => $GetParam{Assignee} || $Param{Assignee} || $DefaultAssignee,
            };

            my $IssueTypes = $UtilsObject->IssueTypesGet(
                Project => $GetParam{Project},
            );

            my $DefaultIssueType = $Defaults->{IssueType};

            push @{ $JSONData }, {
                Name         => 'IssueType',
                Data         => $IssueTypes,
                PossibleNone => 1,
                Class        => 'Validate Modernize',
                SelectedID   => $GetParam{IssueType} || $Param{IssueType} || $DefaultIssueType,
            };

            my $States = $UtilsObject->StatesGet(
                Project => $GetParam{Project},
            );

            my $DefaultState = $Defaults->{State};

            push @{ $JSONData }, {
                Name         => 'State',
                Data         => $States,
                PossibleNone => 1,
                Class        => 'Validate Modernize',
                SelectedID   => $GetParam{State} || $Param{State} || $DefaultState,
            };
        }

        my $JSON = $LayoutObject->BuildSelectionJSON( $JSONData );

        return $LayoutObject->Attachment(
            ContentType => 'application/json; charset=' . $LayoutObject->{Charset},
            Content     => $JSON,
            Type        => 'inline',
            NoCache     => 1,
        );
    }
    elsif ( $Self->{Subaction} eq 'Create' ) {
        $LayoutObject->Block(
            Name => 'CreateResult',
        );

        my $Block = 'Success';

        my %Ticket = $TicketObject->TicketGet(
            TicketID => $Self->{TicketID},
            UserID   => $Self->{UserID},
        );

        if ( $Self->{Debug} ) {
            $LogObject->Log(
                Priority => 'info',
                Message  => __PACKAGE__ . "::Run(): TicketID: $Ticket{TicketID} Title: $Ticket{Title} State: $Ticket{State}"
            );
        }

        my $JIRAConfig = $ConfigObject->Get('TicketJIRA') || {};

        my $AllowedQueuesList         = $JIRAConfig->{OTRSTicketAllowedQueues4Create};
        my $AllowCreationInAllQueues  = $JIRAConfig->{OTRSTicketAllowCreationInAllQueues};

        if ( $AllowCreationInAllQueues
             || (@{$AllowedQueuesList} && grep { $Ticket{Queue} eq $_ } @{$AllowedQueuesList}) ) {
            my $JIRAIssue = $TicketJIRAObject->IssueCreate(
                TicketID  => $Self->{TicketID},
                ArticleID => $GetParam{ArticleID},
                UserID    => $Self->{UserID},
                Project   => $GetParam{Project},
                Assignee  => $GetParam{Assignee},
                IssueType => $GetParam{IssueType},
            );

            if ( $Self->{Debug} ) {
                $LogObject->Log(
                    Priority => 'info',
                    Message  => __PACKAGE__ . "::Run(): " . $MainObject->Dump($JIRAIssue),
                );
            }

            if ( $JIRAIssue->{error} ) {
                $Block       = 'Error';
                $Data{error} = $JIRAIssue->{error};

                $Data{TicketID}     = $GetParam{TicketID};
                $Data{TicketNumber} = $Ticket{TicketNumber};
            }
            else {
                # load new URL in parent window and close popup
                my $ReturnURL = "Action=AgentTicketZoom;TicketID=$GetParam{TicketID}";

                return $LayoutObject->PopupClose(
                    URL => $ReturnURL,
                );
            }

            my $URL = $JIRAConfig->{URL};
	      
            $Data{id}  = $JIRAIssue->{id};
            $Data{key} = $JIRAIssue->{key};
            $Data{url} = "$URL/browse/" . $JIRAIssue->{key};

            if ( $Self->{Debug} ) {
                $LogObject->Log(
                    Priority => 'info',
                    Message  => __PACKAGE__ . "::Run(): Ticket->State $Ticket{State} allows fetching "
                        . "of the JIRAIssue->key: " . $JIRAIssue->{key},
                );
            }
        }
        else {
            my $AllowedQueuesNames = join(', ', @{$AllowedQueuesList});
            $Block        = 'Error';
            $Data{error}  = $LayoutObject->{LanguageObject}->Translate(
                "Ticket not in $AllowedQueuesNames: " . $Ticket{Queue} . "."
            );

            if ( $Self->{Debug} ) {
                $LogObject->Log(
                    Priority => 'info',
                    Message  => __PACKAGE__ . "::Run(): OTRS Ticket not in $AllowedQueuesNames: " . $Ticket{Queue}
                );
            }
        }

        $Data{TicketID}     = $Self->{TicketID};
        $Data{TicketNumber} = $Ticket{TicketNumber};

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

    # user wants to link ticket with an existing issue
    # needs on other form
    elsif ( $Self->{Subaction} eq 'LinkExistingIssue' ) {
        $LayoutObject->Block(
            Name => 'Form',
            Data => {
                %GetParam,
                %Param,
            },
        );

        $TemplateFile = 'AgentTicketJIRADialogExistingIssue';

        my $Projects       = $UtilsObject->ProjectsGet();
        my $DefaultProject = $Defaults->{Project};

        if ( !$ConfigObject->Get('JIRA::ShowProject') ) {
            $LayoutObject->Block(
                Name => 'Project',
                Data => {
                    %GetParam,
                    %Param,
                    Project => $DefaultProject,
                },
            );
        }
        else {
            $Param{ProjectSelect} = $LayoutObject->BuildSelection(
                Name         => 'Project',
                Data         => $Projects,
                PossibleNone => 1,
                Translation  => 0,
                Class        => 'Validate Modernize',
                SelectedID   => $GetParam{Project} || $Param{Project} || $DefaultProject,
            );

            $LayoutObject->Block(
                Name => 'ProjectSelect',
                Data => {
                    %GetParam,
                    %Param,
                },
            );
        }
    }

    # user wants to link ticket with an existing issue
    #   -> create link (set dynamic field in otrs and custom field in JIRA)
    elsif ( $Self->{Subaction} eq 'CreateLinkWithIssue' ) {
        $TemplateFile = 'AgentTicketJIRADialogExistingIssue';

        my %Ticket = $TicketObject->TicketGet(
            TicketID => $Self->{TicketID},
            UserID   => $Self->{UserID},
        );

        $LayoutObject->Block(
            Name => 'CreateResult',
        );

        my $Block = 'Success';

        my $JIRAIssue = $TicketJIRAObject->IssueLinkToOTRS(
            TicketID  => $Self->{TicketID},
            UserID    => $Self->{UserID},
            IssueID   => $GetParam{IssueID},
            Project   => $GetParam{Project},
        );

        if ( $Self->{Debug} ) {
            $LogObject->Log(
                Priority => 'info',
                Message  => __PACKAGE__ . "::Run(): " . $MainObject->Dump($JIRAIssue),
            );
        }

        if ( $JIRAIssue->{error} ) {
            $Block       = 'Error';
            $Data{error} = $JIRAIssue->{error};

            $Data{TicketID}     = $GetParam{TicketID};
            $Data{TicketNumber} = $Ticket{TicketNumber};
        }
        else {
            # load new URL in parent window and close popup
            my $ReturnURL = "Action=AgentTicketZoom;TicketID=$GetParam{TicketID}";

            return $LayoutObject->PopupClose(
                URL => $ReturnURL,
            );
        }

        my $JIRAConfig = $ConfigObject->Get('TicketJIRA') || {};
        my $URL        = $JIRAConfig->{URL};
	      
        $Data{id}  = $JIRAIssue->{id};
        $Data{key} = $JIRAIssue->{key};
        $Data{url} = "$URL/browse/" . $JIRAIssue->{key};
    }

    # user wants to link ticket with an existing issue
    #   -> search for existing issues
    elsif ( $Self->{Subaction} eq 'IssueSearch' ) {

        # build result list
        my $Issues;

        my $Term = $ParamObject->GetParam( Param => 'Term' );

        if ( $GetParam{Project} && $Term ) {
            $Issues = $UtilsObject->IssueList(
                Project => $GetParam{Project},
                Term    => $Term,
            );
        }

        my $JSON = $LayoutObject->JSONEncode(
            Data => $Issues,
        );

        return $LayoutObject->Attachment(
            ContentType => 'application/json; charset=' . $LayoutObject->{Charset},
            Content     => $JSON || '',
            Type        => 'inline',
            NoCache     => 1,
        );
    }

    # build output
    my $Output = $LayoutObject->Header(
        Type      => 'Small',
        BodyClass => 'Popup',
    );

    $Output .= $LayoutObject->Output(
        TemplateFile => $TemplateFile,
        Data         => {
            %GetParam,
            %Data,
        },
    );

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

    return $Output;
}

1;

IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9GaWx0ZXJFbGVtZW50UG9zdC9PVFJTVG9KSVJBTGluay5wbQojIENvcHlyaWdodCAoQykgMjAxNiBjYXR3b3JreCBHbWJIIGh0dHA6Ly93d3cuY2F0d29ya3guZGUKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkZpbHRlckVsZW1lbnRQb3N0OjpPVFJTVG9KSVJBTGluazsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gcXcoCiAgICBLZXJuZWw6OkNvbmZpZwogICAgS2VybmVsOjpMYW5ndWFnZQogICAgS2VybmVsOjpTeXN0ZW06OkxvZwogICAgS2VybmVsOjpTeXN0ZW06Ok1haW4KICAgIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQKICAgIEtlcm5lbDo6U3lzdGVtOjpXZWI6OlJlcXVlc3QKKTsKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJExhbmd1YWdlT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpMYW5ndWFnZScpOwogICAgbXkgJExheW91dE9iamVjdCAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkxheW91dCcpOwogICAgbXkgJFBhcmFtT2JqZWN0ICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OldlYjo6UmVxdWVzdCcpOwogICAgbXkgJFRpY2tldE9iamVjdCAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlRpY2tldCcpOwoKICAgIG15ICRUaXRsZSAgICA9ICRMYW5ndWFnZU9iamVjdC0+VHJhbnNsYXRlKCdPVFJTIHRvIEppcmEnKTsKICAgIG15ICRCYXNlbGluayA9ICRMYXlvdXRPYmplY3QtPntCYXNlbGlua307CiAgICBteSAkVGlja2V0SUQgPSAkUGFyYW1PYmplY3QtPkdldFBhcmFtKCBQYXJhbSA9PiAnVGlja2V0SUQnICk7CgogICAgbXkgJVRpY2tldCA9ICRUaWNrZXRPYmplY3QtPlRpY2tldEdldCgKICAgICAgICBUaWNrZXRJRCAgICAgID0+ICRUaWNrZXRJRCwKICAgICAgICBEeW5hbWljRmllbGRzID0+IDEsCiAgICApOwoKICAgIHJldHVybiAxIGlmICRUaWNrZXR7RHluYW1pY0ZpZWxkX0pJUkFJc3N1ZUlEfTsKCiAgICBteSBAQXJ0aWNsZUlEcyA9ICR7ICRQYXJhbXtEYXRhfSB9ID1+IG17PGEgXHMgbmFtZT0iQXJ0aWNsZShcZCspIn14bXM7CgogICAgJHsgJFBhcmFte0RhdGF9IH0gPX4gc3sgKCA8YSBccyBuYW1lPSJBcnRpY2xlKFxkKykiIC4qPyA8dWwgXHMgY2xhc3M9IkFjdGlvbnMiPiApIFxzKyA8bGk+fXskMSAuICRTZWxmLT5fX0xpbmtpZnkoICRCYXNlbGluaywgJFRpY2tldElELCAkMiwgJFRpdGxlICkgLiAnPGxpPic7fWV4bXNnOwoKICAgIHJldHVybiAxOwp9CgpzdWIgX19MaW5raWZ5IHsKICAgIG15ICgkU2VsZiwgJEJhc2VsaW5rLCAkVGlja2V0SUQsICRBcnRpY2xlSUQsICRUaXRsZSApID0gQF87CgogICAgbXkgJENvbmZpZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyk7CgogICAgbXkgJERpYWxvZyA9ICRDb25maWdPYmplY3QtPkdldCgnSklSQTo6U2hvd0RpYWxvZycpOwoKICAgIG15ICRBY3Rpb24gPSAnQWdlbnRUaWNrZXRKSVJBJzsKICAgICRBY3Rpb24gICAuPSAnRGlhbG9nJyBpZiAkRGlhbG9nOwoKICAgIG15ICRMaW5rID0gcXF+CiAgICAgICAgPGxpIGNsYXNzPSJPVFJTVG9KSVJBTGlua0xpbmsiPgogICAgICAgICAgICA8YSBocmVmPSIke0Jhc2VsaW5rfUFjdGlvbj0kQWN0aW9uO1RpY2tldElEPSRUaWNrZXRJRDtBcnRpY2xlSUQ9JEFydGljbGVJRCIgdGl0bGU9IiRUaXRsZSIgY2xhc3M9IkFzUG9wdXAgUG9wdXBfVHlwZV9UaWNrZXRBY3Rpb24iPiRUaXRsZTwvYT4KICAgICAgICA8L2xpPgogICAgfjsKCiAgICByZXR1cm4gJExpbms7Cn0KCjE7Cg==
# --
# Copyright (C) 2017 Perl-Services.de, http://www.perl-services.de
# --
# 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>[% Translate("Jira Queue Settings") | html %]</h1>
    <div class="SidebarColumn">
        <div class="WidgetSimple">
            <div class="Header">
                <h2>[% Translate("Actions") | html %]</h2>
            </div>
            <div class="Content">
                <ul class="ActionList">
                    <li>
[% RenderBlockStart("AddAction") %]
                        <a href="[% Env("Baselink") %]Action=[% Env("Action") %];Subaction=Edit" class="CallForAction Fullsize Center"><span><i class="fa fa-plus-square"></i>[% Translate("Add Queue Settings") | html %]</span></a>
[% RenderBlockEnd("AddAction") %]
[% RenderBlockStart("OverviewAction") %]
                        <a href="[% Env("Baselink") %]Action=[% Env("Action") %]" class="CallForAction Fullsize Center"><span><i class="fa"></i>[% Translate("Go to overview") | html %]</span></a>
[% RenderBlockEnd("OverviewAction") %]
                    </li>
                </ul>
            </div>
        </div>
    </div>

[% RenderBlockStart("SettingsOverview") %]
    <div class="ContentColumn">
        <div class="WidgetSimple">
            <div class="Header">
                <h2>[% Translate("List") | html %]</h2>
            </div>
            <div class="Content">
                <table class="DataTable" id="ChecklistItemState">
                    <thead>
                        <tr>
                            <th>[% Translate("Queue") | html %]</th>
                            <th>[% Translate("Project") | html %]</th>
                            <th>[% Translate("Assignee") | html %]</th>
                            <th>[% Translate("Validity") | html %]</th>
                            <th>[% Translate("Actions") | html %]</th>
                        </tr>
                    </thead>
                    <tbody>
[% RenderBlockStart("NoSettingsFound") %]
                        <tr>
                            <td colspan="5">[% Translate("No matches found.") | html %]</td>
                        </tr>
[% RenderBlockEnd("NoSettingsFound") %]
[% RenderBlockStart("Setting") %]
                        <tr>
                            <td>
                                [% Data.QueueName | html %]
                            </td>
                            <td>
                                [% Data.Project | html %] ([% Data.ProjectID | html %])
                            </td>
                            <td>
                                [% Data.Assignee | html %]
                            </td>
                            <td>
                                [% Translate(Data.Valid) | html %]
                            </td>
                            <td>
                                <a href="[% Env("Baselink") %]Action=[% Env("Action") %]&Subaction=Edit&QueueID=[% Data.QueueID | uri %]">
                                [% Translate("edit") | html %]
                                </a> |
                                <a href="[% Env("Baselink") %]Action=[% Env("Action") %]&Subaction=Delete&QueueID=[% Data.QueueID | uri %];[% Env("ChallengeTokenParam") | html %]">
                                [% Translate("delete") | html %]
                                </a>
                            </td>
                        </tr>
[% RenderBlockEnd("Setting") %]
                    </tbody>
                </table>
            </div>
        </div>
    </div>
[% RenderBlockEnd("SettingsOverview") %]


[% RenderBlockStart("SettingsMask") %]
    <div class="ContentColumn">
        <div class="WidgetSimple">
            <div class="Header">
                <h2>[% Translate("Edit/Add Jira Queue Settings") | html %]</h2>
            </div>
            <div class="Content">
                <form action="[% Env("CGIHandle") %]" method="post" class="Validate" id="NewEditSetting">

                    <input type="hidden" name="Action" value="[% Env("Action") %]"/>
                    <input type="hidden" name="Subaction" value="Save" />
                    <input type="hidden" name="ID" value="[% Data.ID | html %]"/>

                    <fieldset class="TableLike">
                        <label for="QueueID" class="Mandatory">
                            <span class="Marker">*</span>
                            [% Translate("Queue") | html %]:
                        </label>
                        <div class="Field">
                            [% Data.QueueSelect %]
                            <div id="QueueError" class="TooltipErrorMessage">
                                <p>[% Translate("A Queue is required.") | html %]</p>
                            </div>
                            <div id="QueueServerError" class="TooltipErrorMessage">
                                <p>[% Translate("A Queue is required.") | html %]</p>
                            </div>
                        </div>
                        <div class="Clear"></div>

                        <label for="ProjectID" class="">
                            [% Translate("Project") | html %]:
                        </label>
                        <div class="Field">
                            [% Data.ProjectSelect %]
                        </div>
                        <div class="Clear"></div>

                        <label for="AssigneeID" class="">
                            [% Translate("Assignee") | html %]:
                        </label>
                        <div class="Field">
                            [% Data.UsersSelect %]
                        </div>
                        <div class="Clear"></div>

                        <label for="ValidID" class="Mandatory">
                            <span class="Marker">*</span>
                            [% Translate("Valid") | html %]:
                        </label>
                        <div class="Field">
                            [% Data.ValidSelect %]
                        </div>
                        <div class="Clear"></div>

                        <div class="Field">
                            <button class="Primary" type="submit" value="[% Translate("Save") | html %]">[% Translate("Save") | html %]</button>
                            [% Translate("or") | html %]
                            <a href="[% Env("Baselink") %]Action=[% Env("Action") %]">[% Translate("Cancel") | html %]</a>
                        </div>
                        <div class="Clear"></div>
                    </fieldset>
                </form>
            </div>
        </div>
    </div>
[% WRAPPER JSOnDocumentComplete %]
    <script type="text/javascript">//<![CDATA[
        $('#ProjectID').on('change', function() {
            Core.AJAX.FormUpdate( $('#NewEditSetting'), 'AJAXUpdate', 'ProjectID', [ 'Assignee' ] );
        });
    //]]></script>
[% END %]
[% RenderBlockEnd("SettingsMask") %]
    <div class="Clear"></div>
</div>

# --
# Copyright (C) 2016 Perl-Services.de, http://perl-services.de
# --
# 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">
[% RenderBlockStart("Steps") %]
    <div class="Center">
        <ul id="ProgressBar" class="ProgressBarElements[% Data.Steps | html %] SpacingTop SpacingBottom">
[% RenderBlockStart("StepJIRA") %]
            <li class="First Highlighted NoLink [% Data.Active | html %]" style="width: 315px">
                <span>
                    <a href="#">[% Translate("Step %s", Data.Step) | html %]<span>[% Translate("Jira Settings") | html %]</span></a>
                </span>
            </li>
[% RenderBlockEnd("StepJIRA") %]
[% RenderBlockStart("StepIssue") %]
            <li class="Highlighted NoLink [% Data.Active | html %]" style="width: 315px">
                <span>
                    <a href="#">[% Translate("Step %s", Data.Step) | html %]<span>[% Translate("Issue Settings") | html %]</span></a>
                </span>
            </li>
[% RenderBlockEnd("StepIssue") %]
[% RenderBlockStart("StepMapping") %]
            <li class="Last Highlighted NoLink [% Data.Active | html %]" style="width: 315px">
                <span>
                    <a href="#">[% Translate("Step %s", Data.Step) | html %]<span>[% Translate("Fieldmapping") | html %]</span></a>
                </span>
            </li>
[% RenderBlockEnd("StepMapping") %]
        </ul>
    </div>
[% RenderBlockEnd("Steps") %]

[% RenderBlockStart("Step1") %]
<div class="W950px SpacingTop SpacingBottom CenterBox">
    <form action="[% Env("CGIHandle") %]" method="post" class="PreventMultipleSubmits">
        <input type="hidden" name="Action" value="AdminJIRAWizard">
        <input type="hidden" name="Subaction" value="Step2">
        <input type="hidden" name="Step1Save" value="1">

        <div class="WidgetSimple">
            <div class="Header">
                <h2>[% Translate(Data.Item) | html %] ([% Data.Step %])</h2>
            </div>

            <div class="Content">
[% RenderBlockStart("Step1HasError") %]
                        <div class="Error">
                            [% Translate( Data.ErrorMessage ) | html %]
                        </div>
[% RenderBlockEnd("Step1HasError") %]
                <fieldset class="TableLike">
[% RenderBlockStart("Field") %]
                    <div class="Row [% Data.FieldClass | html %]">
                    <label for="[% Data.Field | html %]">[% Translate(Data.Label) | html %]:</label>
                    <div class="Field">
[% RenderBlockStart("FieldInput") %]
                        <input type="text" name="[% Data.Field | html %]" id="[% Data.Field | html %]" value="[% Data.Value %]">
[% RenderBlockEnd("FieldInput") %]
[% RenderBlockStart("FieldSelect") %]
                        [% Data.Field %]
[% RenderBlockEnd("FieldSelect") %]
[% RenderBlockStart("FieldHint") %]
                        <p class="FieldExplanation">
                            [% Translate(Data.Hint) | html %]
                        </p>
[% RenderBlockEnd("FieldHint") %]
                    </div>
                    </div>
                    <div class="Clear"></div>
[% RenderBlockEnd("Field") %]

                </fieldset>

                <div class="SpacingTop Right">
                    <button class="Primary CallForAction" type="submit"><span>[% Translate("Next") | html %]</span></button>
                </div>
            </div>
        </div>
    </form>
</div>
[% RenderBlockEnd("Step1") %]


[% RenderBlockStart("Step2") %]
<div class="W950px SpacingTop SpacingBottom CenterBox">
    <form action="[% Env("CGIHandle") %]" method="post" class="PreventMultipleSubmits">
        <input type="hidden" name="Action" value="AdminJIRAWizard">
        <input type="hidden" name="Subaction" value="Step3">
        <input type="hidden" name="Step2Save" value="1">

        <div class="WidgetSimple">
            <div class="Header">
                <h2>[% Translate(Data.Item) | html %] ([% Data.Step %])</h2>
            </div>

            <div class="Content">
[% RenderBlockStart("Step2HasError") %]
                        <div class="Error">
                            [% Translate( Data.ErrorMessage ) | html %]
                        </div>
[% RenderBlockEnd("Step2HasError") %]
                <fieldset class="TableLike">
[% RenderBlockStart("FieldStep2") %]
                    <div class="Row [% Data.FieldClass | html %]">
                    <label for="[% Data.Field | html %]">[% Translate(Data.Label) | html %]:</label>
                    <div class="Field">
[% RenderBlockStart("FieldInput2") %]
                        <input type="text" name="[% Data.Field | html %]" id="[% Data.Field | html %]" value="[% Data.Value %]">
[% RenderBlockEnd("FieldInput2") %]
[% RenderBlockStart("FieldSelect2") %]
                        [% Data.Field %]
[% RenderBlockEnd("FieldSelect2") %]
[% RenderBlockStart("FieldHintStep2") %]
                        <p class="FieldExplanation">
                            [% Translate(Data.Hint) | html %]
                        </p>
[% RenderBlockEnd("FieldHintStep2") %]
[% RenderBlockStart("FieldChangeStep2") %]
[% WRAPPER JSOnDocumentComplete %]
    <script type="text/javascript">//<![CDATA[
        JIRA.AdminWizard.InitSelect( "[% Data.Field | html %]" );
    //]]></script>
[% END %]
[% RenderBlockEnd("FieldChangeStep2") %]
                    </div>
                    </div>
                    <div class="Clear"></div>
[% RenderBlockEnd("FieldStep2") %]

                </fieldset>

                <div class="SpacingTop Right">
                    <button class="Primary CallForAction" type="submit"><span>[% Translate("Next") | html %]</span></button>
                </div>
            </div>
        </div>
    </form>
</div>
[% RenderBlockEnd("Step2") %]

[% RenderBlockStart("Step3") %]
<div class="W950px SpacingTop SpacingBottom CenterBox">
    <form action="[% Env("CGIHandle") %]" method="post" class="PreventMultipleSubmits">
        <input type="hidden" name="Action" value="AdminJIRAWizard">
        <input type="hidden" name="Subaction" value="Step3Save">
        <input type="hidden" name="Step3Save" value="1">

        <div class="WidgetSimple">
            <div class="Header">
                <h2>[% Translate(Data.Item) | html %] ([% Data.Step %])</h2>
            </div>

            <div class="Content">
                <fieldset class="TableLike">
                    <table class="DataTable">
                        <thead>
                            <tr>
                                <th>[% Translate("OTRS Ticket Field") | html %]</th>
                                <th>[% Translate("Jira Field") | html %]</th>
                            </tr>
                        </thead>
                        <tbody>
[% RenderBlockStart("Row") %]
                            <tr>
                                <td>[% Data.OTRS %]</td>
                                <td>[% Data.JIRA %]</td>
                            </tr>
[% RenderBlockEnd("Row") %]
                        </tbody>
                    </table>

                    <div class="WidgetSimple Collapsed">
                        <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("Information") | html %]</h2>
                        </div>
                        <div class="Content">

                           <fieldset class="TableLike">
                               <div class="Row">
                                   <p>
                                   [% Translate("This mapping defines which information are transferred from OTRS to Jira. On the left you see all the available information in OTRS and on the right you see the fields in Jira. Please keep in mind, that the type of the fields should match (date/time information in OTRS need to be stored in a date/time field in Jira)") | html %]
                                   [% Translate("These information are transferred automatically:") | html %]
                                   </p>
                                   <div class="Field">
                                       <ol>
                                           <li>[% Translate("Project (standard value from SysConfig or selected project)") | html %]</li>
                                           <li>[% Translate("Issuetype (standard value from SysConfig or selected issuetype)") | html %]</li>
                                           <li>[% Translate("Summary (ticket title)") | html %]</li>
                                           <li>[% Translate("Description (article body)") | html %]</li>
                                           <li>[% Translate("Reporter (standard value from SysConfig or selected reporter)") | html %]</li>
                                           <li>[% Translate("Assignee (standard value from SysConfig or selected assignee)") | html %]</li>
                                           <li>[% Translate("OTRSTicketID customfield (ticket id)") | html %]</li>
                                       </ol>
                                   </div>
                               </div>
                           </fieldset>
                       </div>
                   </div>

                </fieldset>

                <div class="SpacingTop Right">
                    <button class="Primary CallForAction" type="submit"><span>[% Translate("Next") | html %]</span></button>
                </div>
            </div>
        </div>
    </form>
</div>
[% RenderBlockEnd("Step3") %]

[% RenderBlockStart("ThankYou") %]
<div class="W950px SpacingTop SpacingBottom CenterBox">
    <div class="WidgetSimple">
        <div class="Header">
            <h2>[% Translate("Configuration done") | html %] ([% Data.Step %])</h2>
        </div>

        <div class="Content">
            <form action="[% Env("CGIHandle") %]" method="post" class="PreventMultipleSubmits">
                <input type="hidden" name="Action" value="Admin">
                <div class="SpacingTop Center">
                    <button class="Primary CallForAction" type="submit"><span>[% Translate("Go to Admin area") | html %]</span></button>
                </div>
            </form>
        </div>
    </div>
</div>
[% RenderBlockEnd("ThankYou") %]
</div>


IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9TdGFuZGFyZC9BZ2VudFRpY2tldEpJUkEudHQgLSBvdmVydmlldwojIENvcHlyaWdodCAoQykgMjAxNSBjYXR3b3JreCBHbWJoLCBodHRwOi8vd3d3LmNhdHdvcmt4LmRlCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCjxkaXYgY2xhc3M9Ik1haW5Cb3ggQVJJQVJvbGVNYWluIj4KICA8ZGl2IGNsYXNzPSJDbGVhckxlZnQiPjwvZGl2PgogIDxkaXYgY2xhc3M9IkNvbnRlbnQiPgoKWyUgUmVuZGVyQmxvY2tTdGFydCgiU3VjY2VzcyIpICVdCiAgICA8IS0tIHN0YXJ0IGZvcm0gLS0+CiAgICA8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iMTAwJSIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIzIiBzdHlsZT0icGFkZGluZy1sZWZ0OiAxY207cGFkZGluZy10b3A6IDFjbTsiPgogICAgICA8dHI+CiAgICAgICAgPHRoIGNsYXNzPSJtYWluaGVhZCI+CiAgICAgICAgICA8aDE+WyUgRW52KCJCb3gwIikgJV1bJSBUcmFuc2xhdGUoIkEgbmV3IEppcmEgSXNzdWUgaGFzIGJlZW4gY3JlYXRlZCIpIHwgaHRtbCAlXSFbJSBFbnYoIkJveDAiKSAlXTwvaDE+CiAgICAgICAgPC90aD4KICAgICAgPC90cj4KICAgICAgPHRyPgogICAgICAgIDx0ZCBjbGFzcz0ibWFpbmJvZHkiPgogICAgICAgICAgWyUgVHJhbnNsYXRlKCJUbyB0aGUgaXNzdWUgKEppcmEpIikgfCBodG1sICVdOiA8YSBocmVmPSJbJSBEYXRhLnVybCB8IGh0bWwgJV0iPlslIERhdGEua2V5IHwgaHRtbCAlXSAoWyUgRGF0YS5pZCB8IGh0bWwgJV0pPC9hPgogICAgICAgIDwvdGQ+CiAgICAgIDwvdHI+CiAgICAgIDx0cj4KICAgICAgICA8dGQgY2xhc3M9Im1haW5ib2R5Ij4KICAgICAgICAgIFslIFRyYW5zbGF0ZSgiQmFjayB0byB0aGUgdGlja2V0IChPVFJTKSIpIHwgaHRtbCAlXTogPGEgaHJlZj0iWyUgRW52KCJCYXNlbGluayIpICVdQWN0aW9uPUFnZW50VGlja2V0Wm9vbSZUaWNrZXRJRD1bJSBEYXRhLlRpY2tldElEIHwgaHRtbCAlXSI+WyUgRGF0YS5UaWNrZXROdW1iZXIgfCBodG1sICVdIChbJSBEYXRhLlRpY2tldElEIHwgaHRtbCAlXSk8L2E+CiAgICAgICAgPC90ZD4KICAgICAgPC90cj4KICAgIDwvdGFibGU+CiAgICA8IS0tIGVuZCBmb3JtIC0tPgpbJSBSZW5kZXJCbG9ja0VuZCgiU3VjY2VzcyIpICVdCgpbJSBSZW5kZXJCbG9ja1N0YXJ0KCJFcnJvciIpICVdCiAgICA8IS0tIHN0YXJ0IGZvcm0gLS0+CiAgICA8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iMTAwJSIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIzIiBzdHlsZT0icGFkZGluZy1sZWZ0OiAxY207cGFkZGluZy10b3A6IDFjbTsiPgogICAgICA8dHI+CiAgICAgICAgPHRoIGNsYXNzPSJtYWluaGVhZCI+CiAgICAgICAgICA8aDE+WyUgRW52KCJCb3gwIikgJV1bJSBUcmFuc2xhdGUoIkEgSmlyYSBJc3N1ZSBjb3VsZCBub3QgYmUgY3JlYXRlZCIpIHwgaHRtbCAlXSFbJSBFbnYoIkJveDAiKSAlXTwvaDE+CiAgICAgICAgPC90aD4KICAgICAgPC90cj4KICAgICAgPHRyPgogICAgICAgIDx0ZCBjbGFzcz0ibWFpbmJvZHkiPgogICAgICAgICAgWyUgRW52KCJCb3gwIikgJV1bJSBEYXRhLmVycm9yIHwgaHRtbCAlXVslIEVudigiQm94MCIpICVdCiAgICAgICAgPC90ZD4KICAgICAgPC90cj4KICAgICAgPHRyPgogICAgICAgIDx0ZCBjbGFzcz0ibWFpbmJvZHkiPgogICAgICAgICAgWyUgVHJhbnNsYXRlKCJCYWNrIHRvIHRoZSB0aWNrZXQgKE9UUlMpIikgfCBodG1sICVdOiA8YSBocmVmPSJbJSBFbnYoIkJhc2VsaW5rIikgJV1BY3Rpb249QWdlbnRUaWNrZXRab29tJlRpY2tldElEPVslIERhdGEuVGlja2V0SUQgfCBodG1sICVdIj5bJSBEYXRhLlRpY2tldE51bWJlciB8IGh0bWwgJV0gKFslIERhdGEuVGlja2V0SUQgfCBodG1sICVdKTwvYT4KICAgICAgICA8L3RkPgogICAgICA8L3RyPgogICAgPC90YWJsZT4KICAgIDwhLS0gZW5kIGZvcm0gLS0+ClslIFJlbmRlckJsb2NrRW5kKCJFcnJvciIpICVdCgogIDwvZGl2Pgo8L2Rpdj4K
# --
# Kernel/Output/HTML/Standard/AgentTicketJIRADialog.tt - overview
# Copyright (C) 2016 catworkx Gmbh, http://www.catworkx.de
# --
# 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.
# --
<form action="[% Env("CGIHandle") %]" method="POST" id="JIRAIssueForm" class="Validate PreventMultipleSubmits">
    <input type="hidden" name="Action" value="[% Env("Action") | html %]" />
    <input type="hidden" name="Subaction" value="Create" />
    <input type="hidden" name="TicketID" value="[% Data.TicketID | html %]" />
    <input type="hidden" name="ArticleID" value="[% Data.ArticleID | html %]" />
    <div class="LayoutPopup ARIARoleMain">
        <div class="Header">
            <h1>[% Translate("Create issue in Jira") | html %]</h1>
    
            <p class="AsteriskExplanation">[% Translate("All fields marked with an asterisk (*) are mandatory.") | html %]</p>
    
            <p>
                <a class="CancelClosePopup" href="#">[% Translate("Cancel & close") | html %]</a>
                [% Translate("or") | html %]
                <a href="[% Env("Baselink") %]Action=[% Env("Action") %];Subaction=LinkExistingIssue;TicketID=[% Data.TicketID | uri %];ArticleID=[% Data.ArticleID | uri %]"
                    >[% Translate("Link with existing issue") | html %]</a>
            </p>
        </div>

[% RenderBlockStart("Error") %]
        <div class="Content">
            <h2>[% Env("Box0") %][% Translate("A Jira Issue could not be created") | html %]![% Env("Box0") %]</h2>
            [% Data.error | html %]
        </div>
[% RenderBlockEnd("Error") %]

[% RenderBlockStart("Form") %]
        <div class="Content">
            <fieldset class="TableLike FixedLabel">

[% RenderBlockStart("Project") %]
                <label for="Project">[% Translate("Project") | html %]:</label>
                <div class="Field">
                    [% Data.ProjectSelect %]
                </div>
                <div class="Clear"></div>
[% RenderBlockEnd("Project") %]
[% RenderBlockStart("ProjectSelect") %]
                <label class="Mandatory" for="Project"><span class="Marker">*</span>[% Translate("Project") | html %]:</label>
                <div class="Field">
                    [% Data.ProjectSelect %]
                    <div id="ProjectError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                    <div id="ProjectServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                </div>
                <div class="Clear"></div>
[% WRAPPER JSOnDocumentComplete %]
<script type="text/javascript">//<![CDATA[
    $('#Project').bind( 'change', function (Event) {
        Core.AJAX.FormUpdate($('#JIRAIssueForm'), 'AJAXUpdate', 'Project', [ 'Assignee', 'IssueType', 'State' ]);
    });
//]]></script>
[% END %]
[% RenderBlockEnd("Project") %]

[% RenderBlockStart("Assignee") %]
                <label for="Assignee">[% Translate("Assignee") | html %]:</label>
                <div class="Field">
                    [% Data.Assignee %]
                </div>
                <div class="Clear"></div>
[% RenderBlockEnd("Assignee") %]
[% RenderBlockStart("AssigneeSelect") %]
                <label class="Mandatory" for="Assignee"><span class="Marker">*</span>[% Translate("Assignee") | html %]:</label>
                <div class="Field">
                    <input type="text" name="AssigneeSearchField" id="AssigneeSearchField" class="Validate Validate_Required" value="[% Data.Value | html %]" />
                    <input type="hidden" name="Assignee" id="Assignee" value="[% Data.Value | html %]" />
[% WRAPPER JSOnDocumentComplete %]
<script type="text/javascript">//<![CDATA[
    JIRA.Autocomplete.Init( $('#AssigneeSearchField'), 'AssigneeSearch' );
//]]></script>
[% END %]
                    <div id="AssigneeSearchFieldError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                    <div id="AssigneeSearchFieldServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                </div>
                <div class="Clear"></div>
[% RenderBlockEnd("AssigneeSelect") %]

[% RenderBlockStart("IssueType") %]
                <label for="IssueType">[% Translate("IssueType") | html %]:</label>
                <div class="Field">
                    [% Data.IssueType %]
                </div>
                <div class="Clear"></div>
[% RenderBlockEnd("IssueType") %]
[% RenderBlockStart("IssueTypeSelect") %]
                <label class="Mandatory" for="IssueType"><span class="Marker">*</span>[% Translate("IssueType") | html %]:</label>
                <div class="Field">
                    [% Data.IssueTypeSelect %]
                    <div id="IssueTypeError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                    <div id="IssueTypeServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                </div>
                <div class="Clear"></div>
[% RenderBlockEnd("IssueTypeSelect") %]

[% RenderBlockStart("State") %]
                <label for="State">[% Translate("State") | html %]:</label>
                <div class="Field">
                    [% Data.State %]
                </div>
                <div class="Clear"></div>
[% RenderBlockEnd("State") %]
[% RenderBlockStart("StateSelect") %]
                <label class="Mandatory" for="State"><span class="Marker">*</span>[% Translate("State") | html %]:</label>
                <div class="Field">
                    [% Data.StateSelect %]
                    <div id="StateError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                    <div id="StateServerError" class="TooltipErrorMessage"><p>[% Translate("This field is required.") | html %]</p></div>
                </div>
                <div class="Clear"></div>
[% RenderBlockEnd("StateSelect") %]

            </fieldset>
        </div>
        <div class="Footer">
            <button class="CallForAction Primary" id="submitRichText" accesskey="g" title="[% Translate("Submit") | html %] (g)" type="submit" value="[% Translate("Submit") | html %]"><span><i class="fa fa-check-square-o"></i> [% Translate("Submit") | html %]</span></button>
        </div>
[% RenderBlockEnd("Form") %]
    </div>
</form>

IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9TdGFuZGFyZC9BZ2VudFRpY2tldEpJUkFEaWFsb2dFeGlzdGluZ0lzc3VlLnR0IC0gb3ZlcnZpZXcKIyBDb3B5cmlnaHQgKEMpIDIwMTYgY2F0d29ya3ggR21iaCwgaHR0cDovL3d3dy5jYXR3b3JreC5kZQojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQo8Zm9ybSBhY3Rpb249IlslIEVudigiQ0dJSGFuZGxlIikgJV0iIG1ldGhvZD0iUE9TVCIgaWQ9IkpJUkFJc3N1ZUZvcm0iIGNsYXNzPSJWYWxpZGF0ZSBQcmV2ZW50TXVsdGlwbGVTdWJtaXRzIj4KICAgIDxpbnB1dCB0eXBlPSJoaWRkZW4iIG5hbWU9IkFjdGlvbiIgdmFsdWU9IlslIEVudigiQWN0aW9uIikgfCBodG1sICVdIiAvPgogICAgPGlucHV0IHR5cGU9ImhpZGRlbiIgbmFtZT0iU3ViYWN0aW9uIiB2YWx1ZT0iQ3JlYXRlTGlua1dpdGhJc3N1ZSIgLz4KICAgIDxpbnB1dCB0eXBlPSJoaWRkZW4iIG5hbWU9IlRpY2tldElEIiB2YWx1ZT0iWyUgRGF0YS5UaWNrZXRJRCB8IGh0bWwgJV0iIC8+CiAgICA8aW5wdXQgdHlwZT0iaGlkZGVuIiBuYW1lPSJBcnRpY2xlSUQiIHZhbHVlPSJbJSBEYXRhLkFydGljbGVJRCB8IGh0bWwgJV0iIC8+CiAgICA8ZGl2IGNsYXNzPSJMYXlvdXRQb3B1cCBBUklBUm9sZU1haW4iPgogICAgICAgIDxkaXYgY2xhc3M9IkhlYWRlciI+CiAgICAgICAgICAgIDxoMT5bJSBUcmFuc2xhdGUoIkxpbmsgd2l0aCBKaXJhIGlzc3VlIikgfCBodG1sICVdPC9oMT4KICAgIAogICAgICAgICAgICA8cCBjbGFzcz0iQXN0ZXJpc2tFeHBsYW5hdGlvbiI+WyUgVHJhbnNsYXRlKCJBbGwgZmllbGRzIG1hcmtlZCB3aXRoIGFuIGFzdGVyaXNrICgqKSBhcmUgbWFuZGF0b3J5LiIpIHwgaHRtbCAlXTwvcD4KICAgIAogICAgICAgICAgICA8cD4KICAgICAgICAgICAgICAgIDxhIGNsYXNzPSJDYW5jZWxDbG9zZVBvcHVwIiBocmVmPSIjIj5bJSBUcmFuc2xhdGUoIkNhbmNlbCAmIGNsb3NlIikgfCBodG1sICVdPC9hPgogICAgICAgICAgICAgICAgWyUgVHJhbnNsYXRlKCJvciIpIHwgaHRtbCAlXQogICAgICAgICAgICAgICAgPGEgaHJlZj0iWyUgRW52KCJCYXNlbGluayIpICVdQWN0aW9uPVslIEVudigiQWN0aW9uIikgJV07VGlja2V0SUQ9WyUgRGF0YS5UaWNrZXRJRCB8IHVyaSAlXTtBcnRpY2xlSUQ9WyUgRGF0YS5BcnRpY2xlSUQgfCB1cmkgJV0iCiAgICAgICAgICAgICAgICAgICAgPlslIFRyYW5zbGF0ZSgiQ3JlYXRlIG5ldyBpc3N1ZSIpIHwgaHRtbCAlXTwvYT4KICAgICAgICAgICAgPC9wPgogICAgICAgIDwvZGl2PgoKWyUgUmVuZGVyQmxvY2tTdGFydCgiRXJyb3IiKSAlXQogICAgICAgIDxkaXYgY2xhc3M9IkNvbnRlbnQiPgogICAgICAgICAgICA8aDI+WyUgRW52KCJCb3gwIikgJV1bJSBUcmFuc2xhdGUoIlRoZSBKaXJhIElzc3VlIGNvdWxkIG5vdCBiZSBsaW5rZWQiKSB8IGh0bWwgJV0hWyUgRW52KCJCb3gwIikgJV08L2gyPgogICAgICAgICAgICBbJSBEYXRhLmVycm9yIHwgaHRtbCAlXQogICAgICAgIDwvZGl2PgpbJSBSZW5kZXJCbG9ja0VuZCgiRXJyb3IiKSAlXQoKWyUgUmVuZGVyQmxvY2tTdGFydCgiRm9ybSIpICVdCiAgICAgICAgPGRpdiBjbGFzcz0iQ29udGVudCI+CiAgICAgICAgICAgIDxmaWVsZHNldCBjbGFzcz0iVGFibGVMaWtlIEZpeGVkTGFiZWwiPgoKWyUgUmVuZGVyQmxvY2tTdGFydCgiUHJvamVjdCIpICVdCiAgICAgICAgICAgICAgICA8bGFiZWwgZm9yPSJQcm9qZWN0Ij5bJSBUcmFuc2xhdGUoIlByb2plY3QiKSB8IGh0bWwgJV06PC9sYWJlbD4KICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9IkZpZWxkIj4KICAgICAgICAgICAgICAgICAgICBbJSBEYXRhLlByb2plY3RTZWxlY3QgJV0KICAgICAgICAgICAgICAgIDwvZGl2PgogICAgICAgICAgICAgICAgPGRpdiBjbGFzcz0iQ2xlYXIiPjwvZGl2PgpbJSBSZW5kZXJCbG9ja0VuZCgiUHJvamVjdCIpICVdClslIFJlbmRlckJsb2NrU3RhcnQoIlByb2plY3RTZWxlY3QiKSAlXQogICAgICAgICAgICAgICAgPGxhYmVsIGNsYXNzPSJNYW5kYXRvcnkiIGZvcj0iUHJvamVjdCI+PHNwYW4gY2xhc3M9Ik1hcmtlciI+Kjwvc3Bhbj5bJSBUcmFuc2xhdGUoIlByb2plY3QiKSB8IGh0bWwgJV06PC9sYWJlbD4KICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9IkZpZWxkIj4KICAgICAgICAgICAgICAgICAgICBbJSBEYXRhLlByb2plY3RTZWxlY3QgJV0KICAgICAgICAgICAgICAgICAgICA8ZGl2IGlkPSJQcm9qZWN0RXJyb3IiIGNsYXNzPSJUb29sdGlwRXJyb3JNZXNzYWdlIj48cD5bJSBUcmFuc2xhdGUoIlRoaXMgZmllbGQgaXMgcmVxdWlyZWQuIikgfCBodG1sICVdPC9wPjwvZGl2PgogICAgICAgICAgICAgICAgICAgIDxkaXYgaWQ9IlByb2plY3RTZXJ2ZXJFcnJvciIgY2xhc3M9IlRvb2x0aXBFcnJvck1lc3NhZ2UiPjxwPlslIFRyYW5zbGF0ZSgiVGhpcyBmaWVsZCBpcyByZXF1aXJlZC4iKSB8IGh0bWwgJV08L3A+PC9kaXY+CiAgICAgICAgICAgICAgICA8L2Rpdj4KICAgICAgICAgICAgICAgIDxkaXYgY2xhc3M9IkNsZWFyIj48L2Rpdj4KWyUgUmVuZGVyQmxvY2tFbmQoIlByb2plY3QiKSAlXQoKICAgICAgICAgICAgICAgIDxsYWJlbCBjbGFzcz0iTWFuZGF0b3J5IiBmb3I9Iklzc3VlSUQiPjxzcGFuIGNsYXNzPSJNYXJrZXIiPio8L3NwYW4+WyUgVHJhbnNsYXRlKCJJc3N1ZSIpIHwgaHRtbCAlXTo8L2xhYmVsPgogICAgICAgICAgICAgICAgPGRpdiBjbGFzcz0iRmllbGQiPgogICAgICAgICAgICAgICAgICAgIDxpbnB1dCB0eXBlPSJ0ZXh0IiBuYW1lPSJJc3N1ZUlEU2VhcmNoRmllbGQiIGlkPSJJc3N1ZUlEU2VhcmNoRmllbGQiIGNsYXNzPSJWYWxpZGF0ZSBWYWxpZGF0ZV9SZXF1aXJlZCIgdmFsdWU9IlslIERhdGEuVmFsdWUgfCBodG1sICVdIiAvPgogICAgICAgICAgICAgICAgICAgIDxpbnB1dCB0eXBlPSJoaWRkZW4iIG5hbWU9Iklzc3VlSUQiIGlkPSJJc3N1ZUlEIiB2YWx1ZT0iWyUgRGF0YS5WYWx1ZSB8IGh0bWwgJV0iIC8+ClslIFdSQVBQRVIgSlNPbkRvY3VtZW50Q29tcGxldGUgJV0KPHNjcmlwdCB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPi8vPCFbQ0RBVEFbCiAgICBKSVJBLkF1dG9jb21wbGV0ZS5Jbml0KCAkKCcjSXNzdWVJRFNlYXJjaEZpZWxkJyksICdJc3N1ZVNlYXJjaCcsICdJc3N1ZUlEJyApOwovL11dPjwvc2NyaXB0PgpbJSBFTkQgJV0KICAgICAgICAgICAgICAgICAgICA8ZGl2IGlkPSJJc3N1ZUlEU2VhcmNoRmllbGRFcnJvciIgY2xhc3M9IlRvb2x0aXBFcnJvck1lc3NhZ2UiPjxwPlslIFRyYW5zbGF0ZSgiVGhpcyBmaWVsZCBpcyByZXF1aXJlZC4iKSB8IGh0bWwgJV08L3A+PC9kaXY+CiAgICAgICAgICAgICAgICAgICAgPGRpdiBpZD0iSXNzdWVJRFNlYXJjaEZpZWxkU2VydmVyRXJyb3IiIGNsYXNzPSJUb29sdGlwRXJyb3JNZXNzYWdlIj48cD5bJSBUcmFuc2xhdGUoIlRoaXMgZmllbGQgaXMgcmVxdWlyZWQuIikgfCBodG1sICVdPC9wPjwvZGl2PgogICAgICAgICAgICAgICAgPC9kaXY+CiAgICAgICAgICAgICAgICA8ZGl2IGNsYXNzPSJDbGVhciI+PC9kaXY+CgogICAgICAgICAgICA8L2ZpZWxkc2V0PgogICAgICAgIDwvZGl2PgogICAgICAgIDxkaXYgY2xhc3M9IkZvb3RlciI+CiAgICAgICAgICAgIDxidXR0b24gY2xhc3M9IkNhbGxGb3JBY3Rpb24gUHJpbWFyeSIgaWQ9InN1Ym1pdFJpY2hUZXh0IiBhY2Nlc3NrZXk9ImciIHRpdGxlPSJbJSBUcmFuc2xhdGUoIlN1Ym1pdCIpIHwgaHRtbCAlXSAoZykiIHR5cGU9InN1Ym1pdCIgdmFsdWU9IlslIFRyYW5zbGF0ZSgiU3VibWl0IikgfCBodG1sICVdIj48c3Bhbj48aSBjbGFzcz0iZmEgZmEtY2hlY2stc3F1YXJlLW8iPjwvaT4gWyUgVHJhbnNsYXRlKCJTdWJtaXQiKSB8IGh0bWwgJV08L3NwYW4+PC9idXR0b24+CiAgICAgICAgPC9kaXY+ClslIFJlbmRlckJsb2NrRW5kKCJGb3JtIikgJV0KICAgIDwvZGl2Pgo8L2Zvcm0+Cg==
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9UaWNrZXRNZW51L0pJUkEucG0gLSBtZW51IG1vZHVsZQojICBDb3B5cmlnaHQgKEMpIDIwMDgtMjAxNiBjYXR3b3JreCBHbWJIIGh0dHA6Ly93d3cuY2F0d29ya3guZGUKIyAgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpUaWNrZXRNZW51OjpKSVJBOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSBxdygKICAgIEtlcm5lbDo6Q29uZmlnCiAgICBLZXJuZWw6OlN5c3RlbTo6TG9nCiAgICBLZXJuZWw6OlN5c3RlbTo6VGlja2V0CiAgICBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0Cik7CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKCiAgICBteSAkU2VsZiA9IHsgJVBhcmFtIH07CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJFRpY2tldE9iamVjdCAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VGlja2V0Jyk7CiAgICBteSAkQ29uZmlnT2JqZWN0ICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyk7CiAgICBteSAkTG9nT2JqZWN0ICAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKTsKICAgIG15ICRMYXlvdXRPYmplY3QgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkxheW91dCcpOwogICAgbXkgJExhbmd1YWdlT2JqZWN0ICAgPSAkTGF5b3V0T2JqZWN0LT57TGFuZ3VhZ2VPYmplY3R9OwoKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgaWYgKCAhJFBhcmFte1RpY2tldH0gKSB7CiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIFRpY2tldCEnLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIG15ICRBY3Rpb24gPSAkUGFyYW17Q29uZmlnfS0+e0FjdGlvbn07CgogICAgbXkgJFNob3dEaWFsb2cgPSAkQ29uZmlnT2JqZWN0LT5HZXQoJ0pJUkE6OlNob3dEaWFsb2cnKTsKICAgIGlmICggJFNob3dEaWFsb2cgKSB7CiAgICAgICAgJEFjdGlvbiA9ICdBZ2VudFRpY2tldEpJUkFEaWFsb2cnOwoKICAgICAgICAkUGFyYW17UG9wdXBUeXBlfSAgICAgICAgICAgPSAnVGlja2V0QWN0aW9uJzsKICAgICAgICAkUGFyYW17Q29uZmlnfS0+e1BvcHVwVHlwZX0gPSAnVGlja2V0QWN0aW9uJzsKICAgIH0KCiAgICAjIGNoZWNrIGlmIGZyb250ZW5kIG1vZHVsZSByZWdpc3RlcmVkLCBpZiBub3QsIGRvIG5vdCBzaG93IGFjdGlvbgogICAgaWYgKCAkQWN0aW9uICkgewogICAgICAgIG15ICRNb2R1bGUgPSAkQ29uZmlnT2JqZWN0LT5HZXQoJ0Zyb250ZW5kOjpNb2R1bGUnKS0+eyRBY3Rpb259OwoKICAgICAgICByZXR1cm4gaWYgISRNb2R1bGU7CiAgICB9CgogICAgIyBjaGVjayBwZXJtaXNzaW9uCiAgICBteSAkQWNjZXNzT2sgPSAkVGlja2V0T2JqZWN0LT5QZXJtaXNzaW9uKAogICAgICAgIFR5cGUgICAgID0+ICdydycsCiAgICAgICAgVGlja2V0SUQgPT4gJFBhcmFte1RpY2tldH0tPntUaWNrZXRJRH0sCiAgICAgICAgVXNlcklEICAgPT4gJFNlbGYtPntVc2VySUR9LAogICAgICAgIExvZ05vICAgID0+IDEsCiAgICApOwoKICAgIHJldHVybiBpZiAhJEFjY2Vzc09rOwoKICAgICMgY2hlY2sgcGVybWlzc2lvbgogICAgaWYgKCAkVGlja2V0T2JqZWN0LT5UaWNrZXRMb2NrR2V0KCBUaWNrZXRJRCA9PiAkUGFyYW17VGlja2V0fS0+e1RpY2tldElEfSApICkgewogICAgICAgIG15ICRBY2Nlc3NPayA9ICRUaWNrZXRPYmplY3QtPk93bmVyQ2hlY2soCiAgICAgICAgICAgIFRpY2tldElEID0+ICRQYXJhbXtUaWNrZXR9LT57VGlja2V0SUR9LAogICAgICAgICAgICBPd25lcklEICA9PiAkU2VsZi0+e1VzZXJJRH0sCiAgICAgICAgKTsKCiAgICAgICAgcmV0dXJuIGlmICEkQWNjZXNzT2s7CiAgICB9CgogICAgIyBjaGVjayBhY2wKICAgIG15ICVBQ0xMb29rdXAgPSByZXZlcnNlICV7ICRQYXJhbXtBQ0x9IHx8IHt9IH07CiAgICBpZiAoICEkQUNMTG9va3VweyRBY3Rpb259ICkgewogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAjIGlmIHRoZXJlIGlzIGFscmVhZHkgYSBKSVJBIGlzc3VlIGZvciB0aGF0IHRpY2tldCwgZG8gbm90IHNob3cgdGhlIG1lbnUgaXRlbQogICAgaWYgKCAkUGFyYW17VGlja2V0fS0+eyJEeW5hbWljRmllbGRfSklSQUlzc3VlSUQifSApIHsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgbXkgJEpJUkFDb25maWcgICAgICAgICAgICAgICAgPSAkQ29uZmlnT2JqZWN0LT5HZXQoJ1RpY2tldEpJUkEnKTsKICAgIG15ICRBbGxvd2VkUXVldWVzTGlzdCAgICAgICAgID0gJEpJUkFDb25maWctPntPVFJTVGlja2V0QWxsb3dlZFF1ZXVlczRDcmVhdGV9OwogICAgbXkgJEFsbG93Q3JlYXRpb25JbkFsbFF1ZXVlcyAgPSAkSklSQUNvbmZpZy0+e09UUlNUaWNrZXRBbGxvd0NyZWF0aW9uSW5BbGxRdWV1ZXN9OwoKICAgIGlmICggJEFsbG93Q3JlYXRpb25JbkFsbFF1ZXVlcwogICAgICAgICB8fCAoQHskQWxsb3dlZFF1ZXVlc0xpc3R9ICYmIGdyZXAgeyAkUGFyYW17VGlja2V0fS0+e1F1ZXVlfSBlcSAkXyB9IEB7JEFsbG93ZWRRdWV1ZXNMaXN0fSkgKSB7CiAgICAgICAgcmV0dXJuIHsKICAgICAgICAgICAgJXsgJFBhcmFte0NvbmZpZ30gfSwKICAgICAgICAgICAgJXsgJFBhcmFte1RpY2tldH0gfSwKICAgICAgICAgICAgJVBhcmFtLAogICAgICAgICAgICBOYW1lICAgICAgICAgPT4gJExhbmd1YWdlT2JqZWN0LT5UcmFuc2xhdGUoJ09UUlMgdG8gSmlyYScpLAogICAgICAgICAgICBEZXNjcmlwdGlvbiAgPT4gJExhbmd1YWdlT2JqZWN0LT5UcmFuc2xhdGUoJ0VzY2FsYXRlIHRoaXMgdGlja2V0IHRvIEppcmEnKSwKICAgICAgICAgICAgTGluayAgICAgICAgID0+ICdBY3Rpb249JyAuICRBY3Rpb24gLiAnO1RpY2tldElEPVslIERhdGEuVGlja2V0SUQgJV0nLAogICAgICAgIH07CiAgICB9CgogICAgcmV0dXJuOwp9CgoxOwo=
# --
# Copyright (C) 2017 Perl-Services.de, http://www.perl-services.de
# --
# 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::JIRA::QueueSettings;

use strict;
use warnings;

our @ObjectDependencies = qw(
    Kernel::System::Log
    Kernel::System::DB
    Kernel::System::User
    Kernel::System::Valid
    Kernel::System::Ticket
);

=head1 NAME

Kernel::System::JIRA::QueueSettings

=head1 PUBLIC INTERFACE

=over 4

=cut

=item new()

create an object

=cut

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

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

    return $Self;
}

=item SettingsAdd()

Add time tracking 

    my $ID = $SettingsObject->SettingsAdd(
        QueueID   => 1235,
        ProjectID => 'tracking title',
        Assignee  => 'open',  # or StatusID => 1,
        ValidID   => 1,
        UserID    => 123,
    );

=cut

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

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');
    my $DBObject  = $Kernel::OM->Get('Kernel::System::DB');

    # check needed stuff
    for my $Needed ( qw(QueueID UserID) ) {
        if ( !$Param{$Needed} ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );
            return;
        }
    }

    $Param{ValidID}  ||= 0;
    $Param{Assignee} ||= '';

    # insert new item
    return if !$DBObject->Do(
        SQL => 'INSERT INTO cat_jira_queue_settings '
            . '(queue_id, project_id, assignee, valid_id, create_time, create_by, change_time, change_by )'
            . 'VALUES (?, ?, ?, ?, current_timestamp, ?, current_timestamp, ?)',
        Bind => [
            \$Param{QueueID},
            \$Param{ProjectID},
            \$Param{Assignee},
            \$Param{ValidID},
            \$Param{UserID},
            \$Param{UserID},
        ],
    );

    return $Param{QueueID};
}


=item SettingsUpdate()

to update news 

    my $Success = $SettingsObject->SettingsUpdate(
        ID       => 3,
        TicketID => 1235,
        Title    => 'tracking title', # (optional)
        Status   => 'open',  # or StatusID => 1,
    );

=cut

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

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');
    my $DBObject  = $Kernel::OM->Get('Kernel::System::DB');

    # check needed stuff
    for my $Needed ( qw(QueueID UserID) ) {
        if ( !$Param{$Needed} ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );
            return;
        }
    }

    $Param{ValidID}  ||= 0;
    $Param{Assignee} ||= '';

    # insert new item
    return if !$DBObject->Do(
        SQL => 'UPDATE cat_jira_queue_settings '
            . 'SET queue_id = ?, project_id = ?, assignee = ?, valid_id = ?, '
            . ' change_time = current_timestamp, change_by = ?'
            . ' WHERE queue_id = ?',
        Bind => [
            \$Param{QueueID},
            \$Param{ProjectID},
            \$Param{Assignee},
            \$Param{ValidID},
            \$Param{UserID},
            \$Param{ID},
        ],
    );

    return 1;
}

=item ValidSettingsGet()

=cut

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

    my %Settings = $Self->SettingsGet( %Param );

    return if !%Settings;
    return if !$Settings{Valid} eq 'valid';

    return %Settings;
}

=item SettingsGet()

returns a hash with news data

    my %SettingsData = $SettingsObject->SettingsGet( ID => 2 );

This returns something like:

    %SettingsData = (
        ID            => 3,
        Creator       => 'Agent Name',
        CreateBy      => 354,
        CreateTime    => '',
        ChangeBy      => 35,
        ChangeTime    => '',
    );

=cut

sub SettingsGet {
    my ( $Self, %Param ) = @_;
 
    my $LogObject  = $Kernel::OM->Get('Kernel::System::Log');
    my $DBObject   = $Kernel::OM->Get('Kernel::System::DB');
    my $UserObject = $Kernel::OM->Get('Kernel::System::User');

    # check needed stuff
    if ( !$Param{QueueID} ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => 'Need ID!',
        );
        return;
    }


    return if $Param{QueueID} == -1;

    # sql
    return if !$DBObject->Prepare(
        SQL => 'SELECT queue_id, project_id, assignee, s.valid_id, '
            . 's.create_by, s.create_time, s.change_by, s.change_time, q.name, v.name '
            . 'FROM cat_jira_queue_settings s '
            . '  LEFT OUTER JOIN queue q on queue_id = q.id '
            . '  LEFT OUTER JOIN valid v on s.valid_id = v.id '
            . 'WHERE queue_id = ?',
        Bind  => [ \$Param{QueueID} ],
        Limit => 1,
    );

    my %Settings;
    while ( my @Data = $DBObject->FetchrowArray() ) {
        %Settings = (
            QueueID    => $Data[0],
            ProjectID  => $Data[1],
            Assignee   => $Data[2],
            ValidID    => $Data[3],
            CreateBy   => $Data[4],
            CreateTime => $Data[5],
            ChangeBy   => $Data[6],
            ChangeTime => $Data[7],
            QueueName  => $Data[8],
            Valid      => $Data[9],
        );
    }

    if ( $Settings{CreateBy} ) {
        $Settings{Creator} = $UserObject->UserLookup( UserID => $Settings{CreateBy} );
    }

    return %Settings;
}

=item SettingsDelete()

deletes a queue setting

    my $Success = $SettingsObject->SettingsDelete(
        QueueID => 123,
    );

=cut

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

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');
    my $DBObject  = $Kernel::OM->Get('Kernel::System::DB');

    # check needed stuff
    if ( !$Param{QueueID} ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => 'Need either QueueID!',
        );
        return;
    }

    my $SQL  = 'DELETE FROM cat_jira_queue_settings WHERE queue_id = ?';

    return if !$DBObject->Do(
        SQL  => $SQL,
        Bind => [ \$Param{QueueID} ],
    );

    return 1;
}

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

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');
    my $DBObject  = $Kernel::OM->Get('Kernel::System::DB');

    my $SQL  = 'SELECT queue_id, q.name FROM cat_jira_queue_settings INNER JOIN queue q ON queue_id = q.id'; 

    if ( $Param{Valid} ) {
        $SQL .= ' WHERE valid_id = 1';
    }

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

    my %List;
    while ( my @Row = $DBObject->FetchrowArray() ) {
        $List{ $Row[0] } = $Row[1],
    }

    return %List;
}

1;

=back

=head1 TERMS AND CONDITIONS

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 L<http://www.gnu.org/licenses/agpl.txt>.

=cut


#  --
#  Kernel/System/JIRA/Utils.pm - core module
#  Copyright (C) 2016 catworkx GmbH http://www.catworkx.de
#  --
# 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::JIRA::Utils;

use strict;
use warnings;

use JIRA::REST;
use List::Util qw(first);

use constant P => __PACKAGE__;

our @ObjectDependencies = qw(
    Kernel::Config
    Kernel::System::Log
    Kernel::System::Main
    Kernel::System::JSON
    Kernel::System::Cache
);


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

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

    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');
    my $LogObject    = $Kernel::OM->Get('Kernel::System::Log');

    $Self->{Debug}      = $ConfigObject->Get('JIRA::Debug');
    $Self->{Config}     = $ConfigObject->Get('TicketJIRA') || {};

    my $Host = $Param{JIRAURL} || $Self->{Config}->{URL} . $Self->{Config}->{Path};
    $Host =~ s{/\z}{}xmsg;

    my $User     = $Param{JIRAUSER} || $Self->{Config}->{User};
    my $Password = $Param{JIRAPWD}  || $Self->{Config}->{Password};

    $Self->{JIRAClient} = JIRA::REST->new(
        $Host,
        $User,
        $Password,
    );

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => "debug",
            Message  => "$Host // $User // $Password",
        );
    }

    my $VerifySSL = $ConfigObject->Get('JIRA::SSLVerify');
    if ( !$VerifySSL ) {
        my $Loaded = $Kernel::OM->Get('Kernel::System::Main')->Require(
            'Net::SSLeay',
            Silent => 1,
        );
        if ($Loaded) {
            $Self->{JIRAClient}->{rest}->getUseragent()->ssl_opts( verify_hostname => 0 );
            $Self->{JIRAClient}->{rest}->getUseragent()->ssl_opts( SSL_verfiy_mode => Net::SSLeay::VERIFY_NONE() );
        }
    }

    $Self->{CacheType} = 'JIRAUtils';
    $Self->{CacheTTL}  = $ConfigObject->Get('JIRA::CacheTTL');

    return $Self;
}

=item Check()

Checks if the given credentials are correct. This is done with the API call of 'mypreferences'

Returns  HTTP status:

  200 -> URL OK, Credentials OK
  404 -> URL Not OK
  401 -> URL OK, Credentials Not OK

=cut

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

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

    my $ErrorGet = '';

    eval{
        my $Preferences = $Self->{JIRAClient}->GET( '/mypreferences' );
    } or $ErrorGet = $@;

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'debug',
            Message  => P .  "::Check: "
                . $MainObject->Dump( [ $Self->{JIRAClient}->{rest}->getHost() . '/mypreferences', $ErrorGet ] ),
        );
    }

    return 200 if !$ErrorGet;

    my ($ErrorCode) = $ErrorGet =~ m{([0-9]+)}xms;

    return ( $ErrorCode, $ErrorGet );
}

=item ProjectsGet()

=cut

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

    my $LogObject    = $Kernel::OM->Get('Kernel::System::Log');
    my $MainObject   = $Kernel::OM->Get('Kernel::System::Main');
    my $CacheObject  = $Kernel::OM->Get('Kernel::System::Cache');
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    my $Projects = $CacheObject->Get(
        Type => $Self->{CacheType} . '_Projects',
        Key  => 'All',
    );

    if ( $Projects && !$Param{Force} ) {
        return $Projects;
    }

    my $MetaJSON;
    my $ErrorGet;

    eval{
        $MetaJSON = $Self->{JIRAClient}->GET( '/issue/createmeta' );
    } or $ErrorGet = $@;

    if ( $ErrorGet ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => sprintf P .  "::ProjectsGet: cannot metadata: %s",
                $ErrorGet,
        );

        return +{};
    }

    my @Whitelist = @{ $ConfigObject->Get('JIRA::ProjectWhitelist') || [] };
    my @Blacklist = @{ $ConfigObject->Get('JIRA::ProjectBlacklist') || [] };

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'debug',
            Message  => P .  "::ProjectsGet: "
                . $MainObject->Dump( [ \@Whitelist, \@Blacklist, $MetaJSON->{projects} ] ),
        );
    }

    PROJECTITEM:
    for my $ProjectItem ( @{ $MetaJSON->{projects} || [] } ) {
        my $Key  = $ProjectItem->{key};
        my $Name = $ProjectItem->{name};

        if ( $Self->{Debug} ) {
            $LogObject->Log(
                Priority => 'debug',
                Message  => P .  "::ProjectsGet: Check Project $Key -> $Name"
            );
        }

        if ( @Whitelist ) {
            next PROJECTITEM if !(first{ $Key =~ m{$_}xms }@Whitelist);
        }

        if ( @Blacklist ) {
            next PROJECTITEM if (first{ $Key =~ m{$_}xms }@Blacklist);
        }

        $Projects->{$Key} = $Name;

        # save issuetypes
        my $IssueTypesList;

        my $IssueTypes = $ProjectItem->{issuetypes} || [];

        ISSUETYPE:
        for my $IssueType ( @{ $IssueTypes } ) {

            next ISSUETYPE if $IssueType->{subtask};

            my $IssueTypeName = $IssueType->{name};
            $IssueTypesList->{$IssueTypeName} = $IssueTypeName;
        }

        $CacheObject->Set(
            Type  => $Self->{CacheType} . '_IssueTypes',
            Key   => $Key,
            Value => $IssueTypesList,
            TTL   => $Self->{CacheTTL},
        );
    }

    $Projects //= {};

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'debug',
            Message  => P .  "::ProjectsGet: "
                . $MainObject->Dump( $Projects ),
        );
    }

    $CacheObject->Set(
        Type  => $Self->{CacheType} . '_Projects',
        Key   => 'All',
        Value => $Projects,
        TTL   => $Self->{CacheTTL},
    );

    return $Projects;
}

=item IssueTypesGet()

Get all issue types that belong to a project from JIRA

    my $IssueTypes = $Object->IssueTypesGet(
        Project => $Project,
    );

=cut

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

    my $LogObject   = $Kernel::OM->Get('Kernel::System::Log');
    my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');

    # check for needed data
    for my $Needed ( qw/Project/ ) {
        if ( !$Param{$Needed} ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );

            return;
        }
    }

    my $IssueTypes = $CacheObject->Get(
        Type  => $Self->{CacheType} . '_IssueTypes',
        Key   => $Param{Project},
    );

    if ( $IssueTypes ) {
        return $IssueTypes;
    }

    $Self->ProjectsGet( Force => 1 );

    $IssueTypes = $CacheObject->Get(
        Type  => $Self->{CacheType} . '_IssueTypes',
        Key   => $Param{Project},
    );

    return $IssueTypes || {};
}

=item UsersGet()

Get all issue types that belong to a project from JIRA

    my $Users = $Object->UsersGet(
        Project => $Project,
    );

=cut

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

    my $LogObject   = $Kernel::OM->Get('Kernel::System::Log');
    my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');

    # check for needed data
    for my $Needed ( qw/Project/ ) {
        if ( !$Param{$Needed} ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );

            return;
        }
    }

    # implement caching
    my $Users = $CacheObject->Get(
        Type  => $Self->{CacheType} . '_Users',
        Key   => $Param{Project},
    );


    my $UsersJSON;
    my $ErrorGet;

    # get users
    eval{
        $UsersJSON = $Self->{JIRAClient}->GET(
            sprintf '/user/assignable/search?maxResults=%d&project=%s',
                1000,
                $Param{Project}
        );
    } or $ErrorGet = $@;

    if ( $ErrorGet ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => sprintf P .  "::UsersGet: cannot assignable users for project %s: %s",
                $Param{Project}, $ErrorGet,
        );

        return +{};
    }

    for my $User ( @{ $UsersJSON || [] } ) {
        my $Key   = $User->{name};
        my $Value = $User->{displayName};   

        $Users->{$Key} = $Value;
    }

    $CacheObject->Set(
        Type  => $Self->{CacheType} . '_Users',
        Key   => $Param{Project},
        Value => $Users,
        TTL   => $Self->{CacheTTL},
    );

    return $Users;
}

=item StatesGet()

Get all issue types that belong to a project from JIRA

    my $States = $Object->StatesGet(
        Project => $Project,
    );

=cut

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

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    # check for needed data
    for my $Needed ( qw/Project/ ) {
        if ( !$Param{$Needed} ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );

            return;
        }
    }

    # TODO: implement caching

    my $States = {};
    my $ErrorGet;

    # TODO: URL to get states
    #eval{
    #    $States = $Self->{JIRAClient}->GET( '/issue/' . $Param{Project} );
    #} or $ErrorGet = $@;
    #
    #if ( $ErrorGet ) {
    #    $LogObject->Log(
    #        Priority => 'error',
    #        Message  => sprintf P .  "::StatesGet: cannot issue types for project %s: %s",
    #            $Param{Project}, $ErrorGet,
    #    );
    #
    #    return +{ error => $ErrorGet };
    #}

    return $States;
}

=item IssueList()

=cut

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

    my $LogObject    = $Kernel::OM->Get('Kernel::System::Log');
    my $MainObject   = $Kernel::OM->Get('Kernel::System::Main');
    my $CacheObject  = $Kernel::OM->Get('Kernel::System::Cache');
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    # check for needed data
    for my $Needed ( qw/Project/ ) {
        if ( !$Param{$Needed} ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );

            return;
        }
    }

    $Param{Term} //= '';

    my $TypeTerm = $Param{Term};
    $TypeTerm    =~ s{[^A-Za-z0-9_]}{_}g;

    my $CacheType =  $Self->{CacheType} . '_Issues_' . $Param{Project} . '_' . $TypeTerm;

    my $Issues = $CacheObject->Get(
        Type => $CacheType,
        Key  => 'All',
    );

    if ( $Issues && !$Param{Force} ) {
        return $Issues;
    }

    my $IssuesJSON;
    my $ErrorGet;

    my @States = @{ $ConfigObject->Get('JIRA::IssueSearchStates') || [] };

    my $StatesString = '';
    if ( @States ) {
        $StatesString = ' AND status IN (' . join(',', map{ qq~"$_"~ }@States ) . ')';
    }

    my $Params = sprintf "project=%s%s order by issueKey desc", $Param{Project}, $StatesString;

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'debug',
            Message  => P .  "::IssueList: Params -> " . $Params,
        );
    }

    eval{
        my $MaxResults = $ConfigObject->Get('JIRA::IssueSearchMaxResults') || 300;
        $IssuesJSON    = $Self->{JIRAClient}->POST(
            '/search',
            undef,
            {
                jql        => $Params,
                maxResults => $MaxResults,
                fields     => [qw/id key description/],
            }
        );
    } or $ErrorGet = $@;

    if ( $ErrorGet ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => sprintf P .  "::IssueList: cannot get issue list: %s",
                $ErrorGet,
        );

        return +{};
    }

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'debug',
            Message  => P .  "::IssueList: JSON -> "
                . $MainObject->Dump( $IssuesJSON ),
        );
    }

    my @IssueList;

    my $RE;
    if ( $Param{Term} ) {
        $Param{Term} =~ s{\*}{.*}xmsg;
        $RE          = qr/$Param{Term}/i;
    }

    ISSUEITEM:
    for my $IssueItem ( @{ $IssuesJSON->{issues} || [] } ) {
        my $Key  = $IssueItem->{key};
        my $Desc = $IssueItem->{fields}->{description};
        my $ID   = $IssueItem->{id};

        if ( $Param{Term} ) {
            next ISSUEITEM if $Key !~ $RE and $Desc !~ $RE;
        }

        push @IssueList, {
            Value => $ID,
            Label => sprintf( "%s - %s", $Key, $Desc ),
        }
    }

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'debug',
            Message  => P .  "::IssueList: " .
                $MainObject->Dump( \@IssueList ),
        );
    }

    $CacheObject->Set(
        Type  => $CacheType,
        Key   => 'All',
        Value => \@IssueList,
        TTL   => $Self->{CacheTTL},
    );

    return \@IssueList;
}

1;


IyAtLQojIENvcHlyaWdodCAoQykgMjAxNiAtIDIwMTggUGVybC1TZXJ2aWNlcy5kZSwgaHR0cDovL3Blcmwtc2VydmljZXMuZGUKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OlRpY2tldDo6Q3VzdG9tOjpBdHRhY2htZW50RXZlbnRzOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKb3VyICRPYmplY3RNYW5hZ2VyRGlzYWJsZWQgPSAxOwoKdXNlIEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyOwoKbXkgJExvYWRlZDsKbXkgJUFscmVhZHlSdW47CgpJbml0KCk7CgojIGRpc2FibGUgcmVkZWZpbmUgd2FybmluZ3MgaW4gdGhpcyBzY29wZQpzdWIgSW5pdCB7CiAgICBubyB3YXJuaW5ncyAncmVkZWZpbmUnOwoKICAgIHJldHVybiBpZiAkTG9hZGVkKys7CgogICAgewogICAgICAgIGxvY2FsICRLZXJuZWw6Ok9NID0gS2VybmVsOjpTeXN0ZW06Ok9iamVjdE1hbmFnZXItPm5ldygpOwogICAgICAgIG15ICRNSU1FT2JqZWN0ICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlRpY2tldDo6QXJ0aWNsZTo6QmFja2VuZDo6TUlNRUJhc2U6OkFydGljbGVTdG9yYWdlREInKTsKICAgICAgICAgICAkTUlNRU9iamVjdCAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpUaWNrZXQ6OkFydGljbGU6OkJhY2tlbmQ6Ok1JTUVCYXNlOjpBcnRpY2xlU3RvcmFnZUZTJyk7CiAgICB9CgogICAgbXkgJE9yaWdSdW5EQiA9ICpLZXJuZWw6OlN5c3RlbTo6VGlja2V0OjpBcnRpY2xlOjpCYWNrZW5kOjpNSU1FQmFzZTo6QXJ0aWNsZVN0b3JhZ2VEQjo6QXJ0aWNsZVdyaXRlQXR0YWNobWVudHtDT0RFfTsKICAgICpLZXJuZWw6OlN5c3RlbTo6VGlja2V0OjpBcnRpY2xlOjpCYWNrZW5kOjpNSU1FQmFzZTo6QXJ0aWNsZVN0b3JhZ2VEQjo6QXJ0aWNsZVdyaXRlQXR0YWNobWVudCA9IHN1YiB7CiAgICAgICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAgICAgbXkgJFN1Y2Nlc3MgPSAkU2VsZi0+JE9yaWdSdW5EQiggJVBhcmFtICk7CiAgICAgICAgX1RyaWdnZXJFdmVudCggJFNlbGYsICVQYXJhbSwgU3VjY2VzcyA9PiAkU3VjY2VzcyApOwoKICAgICAgICByZXR1cm4gJFN1Y2Nlc3M7CiAgICB9OwoKICAgIG15ICRPcmlnUnVuRlMgPSAqS2VybmVsOjpTeXN0ZW06OlRpY2tldDo6QXJ0aWNsZTo6QmFja2VuZDo6TUlNRUJhc2U6OkFydGljbGVTdG9yYWdlRlM6OkFydGljbGVXcml0ZUF0dGFjaG1lbnR7Q09ERX07CiAgICAqS2VybmVsOjpTeXN0ZW06OlRpY2tldDo6QXJ0aWNsZTo6QmFja2VuZDo6TUlNRUJhc2U6OkFydGljbGVTdG9yYWdlRlM6OkFydGljbGVXcml0ZUF0dGFjaG1lbnQgPSBzdWIgewogICAgICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgICAgIG15ICRTdWNjZXNzID0gJFNlbGYtPiRPcmlnUnVuRlMoICVQYXJhbSApOwogICAgICAgIF9UcmlnZ2VyRXZlbnQoICRTZWxmLCAlUGFyYW0sIFN1Y2Nlc3MgPT4gJFN1Y2Nlc3MgKTsKCiAgICAgICAgcmV0dXJuICRTdWNjZXNzOwogICAgfTsKfQoKc3ViIF9UcmlnZ2VyRXZlbnQgewogICAgbXkgKCRTZWxmLCAlUGFyYW0pID0gQF87CgogICAgbXkgJFRpY2tldE9iamVjdCAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VGlja2V0Jyk7CiAgICBteSAkQXJ0aWNsZU9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpUaWNrZXQ6OkFydGljbGUnKTsKCiAgICAjIHRyaWdnZXIgZXZlbnQKICAgIGlmICggJFBhcmFte1N1Y2Nlc3N9ICkgewoKICAgICAgICBteSAkVGlja2V0SUQgPSAkQXJ0aWNsZU9iamVjdC0+VGlja2V0SURMb29rdXAoIEFydGljbGVJRCA9PiAkUGFyYW17QXJ0aWNsZUlEfSApOwogICAgICAgIG15ICVUaWNrZXQgICA9ICRUaWNrZXRPYmplY3QtPlRpY2tldEdldCgKICAgICAgICAgICAgVGlja2V0SUQgPT4gJFRpY2tldElELAogICAgICAgICk7CgogICAgICAgIG15ICRLZXkgPSBzcHJpbnRmICIlc18lcyIsICRUaWNrZXRJRCwgJFBhcmFte0FydGljbGVJRH07CiAgICAgICAgcmV0dXJuIGlmICRBbHJlYWR5UnVueyRLZXl9Kys7CgogICAgICAgIG15ICRCYWNrZW5kT2JqZWN0ID0gJEFydGljbGVPYmplY3QtPkJhY2tlbmRGb3JBcnRpY2xlKAogICAgICAgICAgICBUaWNrZXRJRCAgPT4gJFRpY2tldElELAogICAgICAgICAgICBBcnRpY2xlSUQgPT4gJFBhcmFte0FydGljbGVJRH0sCiAgICAgICAgKTsKCiAgICAgICAgbXkgJUF0dGFjaG1lbnRJbmRleCA9ICRCYWNrZW5kT2JqZWN0LT5BcnRpY2xlQXR0YWNobWVudEluZGV4KAogICAgICAgICAgICBBcnRpY2xlSUQgPT4gJFBhcmFte0FydGljbGVJRH0sCiAgICAgICAgKTsKCiAgICAgICAgbXkgKCRGaWxlSUQpID0gZ3JlcHsKICAgICAgICAgICAgJEF0dGFjaG1lbnRJbmRleHskX30tPntGaWxlbmFtZX0gZXEgJFBhcmFte0ZpbGVuYW1lfQogICAgICAgIH0ga2V5cyAlQXR0YWNobWVudEluZGV4OwoKICAgICAgICAkVGlja2V0T2JqZWN0LT5FdmVudEhhbmRsZXIoCiAgICAgICAgICAgIEV2ZW50ID0+ICdBdHRhY2htZW50V3JpdGUnLAogICAgICAgICAgICBEYXRhICA9PiB7CiAgICAgICAgICAgICAgICBBcnRpY2xlSUQgPT4gJFBhcmFte0FydGljbGVJRH0sCiAgICAgICAgICAgICAgICBGaWxlbmFtZSAgPT4gJFBhcmFte0ZpbGVuYW1lfSwKICAgICAgICAgICAgICAgIEZpbGVJRCAgICA9PiAkRmlsZUlELAogICAgICAgICAgICAgICAgVGlja2V0SUQgID0+ICRUaWNrZXRJRCwKICAgICAgICAgICAgfSwKICAgICAgICAgICAgVXNlcklEID0+ICRQYXJhbXtVc2VySUR9LAogICAgICAgICk7CiAgICB9Cn0KCjE7Cgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIEw8aHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0Pi4KCj1jdXQK
IyAtLQojIEtlcm5lbC9TeXN0ZW0vVGlja2V0L0V2ZW50L0FkZEFydGljbGVUb0pJUkFJc3N1ZS5wbSAtIGV2ZW50IG1vZHVsZQojIENvcHlyaWdodCAoQykgMjAwOC0yMDE4IGNhdHdvcmt4IEdtYkggaHR0cDovL3d3dy5jYXR3b3JreC5kZQojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6VGlja2V0OjpFdmVudDo6QWRkQXJ0aWNsZVRvSklSQUlzc3VlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIExpc3Q6OlV0aWwgcXcoZmlyc3QpOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSBxdygKICAgIEtlcm5lbDo6Q29uZmlnCiAgICBLZXJuZWw6OlN5c3RlbTo6TG9nCiAgICBLZXJuZWw6OlN5c3RlbTo6TWFpbgogICAgS2VybmVsOjpTeXN0ZW06OlRpY2tldAogICAgS2VybmVsOjpTeXN0ZW06OlRpY2tldDo6QXJ0aWNsZQogICAgS2VybmVsOjpTeXN0ZW06OlVzZXIKICAgIEtlcm5lbDo6U3lzdGVtOjpUaWNrZXRKSVJBCik7CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHt9OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgIG15ICRDb25maWdPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OkNvbmZpZycpOwogICAgJFNlbGYtPntEZWJ1Z30gICA9ICRDb25maWdPYmplY3QtPkdldCgnSklSQTo6RGVidWcnKTsKCiAgICByZXR1cm4gJFNlbGY7Cn0KCnN1YiBSdW4gewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICBteSAkVGlja2V0T2JqZWN0ICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpUaWNrZXQnKTsKICAgIG15ICRBcnRpY2xlT2JqZWN0ICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlRpY2tldDo6QXJ0aWNsZScpOwogICAgbXkgJENvbmZpZ09iamVjdCAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OkNvbmZpZycpOwogICAgbXkgJE1haW5PYmplY3QgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TWFpbicpOwogICAgbXkgJFVzZXJPYmplY3QgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpOwogICAgbXkgJExvZ09iamVjdCAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyk7CiAgICBteSAkVGlja2V0SklSQU9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpUaWNrZXRKSVJBJyk7CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGZvciBteSAkTmVlZGVkIChxdyhEYXRhIEV2ZW50IENvbmZpZyBVc2VySUQpKSB7CiAgICAgICAgaWYgKCAhJFBhcmFteyROZWVkZWR9ICkgewogICAgICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICAgICAgTWVzc2FnZSAgPT4gIk5lZWQgJE5lZWRlZCFcbiIsCiAgICAgICAgICAgICk7CgogICAgICAgICAgICByZXR1cm47CiAgICAgICAgfQogICAgfQoKICAgIGZvciBteSAkTmVlZGVkRGF0YSAocXcoVGlja2V0SUQpKSB7CiAgICAgICAgaWYgKCAhJFBhcmFte0RhdGF9LT57JE5lZWRlZERhdGF9ICkgewogICAgICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICAgICAgTWVzc2FnZSAgPT4gIk5lZWQgJE5lZWRlZERhdGEgaW4gRGF0YSFcbiIsCiAgICAgICAgICAgICk7CgogICAgICAgICAgICByZXR1cm47CiAgICAgICAgfQogICAgfQoKICAgIG15ICRVc2VySUQgPSAkUGFyYW17VXNlcklEfTsKCiAgICAjIEdldCBjdXJyZW50IHVzZXIgZGF0YQogICAgbXkgJUN1cnJlbnRVc2VyID0gJFVzZXJPYmplY3QtPkdldFVzZXJEYXRhKAogICAgICAgIFVzZXJJRCA9PiAkVXNlcklELAogICAgKTsKCiAgICBpZiAoICRTZWxmLT57RGVidWd9ICkgewogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2RlYnVnJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gX19QQUNLQUdFX18gLiAiOjpSdW4oKTogVXNlcklEOiAkVXNlcklEIFVzZXJMb2dpbjogJEN1cnJlbnRVc2Vye1VzZXJMb2dpbn1cbiIKICAgICAgICApOwogICAgfQoKICAgIGlmICggJEN1cnJlbnRVc2Vye1VzZXJMb2dpbn0gZXEgJENvbmZpZ09iamVjdC0+R2V0KCdUaWNrZXRKSVJBJyktPntJbmNvbWluZ1VzZXJ9ICkgewogICAgCXJldHVybiAxOwogICAgfQoKICAgIG15ICRUaWNrZXRJRCA9ICRQYXJhbXtEYXRhfS0+e1RpY2tldElEfTsKCiAgICBpZiAoICRTZWxmLT57RGVidWd9ICkgewogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2RlYnVnJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gX19QQUNLQUdFX18gLiAiOjpSdW4oKTogVGlja2V0SUQ6ICRUaWNrZXRJRCBFdmVudDogJFBhcmFte0V2ZW50fVxuIgogICAgICAgICk7CiAgICB9CgojICAgIHJldHVybiAxIGlmICRQYXJhbXtFdmVudH0gbmUgJ0FydGljbGVDcmVhdGUnOwoKICAgIG15ICVUaWNrZXQgPSAkVGlja2V0T2JqZWN0LT5UaWNrZXRHZXQoCiAgICAgICAgVGlja2V0SUQgICAgICA9PiAkVGlja2V0SUQsCiAgICAgICAgVXNlcklEICAgICAgICA9PiAxLAogICAgICAgIER5bmFtaWNGaWVsZHMgPT4gMSwKICAgICk7CgogICAgcmV0dXJuIDEgaWYgISVUaWNrZXQ7CgogICAgbXkgJElzc3VlSUQgPSAkVGlja2V0e0R5bmFtaWNGaWVsZF9KSVJBSXNzdWVJRH07CiAgICByZXR1cm4gMSBpZiAhJElzc3VlSUQ7CgogICAgaWYgKCAkU2VsZi0+e0RlYnVnfSApIHsKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdkZWJ1ZycsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+IF9fUEFDS0FHRV9fIC4gIjo6UnVuKCk6IFRpY2tldElEOiAkVGlja2V0e1RpY2tldElEfSBUaXRsZTogJFRpY2tldHtUaXRsZX0gIgogICAgICAgICAgICAgICAgLiAiU3RhdGU6ICRUaWNrZXR7U3RhdGV9XG4iLAogICAgICAgICk7CiAgICB9CgogICAgbXkgQEFydGljbGVJbmRleCA9ICRBcnRpY2xlT2JqZWN0LT5BcnRpY2xlTGlzdCgKICAgICAgICBUaWNrZXRJRCA9PiAkVGlja2V0SUQsCiAgICAgICAgVXNlcklEICAgPT4gMSwKICAgICk7CgogICAgcmV0dXJuIDEgaWYgIUBBcnRpY2xlSW5kZXg7CgogICAgbXkgJEJhY2tlbmRPYmplY3QgPSAkQXJ0aWNsZU9iamVjdC0+QmFja2VuZEZvckFydGljbGUoICV7ICRBcnRpY2xlSW5kZXhbLTFdIH0gKTsKICAgIG15ICVBcnRpY2xlICAgICAgID0gJEJhY2tlbmRPYmplY3QtPkFydGljbGVHZXQoCiAgICAgICAgQXJ0aWNsZUlEICAgICA9PiAkQXJ0aWNsZUluZGV4Wy0xXS0+e0FydGljbGVJRH0sCiAgICAgICAgVGlja2V0SUQgICAgICA9PiAkVGlja2V0SUQsCiAgICAgICAgRHluYW1pY0ZpZWxkcyA9PiAxLAogICAgKTsKCiAgICBpZiAoICRTZWxmLT57RGVidWd9ICkgewogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2RlYnVnJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gX19QQUNLQUdFX18gLiAiOjpSdW4oKTogQXJ0aWNsZUlEOiAkQXJ0aWNsZXtBcnRpY2xlSUR9IFRpdGxlOiAkVGlja2V0e1RpdGxlfVxuIiwKICAgICAgICApOwogICAgfQoKICAgIHJldHVybiAxIGlmICElQXJ0aWNsZTsKCiAgICBteSAkQWxsb3dlZFN0YXRlc05hbWVzID0gJENvbmZpZ09iamVjdC0+R2V0KCdUaWNrZXRKSVJBJyktPntPVFJTVGlja2V0QWxsb3dlZFN0YXRlczRVcGRhdGV9OwogICAgbXkgQEFsbG93ZWRTdGF0ZXNMaXN0ICA9IHNwbGl0IC8sLywgJEFsbG93ZWRTdGF0ZXNOYW1lczsKCiAgICBpZiAoICEoIGZpcnN0IHsgJF8gZXEgJFRpY2tldHtTdGF0ZX0gfSBAQWxsb3dlZFN0YXRlc0xpc3QgKSApIHsKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdpbmZvJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gX19QQUNLQUdFX18gLiAiOjpSdW4oKTogT1RSUyBUaWNrZXQgbm90IGluICRBbGxvd2VkU3RhdGVzTmFtZXM6ICRUaWNrZXR7U3RhdGV9XG4iLAogICAgICAgICk7CgogICAgICAgIHJldHVybiAxOwogICAgfQoKICAgIG15ICRKSVJBSXNzdWUgPSAkVGlja2V0SklSQU9iamVjdC0+SXNzdWVDb21tZW50KAogICAgICAgIElzc3VlSUQgICA9PiAkSXNzdWVJRCwKICAgICAgICBBcnRpY2xlSUQgPT4gJEFydGljbGV7QXJ0aWNsZUlEfSwKICAgICAgICBVc2VySUQgICAgPT4gJFBhcmFte1VzZXJJRH0gLy8gMSwKICAgICAgICBDb21tZW50ICAgPT4gJEFydGljbGV7Qm9keX0sCiAgICAgICAgQXJ0aWNsZSAgID0+IFwlQXJ0aWNsZSwKICAgICk7CgogICAgIyBUT0RPOiBJZiBlcnJvciBvY2N1cmVkIHNob3cgbm90aWZpY2F0aW9uIGJhcgoKICAgIGlmICggJFNlbGYtPntEZWJ1Z30gKSB7CiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZGVidWcnLAogICAgICAgICAgICBNZXNzYWdlICA9PiBzcHJpbnRmIF9fUEFDS0FHRV9fIC4gIjogVXBkYXRlIHJlc3BvbnNlICVzXG4iLAogICAgICAgICAgICAgICAgJE1haW5PYmplY3QtPkR1bXAoICRKSVJBSXNzdWUgKSwKICAgICAgICApOwogICAgfQoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIEtlcm5lbC9TeXN0ZW0vVGlja2V0L0V2ZW50L0Nsb3NlSklSQUlzc3VlLnBtIC0gZXZlbnQgbW9kdWxlCiMgQ29weXJpZ2h0IChDKSAyMDA4LTIwMTggY2F0d29ya3ggR21iSCBodHRwOi8vd3d3LmNhdHdvcmt4LmRlCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6U3lzdGVtOjpUaWNrZXQ6OkV2ZW50OjpDbG9zZUpJUkFJc3N1ZTsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSBMaXN0OjpVdGlsIHF3KGZpcnN0KTsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gcXcoCiAgICBLZXJuZWw6OlN5c3RlbTo6TG9nCiAgICBLZXJuZWw6OlN5c3RlbTo6Q29uZmlnCiAgICBLZXJuZWw6OlN5c3RlbTo6SlNPTgogICAgS2VybmVsOjpTeXN0ZW06OlRpY2tldAogICAgS2VybmVsOjpTeXN0ZW06OlRpY2tldDo6QXJ0aWNsZQogICAgS2VybmVsOjpTeXN0ZW06OlRpY2tldEpJUkEKKTsKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgbXkgJENvbmZpZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyk7CiAgICAkU2VsZi0+e0RlYnVnfSAgID0gJENvbmZpZ09iamVjdC0+R2V0KCdKSVJBOjpEZWJ1ZycpOwoKICAgIHJldHVybiAkU2VsZjsKfQoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRUaWNrZXRPYmplY3QgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlRpY2tldCcpOwogICAgbXkgJEFydGljbGVPYmplY3QgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VGlja2V0OjpBcnRpY2xlJyk7CiAgICBteSAkQ29uZmlnT2JqZWN0ICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyk7CiAgICBteSAkTG9nT2JqZWN0ICAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKTsKICAgIG15ICRUaWNrZXRKSVJBT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlRpY2tldEpJUkEnKTsKICAgIG15ICRVc2VyT2JqZWN0ICAgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVzZXInKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgZm9yIG15ICROZWVkZWQgKHF3KERhdGEgRXZlbnQgQ29uZmlnIFVzZXJJRCkpIHsKICAgICAgICBpZiAoICEkUGFyYW17JE5lZWRlZH0gKSB7CiAgICAgICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkTmVlZGVkIVxuIiwKICAgICAgICAgICAgKTsKCiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICB9CgogICAgZm9yIG15ICROZWVkZWREYXRhIChxdyhUaWNrZXRJRCkpIHsKICAgICAgICBpZiAoICEkUGFyYW17RGF0YX0tPnskTmVlZGVkRGF0YX0gKSB7CiAgICAgICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkTmVlZGVkRGF0YSBpbiBEYXRhIVxuIiwKICAgICAgICAgICAgKTsKCiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICB9CgogICAgbXkgJFRpY2tldElEID0gJFBhcmFte0RhdGF9LT57VGlja2V0SUR9OwoKICAgIGlmICggJFNlbGYtPntEZWJ1Z30gKSB7CiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZGVidWcnLAogICAgICAgICAgICBNZXNzYWdlICA9PiBfX1BBQ0tBR0VfXyAuICI6OlJ1bigpOiBUaWNrZXRJRDogJFRpY2tldElEIEV2ZW50OiAkUGFyYW17RXZlbnR9XG4iLAogICAgICAgICk7CiAgICB9CgojICAgIHJldHVybiAxIGlmICRQYXJhbXtFdmVudH0gbmUgJ1RpY2tldFN0YXRlVXBkYXRlJzsKCiAgICBteSAlVGlja2V0ID0gJFRpY2tldE9iamVjdC0+VGlja2V0R2V0KAogICAgICAgIFRpY2tldElEICAgICAgPT4gJFRpY2tldElELAogICAgICAgIFVzZXJJRCAgICAgICAgPT4gMSwKICAgICAgICBEeW5hbWljRmllbGRzID0+IDEsCiAgICApOwoKICAgIHJldHVybiAxIGlmICElVGlja2V0OwoKICAgIG15ICRVc2VySUQgPSAkUGFyYW17VXNlcklEfTsKCiAgICAjIEdldCBjdXJyZW50IHVzZXIgZGF0YQogICAgbXkgJUN1cnJlbnRVc2VyID0gJFVzZXJPYmplY3QtPkdldFVzZXJEYXRhKAogICAgICAgIFVzZXJJRCA9PiAkVXNlcklELAogICAgKTsKCiAgICBpZiAoICRTZWxmLT57RGVidWd9ICkgewogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2RlYnVnJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gX19QQUNLQUdFX18gLiAiOjpSdW4oKTogVXNlcklEOiAkVXNlcklEIFVzZXJMb2dpbjogJEN1cnJlbnRVc2Vye1VzZXJMb2dpbn1cbiIsCiAgICAgICAgKTsKICAgIH0KCiAgICBpZiAoICRDdXJyZW50VXNlcntVc2VyTG9naW59IGVxICRDb25maWdPYmplY3QtPkdldCgnVGlja2V0SklSQScpLT57SW5jb21pbmdVc2VyfSApIHsKICAgICAgICByZXR1cm4gMTsKICAgIH0KCgogICAgbXkgJEpJUkFJc3N1ZUlEID0gJFRpY2tldHtEeW5hbWljRmllbGRfSklSQUlzc3VlSUR9OwogICAgcmV0dXJuIDEgaWYgISRKSVJBSXNzdWVJRDsKCiAgICBpZiAoICRTZWxmLT57RGVidWd9ICkgewogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2RlYnVnJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gX19QQUNLQUdFX18gLiAiOjpSdW4oKTogVGlja2V0SUQ6ICRUaWNrZXR7VGlja2V0SUR9IFRpdGxlOiAkVGlja2V0e1RpdGxlfSAiCiAgICAgICAgICAgICAgICAuICJTdGF0ZTogJFRpY2tldHtTdGF0ZX1cbiIsCiAgICAgICAgKTsKICAgIH0KCiAgICBteSAkQ29uZmlnID0gJENvbmZpZ09iamVjdC0+R2V0KCdUaWNrZXRKSVJBJykgfHwgIHt9OwoKICAgIG15ICRBbGxvd2VkU3RhdGVzTmFtZXMgPSAkQ29uZmlnLT57T1RSU1RpY2tldEFsbG93ZWRTdGF0ZXM0Q2xvc2V9OwogICAgbXkgQEFsbG93ZWRTdGF0ZXNMaXN0ICA9IHNwbGl0IC8sLywgJEFsbG93ZWRTdGF0ZXNOYW1lczsKCiAgICBpZiAoICEoIGZpcnN0eyAkXyBlcSAkVGlja2V0e1N0YXRlfSB9IEBBbGxvd2VkU3RhdGVzTGlzdCApICkgewogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2luZm8nLAogICAgICAgICAgICBNZXNzYWdlICA9PiBfX1BBQ0tBR0VfXyAuICI6OlJ1bigpOiBPVFJTIFRpY2tldCBub3QgaW4gJEFsbG93ZWRTdGF0ZXNOYW1lczogJFRpY2tldHtTdGF0ZX1cbiIsCiAgICAgICAgKTsKCiAgICAgICAgcmV0dXJuIDE7CiAgICB9CgogICAgbXkgJFVSTCA9ICRDb25maWctPntVUkx9OwoKICAgIG15ICRWYWxpZExpY2Vuc2UgPSAkVGlja2V0SklSQU9iamVjdC0+Q2hlY2tWYWxpZExpY2Vuc2UoKTsKICAgIGlmICggJFZhbGlkTGljZW5zZSAhPSAxICkgewogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gX19QQUNLQUdFX18gLiAiOjpSdW4oKTogaW52YWxpZCBsaWNlbnNlIGF0ICRVUkwsIHN0YXR1cyAkVmFsaWRMaWNlbnNlIiwKICAgICAgICApOwoKICAgICAgICByZXR1cm4gMTsgIyBza2lwIHRoZSByZXN0CiAgICB9CgogICAgbXkgJElzc3VlID0gJFRpY2tldEpJUkFPYmplY3QtPklzc3VlR2V0KCBJc3N1ZSA9PiAkSklSQUlzc3VlSUQgKTsKCiAgICBpZiAoICRJc3N1ZS0+e2Vycm9yfSApIHsKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+IF9fUEFDS0FHRV9fIC4gIjo6UnVuKCk6IGNhbm5vdCBmZXRjaCB0aGUgSmlyYSBJc3N1ZSAiCiAgICAgICAgICAgICAgICAuICRKSVJBSXNzdWVJRCAuICIgZm9yICIgLiAkVGlja2V0e1RpY2tldElEfSAuICI6ICRAIiwKICAgICAgICApOwoKICAgICAgICByZXR1cm4gMTsKICAgIH0KCiAgICBpZiAoICRTZWxmLT57RGVidWd9ICkgewogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2RlYnVnJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gX19QQUNLQUdFX18gLiAiOjpSdW4oKTogaXNzdWUgZXhpc3RzXG4iLAogICAgICAgICk7CiAgICB9CgogICAgbXkgQEFydGljbGVJbmRleCA9ICRBcnRpY2xlT2JqZWN0LT5BcnRpY2xlTGlzdCgKICAgICAgICBUaWNrZXRJRCA9PiAkVGlja2V0e1RpY2tldElEfSwKICAgICAgICBVc2VySUQgICA9PiAxLAogICAgICAgIE9ubHlMYXN0ID0+IDEsCiAgICApOwoKICAgIG15ICRCYWNrZW5kT2JqZWN0ID0gJEFydGljbGVPYmplY3QtPkJhY2tlbmRGb3JBcnRpY2xlKCAleyAkQXJ0aWNsZUluZGV4WzBdIH0gKTsKICAgIG15ICVBcnRpY2xlICAgICAgID0gJEJhY2tlbmRPYmplY3QtPkFydGljbGVHZXQoCiAgICAgICAgVGlja2V0SUQgID0+ICRUaWNrZXR7VGlja2V0SUR9LAogICAgICAgIEFydGljbGVJRCA9PiAkQXJ0aWNsZUluZGV4WzBdLT57QXJ0aWNsZUlEfSwKICAgICk7CgogICAgbXkgJFJlc3VsdCA9ICRUaWNrZXRKSVJBT2JqZWN0LT5Jc3N1ZUNsb3NlKAogICAgICAgIElzc3VlSUQgPT4gJEpJUkFJc3N1ZUlELAogICAgICAgIENvbW1lbnQgPT4gIk9UUlM6ICRUaWNrZXR7U3RhdGV9XG5cbi0tLS0tLS0tXG4iIC4gJEFydGljbGV7Qm9keX0sCiAgICApOwoKICAgIGlmICggJFJlc3VsdC0+e2Vycm9yfSApIHsKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+IF9fUEFDS0FHRV9fIC4gIjogQ2Fubm90IGNsb3NlIGlzc3VlICRKSVJBSXNzdWVJRDogJFJlc3VsdC0+e2Vycm9yfVxuIiwKICAgICAgICApOwogICAgfQoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIEtlcm5lbC9TeXN0ZW0vVGlja2V0L0V2ZW50L0NyZWF0ZUpJUkFJc3N1ZS5wbSAtIGV2ZW50IG1vZHVsZQojIENvcHlyaWdodCAoQykgMjAwOC0yMDE1IGNhdHdvcmt4IEdtYkggaHR0cDovL3d3dy5jYXR3b3JreC5kZQojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6VGlja2V0OjpFdmVudDo6Q3JlYXRlSklSQUlzc3VlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIExpc3Q6OlV0aWwgcXcoZmlyc3QpOwoKb3VyICRWRVJTSU9OID0gJzQuMDAxJzsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gcXcoCiAgICBLZXJuZWw6OkNvbmZpZwogICAgS2VybmVsOjpTeXN0ZW06OkxvZwogICAgS2VybmVsOjpTeXN0ZW06OlRpY2tldAogICAgS2VybmVsOjpTeXN0ZW06OlRpY2tldEpJUkEKKTsKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgbXkgJENvbmZpZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyk7CiAgICAkU2VsZi0+e0RlYnVnfSAgID0gJENvbmZpZ09iamVjdC0+R2V0KCdKSVJBOjpEZWJ1ZycpOwoKICAgIHJldHVybiAkU2VsZjsKfQoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRUaWNrZXRPYmplY3QgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlRpY2tldCcpOwogICAgbXkgJENvbmZpZ09iamVjdCAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OkNvbmZpZycpOwogICAgbXkgJExvZ09iamVjdCAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyk7CiAgICBteSAkVGlja2V0SklSQU9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpUaWNrZXRKSVJBJyk7CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGZvciBteSAkTmVlZGVkIChxdyhEYXRhIEV2ZW50IENvbmZpZykpIHsKICAgICAgICBpZiAoICEkUGFyYW17JE5lZWRlZH0gKSB7CiAgICAgICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkTmVlZGVkIVxuIiwKICAgICAgICAgICAgKTsKCiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICB9CgogICAgZm9yIG15ICROZWVkZWREYXRhIChxdyhUaWNrZXRJRCkpIHsKICAgICAgICBpZiAoICEkUGFyYW17RGF0YX0tPnskTmVlZGVkRGF0YX0gKSB7CiAgICAgICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkTmVlZGVkRGF0YSBpbiBEYXRhIVxuIiwKICAgICAgICAgICAgKTsKCiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICB9CgogICAgbXkgJFRpY2tldElEID0gJFBhcmFte0RhdGF9LT57VGlja2V0SUR9OwoKICAgIGlmICggJFNlbGYtPntEZWJ1Z30gKSB7CiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZGVidWcnLAogICAgICAgICAgICBNZXNzYWdlICA9PiBfX1BBQ0tBR0VfXyAuICI6OlJ1bigpOiBUaWNrZXRJRDogJFRpY2tldElEXG4iLAogICAgICAgICk7CgogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2RlYnVnJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gX19QQUNLQUdFX18gLiAiOjpSdW4oKTogVGlja2V0SUQ6ICRUaWNrZXRJRCBFdmVudDogJFBhcmFte0V2ZW50fVxuIiwKICAgICAgICApOwogICAgfQoKIyAgICByZXR1cm4gMSBpZiAkUGFyYW17RXZlbnR9IG5lICdUaWNrZXRTdGF0ZVVwZGF0ZSc7CgogICAgbXkgJVRpY2tldCA9ICRUaWNrZXRPYmplY3QtPlRpY2tldEdldCgKICAgICAgICBUaWNrZXRJRCAgICAgID0+ICRUaWNrZXRJRCwKICAgICAgICBVc2VySUQgICAgICAgID0+IDEsCiAgICAgICAgRHluYW1pY0ZpZWxkcyA9PiAxLAogICAgKTsKCiAgICByZXR1cm4gMSBpZiAhJVRpY2tldDsKCiAgICBteSAkSklSQUlzc3VlSUQgPSAkVGlja2V0e0R5bmFtaWNGaWVsZF9KSVJBSXNzdWVJRH07CiAgICByZXR1cm4gMSBpZiAkSklSQUlzc3VlSUQ7CgogICAgaWYgKCAkU2VsZi0+e0RlYnVnfSApIHsKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdkZWJ1ZycsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+IF9fUEFDS0FHRV9fIC4gIjo6UnVuKCk6IFRpY2tldElEOiAkVGlja2V0e1RpY2tldElEfSBUaXRsZTogJFRpY2tldHtUaXRsZX0gIgogICAgICAgICAgICAgICAgLiAiU3RhdGU6ICRUaWNrZXR7U3RhdGV9XG4iLAogICAgICAgICk7CiAgICB9CgogICAgbXkgJEFsbG93ZWRTdGF0ZXNOYW1lcyA9ICRDb25maWdPYmplY3QtPkdldCgnVGlja2V0SklSQScpLT57T1RSU1RpY2tldFN0YXRlc0VuZm9yY2luZ0NyZWF0ZX07CiAgICBteSBAQWxsb3dlZFN0YXRlc0xpc3QgID0gc3BsaXQgLywvLCAkQWxsb3dlZFN0YXRlc05hbWVzOwoKICAgIGlmICggISBmaXJzdHsgJF8gZXEgJFRpY2tldHtTdGF0ZX0gfSBAQWxsb3dlZFN0YXRlc0xpc3QgKSB7CiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnaW5mbycsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+IF9fUEFDS0FHRV9fIC4gIjo6UnVuKCk6IE9UUlMgVGlja2V0IG5vdCBpbiAkQWxsb3dlZFN0YXRlc05hbWVzOiAkVGlja2V0e1N0YXRlfVxuIiwKICAgICAgICApOwoKICAgICAgICByZXR1cm4gMTsKICAgIH0KCiAgICBteSAkSklSQUlzc3VlID0gJFRpY2tldEpJUkFPYmplY3QtPklzc3VlQ3JlYXRlKAogICAgICAgIFRpY2tldElEID0+ICRUaWNrZXR7VGlja2V0SUR9LAogICAgICAgIFVzZXJJRCAgID0+IDEsCiAgICApOyAjIEZJWE1FOiBVc2VySUQgPwoKICAgIGlmICggJFNlbGYtPntEZWJ1Z30gKSB7CiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiBfX1BBQ0tBR0VfXyAuICI6IFVwZGF0ZSByZXNwb25zZSAkSklSQUlzc3VlXG4iLAogICAgICAgICk7CiAgICB9CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwOC0yMDE4IGNhdHdvcmt4IEdtYkggaHR0cDovL3d3dy5jYXR3b3JreC5kZQojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6VGlja2V0OjpFdmVudDo6U2VuZEF0dGFjaG1lbnRzVG9KSVJBSXNzdWU7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9IHF3KAogICAgS2VybmVsOjpDb25maWcKICAgIEtlcm5lbDo6U3lzdGVtOjpMb2cKICAgIEtlcm5lbDo6U3lzdGVtOjpNYWluCiAgICBLZXJuZWw6OlN5c3RlbTo6VGlja2V0CiAgICBLZXJuZWw6OlN5c3RlbTo6VGlja2V0OjpBcnRpY2xlCiAgICBLZXJuZWw6OlN5c3RlbTo6VXNlcgogICAgS2VybmVsOjpTeXN0ZW06OlRpY2tldEpJUkEKKTsKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgbXkgJENvbmZpZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyk7CiAgICAkU2VsZi0+e0RlYnVnfSAgID0gJENvbmZpZ09iamVjdC0+R2V0KCdKSVJBOjpEZWJ1ZycpOwoKICAgIHJldHVybiAkU2VsZjsKfQoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRUaWNrZXRPYmplY3QgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlRpY2tldCcpOwogICAgbXkgJEFydGljbGVPYmplY3QgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VGlja2V0OjpBcnRpY2xlJyk7CiAgICBteSAkQ29uZmlnT2JqZWN0ICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyk7CiAgICBteSAkTWFpbk9iamVjdCAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpNYWluJyk7CiAgICBteSAkVXNlck9iamVjdCAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpVc2VyJyk7CiAgICBteSAkTG9nT2JqZWN0ICAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKTsKICAgIG15ICRUaWNrZXRKSVJBT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlRpY2tldEpJUkEnKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgZm9yIG15ICROZWVkZWQgKHF3KERhdGEgRXZlbnQgQ29uZmlnIFVzZXJJRCkpIHsKICAgICAgICBpZiAoICEkUGFyYW17JE5lZWRlZH0gKSB7CiAgICAgICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgICAgICBNZXNzYWdlICA9PiAiTmVlZCAkTmVlZGVkIVxuIiwKICAgICAgICAgICAgKTsKCiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICB9CgogICAgZm9yIG15ICROZWVkZWREYXRhIChxdyhUaWNrZXRJRCBBcnRpY2xlSUQpKSB7CiAgICAgICAgaWYgKCAhJFBhcmFte0RhdGF9LT57JE5lZWRlZERhdGF9ICkgewogICAgICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICAgICAgTWVzc2FnZSAgPT4gIk5lZWQgJE5lZWRlZERhdGEgaW4gRGF0YSFcbiIsCiAgICAgICAgICAgICk7CgogICAgICAgICAgICByZXR1cm47CiAgICAgICAgfQogICAgfQoKICAgIG15ICRVc2VySUQgPSAkUGFyYW17VXNlcklEfTsKCiAgICAjIEdldCBjdXJyZW50IHVzZXIgZGF0YQogICAgbXkgJFRpY2tldElEID0gJFBhcmFte0RhdGF9LT57VGlja2V0SUR9OwoKICAgIGlmICggJFNlbGYtPntEZWJ1Z30gKSB7CiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZGVidWcnLAogICAgICAgICAgICBNZXNzYWdlICA9PiBzcHJpbnRmIF9fUEFDS0FHRV9fIC4gIjo6UnVuKCk6IFRpY2tldElEOiAlcyBBcnRpY2xlSUQ6ICVzIEZpbGVuYW1lOiAlc1xuIiwKICAgICAgICAgICAgICAgICRUaWNrZXRJRCwgJFBhcmFte0RhdGF9LT57QXJ0aWNsZUlEfSwgJFBhcmFte0RhdGF9LT57RmlsZW5hbWV9CiAgICAgICAgKTsKICAgIH0KCiAgICBteSAkQmFja2VuZE9iamVjdCA9ICRBcnRpY2xlT2JqZWN0LT5CYWNrZW5kRm9yQXJ0aWNsZSgKICAgICAgICBBcnRpY2xlSUQgPT4gJFBhcmFte0RhdGF9LT57QXJ0aWNsZUlEfSwKICAgICAgICBUaWNrZXRJRCAgPT4gJFBhcmFte0RhdGF9LT57VGlja2V0SUR9LAogICAgKTsKCiAgICBteSAlQXJ0aWNsZSA9ICRCYWNrZW5kT2JqZWN0LT5BcnRpY2xlR2V0KAogICAgICAgIEFydGljbGVJRCAgICAgPT4gJFBhcmFte0RhdGF9LT57QXJ0aWNsZUlEfSwKICAgICAgICBUaWNrZXRJRCAgICAgID0+ICRQYXJhbXtEYXRhfS0+e1RpY2tldElEfSwKICAgICAgICBVc2VySUQgICAgICAgID0+IDEsCiAgICAgICAgRHluYW1pY0ZpZWxkcyA9PiAxLAogICAgKTsKCiAgICBteSAlVGlja2V0ID0gJFRpY2tldE9iamVjdC0+VGlja2V0R2V0KAogICAgICAgIFRpY2tldElEICAgICAgPT4gJFBhcmFte0RhdGF9LT57VGlja2V0SUR9LAogICAgICAgIFVzZXJJRCAgICAgICAgPT4gMSwKICAgICAgICBEeW5hbWljRmllbGRzID0+IDEsCiAgICApOwoKICAgIHJldHVybiAxIGlmICElQXJ0aWNsZTsKICAgIHJldHVybiAxIGlmICEkQXJ0aWNsZXtEeW5hbWljRmllbGRfQXJ0aWNsZVNlbnRUb0pJUkF9OwoKICAgIG15ICRJc3N1ZUlEID0gJFRpY2tldHtEeW5hbWljRmllbGRfSklSQUlzc3VlSUR9OwogICAgcmV0dXJuIDEgaWYgISRJc3N1ZUlEOwoKICAgICRUaWNrZXRKSVJBT2JqZWN0LT5BdHRhY2htZW50c1NlbmQoCiAgICAgICAgSXNzdWUgICAgID0+ICRJc3N1ZUlELAogICAgICAgIEFydGljbGVJRCA9PiAkQXJ0aWNsZXtBcnRpY2xlSUR9LAogICAgICAgIFVzZXJJRCAgICA9PiAkUGFyYW17VXNlcklEfSAvLyAxLAogICAgICAgIEZpbGVJRCAgICA9PiAkUGFyYW17RGF0YX0tPntGaWxlSUR9LAogICAgICAgIEZpbGVuYW1lICA9PiAkUGFyYW17RGF0YX0tPntGaWxlbmFtZX0sCiAgICApOwoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwOC0yMDE4IGNhdHdvcmt4IEdtYkggaHR0cDovL3d3dy5jYXR3b3JreC5kZQojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6OlN5c3RlbTo6VGlja2V0OjpFdmVudDo6VXBkYXRlSklSQUlzc3VlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIExpc3Q6OlV0aWwgcXcoZmlyc3QpOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSBxdygKICAgIEtlcm5lbDo6Q29uZmlnCiAgICBLZXJuZWw6OlN5c3RlbTo6TG9nCiAgICBLZXJuZWw6OlN5c3RlbTo6VGlja2V0CiAgICBLZXJuZWw6OlN5c3RlbTo6VXNlcgogICAgS2VybmVsOjpTeXN0ZW06OlRpY2tldEpJUkEKKTsKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgbXkgJENvbmZpZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyk7CiAgICAkU2VsZi0+e0RlYnVnfSAgID0gJENvbmZpZ09iamVjdC0+R2V0KCdKSVJBOjpEZWJ1ZycpOwoKICAgIHJldHVybiAkU2VsZjsKfQoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRUaWNrZXRPYmplY3QgICAgID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlRpY2tldCcpOwogICAgbXkgJENvbmZpZ09iamVjdCAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OkNvbmZpZycpOwogICAgbXkgJFVzZXJPYmplY3QgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VXNlcicpOwogICAgbXkgJExvZ09iamVjdCAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyk7CiAgICBteSAkVGlja2V0SklSQU9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpUaWNrZXRKSVJBJyk7CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGZvciBteSAkTmVlZGVkIChxdyhEYXRhIEV2ZW50IENvbmZpZyBVc2VySUQpKSB7CiAgICAgICAgaWYgKCAhJFBhcmFteyROZWVkZWR9ICkgewogICAgICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICAgICAgTWVzc2FnZSAgPT4gIk5lZWQgJE5lZWRlZCFcbiIsCiAgICAgICAgICAgICk7CgogICAgICAgICAgICByZXR1cm47CiAgICAgICAgfQogICAgfQoKICAgIGZvciBteSAkTmVlZGVkRGF0YSAocXcoVGlja2V0SUQpKSB7CiAgICAgICAgaWYgKCAhJFBhcmFte0RhdGF9LT57JE5lZWRlZERhdGF9ICkgewogICAgICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICAgICAgTWVzc2FnZSAgPT4gIk5lZWQgJE5lZWRlZERhdGEgaW4gRGF0YSFcbiIsCiAgICAgICAgICAgICk7CgogICAgICAgICAgICByZXR1cm47CiAgICAgICAgfQogICAgfQoKICAgIG15ICRVc2VySUQgPSAkUGFyYW17VXNlcklEfTsKCiAgICAjIEdldCBjdXJyZW50IHVzZXIgZGF0YQogICAgbXkgJUN1cnJlbnRVc2VyID0gJFVzZXJPYmplY3QtPkdldFVzZXJEYXRhKAogICAgICAgIFVzZXJJRCA9PiAkVXNlcklELAogICAgKTsKCiAgICBpZiAoICRTZWxmLT57RGVidWd9ICkgewogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2RlYnVnJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gX19QQUNLQUdFX18gLiAiOjpSdW4oKTogVXNlcklEOiAkVXNlcklEIFVzZXJMb2dpbjogJEN1cnJlbnRVc2Vye1VzZXJMb2dpbn1cbiIsCiAgICAgICAgKTsKICAgIH0KCiAgICBpZiAoICRDdXJyZW50VXNlcntVc2VyTG9naW59IGVxICRDb25maWdPYmplY3QtPkdldCgnVGlja2V0SklSQScpLT57SW5jb21pbmdVc2VyfSApIHsKICAgIAlyZXR1cm4gMTsKICAgIH0KCiAgICBteSAkVGlja2V0SUQgPSAkUGFyYW17RGF0YX0tPntUaWNrZXRJRH07CgogICAgaWYgKCAkU2VsZi0+e0RlYnVnfSApIHsKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdkZWJ1ZycsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+IF9fUEFDS0FHRV9fIC4gIjo6UnVuKCk6IFRpY2tldElEOiAkVGlja2V0SUQgRXZlbnQ6ICRQYXJhbXtFdmVudH1cbiIsCiAgICAgICAgKTsKICAgIH0KCiAgICAjIyBjYW5ub3QgdG8gKnRoYXQqIGVhc2lseSwgbWF5YmUgd2UnbGwgY2hlY2sgdGhlIFhNTC1Qcm9wcyBsYXRlciBvbiAjI3JldHVybiAxIGlmICRQYXJhbXtFdmVudH0gbmUgJ1RpY2tldFN0YXRlVXBkYXRlJzsKCiAgICBteSAlVGlja2V0ID0gJFRpY2tldE9iamVjdC0+VGlja2V0R2V0KAogICAgICAgIFRpY2tldElEICAgICAgPT4gJFRpY2tldElELAogICAgICAgIFVzZXJJRCAgICAgICAgPT4gMSwKICAgICAgICBEeW5hbWljRmllbGRzID0+IDEsCiAgICApOwoKICAgIHJldHVybiAxIGlmICElVGlja2V0OwoKICAgIG15ICRKSVJBSXNzdWVJRCA9ICRUaWNrZXR7RHluYW1pY0ZpZWxkX0pJUkFJc3N1ZUlEfTsKICAgIHJldHVybiAxIGlmICEkSklSQUlzc3VlSUQ7CgogICAgaWYgKCAkU2VsZi0+e0RlYnVnfSApIHsKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdkZWJ1ZycsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+IF9fUEFDS0FHRV9fIC4gIjo6UnVuKCk6IFRpY2tldElEOiAkVGlja2V0e1RpY2tldElEfSBUaXRsZTogJFRpY2tldHtUaXRsZX0gIgogICAgICAgICAgICAgICAgLiAiU3RhdGU6ICRUaWNrZXR7U3RhdGV9XG4iLAogICAgICAgICk7CiAgICB9CgogICAgbXkgJEFsbG93ZWRTdGF0ZXNOYW1lcyA9ICRDb25maWdPYmplY3QtPkdldCgnVGlja2V0SklSQScpLT57T1RSU1RpY2tldEFsbG93ZWRTdGF0ZXM0VXBkYXRlfTsKICAgIG15IEBBbGxvd2VkU3RhdGVzTGlzdCAgPSBzcGxpdCAvLC8sICRBbGxvd2VkU3RhdGVzTmFtZXM7CgogICAgaWYgKCAhKCBmaXJzdHsgJF8gZXEgJFRpY2tldHtTdGF0ZX0gfSBAQWxsb3dlZFN0YXRlc0xpc3QgKSApIHsKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdpbmZvJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gX19QQUNLQUdFX18gLiAiOjpSdW4oKTogT1RSUyBUaWNrZXQgbm90IGluICRBbGxvd2VkU3RhdGVzTmFtZXM6ICRUaWNrZXR7U3RhdGV9XG4iLAogICAgICAgICk7CgogICAgICAgIHJldHVybiAxOwogICAgfQoKICAgIG15ICRKSVJBSXNzdWUgPSAkVGlja2V0SklSQU9iamVjdC0+SXNzdWVVcGRhdGUoCiAgICAgICAgVGlja2V0SUQgPT4gJFRpY2tldHtUaWNrZXRJRH0sCiAgICAgICAgVXNlcklEICAgPT4gMSwKICAgICk7ICMgRklYTUU6IFVzZXJJRCA/CgogICAgaWYgKCAkU2VsZi0+e0RlYnVnfSApIHsKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdkZWJ1ZycsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+IF9fUEFDS0FHRV9fIC4gIjogVXBkYXRlIHJlc3BvbnNlICRKSVJBSXNzdWVcbiIsCiAgICAgICAgKTsKICAgIH0KCiAgICByZXR1cm4gMTsKfQoKMTsK
#  --
#  Copyright (C) 2008-2018 catworkx GmbH http://www.catworkx.de
#  --
# 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::TicketJIRA;

use strict;
use warnings;

use REST::Client;
use JIRA::REST;
use Time::Piece;
use Scalar::Util qw(looks_like_number);
use List::Util qw(first);

use constant P => __PACKAGE__;
our @ObjectDependencies = qw(
    Kernel::Config
    Kernel::System::Log
    Kernel::System::Main
    Kernel::System::Ticket
    Kernel::System::Ticket::Article
    Kernel::System::User
    Kernel::System::DynamicField
    Kernel::System::DynamicField::Backend
    Kernel::System::DynamicFieldValue
    Kernel::System::JSON
    Kernel::System::Cache
    Kernel::System::JIRA::QueueSettings
);


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

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

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

    $Self->{Debug}      = $ConfigObject->Get('JIRA::Debug');
    $Self->{Config}     = $ConfigObject->Get('TicketJIRA') || {};

    my $Host = $Self->{Config}->{URL} . $Self->{Config}->{Path};
    $Host =~ s{/\z}{}xmsg;

    $Self->{Host} = $Host;

    $Self->{JIRAClient} = JIRA::REST->new(
        $Host,
        $Self->{Config}->{User},
        $Self->{Config}->{Password},
    );

    my $VerifySSL = $ConfigObject->Get('JIRA::SSLVerify');
    if ( !$VerifySSL ) {
        my $Loaded = $Kernel::OM->Get('Kernel::System::Main')->Require(
            'Net::SSLeay',
            Silent => 1,
        );
        if ($Loaded) {
            $Self->{JIRAClient}->{rest}->getUseragent()->ssl_opts( verify_hostname => 0 );
            $Self->{JIRAClient}->{rest}->getUseragent()->ssl_opts( SSL_verfiy_mode => Net::SSLeay::VERIFY_NONE() );
        }
    }

    $Self->{JIRADynamicField} = $DynamicFieldObject->DynamicFieldGet( Name => 'JIRAIssueID' );

    $Self->{CacheType} = 'JIRAAPI';
    $Self->{CacheTTL}  = $ConfigObject->Get('JIRA::CacheTTL');

    return $Self;
}

=item IssueClose()

Close an issue in Jira

=cut

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

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

    # check for needed data
    for my $Needed ( qw/IssueID/ ) {
        if ( !$Param{$Needed} ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );

            return;
        }
    }

    my $CloseTransition = $Self->{Config}->{WorkflowCloseActionName} // 'Done';

    my $Transitions;
    my $ErrorTransitions;

    eval{
        $Transitions = $Self->{JIRAClient}->GET(
            '/issue/' . $Param{IssueID} . '/transitions',
            {
                expand => 'transition.fields',
            },
        );

        1;
    } or $ErrorTransitions = $@;

    if ( $ErrorTransitions ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => sprintf P .  "::IssueClose: cannot get transitions the Jira Issue for %s: %s",
                $Param{IssueID}, $ErrorTransitions,
        );

        return +{ error => $ErrorTransitions };
    }

    my ($Transition) = grep{ $_->{name} eq $CloseTransition }@{ $Transitions->{transitions} || [] };

    if ( !$Transition ) {
        return +{ error => 'No transition found for ' . $CloseTransition };
    }

    my $ErrorClose;

    my %Opts = (
        transition => {
            id => $Transition->{id},
        }
    );

    if ( $Param{Comment} ) {
        $Opts{update} = {
            comment => [
                { add => { body => $Param{Comment} } },
            ],
        };
    }

    eval{
        $Self->{JIRAClient}->POST(
            '/issue/' . $Param{IssueID} . '/transitions',
            undef,
            \%Opts,
        );

        1;
    } or $ErrorClose = $@;

    if ( $ErrorClose ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => sprintf P .  "::IssueClose: cannot close issue %s: %s for %s",
                $Param{IssueID}, $ErrorTransitions,
                $MainObject->Dump(\%Opts),
        );

        return +{ error => $ErrorClose };
    }

    return +{};
}

=item IssueGet()

Get an issue from Jira

    my $IssueData = $Object->IssueGet(
        Issue => $IssueKey,
    );

=cut

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

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    # check for needed data
    for my $Needed ( qw/Issue/ ) {
        if ( !$Param{$Needed} ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );

            return;
        }
    }

    my $Issue;
    my $ErrorGet;

    eval{
        $Issue = $Self->{JIRAClient}->GET( '/issue/' . $Param{Issue} );
    } or $ErrorGet = $@;

    if ( $ErrorGet ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => sprintf P .  "::IssueClose: cannot close issue %s: %s",
                $Param{IssueID}, $ErrorGet,
        );

        return +{ error => $ErrorGet };
    }

    return $Issue;
}

=item IssueCreate()

creates an issue in the configured Jira system using REST

    my $Issue = $TicketJIRAObject->IssueCreate(
        TicketID       => $TicketID,                  # the TicketID, mandatory
        ArticleID      => $ArticleID,                 # optional, to define which article is sent to Jira
        UserID         => $UserID                     # the UserID, mandatory
    );

returns:

    $Issue = {
        id => 10000,                               # issue id in Jira, starts from 10000
        key => PROJ-1,                             # issue key as used in the webinterface, <Proj>-<Number>, starts from 1
        [...]                                      # plenty of other fields as defined in the API
    }

=cut

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

    my $LogObject      = $Kernel::OM->Get('Kernel::System::Log');
    my $MainObject     = $Kernel::OM->Get('Kernel::System::Main');
    my $ConfigObject   = $Kernel::OM->Get('Kernel::Config');
    my $TicketObject   = $Kernel::OM->Get('Kernel::System::Ticket');
    my $DFValueObject  = $Kernel::OM->Get('Kernel::System::DynamicFieldValue');
    my $BackendObject  = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');
    my $SettingsObject = $Kernel::OM->Get('Kernel::System::JIRA::QueueSettings');
    my $ArticleObject  = $Kernel::OM->Get('Kernel::System::Ticket::Article');


    # check for needed data
    for my $Needed ( qw/TicketID UserID/ ) {
        if ( !$Param{$Needed} ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );

            return;
        }
    }

    my %Article;

    if ( $Param{ArticleID} ) {
        my $BackendObject = $ArticleObject->BackendForArticle(
            TicketID      => $Param{TicketID},
            ArticleID     => $Param{ArticleID},
        );

        %Article = $BackendObject->ArticleGet(
            TicketID      => $Param{TicketID},
            ArticleID     => $Param{ArticleID},
            DynamicFields => 1,
        );
    }
    else {
        my ($FirstArticle) = $ArticleObject->ArticleList(
            TicketID  => $Param{TicketID},
            OnlyFirst => 1,
        );

        my $BackendObject = $ArticleObject->BackendForArticle( %{$FirstArticle} );

        %Article = $BackendObject->ArticleGet(
            TicketID      => $Param{TicketID},
            ArticleID     => $FirstArticle->{ArticleID},
            DynamicFields => 1,
        );
    }

    my %Ticket = $TicketObject->TicketGet(
        TicketID      => $Param{TicketID},
        DynamicFields => 1,
    );

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'info',
            Message  => sprintf P . "::IssueCreate: TicketID: %s == %s Title: %s Text: %s",
                $Param{TicketID}, $Article{TicketID}, $Ticket{Title}, $Article{Body},
        );
    }

    my $ValidLicense = $Self->CheckValidLicense();
    if ( $ValidLicense != 1 ) {
        my $URL = $Self->{Config}->{URL};
        $LogObject->Log(
            Priority => 'error',
            Message  => P . "::IssueCreate: invalid license at $URL, status $ValidLicense",
        );

        return +{ error => 'No valid license' }; # skip the rest
    }

    my %Settings;
    if ( $Param{QueueID} ) {
        %Settings = $SettingsObject->ValidSettingsGet( QueueID => $Param{QueueID} );
    }

    my $Reporter = $Self->{Config}->{Reporter};
    my $Assignee = $Settings{Assignee} || $Param{Assignee} || $Self->{Config}->{Assignee};

    # read 'Project' feld from  "SysConfig" in OTRS
    my $Project = $Settings{ProjectID} || $Param{Project} || $Self->{Config}->{Project};

    # read 'IssueType' feld from  "SysConfig" in OTRS
    my $IssueType   = $Param{IssueType} || $Self->{Config}->{IssueType};
    my $FieldName   = 'JIRAIssueID';
    my $JIRAIssueID = $Article{"DynamicField_" . $FieldName};

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'info',
            Message  => sprintf P . "::IssueCreate: JIRAIssueID(pre): '%s' DynamicFieldID: %s ",
                ( $JIRAIssueID || '' ), $FieldName,
        );
    }

    my $Issue = {};
    my $UsedArticleID;

    if ($JIRAIssueID) {

        $Issue = $Self->IssueGet(
            Issue => $JIRAIssueID,
        );

        if ( !$Issue && $Self->{Debug} ) {
            $LogObject->Log(
                Priority => 'error',
                Message  =>  sprintf P . "::IssueCreate: cannot fetch the Jira Issue %s for %s",
                    $JIRAIssueID, $Param{TicketID},
            );

            return +{ error => 'Cannot fetch Issue' };
        }

    }
    else {
        my $CustomField = 'customfield_' . ( $ConfigObject->Get('JIRA::OTRSTicketIDField') || 10000 );
        my $JIRACreateParams = {
            project       => { key => $Project },
            issuetype     => { name => $IssueType },
            summary       => $Ticket{Title},
            description   => $Article{Body},
            reporter      => { name => $Reporter },
            assignee      => { name => $Assignee },
            $CustomField  => $Param{TicketID},
        };

        if ( $ConfigObject->Get('JIRA::NoAssignee') ) {
            delete $JIRACreateParams->{assignee};
        }

        if ( !$ConfigObject->Get('JIRA::SendReporter') ) {
            delete $JIRACreateParams->{reporter};
        }

        $JIRACreateParams = $Self->MapFields( $JIRACreateParams, %Ticket, %Article);

        # clean up summary - all vertical spaces are removed
        # Jira reports an error when the summary contains "newline characters"
        $JIRACreateParams->{summary} =~ s{\v}{ }g;

        if ( $Self->{Debug} ) {
            $LogObject->Log(
                Priority => 'notice',
                Message  => sprintf P .  "::IssueCreate: Parameters: %s",
                    $MainObject->Dump($JIRACreateParams),
            );
        }

        my $ErrorCreate;
        eval {
            $Issue = $Self->{JIRAClient}->POST(
                '/issue',
                undef,
                +{ fields => $JIRACreateParams },
            );

            1;
        } or $ErrorCreate = $@;
 
        if ( $ErrorCreate ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => sprintf P .  "::IssueCreate: cannot create the Jira Issue for %s: %s",
                    $Param{TicketID}, $ErrorCreate,
            );

            return +{ error => $ErrorCreate };
        }

        $UsedArticleID = $Article{ArticleID};

        $Self->AttachmentsSend(
            ArticleID => $Article{ArticleID},
            Issue     => $Issue->{key},
            UserID    => $Param{UserID},
        );

        my $Success = $BackendObject->ValueSet(
            DynamicFieldConfig => $Self->{JIRADynamicField},
            ObjectID           => $Article{TicketID},
            Value              => $Issue->{key},
            UserID             => $Param{UserID},
        );

        if ( $Self->{Debug} ) {
            $LogObject->Log(
                Priority => 'info',
                Message  => sprintf P . "::IssueCreate: JIRAIssueID(post): %s DynamicFieldID: %s",
                    $Issue->{key}, $FieldName,
            );

            $LogObject->Log(
                Priority => 'info',
                Message  => sprintf P . "::IssueCreate: Success: %s", $Success,
            );
        }

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

        my $Data   = {
            object => {
                url   => sprintf(
                    "%s://%s/%sindex.pl?Action=AgentTicketZoom;TicketID=%s",
                    $ConfigObject->Get('HttpType'),
                    $ConfigObject->Get('FQDN'),
                    $ConfigObject->Get('ScriptAlias'),
                    $Param{TicketID},
                ),
                title => sprintf( "OTRS Ticket#%s (%s)", $TicketNumber, $Param{TicketID} ),
            },
        };

        my $Result;
        eval {
            $Result = $Self->{JIRAClient}->POST(
                '/issue/' . $Issue->{key} . '/remotelink',
                undef,
                $Data,
            );
        };

        if ( $Self->{Debug} ) {
            $LogObject->Log(
                Priority => 'info',
                Message  => sprintf P . "::IssueCreate: Remotelink (%s/%s) result %s",
                    $Issue->{key}, $Param{TicketID},
                    $MainObject->Dump($Result),
            );
        }

        # transfer ticket history - if configured and issue is not created based
        # on a single article
        my $ShouldTransferHistory = $ConfigObject->Get('JIRA::TransferTicketHistory');
        if ( $ShouldTransferHistory && !$Param{ArticleID} && $UsedArticleID ) {

            my @ArticleIndex = $ArticleObject->ArticleList(
                TicketID => $Param{TicketID},
                UserID   => 1,
            );

            COMMENT:
            for my $Comment ( @ArticleIndex ) {
                next COMMENT if $Comment->{ArticleID} == $UsedArticleID;

                my $BackendObject = $ArticleObject->BackendForArticle( %{$Comment} );
                my %Article       = $BackendObject->ArticleGet(
                    ArticleID => $Comment->{ArticleID},
                    TicketID  => $Param{TicketID},
                    UserID    => 1,
                );

                my $JIRAIssue = $Self->IssueComment(
                    IssueID   => $Issue->{key},
                    ArticleID => $Comment->{ArticleID},
                    Article   => \%Article,
                    UserID    => $Param{UserID} // 1,
                    Comment   => $Article{Body},
                );

                $Self->AttachmentsSend(
                    ArticleID => $Comment->{ArticleID},
                    Issue     => $Issue->{key},
                    UserID    => $Param{UserID},
                );
            }
        }
    }

    return $Issue;
}

=item IssueUpdate()

updates an issue in the configured Jira system using REST

    my $Issue = $TicketJIRAObject->IssueUpdate(
        TicketID       => $TicketID,                  # the TicketID, mandatory
        UserID         => $UserID                     # the UserID, mandatory
    );

returns:

    $Issue = {
        id => 10000,                               # issue id in Jira, starts from 10000
        key => PROJ-1,                             # issue key as used in the webinterface, <Proj>-<Number>, starts from 1
        [...]                                      # plenty of other fields as defined in the API
    }
=cut

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

    my $LogObject     = $Kernel::OM->Get('Kernel::System::Log');
    my $ConfigObject  = $Kernel::OM->Get('Kernel::Config');
    my $TicketObject  = $Kernel::OM->Get('Kernel::System::Ticket');
    my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');

    # check for needed data
    for my $Needed ( qw/TicketID UserID/ ) {
        if ( !$Param{$Needed} ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );

            return;
        }
    }

    my %Ticket = $TicketObject->TicketGet(
        TicketID      => $Param{TicketID},
        DynamicFields => 1
    );

    my @Index = $ArticleObject->ArticleList(
        TicketID   => $Param{TicketID},
    );

    my %Article;

    if (@Index) {
        my $BackendObject = $ArticleObject->BackendForArticle( %{ $Index[-1] } );
        %Article = $BackendObject->ArticleGet(
            ArticleID     => $Index[-1]->{ArticleID},
            DynamicFields => 1,
        );
    }

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'info',
            Message  => sprintf P . "::IssueUpdate: TicketID: %s == %s Title: %s",
                $Param{TicketID}, $Ticket{TicketID}, $Ticket{Title},
        );
    }

    my $URL = $Self->{Config}->{URL};

    my $ValidLicense = $Self->CheckValidLicense();
    if ( $ValidLicense != 1 ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => sprintf P . "::IssueUpdate: invalid license at %s, status %s",
                $URL, $ValidLicense,
        );

        return {}; # skip the rest
    }

    my $Reporter = $Self->{Config}->{Reporter};
    my $Assignee = $Self->{Config}->{Assignee};

    # read 'Project' feld from  "SysConfig" in OTRS
    my $Project = $Self->{Config}->{Project};

    # read 'IssueType' feld from  "SysConfig" in OTRS
    my $IssueType = $Self->{Config}->{IssueType};
    my $FieldName = 'JIRAIssueID';

    my $JIRAIssueID = $Ticket{'DynamicField_' . $FieldName};

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'info',
            Message  => sprintf P . "::IssueUpdate: JIRAIssueID(pre): '%s' DynamicFieldID: %s",
                $JIRAIssueID, $FieldName,
        );
    }

    my $Issue = {};

    if ( $JIRAIssueID ) {
        my $CustomField = 'customfield_' . ( $ConfigObject->Get('JIRA::OTRSTicketIDField') || 10000 );

        my $JIRAUpdateParams = {
            $CustomField => $Param{TicketID},
        };

        $JIRAUpdateParams = $Self->MapFields( $JIRAUpdateParams, %Ticket, %Article );

        my $UpdateError;

        eval{
            $Issue = $Self->{JIRAClient}->PUT(
                "/issue/" . $JIRAIssueID,
                undef,
                { fields => $JIRAUpdateParams },
            )
        } or $UpdateError = $@;

        if ( $UpdateError ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => sprintf P . "::IssueUpdate: Error on update: %s\n",
                    $UpdateError,
            );
        }
    }

    return $Issue || 'Success';
}

=item IssueLinkToOTRS()

link an existing issue to an OTRS ticket. This operation updates the custom field holding the otrs ticket id on the Jira side
and the dynamic field with the issue id on the OTRS side

    my $Issue = $TicketJIRAObject->IssueLinkToOTRS(
        TicketID       => $TicketID,                  # the TicketID, mandatory
        UserID         => $UserID                     # the UserID, mandatory
    );

returns:

    $Issue = {
        id => 10000,                               # issue id in Jira, starts from 10000
        key => PROJ-1,                             # issue key as used in the webinterface, <Proj>-<Number>, starts from 1
        [...]                                      # plenty of other fields as defined in the API
    }
=cut

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

    my $LogObject     = $Kernel::OM->Get('Kernel::System::Log');
    my $MainObject    = $Kernel::OM->Get('Kernel::System::Main');
    my $ConfigObject  = $Kernel::OM->Get('Kernel::Config');
    my $TicketObject  = $Kernel::OM->Get('Kernel::System::Ticket');
    my $BackendObject = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

    # check for needed data
    for my $Needed ( qw/TicketID UserID IssueID Project/ ) {
        if ( !$Param{$Needed} ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );

            return;
        }
    }

    my $URL = $Self->{Config}->{URL};

    my $ValidLicense = $Self->CheckValidLicense();
    if ( $ValidLicense != 1 ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => sprintf P . "::IssueLinkToOTRS: invalid license at %s, status %s",
                $URL, $ValidLicense,
        );

        return {}; # skip the rest
    }

    # read 'Project' feld from  "SysConfig" in OTRS
    my $Project     = $Param{Project};
    my $JIRAIssueID = $Param{IssueID};

    my $FieldName = 'JIRAIssueID';

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'info',
            Message  => sprintf P . "::IssueLinkToOTRS: TicketID: '%s' DynamicFieldID: %s",
                $Param{TicketID}, $FieldName,
        );
     }

    my $CustomField = 'customfield_' . ( $ConfigObject->Get('JIRA::OTRSTicketIDField') || 10000 );

    my $JIRAUpdateParams = {
        $CustomField => $Param{TicketID},
    };

    $JIRAUpdateParams = $Self->MapFields( $JIRAUpdateParams );

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'info',
            Message  => sprintf P . "::IssueLinkToOTRS: UpdateParams -> "
                . $MainObject->Dump( $JIRAUpdateParams ),
        );
    }

    my $UpdateError;

    my $Issue = {};
    eval{
        $Issue = $Self->{JIRAClient}->PUT(
            "/issue/" . $JIRAIssueID,
            undef,
            { fields => $JIRAUpdateParams },
        );
    } or $UpdateError = $@;

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

    my $Data   = {
        object => {
            url   => sprintf(
                "%s://%s/%sindex.pl?Action=AgentTicketZoom;TicketID=%s",
                $ConfigObject->Get('HttpType'),
                $ConfigObject->Get('FQDN'),
                $ConfigObject->Get('ScriptAlias'),
                $Param{TicketID},
            ),
            title => sprintf( "OTRS Ticket#%s (%s)", $TicketNumber, $Param{TicketID} ),
        },
    };

    my $Result;
    eval {
        $Result = $Self->{JIRAClient}->POST(
            '/issue/' . $JIRAIssueID . '/remotelink',
            undef,
            $Data,
        );
    };

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'info',
            Message  => sprintf P . "::IssueLinkToOTRS: Issue (from Update) -> "
                . $MainObject->Dump( $Issue ),
        );
    }

    if ( $UpdateError ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => sprintf P . "::IssueLinkToOTRS: Error on update: %s\n",
                $UpdateError,
        );

        return { error => $UpdateError };
    }

    my $IssueInfo = $Self->IssueGet( Issue => $JIRAIssueID );

    my $Success = $BackendObject->ValueSet(
        DynamicFieldConfig => $Self->{JIRADynamicField},
        ObjectID           => $Param{TicketID},
        Value              => $IssueInfo->{key},
        UserID             => $Param{UserID},
    );

    return $Issue || {};
}

=item IssueComment()

creates an issue in the configured Jira system using SOAP

    my $Issue = $TicketJIRAObject->IssueComment(
        IssueID => $TicketID, # the IssueID, mandatory
        Comment => $ArticleID # the comment, mandatory
    );

returns:

    $Issue = {
        id => 10000,                               # issue id in Jira, starts from 10000
        key => PROJ-1,                             # issue key as used in the webinterface, <Proj>-<Number>, starts from 1
        [...]                                      # plenty of other fields as defined in the API
    }

=cut

sub IssueComment {
    my ( $Self, %Param ) = @_; ## FIXME implement

    my $ConfigObject       = $Kernel::OM->Get('Kernel::Config');
    my $LogObject          = $Kernel::OM->Get('Kernel::System::Log');
    my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');
    my $BackendObject      = $Kernel::OM->Get('Kernel::System::DynamicField::Backend');

    # check for needed data
    for my $Needed ( qw/IssueID Comment ArticleID UserID/ ) {
        if ( !$Param{$Needed} ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => "Need $Needed!",
            );

            return;
        }
    }

    my $CommentText   = $Param{Comment};
    my $ArticleFormat = $ConfigObject->Get('JIRA::ArticleFormat');
    if ( $ArticleFormat && !$Param{UseComment} ) {
        $CommentText = $Self->_FormatArticle(
            Article       => $Param{Article},
            ArticleFormat => $ArticleFormat,
        );
    }

    my $URL = $Self->{Config}->{URL};

    my $ValidLicense = $Self->CheckValidLicense();
    if ( $ValidLicense != 1 ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => sprintf P . "::IssueComment: invalid license at %s, status %s",
                $URL, $ValidLicense,
        );

        return {}; # skip the rest
    }

    my $Issue = {};

    if ( $Param{IssueID} ) {
        my $ErrorComment;
        eval {
            $Issue = $Self->{JIRAClient}->POST(
                "/issue/" . $Param{IssueID} . "/comment",
                undef,
                { body => $CommentText },
            );

            1;
        } or $ErrorComment = $@;

        if ( $ErrorComment ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => sprintf P .  "::IssueComment: cannot create comment for issue %s: %s",
                    $Param{IssueID}, $ErrorComment,
            );

            return +{ error => $ErrorComment };
        }

    }

    my $SentField = $DynamicFieldObject->DynamicFieldGet( Name => 'ArticleSentToJIRA' );
    my $Success   = $BackendObject->ValueSet(
        DynamicFieldConfig => $SentField,
        ObjectID           => $Param{ArticleID},
        Value              => 1,
        UserID             => $Param{UserID},
    );

    return $Issue;
}

=item AttachmentsSend()

=cut

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

    my $TicketObject  = $Kernel::OM->Get('Kernel::System::Ticket');
    my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');
    my $LogObject     = $Kernel::OM->Get('Kernel::System::Log');
    my $MainObject    = $Kernel::OM->Get('Kernel::System::Main');
    my $EncodeObject  = $Kernel::OM->Get('Kernel::System::Encode');

    for my $Needed ( qw(ArticleID Issue UserID) ) {
        if ( !$Param{$Needed} ) {
            $LogObject->Log(
                Priority => 'notice',
                Message  => "Need $Needed!",
            );

            return;
        }
    }

    my $TicketID      = $ArticleObject->TicketIDLookup( ArticleID => $Param{ArticleID} );
    my $BackendObject = $ArticleObject->BackendForArticle(
        TicketID  => $TicketID,
        ArticleID => $Param{ArticleID},
    );

    return if !$BackendObject;
    return if !$BackendObject->can('ArticleAttachmentIndex');

    # get the list of the attachments
    my %Index = $BackendObject->ArticleAttachmentIndex(
        ArticleID => $Param{ArticleID},
        UserID    => $Param{UserID},
    );

    # get useragent used by the Jira object to do the fileuploads
    my $Rest = $Self->{JIRAClient}->{rest};
    my $UA   = $Rest->getUseragent();

    # get ID of each attachment file, then the corresponding attachment file
    while ( my ( $FileID, $AttachH ) = each %Index ) {

        # retrieve the name and content of each attachment file,
        # store them into @filenames and @attachments respectively.
        my @Filenames;
        my @Attachments;

        my %Attachment = $BackendObject->ArticleAttachment(
            ArticleID => $Param{ArticleID},
            FileID    => $FileID,
            UserID    => $Param{UserID},
        );

        next if $Attachment{Filename} eq 'file-2';

        if ( $Param{Filename} && $Attachment{Filename} ne $Param{Filename} ) {
            next;
        }

        if ( $Self->{Debug} ) {
            $LogObject->Log(
                Priority => 'info',
                Message  => sprintf P . "::AttachmentsSend: Handling attachment: %s", $Attachment{Filename}
            );
        }

        my $IssueKey = $Param{Issue};
        my $FileName = $EncodeObject->EncodeOutput( \$Attachment{Filename} );

        my $Response = $UA->post(
            $Self->{Host} . "/issue/$IssueKey/attachments",
            #$Rest->getHost() . "/issue/$IssueKey/attachments",
            %{$Rest->{_headers}},
            'X-Atlassian-Token' => 'no-check',
            'Content-Type'      => 'form-data',
            Content             => [
                file => [
                    undef,                   # as we have the content alread, we do not pass a filename
                    $Attachment{Filename},   # filename for the server
                   'Content-Type' => $Attachment{ContentType},
                    Content       => $Attachment{Content},
                ],
            ],
        );

        if ( !$Response->is_success ) {
            $LogObject->Log(
                Priority => 'error',
                Message  => sprintf P . "::AttachmentsSend: Cannot attach file $Attachment{Filename} to issue $IssueKey -> "
                    . $Response->status_line,
            );

            $LogObject->Log(
                Priority => 'debug',
                Message  => sprintf P . "::AttachmentsSend: Header -> " . $MainObject->Dump ( $Rest->{_headers} ),
            );
        }
        elsif ( $Self->{Debug} ) {
            $LogObject->Log(
                Priority => 'debug',
                Message  => sprintf P . "::AttachmentsSend: attached file $Attachment{Filename} -> "
                    . $Response->status_line,
            );
        }
    }
}


=item FieldsGet()

Retrieve field definitions from Jira. The result is cached for performance reasons.

    my $Fields = $Object->FieldsGet();

=cut

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

    my $LogObject   = $Kernel::OM->Get('Kernel::System::Log');
    my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');

    if (!$Param{DoNotCache}) {
        my $CacheKey = 'FieldsGet';

        $Param{Return} ||= '';

        if ( $Param{Return} && lc $Param{Return} eq 'types' ) {
            $CacheKey = 'FieldTypes';
        }
        elsif ( $Param{Return} && lc $Param{Return} eq 'standard' ) {
            $CacheKey = 'StandardFields';
        }

        my $Cache = $CacheObject->Get(
            Type => $Self->{CacheType},
            Key  => $CacheKey,
        );

        return $Cache if $Cache;
    }

    my $Fields;
    my $FieldTypes;
    my $StandardFields;

    my $ErrorFields;
    eval {
        my $FieldsData = $Self->{JIRAClient}->GET(
            "/field"
        );

        my %CustomFields = map{
            $_->{name} => $_->{schema}->{customId}
        } grep {
            $_->{schema}->{customId}
        } @{ $FieldsData || [] };

        $Fields = \%CustomFields;

        for my $Field ( @{ $FieldsData || [] } ) {
            $FieldTypes->{ $Field->{id} } = $Field->{schema};

            if ( !$Field->{custom} ) {
                push @{$StandardFields}, $Field->{id};
            }
        }

        1;
    } or $ErrorFields = $@;

    if ( !$ErrorFields ) {
        $CacheObject->Set(
            Type  => $Self->{CacheType},
            Key   => 'FieldsGet',
            Value => $Fields,
            TTL   => $Self->{CacheTTL} || 1,
        );

        $CacheObject->Set(
            Type  => $Self->{CacheType},
            Key   => 'FieldTypes',
            Value => $FieldTypes,
            TTL   => $Self->{CacheTTL} || 1,
        );

        $CacheObject->Set(
            Type  => $Self->{CacheType},
            Key   => 'StandardFields',
            Value => $StandardFields,
            TTL   => $Self->{CacheTTL} || 1,
        );
    }
    else {
        $LogObject->Log(
            Priority => 'error',
            Message  => sprintf P .  "::FieldsGet: cannot retrieve fields definitions: %s",
                $ErrorFields,
        );

        return;
    }

    if ( lc $Param{Return} eq 'types' ) {
        return $FieldTypes;
    }

    return $Fields;
}

=item MapFields

Maps the OTRS to the Jira custom fields as defined in the configuration.
Returns a hash map.

  my $Fields = $Object->MapFields($Fields, %Ticket, %Article)

return:

  $Fields = {
    standardField1 => value1,
    standardField2 => value2,
    [...]
    customfields => {
        cf1 => value3,
        cf2 => value4,
        [...]
    }
  }

=cut

sub MapFields {
    my ( $Self, $Fields, %Article ) = @_;

    my $LogObject          = $Kernel::OM->Get('Kernel::System::Log');
    my $CacheObject        = $Kernel::OM->Get('Kernel::System::Cache');
    my $MainObject         = $Kernel::OM->Get('Kernel::System::Main');
    my $DynamicFieldObject = $Kernel::OM->Get('Kernel::System::DynamicField');
    my $DFValueObject      = $Kernel::OM->Get('Kernel::System::DynamicFieldValue');

    # retrieves dynamic-custom field mapping
    my $CustomFields   = {};
    my $Mapping        = $Self->{Config}->{'OTRS-JIRA-FieldMapping'};
    my $DateFormat     = $Self->{Config}->{'Jira-DateFormat'};
    my $DateTimeFormat = $Self->{Config}->{'Jira-DateTimeFormat'};

    my $JIRAFields     = $Self->FieldsGet();

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'debug',
            Message  => sprintf P . "::MapFields: Article: %s",
                $MainObject->Dump( \%Article ),
        );

        $LogObject->Log(
            Priority => 'debug',
            Message  => sprintf P . "::MapFields: mapping: %s",
                $MainObject->Dump( $Mapping ),
        );
    }
    

    if ( defined $Mapping ) {

        # all changes required ONLY for Perl 5.10 ...
        KEY:
        for my $Key ( keys %{$Mapping}  ) {
            my $Value = $Mapping->{$Key};

            next KEY if !length $Key || !length $Value;

            my ($KPrefix,$VPrefix,$DBVal) = ("","",undef);
            my ($IsADate, $IsADateTime) = (0,0);

            ($KPrefix, $Key)   = split /:/, $Key, 2;
            ($VPrefix, $Value) = split /:/, $Value, 2;

            if ( $Self->{Debug} ) {
                $LogObject->Log(
                    Priority => 'debug',
                    Message  => P . "::MapFields: checking: kprefix,key,vprefix,val : "
                        . "$KPrefix,$Key,$VPrefix,$Value\n",
                );
            }

            if ( $KPrefix eq 'std' ) {
                $DBVal = $Article{$Key};

                if ( $Self->{Debug} ) {
                    $LogObject->Log(
                        Priority => 'debug',
                        Message  => P . "::MapFields: std field with $DBVal\n",
                    );
                }

                if ( $Key eq 'Created' || $Key eq 'Changed' ) {
                    $IsADateTime = 1;
                }
                elsif ( $Key =~ m{Date\z}xms ) {
                    $IsADate = 1;
                }
            }
            elsif ( $KPrefix eq 'dyn' ) {
                my $DynField  = $DynamicFieldObject->DynamicFieldGet( Name => $Key );
                my $FieldType = $DynField->{FieldType};

                $DBVal = $Article{"DynamicField_$Key"};

                if ( !$DBVal && $DynField ) {
                    $DBVal = $DFValueObject->ValueGet(
                        FieldID  => $DynField->{ID},
                        ObjectID => $Article{TicketID},
                    ) || undef;
                }

                if ( $Self->{Debug} ) {
                    $LogObject->Log(
                        Priority => 'debug',
                        Message  => P . "::MapFields: dyn field Array: "
                            . $MainObject->Dump($DBVal // '') . "\n",
                    );
                }

                my $IsMultiselect;

                # only handle Date, DateTime and Checkbox, for the rest we have nothing to do
                if ( $FieldType eq 'Date' ) {
                    $IsADate = 1;

                    $DBVal = ref $DBVal ? $DBVal->[0]->{ValueDateTime} : $DBVal;
                }
                elsif ( $FieldType eq 'DateTime' ) {
                    $IsADateTime = 1;

                    $DBVal = ref $DBVal ? $DBVal->[0]->{ValueDateTime} : $DBVal;
                }
                elsif ( $FieldType eq 'Checkbox' ) {
                    if ( $Self->{Debug} ) {
                        $LogObject->Log(
                            Priority => 'info',
                            Message  => P . "::MapFields: skipping: Checkbox "
                                . "Dynamicfield $Key with value $DBVal\n",
                        );
                    }

                    $DBVal = undef;
                } # Ignore by default since Jira does not have a checkbox custom field
                elsif ( $FieldType eq 'Multiselect' ) {
                    $DBVal = ref $DBVal ?
                        [ map{ +{ value => $_ } }@{$DBVal} ] :
                        [];

                    $IsMultiselect = 1;
                }
                elsif ( $FieldType eq 'Dropdown' ) {
                    $DBVal = ref $DBVal ?
                        +{ value => $DBVal->[0]->{ValueText} } :
                        +{ value => $DBVal };

                    $IsMultiselect = 1;
                }

                $DBVal = (ref $DBVal && !$IsMultiselect) ?
                    ( $DBVal->[0]->{ValueText} || $DBVal->[0]->{ValueInt} ) :
                    $DBVal;

                if ( $Self->{Debug} ) {
                    $LogObject->Log(
                        Priority => 'debug',
                        Message  => P . "::MapFields: dyn field with $DBVal and type $FieldType\n",
                    );
                }
            }
            if ( $KPrefix eq 'call' ) {

                # calls must be specified as this: e.g. "MIME::Base64,encode_base64"
                my ($Module, $Function) = split /,/, $Key;

                if ( $MainObject->Require( $Module ) ) {
                    my $Code = $Module->can( $Function );
                    $DBVal   = $Code->( $Self, $Key, %Article );
                }
                else {
                    $LogObject->Log(
                        Priority => 'debug',
                        Message  => P . '::MapFields: Could not load ' . $Module,
                    );
                }

                if ( $Self->{Debug} ) {
                    $LogObject->Log(
                        Priority => 'debug',
                        Message  => P . "::MapFields: call to '$Function' resulting in $DBVal\n",
                    );
                }
            }

            if ( $DBVal ) {
                if ( $IsADate == 1 ) {
                    $DBVal = $Self->DateFormatTransform( $DateFormat, $DBVal );
                }

                if ( $IsADateTime == 1 ) {
                    $DBVal = $Self->DateFormatTransform( $DateTimeFormat, $DBVal );
                }

                if ( $Self->{Debug} ) {
                    $LogObject->Log(
                        Priority => 'debug',
                        Message  => P . "::MapFields: got mapped value: kprefix,key,vprefix,val,dbval : "
                            . "$KPrefix,$Key,$VPrefix,$Value,$DBVal\n",
                    );
                }

                if ( $VPrefix eq 'std' ) {
                    $Fields->{$Value} = $DBVal;

                    if ( $Value =~ m{:} ) {
                        my ($Key, $Subkey) = split /:/, $Value;

                        my $RefType = ref $DBVal;
                        if ( $RefType && $RefType eq 'ARRAY' ) {
                            $DBVal = $DBVal->[0]->{value};
                        }
                        elsif ( $RefType && $RefType eq 'HASH' ) {
                            $DBVal = $DBVal->{value};
                        }

                        $Fields->{$Key} = { $Subkey => $DBVal };
                        delete $Fields->{$Value};
                    }
                }
                elsif ( $VPrefix eq 'cust' ) {
                    if ( $JIRAFields->{$Value} ) {
                        $Value = $JIRAFields->{$Value};
                    }

                    $Fields->{"customfield_" . $Value} = $DBVal;
                }
                elsif ( $VPrefix eq 'call' ) {

                    # calls must be specified as this: e.g. "MIME::Base64,encode_base64"
                    my ($Module, $Function) = split /,/, $Key;

                    my $Res = 0;
                    if ( $MainObject->Require( $Module ) ) {
                        my $Code = $Module->can( $Function );
                        $Res     = $Code->( $Self, $DBVal, $Fields, %Article );
                    }
                    else {
                        $LogObject->Log(
                            Priority => 'debug',
                            Message  => P . '::MapFields: Could not load ' . $Module,
                        );
                    } 

                    if ( $Self->{Debug} ) {
                        $LogObject->Log(
                            Priority => 'debug',
                            Message  => P . "::MapFields: call to '$Function' resulting in $Res\n",
                        );
                    }
                }
            }
            elsif ( $Self->{Debug} ) {
                $LogObject->Log(
                    Priority => 'debug',
                    Message  => P . "::MapFields: skipping: kprefix,key,vprefix,val,dbval : "
                        . "$KPrefix,$Key,$VPrefix,$Value,$DBVal\n",
                );
            }
        }
    }

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'debug',
            Message  => P . "::MapFields: fields: " . $MainObject->Dump($Fields) . "\n",
        );
    }

    $Fields = $Self->CheckFieldTypes( $Fields );

    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'debug',
            Message  => P . "::MapFields: fields (after type check): " . $MainObject->Dump($Fields) . "\n",
        );
    }

    return $Fields;
}

=item DateFormatTransform

This function transforms the date value of a otrs (dynamic/standard) field to the Jira format.

It receives the current format from the running system.

  my $Value1 = $Self->DateFormatTransform('%d/%b/%y %I:%M %p',   '2004-08-14 22:45:00')

returns 

  14/Aug/04 10:05 PM

or C<undef> upon failure

=cut

sub DateFormatTransform {
    my ($Self, $DateFormat, $Date) = @_;

    if ( !defined($Date) || ( $Date eq "" ) ) {
        return undef;
    }

    if ( length $Date < 11 ) {
        $Date .= ' 00:00:00';
    }

    my $Time = Time::Piece->strptime( $Date, '%Y-%m-%d %H:%M:%S' );
    if ( ! $Time ) {
        return undef;
    }

    return $Time->strftime('%Y-%m-%dT%H:%M:%S.000+0000');
}

=item CheckFieldTypes

This function ensures that the data transfered to the Jira system have the correct type.
E.g. fields of type "labels" have to be an array of string etc.

On March 10th, 2015 we agreed on those restrictions:

=over 4

=item * For fields of type "user" the "username" is set

=item * For fields of type "select" and the like the value is set

=item * Beside customfieldtypes only type "user" is supported at the moment

=back

=cut

sub CheckFieldTypes {
    my ($Self, $Fields) = @_;

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

    my %FieldsCopy = %{$Fields};

    my $FieldsInfo = $Self->FieldsGet(
        Return => 'Types',
    );

    my $StandardFields = $Self->FieldsGet(
        Return => 'Standard',
    );

    my %Dispatch = (
        string          => [ \&_String,                   ],
        url             => [ \&_String,                   ],
        array           => [ \&_Array,                    ],
        labels          => [ \&_Array,                    ],
        cascadingselect => [ \&_Array,                    ],
        multicheckboxes => [ \&_Array,                    ],
        user            => [ \&_Generic,       'name',    ],
        userpicker      => [ \&_Generic,       'name',    ],
        select          => [ \&_Generic,       'value'    ],
        multiselect     => [ \&_ArrayOfHashes, 'value'    ],
        radiobuttons    => [ \&_Generic,       'value'    ],
        number          => [ \&_Number,                   ],
        float           => [ \&_Number,                   ],
        datetime        => [ \&_Datetime,                 ],
        date            => [ \&_Datetime,      'short',   ],
        datepicker      => [ \&_Datetime,      'short',   ],
        textarea        => [ \&_String,                   ],
        textfield       => [ \&_String,                   ],
        default         => [ \&_String,                   ],
    );

    FIELD:
    for my $Field ( keys %FieldsCopy ) {

        next if first{ $Field eq $_ }@{ $StandardFields || [qw(project issuetype reporter assignee)] };

        my $Schema = $FieldsInfo->{$Field};
        my $Ref    = ref $FieldsCopy{$Field};
        my $Type   = $Schema->{custom} || $Schema->{type};

        if ( $Type && index( $Type, ':' ) != -1 ) {
            $Type = (split /:/, $Type)[-1];
        }

        # string is required and it is already a string - no transformation required
        next FIELD if $Type eq 'string' && !$Ref;

        # array is required and it is already an array - no transformation required
        next FIELD if $Type eq 'array' && $Ref && $Ref eq 'ARRAY';

        my @SubParams = @{ $Dispatch{$Type} || $Dispatch{default} };
        my $Sub       = shift @SubParams;

        if ( $Self->{Debug} ) {
            $LogObject->Log(
                Priority => 'debug',
                Message  => P . "::CheckFieldTypes: " . $MainObject->Dump([
                    $FieldsCopy{$Field}, $Field, $Ref, $Schema, $Type, "$Sub", \@SubParams
                ]) . "\n",
            );
        }

        next FIELD if !$Sub;

        $FieldsCopy{$Field} = $Self->$Sub( $FieldsCopy{$Field}, $Ref, @SubParams );
    }

    return \%FieldsCopy;
}

sub _String {
    my ($Self, $Value, $Ref) = @_;

    if ( $Ref eq 'HASH' ) {
        my ($Key) = sort keys %{$Value};
        return $Value->{$Key};
    }
    elsif ( $Ref eq 'ARRAY' ) {
        return join ', ', @{$Value};
    }

    return "$Value";
}

sub _Array {
    my ($Self, $Value, $Ref) = @_;

    if ( $Ref eq 'HASH' ) {
        my ($Key) = sort keys %{$Value};
        return [ $Value->{$Key} ];
    }

    return [ $Value ];
}

sub _ArrayOfHashes {
    my ($Self, $Value, $Ref, $Key) = @_;

    my $ArrayRef = [];

    if ( !ref $Value ) {
        push @{$ArrayRef}, +{ $Key => $Value };
    }

    if ( $Ref eq 'ARRAY' ) {
        for my $Item ( @{ $Value || [] } ) {
            if ( !ref $Item ) {
                push @{ $ArrayRef}, +{ $Key => $Item };
            }
            elsif ( ref $Item eq 'HASH' ) {
                push @{ $ArrayRef}, $Item;
            }
        }
    }

    return $ArrayRef;
}

sub _Generic {
    my ($Self, $Value, $Ref, $Key) = @_;

    if ( $Ref eq 'HASH' ) {
        if ( !exists $Value->{$Key} ) {
            my ($ExistingKey) = sort keys %{$Value};
            $Value = $Value->{$ExistingKey};
        }
        else {
            $Value = $Value->{$Key};
        }

        if ( !defined $Value ) {
            return undef;
            return +{ set => undef };
        }
    }
    elsif ( $Ref eq 'ARRAY' ) {
        return +{ $Key => join ', ', @{$Value} };
    }

    return +{ $Key => "$Value" };
}

sub _Number {
    my ($Self, $Value, $Ref) = @_;

    my $ValueCopy = $Value;
    if ( $Ref eq 'HASH' ) {
        my ($Key) = sort keys %{$Value};
        $ValueCopy = $Value->{$Key};
    }
    elsif ( $Ref eq 'ARRAY' ) {
        $ValueCopy = join ', ', @{$Value};
    }

    if ( ! looks_like_number( $ValueCopy ) ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Try to cast $ValueCopy to a number",
        );

        no warnings 'numeric';
        $ValueCopy += 0;
    }

    return $ValueCopy;
}

sub _Datetime {
    my ($Self, $Value, $Ref, $Type) = @_;

    my $ValueCopy = $Value;
    if ( $Ref eq 'HASH' ) {
        my ($Key) = sort keys %{$Value};
        $ValueCopy = $Value->{$Key};
    }
    elsif ( $Ref eq 'ARRAY' ) {
        $ValueCopy = join ' ', @{$Value};
    }

    my %Regexes = (
        'Long'  => qr/[0-9]{4}-[0-9]{2}-[0-9]{2}T[0-9]{2}:[0-9]{2}:[0-9]{2}\.[0-9]+\+[0-9]+/,
        'Short' => qr/[0-9]{4}-[0-9]{2}-[0-9]{2}/,
    );

    #my $Regex = $Regexes{$Type} || $Regexes{Long};
    my $Regex = $Regexes{Long};

    if ( $ValueCopy !~ $Regex ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "$ValueCopy is not a date",
        );
    }

    return $ValueCopy;
}

=item This function calls Jira an queries the license status for this installation.

my $status = $Self->CheckValidLicense();

returns:

0 upon failure
1 upon valid license
2 upon expired license

=cut

sub CheckValidLicense {
    my ($Self) = @_;

#    return 1;

    my $LogObject    = $Kernel::OM->Get('Kernel::System::Log');
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

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

    my $URL      = $Config->{URL};
    my $User     = $Config->{User};
    my $Password = $Config->{Password};

    ## maybe move this into a sub
    my $Headers = {
        Accept        => 'application/json',
# The latter has been disabled due to OTRSJIRA-81
#            Authorization => 'Basic '
#              . encode_base64( $user . ':' . $password ),
        "Content-type" => 'application/json',
    };

    my $Client = REST::Client->new();
    $Client->setFollow(1);

    my $VerifySSL = $ConfigObject->Get('JIRA::SSLVerify');

    if ( !$VerifySSL ) {
        my $Loaded = $Kernel::OM->Get('Kernel::System::Main')->Require(
            'Net::SSLeay',
            Silent => 1,
        );
        if ($Loaded) {
            $Client->getUseragent()->ssl_opts( verify_hostname => 0 );
            $Client->getUseragent()->ssl_opts( SSL_verfiy_mode => Net::SSLeay::VERIFY_NONE() );
        }
    }

    my ($Proto, undef, $Host, @BaseURL ) = split '/', $URL;
    my $GetURL = '/'. join('/', @BaseURL) . '/rest/otrs2jira/1.0/info';

    $GetURL =~ s{//}{/}g;
    $Client->setHost("$Proto//$Host");
    $Client->GET( $GetURL, $Headers );

 
    if ( $Self->{Debug} ) {
        $LogObject->Log(
            Priority => 'info',
            Message =>  P . "::CheckValidLicense: REST: host $Proto//$Host geturl $GetURL",
        );

        $LogObject->Log(
            Priority => 'info',
            Message  => sprintf P . "::CheckValidLicense: REST: responseCode %s, responseContent %s",
                $Client->responseCode(),
                $Client->responseContent(),
        );
    }

    # TODO: Fehlermeldung ausgeben wenn Fehler bei der Verbindung auftritt

    return int($Client->responseContent());
}

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

    my $ArticleFormat = $Param{ArticleFormat};

    return '' if !defined $ArticleFormat;

    my $LayoutObject;
    {
        local $Kernel::OM = Kernel::System::ObjectManager->new();
        $LayoutObject     = $Kernel::OM->Get('Kernel::Output::HTML::Layout');
    }

    my $Text = $LayoutObject->Output(
        Template => $ArticleFormat,
        Data     => $Param{Article},
    );

    return $Text;
}


1;


LS0tCkRlYnVnZ2VyOgogIERlYnVnVGhyZXNob2xkOiBkZWJ1ZwogIFRlc3RNb2RlOiAnMCcKRGVzY3JpcHRpb246IFRpY2tldCBDb25uZWN0b3IgUkVTVApGcmFtZXdvcmtWZXJzaW9uOiA0LjAuNQpQcm92aWRlcjoKICBPcGVyYXRpb246CiAgICBTZXNzaW9uQ3JlYXRlOgogICAgICBEZXNjcmlwdGlvbjogQ3JlYXRlcyBhIFNlc3Npb24KICAgICAgTWFwcGluZ0luYm91bmQ6IHt9CiAgICAgIE1hcHBpbmdPdXRib3VuZDoge30KICAgICAgVHlwZTogU2Vzc2lvbjo6U2Vzc2lvbkNyZWF0ZQogICAgVGlja2V0Q3JlYXRlOgogICAgICBEZXNjcmlwdGlvbjogQ3JlYXRlcyBhIFRpY2tldAogICAgICBNYXBwaW5nSW5ib3VuZDoge30KICAgICAgTWFwcGluZ091dGJvdW5kOiB7fQogICAgICBUeXBlOiBUaWNrZXQ6OlRpY2tldENyZWF0ZQogICAgVGlja2V0R2V0OgogICAgICBEZXNjcmlwdGlvbjogUmV0cmlldmVzIFRpY2tldCBkYXRhCiAgICAgIE1hcHBpbmdJbmJvdW5kOiB7fQogICAgICBNYXBwaW5nT3V0Ym91bmQ6IHt9CiAgICAgIFR5cGU6IFRpY2tldDo6VGlja2V0R2V0CiAgICBUaWNrZXRTZWFyY2g6CiAgICAgIERlc2NyaXB0aW9uOiBTZWFyY2ggZm9yIFRpY2tldHMKICAgICAgTWFwcGluZ0luYm91bmQ6IHt9CiAgICAgIE1hcHBpbmdPdXRib3VuZDoge30KICAgICAgVHlwZTogVGlja2V0OjpUaWNrZXRTZWFyY2gKICAgIFRpY2tldFVwZGF0ZToKICAgICAgRGVzY3JpcHRpb246IFVwZGF0ZXMgYSBUaWNrZXQKICAgICAgTWFwcGluZ0luYm91bmQ6IHt9CiAgICAgIE1hcHBpbmdPdXRib3VuZDoge30KICAgICAgVHlwZTogVGlja2V0OjpUaWNrZXRVcGRhdGUKICBUcmFuc3BvcnQ6CiAgICBDb25maWc6CiAgICAgIEtlZXBBbGl2ZTogJycKICAgICAgTWF4TGVuZ3RoOiAnMTAwMDAwMDAwJwogICAgICBSb3V0ZU9wZXJhdGlvbk1hcHBpbmc6CiAgICAgICAgU2Vzc2lvbkNyZWF0ZToKICAgICAgICAgIFJlcXVlc3RNZXRob2Q6CiAgICAgICAgICAtIFBPU1QKICAgICAgICAgIFJvdXRlOiAvU2Vzc2lvbgogICAgICAgIFRpY2tldENyZWF0ZToKICAgICAgICAgIFJlcXVlc3RNZXRob2Q6CiAgICAgICAgICAtIFBPU1QKICAgICAgICAgIFJvdXRlOiAvVGlja2V0CiAgICAgICAgVGlja2V0R2V0OgogICAgICAgICAgUmVxdWVzdE1ldGhvZDoKICAgICAgICAgIC0gR0VUCiAgICAgICAgICBSb3V0ZTogL1RpY2tldC86VGlja2V0SUQKICAgICAgICBUaWNrZXRTZWFyY2g6CiAgICAgICAgICBSZXF1ZXN0TWV0aG9kOgogICAgICAgICAgLSBHRVQKICAgICAgICAgIFJvdXRlOiAvVGlja2V0CiAgICAgICAgVGlja2V0VXBkYXRlOgogICAgICAgICAgUmVxdWVzdE1ldGhvZDoKICAgICAgICAgIC0gUEFUQ0gKICAgICAgICAgIFJvdXRlOiAvVGlja2V0LzpUaWNrZXRJRAogICAgVHlwZTogSFRUUDo6UkVTVApSZW1vdGVTeXN0ZW06ICcnClJlcXVlc3RlcjoKICBUcmFuc3BvcnQ6CiAgICBUeXBlOiAnJwo=
Ly8gLS0KLy8gSklSQS5BZG1pbldpemFyZC5qcyAtIHByb3ZpZGVzIHRoZSBzcGVjaWFsIG1vZHVsZSBmdW5jdGlvbnMgZm9yIHRoZSBDdXN0b21lcklEIHNlYXJjaAovLyBDb3B5cmlnaHQgKEMpIDIwMTYgUGVybC1TZXJ2aWNlcy5kZSwgaHR0cDovL3Blcmwtc2VydmljZXMuZGUKLy8gLS0KLy8gVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKLy8gdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQovLyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgovLyAtLQoKInVzZSBzdHJpY3QiOwoKdmFyIEpJUkEgPSBKSVJBIHx8IHt9OwoKLyoqCiAqIEBuYW1lc3BhY2UKICogQGV4cG9ydHMgVGFyZ2V0TlMgYXMgSklSQS5BZG1pbldpemFyZAogKiBAZGVzY3JpcHRpb24KICogICAgICBUaGlzIG5hbWVzcGFjZSBjb250YWlucyB0aGUgc3BlY2lhbCBtb2R1bGUgZnVuY3Rpb25zIGZvciB0aGUgc2VhcmNoLgogKi8KSklSQS5BZG1pbldpemFyZCA9IChmdW5jdGlvbiAoVGFyZ2V0TlMpIHsKCiAgICAvKioKICAgICAqIEBmdW5jdGlvbgogICAgICogQHBhcmFtIHtqUXVlcnlPYmplY3R9ICRJbnB1dCBJbnB1dCBlbGVtZW50IHRvIGFkZCBhdXRvIGNvbXBsZXRlIHRvCiAgICAgKiBAcGFyYW0ge1N0cmluZ30gU3ViYWN0aW9uIFN1YmFjdGlvbiB0byBleGVjdXRlLCAiU2VhcmNoQ3VzdG9tZXJJRCIgb3IgIlNlYXJjaEN1c3RvbWVyVXNlciIKICAgICAqIEByZXR1cm4gbm90aGluZwogICAgICovCiAgICBUYXJnZXROUy5Jbml0U2VsZWN0ID0gZnVuY3Rpb24gKEZpZWxkTmFtZSkgewogICAgICAgICQoJyMnICsgRmllbGROYW1lKS5iaW5kKCAnY2hhbmdlJywgZnVuY3Rpb24oKSB7CiAgICAgICAgICAgICQoJ2RpdltjbGFzcyo9IicgKyBGaWVsZE5hbWUgKyAnLSJdJykuYWRkQ2xhc3MoJ0hpZGRlbicpOwoKICAgICAgICAgICAgdmFyIFN1YkNsYXNzID0gJCh0aGlzKS52YWwoKTsKCiAgICAgICAgICAgICQoJy4nICsgRmllbGROYW1lICsgJy0nICsgU3ViQ2xhc3MpLnJlbW92ZUNsYXNzKCdIaWRkZW4nKTsKICAgICAgICB9KTsKCiAgICAgICAgJCgnIycgKyBGaWVsZE5hbWUpLnRyaWdnZXIoICdjaGFuZ2UnICk7CiAgICB9OwoKICAgIHJldHVybiBUYXJnZXROUzsKCn0pKCBKSVJBLkFkbWluV2l6YXJkIHx8IHt9ICk7CgoK
Ly8gLS0KLy8gSklSQS5BdXRvY29tcGxldGUuanMgLSBwcm92aWRlcyB0aGUgc3BlY2lhbCBtb2R1bGUgZnVuY3Rpb25zIGZvciB0aGUgQ3VzdG9tZXJJRCBzZWFyY2gKLy8gQ29weXJpZ2h0IChDKSAyMDE2IFBlcmwtU2VydmljZXMuZGUsIGh0dHA6Ly9wZXJsLXNlcnZpY2VzLmRlCi8vIC0tCi8vIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCi8vIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKLy8gZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KLy8gLS0KCiJ1c2Ugc3RyaWN0IjsKCnZhciBKSVJBID0gSklSQSB8fCB7fTsKCi8qKgogKiBAbmFtZXNwYWNlCiAqIEBleHBvcnRzIFRhcmdldE5TIGFzIEpJUkEuQXV0b2NvbXBsZXRlCiAqIEBkZXNjcmlwdGlvbgogKiAgICAgIFRoaXMgbmFtZXNwYWNlIGNvbnRhaW5zIHRoZSBzcGVjaWFsIG1vZHVsZSBmdW5jdGlvbnMgZm9yIHRoZSBzZWFyY2guCiAqLwpKSVJBLkF1dG9jb21wbGV0ZSA9IChmdW5jdGlvbiAoVGFyZ2V0TlMpIHsKCiAgICAvKioKICAgICAqIEBmdW5jdGlvbgogICAgICogQHBhcmFtIHtqUXVlcnlPYmplY3R9ICRJbnB1dCBJbnB1dCBlbGVtZW50IHRvIGFkZCBhdXRvIGNvbXBsZXRlIHRvCiAgICAgKiBAcGFyYW0ge1N0cmluZ30gU3ViYWN0aW9uIFN1YmFjdGlvbiB0byBleGVjdXRlLCAiU2VhcmNoQ3VzdG9tZXJJRCIgb3IgIlNlYXJjaEN1c3RvbWVyVXNlciIKICAgICAqIEByZXR1cm4gbm90aGluZwogICAgICovCiAgICBUYXJnZXROUy5Jbml0ID0gZnVuY3Rpb24gKCRJbnB1dCwgU3ViYWN0aW9uLCBUYXJnZXRGaWVsZCkgewogICAgICAgIHZhciBLZXl3b3JkID0gJ0Fzc2lnbmVlJzsKCiAgICAgICAgaWYgKCBUYXJnZXRGaWVsZCAmJiBUYXJnZXRGaWVsZCAhPSAnJyApIHsKICAgICAgICAgICAgS2V5d29yZCA9IFRhcmdldEZpZWxkOwogICAgICAgIH0KCiAgICAgICAgQ29yZS5VSS5BdXRvY29tcGxldGUuSW5pdCgkSW5wdXQsIGZ1bmN0aW9uIChSZXF1ZXN0LCBSZXNwb25zZSkgewogICAgICAgICAgICAgICAgdmFyIFVSTCA9IENvcmUuQ29uZmlnLkdldCgnQmFzZWxpbmsnKSwgRGF0YSA9IHsKICAgICAgICAgICAgICAgICAgICBBY3Rpb246ICdBZ2VudFRpY2tldEpJUkFEaWFsb2cnLAogICAgICAgICAgICAgICAgICAgIFN1YmFjdGlvbjogU3ViYWN0aW9uLAogICAgICAgICAgICAgICAgICAgIFByb2plY3Q6ICQoJyNQcm9qZWN0JykudmFsKCksCiAgICAgICAgICAgICAgICAgICAgVGVybTogUmVxdWVzdC50ZXJtCiAgICAgICAgICAgICAgICB9OwoKICAgICAgICAgICAgICAgICRJbnB1dC5kYXRhKCdBdXRvQ29tcGxldGVYSFInLCBDb3JlLkFKQVguRnVuY3Rpb25DYWxsKFVSTCwgRGF0YSwgZnVuY3Rpb24gKFJlc3VsdCkgewogICAgICAgICAgICAgICAgICAgIHZhciBEYXRhID0gW107CiAgICAgICAgICAgICAgICAgICAgJElucHV0LnJlbW92ZURhdGEoJ0F1dG9Db21wbGV0ZVhIUicpOwogICAgICAgICAgICAgICAgICAgICQuZWFjaChSZXN1bHQsIGZ1bmN0aW9uICgpIHsKICAgICAgICAgICAgICAgICAgICAgICAgRGF0YS5wdXNoKHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxhYmVsOiB0aGlzLkxhYmVsLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgdmFsdWU6IHRoaXMuVmFsdWUKICAgICAgICAgICAgICAgICAgICAgICAgfSk7CiAgICAgICAgICAgICAgICAgICAgfSk7CiAgICAgICAgICAgICAgICAgICAgUmVzcG9uc2UoRGF0YSk7CiAgICAgICAgICAgICAgICB9KSk7CiAgICAgICAgfSwgZnVuY3Rpb24gKEV2ZW50LCBVSSkgewogICAgICAgICAgICAkKCcjJyArIEtleXdvcmQgKS52YWwoVUkuaXRlbS52YWx1ZSk7CiAgICAgICAgICAgICQoJyMnICsgS2V5d29yZCArICdTZWFyY2hGaWVsZCcpLnZhbChVSS5pdGVtLmxhYmVsKTsKICAgICAgICB9LCBLZXl3b3JkICsgJ1NlYXJjaCcpOwogICAgfTsKCiAgICByZXR1cm4gVGFyZ2V0TlM7Cgp9KSggSklSQS5BdXRvY29tcGxldGUgfHwge30gKTsKCgo=
# --
# Copyright (C) 2015 - 2018 catworkx GmbH, http://catworkx.de
# --
# 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 var::packagesetup::JIRAPackage;

use strict;
use warnings;

use utf8;

use List::Util qw(first);
use File::Basename;

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

our @ObjectDependencies = qw(
    Kernel::Config
    Kernel::System::SysConfig
    Kernel::System::GenericInterface::Webservice
    Kernel::System::YAML
    Kernel::System::Valid
    Kernel::System::DynamicField
);

=head1 NAME

var::packagesetup::JIRAPackage.pm - code to excecute during package installation

=head1 SYNOPSIS

All functions

=head1 PUBLIC INTERFACE

=over 4

=cut

=item new()

create an object

=cut

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

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

    return $Self;
}

=item CodeInstall()

run the code install part

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

=cut

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

    # create dynamic fields 
    $Self->_InstallWebservice();
    $Self->_CreateDynamicFields();
    $Self->_ActivateDynamicFields();

    return 1;
}

=item CodeReinstall()

run the code reinstall part

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

=cut

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

    return 1;
}

=item CodeUpgrade()

run the code upgrade part

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

=cut

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

    $Self->_InstallWebservice();
    $Self->_CreateDynamicFields();
    $Self->_ActivateDynamicFields();

    return 1;
}

=item CodeUninstall()

run the code uninstall part

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

=cut

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

    return 1;
}

=item _InstallWebservice()

=cut

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

    my $YAMLObject = $Kernel::OM->Get('Kernel::System::YAML');
    my $Location   = join '/',
        $Kernel::OM->Get('Kernel::Config')->Get('Home'),
        qw/var GenericInterface GenericTicketConnectorREST.yml/;

    my $Content = $Kernel::OM->Get('Kernel::System::Main')->FileRead(
        Location => $Location,
    );

    my $ImportedConfig = $YAMLObject->Load( Data => ${$Content} );

    # display any YAML error message as a normal otrs error message
    if ( !IsHashRefWithData($ImportedConfig) ) {
        return;
    }

    # check if imported configuration has current framework version otherwise update it
    #if ( $ImportedConfig->{FrameworkVersion} ne $Self->{FrameworkVersion} ) {
    #    $ImportedConfig = $Self->_UpdateConfiguration( Configuration => $ImportedConfig );
    #}

    # remove framework information since is not needed anymore
    delete $ImportedConfig->{FrameworkVersion};

    # get webservice name
    my $WebserviceName   = basename $Location;
    my $WebserviceObject = $Kernel::OM->Get('Kernel::System::GenericInterface::Webservice');

    # remove file extension
    $WebserviceName =~ s{\.[^.]+$}{}g;

    # check if name is duplicated
    my %WebserviceList = reverse %{ $WebserviceObject->WebserviceList() };

    return if $WebserviceList{ $WebserviceName };

    # otherwise save configuration and return to overview screen
    my $Success = $WebserviceObject->WebserviceAdd(
        Name    => $WebserviceName,
        Config  => $ImportedConfig,
        ValidID => 1,
        UserID  => 1,
    );
}

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

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

    my $ValidID = $ValidObject->ValidLookup(
        Valid => 'valid',
    );

    # get all current dynamic fields
    my $DynamicFieldList = $DynamicFieldObject->DynamicFieldListGet(
        Valid => 0,
    );

    # get the list of order numbers (is already sorted).
    my @DynamicfieldOrderList;
    for my $Dynamicfield ( @{$DynamicFieldList} ) {
        push @DynamicfieldOrderList, $Dynamicfield->{FieldOrder};
    }

    # get the last element from the order list and add 1
    my $NextOrderNumber = 1;
    if (@DynamicfieldOrderList) {
        $NextOrderNumber = $DynamicfieldOrderList[-1] + 1;
    }

    # get the definition for all dynamic fields for ITSM
    my @DynamicFields = $Self->_GetDynamicFieldsDefinition();

    # create a dynamic fields lookup table
    my %DynamicFieldLookup;
    for my $DynamicField ( @{$DynamicFieldList} ) {
        next if ref $DynamicField ne 'HASH';
        $DynamicFieldLookup{ $DynamicField->{Name} } = $DynamicField;
    }

    # create or update dynamic fields
    DYNAMICFIELD:
    for my $DynamicField (@DynamicFields) {
        my $CreateDynamicField;

        # check if the dynamic field already exists
        if ( ref $DynamicFieldLookup{ $DynamicField->{Name} } ne 'HASH' ) {
            $CreateDynamicField = 1;
        }

        # if the field exists check if the type match with the needed type
        elsif (
            $DynamicFieldLookup{ $DynamicField->{Name} }->{FieldType}
            ne $DynamicField->{FieldType}
            )
        {

            # rename the field and create a new one
            my $Success = $DynamicFieldObject->DynamicFieldUpdate(
                %{ $DynamicFieldLookup{ $DynamicField->{Name} } },
                Name   => $DynamicFieldLookup{ $DynamicField->{Name} }->{Name} . 'Old',
                UserID => 1,
            );

            $CreateDynamicField = 1;
        }

        # otherwise if the field exists and the type match, update it to the ITSM definition
        else {
#            my $Success = $DynamicFieldObject->DynamicFieldUpdate(
#                %{$DynamicField},
#                ID         => $DynamicFieldLookup{ $DynamicField->{Name} }->{ID},
#                FieldOrder => $DynamicFieldLookup{ $DynamicField->{Name} }->{FieldOrder},
#                ValidID    => $ValidID,
#                Reorder    => 0,
#                UserID     => 1,
#            );
        }

        # check if new field has to be created
        if ($CreateDynamicField) {
            # create a new field
            my $FieldID = $DynamicFieldObject->DynamicFieldAdd(
                Name       => $DynamicField->{Name},
                Label      => $DynamicField->{Label},
                FieldOrder => $NextOrderNumber,
                FieldType  => $DynamicField->{FieldType},
                ObjectType => $DynamicField->{ObjectType},
                Config     => $DynamicField->{Config},
                ValidID    => $ValidID,
                UserID     => 1,
            );
            next DYNAMICFIELD if !$FieldID;

            # increase the order number
            $NextOrderNumber++;
        }

    }

    return 1;
}

=item _GetDynamicFieldsDefinition()

returns the definition for reporting related dynamic fields

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

=cut

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

    # define all dynamic fields for reporting
    my @DynamicFields = (
        {
            Name       => 'JIRAIssueID',
            Label      => 'Jira Issue ID',
            FieldType  => 'Text',
            ObjectType => 'Ticket',
            Config     => {
                DefaultValue  => '',
                Link          => 'http://localhost:8080/browse/[% Data.JIRAIssueID | url %]',
            },
        },
        {
            Name       => 'ArticleSentToJIRA',
            Label      => 'Article was sent to Jira',
            FieldType  => 'Checkbox',
            ObjectType => 'Article',
            Config     => {
                DefaultValue  => 0,
            },
        },
    );

    return @DynamicFields;
}

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


    my %Map = (
        JIRAIssueID => {
            'Zoom'     => 1,
            'FreeText' => 1,
            'Search'   => 1,
        },
    );

    my $ConfigObject    = $Kernel::OM->Get('Kernel::Config');
    my $SysConfigObject = $Kernel::OM->Get('Kernel::System::SysConfig');

    my @Settings;
    for my $Field ( keys %Map ) {
        for my $Screen ( keys %{ $Map{$Field} } ) {
            my $Fullname      = 'Ticket::Frontend::AgentTicket' . $Screen;
            my $Options       = $ConfigObject->Get( $Fullname );
            my $Mapping       = $Options->{DynamicField} || {};

            my %NewMapping = (
                %{ $Mapping },
                $Field => $Map{$Field}->{$Screen},
            );

            push @Settings, {
                Name           => "$Fullname###DynamicField",
                EffectiveValue => \%NewMapping,
            };
        }
    }

    my $Success = $SysConfigObject->SettingsSet(
        Settings => \@Settings,
        UserID   => 1,
    );

    return 1;
}

1;

=back

=head1 TERMS AND CONDITIONS

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<http://www.gnu.org/licenses/gpl-2.0.txt>.

=cut

