ITSMCore
2.0.93
OTRS AG
http://otrs.org/
GNU AFFERO GENERAL PUBLIC LICENSE Version 3, November 2007
Build for OTRS::ITSM 2.1.0 beta3.
Build for OTRS::ITSM 2.1.0 beta2.
Build for OTRS::ITSM 2.1.0 beta1.
Build for OTRS::ITSM 2.1.0 alpha1.
The OTRS::ITSM Core package.
Das OTRS::ITSM Core Paket.
2.4.x
<br>
<b>WELCOME</b>
<br>
<br>
You are about to install the OTRS package ITSMCore.<br>
<br>
<br>
<b>REQUIRED OTRS PACKAGES</b>
<ul>
<li>GeneralCatalog 2.0.93</li>
</ul>
<br>
<b>ATTENTION</b>
<br>
<br>
Make sure your database accepts packages over 5 MB in size. A MySQL database for example accepts packages up to 1 MB by default. In this case, the value for max_allowed_packet must be increased. The recommended maximum size accepted is 20 MB.<br>
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>WILLKOMMEN</b>
<br>
<br>
Sie sind im Begriff das OTRS-Paket ITSMCore zu installieren.<br>
<br>
<br>
<b>BENÖTIGTE OTRS-PAKETE</b>
<ul>
<li>GeneralCatalog 2.0.93</li>
</ul>
<br>
<b>ACHTUNG</b>
<br>
<br>
Vergewissern Sie sich, dass ihre Datenbank Pakete mit einer Größe von mehr als 5 MB akzeptiert. Eine MySQL Datenbank akzeptiert beispielsweise standardmäßig Pakete mit einer Größe von maximal 1 MB. In diesem Fall muss der Wert für max_allowed_packet erhöht werden. Empfohlen werden 20 MB.<br>
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>NOTICE</b>
<br>
<br>
In order to grant users access to the service menu, you need to add them as member to the group 'itsm-service'.
<br>
<br>
The menu items that were added by this package will be visible after you log-in to the system again.
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>HINWEIS</b>
<br>
<br>
Um Benutzern Zugriff auf das Service-Menü zu gewähren, müssen diese Mitglied der neuen Gruppe 'itsm-service' sein.
<br>
<br>
Die von diesem Paket hinzugefügten Menü-Punkte sind erst nach einem erneuten Anmeldevorgang im System sichtbar.
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>ATTENTION</b>
<br>
<br>
If you uninstall this package, all database tables that were created during installation will be deleted.
All data from these tables will be irrevocably lost!
<br>
<br>
The group 'itsm-service' that was created during package installation will be deactivated.
You can activate this group again in the admin area.
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>ACHTUNG</b>
<br>
<br>
Bei der Deinstallation werden die von diesem Paket angelegten Datenbank-Tabellen gelöscht.
Alle darin enthaltenen Daten gehen unwiderruflich verloren!
<br>
<br>
Die von diesem Paket angelegte Gruppe 'itsm-service' wird deaktiviert.
Sie kann jederzeit im Admin-Bereich wieder aktiviert werden.
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>WELCOME</b>
<br>
<br>
You are about to upgrade the OTRS package ITSMCore.<br>
<br>
<br>
<b>REQUIRED OTRS PACKAGES</b>
<ul>
<li>GeneralCatalog 2.0.93</li>
</ul>
<br>
</b>ATTENTION</b>
<br>
<br>
Make sure your database accepts packages over 5 MB in size. A MySQL database for example accepts packages up to 1 MB by default. In this case, the value for max_allowed_packet must be increased. The recommended maximum size accepted is 20 MB.<br>
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>WILLKOMMEN</b>
<br>
<br>
Sie sind im Begriff das OTRS-Paket ITSMCore zu aktualisieren.<br>
<br>
<br>
<b>BENÖTIGTE OTRS-PAKETE</b>
<ul>
<li>GeneralCatalog 2.0.93</li>
</ul>
<br>
<b>ACHTUNG</b>
<br>
<br>
Vergewissern Sie sich, dass ihre Datenbank Pakete mit einer Größe von mehr als 5 MB akzeptiert. Eine MySQL Datenbank akzeptiert beispielsweise standardmäßig Pakete mit einer Größe von maximal 1 MB. In diesem Fall muss der Wert für max_allowed_packet erhöht werden. Empfohlen werden 20 MB.<br>
<br>
<br>
((enjoy))<br>
<br>
GeneralCatalog
# define function name
my $FunctionName = 'CodeInstall';
# create the package name
my $CodeModule = 'var::packagesetup::' . $Param{Structure}->{Name}->{Content};
# load the module
if ( $Self->{MainObject}->Require($CodeModule) ) {
# create new instance
my $CodeObject = $CodeModule->new( %{$Self} );
if ($CodeObject) {
# start methode
if ( !$CodeObject->$FunctionName(%{$Self}) ) {
$Self->{LogObject}->Log(
Priority => 'error',
Message => "Could not call method $FunctionName() on $CodeModule.pm."
);
}
}
# error handling
else {
$Self->{LogObject}->Log(
Priority => 'error',
Message => "Could not call method new() on $CodeModule.pm."
);
}
}
# define function name
my $FunctionName = 'CodeUpgrade';
# create the package name
my $CodeModule = 'var::packagesetup::' . $Param{Structure}->{Name}->{Content};
# The code module has just recently been copied to it's location in the file system.
# In a persistent Perl environment an old version of the module might still be loaded,
# as watchdogs like Apache2::Reload haven't had a chance to reload it.
# So we need to make sure that the new version is being loaded.
# Kernel::System::Main::Require() checks the relative file path, so we need to remove that from %INC.
# This is only needed in persistent Perl environment, but does no harm in a CGI environment.
my $CodeModulePath = $CodeModule;
$CodeModulePath =~ s/::/\//g;
$CodeModulePath .= '.pm';
delete $INC{$CodeModulePath};
# load the module
if ( $Self->{MainObject}->Require($CodeModule) ) {
# create new instance
my $CodeObject = $CodeModule->new( %{$Self} );
if ($CodeObject) {
# start methode
if ( !$CodeObject->$FunctionName(%{$Self}) ) {
$Self->{LogObject}->Log(
Priority => 'error',
Message => "Could not call method $FunctionName() on $CodeModule.pm."
);
}
}
# error handling
else {
$Self->{LogObject}->Log(
Priority => 'error',
Message => "Could not call method new() on $CodeModule.pm."
);
}
}
# define function name
my $FunctionName = 'CodeUninstall';
# create the package name
my $CodeModule = 'var::packagesetup::' . $Param{Structure}->{Name}->{Content};
# load the module
if ( $Self->{MainObject}->Require($CodeModule) ) {
# create new instance
my $CodeObject = $CodeModule->new( %{$Self} );
if ($CodeObject) {
# start methode
if ( !$CodeObject->$FunctionName(%{$Self}) ) {
$Self->{LogObject}->Log(
Priority => 'error',
Message => "Could not call method $FunctionName() on $CodeModule.pm."
);
}
}
# error handling
else {
$Self->{LogObject}->Log(
Priority => 'error',
Message => "Could not call method new() on $CodeModule.pm."
);
}
}
# define function name
my $FunctionName = 'CodeReinstall';
# create the package name
my $CodeModule = 'var::packagesetup::' . $Param{Structure}->{Name}->{Content};
# load the module
if ( $Self->{MainObject}->Require($CodeModule) ) {
# create new instance
my $CodeObject = $CodeModule->new( %{$Self} );
if ($CodeObject) {
# start methode
if ( !$CodeObject->$FunctionName(%{$Self}) ) {
$Self->{LogObject}->Log(
Priority => 'error',
Message => "Could not call method $FunctionName() on $CodeModule.pm."
);
}
}
# error handling
else {
$Self->{LogObject}->Log(
Priority => 'error',
Message => "Could not call method new() on $CodeModule.pm."
);
}
}
2010-09-07 16:21:36
opms.otrs.com
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vLi+vv6pT30qdCEUZH19/bu/+7tDVwHzT7QEABRhb2+vXq/X6/XQhcyKriUov+/6ru96/etf//Wvfz10IRTkc5/73Kc//enQVcD888UaAFCEl19++bnnntva2gpdyKyk0ZKuJQBg0ehaAgBm7tatWzs7O7VabWVlJXQts2JBHFRCq9Xa2NgIXQUFuX37dugSYCGIlgCA2Wq1WlevXo2i6MSJE43G3J57WBAH5RfH8X//93/fd999oQuhOD/xEz8RugSYf3N7egcAlMQrr7zSarWWl5dPnDgRupYZ0rUE5fcDP/ADzWYzdBUA88bZDwAwQ7dv375582YURadPn57vjh5dSwDAYhItAQCz0m63r1y5EkXR2tra6upq6HJmyzbeAMBicvYDAMzKtWvX9vf3G43GyZMnQ9cycxbEAQCLydkPADATOzs7N27ciKLo9OnTixC4WBAHACym+T/PAwCKlyRJuhTu2LFjR44cCV1OESyIAwAWk7MfAGD6rl+/vru7W6/XT506FbqWglgQBwAsJmc/AMCU7e3tra+vR1F06tSper0eupyC1Ov15eXlxXm9AACpRugCAIB58/LLLydJcuTIkWPHjoWupTinT58OXQIAQAC6lgCAabpx48bOzk6tVhO1AAAsAtESADA1u7u7r7zyShRFJ0+ebDQ0RwMAzD/REgAwHUmSXL58OV0Kt7a2FrocAACKIFoCAKbj6tWre3t7jUbj/vvvD10LAAAFES0BAFNw+/btjY2NKIruv/9+V0kDAFgcNkEAAKbg0KFDx44dazQahw8fDl0LAADFES0BAFNQq9XOnDmTJEnoQgAAKJQFcQDA1MRxHLoEAAAKpWsJAGAKLl++3Gw2T506tbKyEroWAIDiiJYAAKZgZ2dnf3+/3W6HLgQAoFCiJQCAKTh9+nSr1VpeXg5dCABAoURLAABT4NJ4AMBiso03ADC2nZ2dVqsVugoAAMLTtQQAjKfZbF68eDFJkoceesiW1QAAC060BACMbWlpKUmSpaWl0IUAABCYaAkAGE+j0XjNa17TbDZrNSvrAQAWnTNCAGASjYYvqAAAEC0BAAAAMCnREgAAAAAT0soOADAF29vbURQdOnQojuPQtQAAFEfXEgDAFJw/f/78+fOtVit0IQAAhRItAQBZNjY22u126CoqQ8sSALBoREsAwFAbGxtXrlx58cUX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H3/88enTp6oLAYAjJVoCAGCPTSaTJEkGg0HVhQDAkRItAQCwxzqdThAEoiUAqIpoCQCAPZZHS0mSTCaTqmsBgGMkWgIAYI9FUdRqtYIgGA6HVdcCAMdItAQAwH7TEwcAFRItAQCw30RLAFAh0RIAAPut3W4HQTAajdI0rboWADg6oiUAAPZbHMfNZjMwcQkAqiBaAgBg7+mJA4CqiJYAANh7oiUAqIpoCQCAvZdHS6PRKMuyqmsBgOMiWgIAYO81m804jrMsGw6HVdcCAMdFtAQAwCHI3xOnJw4Adky0BADAIbDcEgBUQrQEAMAhyKOl4XBouSUA2CXREgAAh6DVakVRlKbpaDSquhYAOCKiJQAADsRs4lLVhQDAEREtAQBwICy3BAC7J1oCAOBAeEkcAOxeXHUBAACwGe12u9Pp9Hq9LMvCMKy6HAA4CqE3aAAAAABQjoY4AAAAAEoSLQEAAABQkmgJAAAAgJJESwAAAACUJFoCAAAAoCTREgAAAAAl/Q/GAOyVx0UjDQAAAABJRU5ErkJggg==
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSIgPz4KPG90cnNfY29uZmlnIHZlcnNpb249IjEuMCIgaW5pdD0iRnJhbWV3b3JrIj4KICAgIDxDVlM+JElkOiBGcmFtZXdvcmtJVFNNQ29yZS54bWwsdiAxLjQ3LjIuMiAyMDEwLzA5LzA3IDEzOjU3OjA4IHViIEV4cCAkPC9DVlM+CiAgICA8Q1ZTPiRPbGRJZDogRnJhbWV3b3JrLnhtbCx2IDEuMjQyLjIuNiAyMDEwLzAzLzA5IDA5OjM3OjQ2IGR6IEV4cCAkPC9DVlM+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJQcm9kdWN0TmFtZSIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlRoaXMgc2V0dGluZyBjb250cm9scyB0aGUgbmFtZSBvZiB0aGUgYXBwbGljYXRpb24gYXMgaXMgc2hvd24gaW4gdGhlIHdlYiBpbnRlcmZhY2UgYXMgd2VsbCBhcyB0aGUgdGFicyBhbmQgdGl0bGUgYmFyIG9mIHlvdXIgd2ViIGJyb3dzZXIuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkltIFdlYkZyb250ZW5kIGFuZ2V6ZWlndGVyIE5hbWUgZGVyIFNvZnR3YXJlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZyYW1ld29yazwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPk9UUlM6OklUU00gMi4xIEJldGEgMzwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkxpbmtPYmplY3Q6OlZpZXdNb2RlIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGV0ZXJtaW5lcyB0aGUgd2F5IHRoZSBsaW5rZWQgb2JqZWN0cyBhcmUgZGlzcGxheWVkIGluIGVhY2ggem9vbSBtYXNrLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5MZWd0IGRpZSBBbnNpY2h0IGRlciB2ZXJsaW5rdGVuIE9iamVrdGUgaW4gZGVuIGpld2VpbGlnZW4gWm9vbS1NYXNrZW4gZmVzdC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5GcmFtZXdvcms8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Db3JlOjpMaW5rT2JqZWN0PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPE9wdGlvbiBTZWxlY3RlZElEPSJDb21wbGV4Ij4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iU2ltcGxlIj5TaW1wbGU8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IkNvbXBsZXgiPkNvbXBsZXg8L0l0ZW0+CiAgICAgICAgICAgIDwvT3B0aW9uPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlBhY2thZ2U6OlJlcG9zaXRvcnlMaXN0IiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+TGlzdCBvZiBvbmxpbmUgcmVwb3NpdG9yaWVzIChmb3IgZXhhbXBsZSB5b3UgYWxzbyBjYW4gdXNlIG90aGVyIGluc3RhbGxhdGlvbnMgYXMgcmVwb3NpdG9yaXkgYnkgdXNpbmcgS2V5PSJodHRwOi8vZXhhbXBsZS5jb20vb3Rycy9wdWJsaWMucGw/QWN0aW9uPVB1YmxpY1JlcG9zaXRvcnkmYW1wO0ZpbGU9IiBhbmQgQ29udGVudD0iU29tZSBOYW1lIikuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkxpc3RlIGRlciB6dXIgVmVyZvxndW5nIHN0ZWhlbmRlbiBPbmxpbmUtUXVlbGxlbiAoZXMga/ZubmVuIHouIEIuIGF1Y2ggYW5kZXJlIEluc3RhbGxhdGlvbmVuIGFscyBPbmxpbmUtUXVlbGxlbiB2ZXJ3ZW5kZXQgd2VyZGVuIG1pdCBkZXIgVmVyd2VuZHVuZyB2b24gS2V5PSJodHRwOi8vZXhhbXBsZS5jb20vb3Rycy9wdWJsaWMucGw/QWN0aW9uPVB1YmxpY1JlcG9zaXRvcnkmYW1wO0ZpbGU9IiBhbmQgQ29udGVudD0iRWluIE5hbWUiKS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5GcmFtZXdvcms8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Db3JlOjpQYWNrYWdlPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Imh0dHA6Ly9mdHAub3Rycy5vcmcvcHViL290cnMvaXRzbS9wYWNrYWdlczIxLyI+Wy0tT1RSUzo6SVRTTSAyLjEgTWFzdGVyLS1dIGh0dHA6Ly9mdHAub3Rycy5vcmcvPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgo8L290cnNfY29uZmlnPgo=
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSIgPz4KPG90cnNfY29uZmlnIHZlcnNpb249IjEuMCIgaW5pdD0iQ29uZmlnIj4KICAgIDxDVlM+JElkOiBJVFNNQ29yZS54bWwsdiAxLjMwIDIwMTAvMDQvMTMgMTI6Mjg6MjggdWIgRXhwICQ8L0NWUz4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZyb250ZW5kOjpNb2R1bGUjIyNBZG1pbklUU01DSVBBbGxvY2F0ZSIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkZyb250ZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBBZG1pbklUU01DSVBBbGxvY2F0ZSBjb25maWd1cmF0aW9uIGluIHRoZSBhZG1pbiBhcmVhLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5Gcm9udGVuZG1vZHVsLVJlZ2lzdHJhdGlvbiBkZXIgQWRtaW5JVFNNQ0lQQWxsb2NhdGUgS29uZmlndXJhdGlvbiBpbSBBZG1pbi1CZXJlaWNoLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPklUU00gQ29yZTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZG1pbjo6TW9kdWxlUmVnaXN0cmF0aW9uPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEZyb250ZW5kTW9kdWxlUmVnPgogICAgICAgICAgICAgICAgPEdyb3VwPmFkbWluPC9Hcm91cD4KICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5BZG1pbjwvRGVzY3JpcHRpb24+CiAgICAgICAgICAgICAgICA8VGl0bGU+Q3JpdGljYWxpdHkgJmx0Oy0mZ3Q7IEltcGFjdCAmbHQ7LSZndDsgUHJpb3JpdHk8L1RpdGxlPgogICAgICAgICAgICAgICAgPE5hdkJhck5hbWU+QWRtaW48L05hdkJhck5hbWU+CiAgICAgICAgICAgICAgICA8TmF2QmFyTW9kdWxlPgogICAgICAgICAgICAgICAgICAgIDxNb2R1bGU+S2VybmVsOjpPdXRwdXQ6OkhUTUw6Ok5hdkJhck1vZHVsZUFkbWluPC9Nb2R1bGU+CiAgICAgICAgICAgICAgICAgICAgPE5hbWU+Q3JpdGljYWxpdHkgJmx0Oy0mZ3Q7IEltcGFjdCAmbHQ7LSZndDsgUHJpb3JpdHk8L05hbWU+CiAgICAgICAgICAgICAgICAgICAgPEJsb2NrPkJsb2NrMzwvQmxvY2s+CiAgICAgICAgICAgICAgICAgICAgPFByaW8+NDMwPC9QcmlvPgogICAgICAgICAgICAgICAgPC9OYXZCYXJNb2R1bGU+CiAgICAgICAgICAgIDwvRnJvbnRlbmRNb2R1bGVSZWc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iSVRTTTo6Q29yZTo6SW5jaWRlbnRMaW5rVHlwZSIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlNldCB0aGUgdHlwZSBvZiBsaW5rIHRvIGJlIHVzZWQgdG8gY2FsY3VsYXRlIHRoZSBpbmNpZGVudCBzdGF0ZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+TGVndCBkZW4gTGlua3R5cCBmZXN0LCBkZXIgenVyIEJlcmVjaG51bmcgZGVzIFZvcmZhbGxzdGF0dXMgdmVyd2VuZGV0IHdpcmQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+SVRTTSBDb3JlPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6SVRTTUNvcmU8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPkRlcGVuZHNPbjwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkxpbmtPYmplY3Q6OlR5cGUjIyNBbHRlcm5hdGl2ZVRvIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+VGhpcyBzZXR0aW5nIGRlZmluZXMgdGhlIGxpbmsgdHlwZSAnQWx0ZXJuYXRpdmVUbycuIElmIHRoZSBzb3VyY2UgbmFtZSBhbmQgdGhlIHRhcmdldCBuYW1lIGNvbnRhaW4gdGhlIHNhbWUgdmFsdWUsIHRoZSByZXN1bHRpbmcgbGluayBpcyBhIG5vbi1kaXJlY3Rpb25hbCBvbmUuIElmIHRoZSB2YWx1ZXMgYXJlIGRpZmZlcmVudCwgdGhlIHJlc3VsdGluZyBsaW5rIGlzIGEgZGlyZWN0aW9uYWwgbGluay48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RGVmaW5pZXJ0IGRlbiBMaW5rdHlwICdBbHRlcm5hdGl2ZVRvJy4gV2lyZCBhbHMgU291cmNlTmFtZSB1bmQgVGFyZ2V0TmFtZSBkZXIgZ2xlaWNoZSBJbmhhbHQgYW5nZWdlYmVuLCBlbnRzdGVodCBlaW4gdW5nZXJpY2h0ZXRlciBMaW5rdHlwLiBXaXJkIGFscyBTb3VyY2VOYW1lIHVuZCBUYXJnZXROYW1lIHZlcnNjaGllZGVuZXIgSW5oYWx0IGFuZ2VnZWJlbiwgZW50c3RlaHQgZWluIGdlcmljaHRldGVyIExpbmt0eXAuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RnJhbWV3b3JrPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6TGlua09iamVjdDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJTb3VyY2VOYW1lIj5BbHRlcm5hdGl2ZSB0bzwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iVGFyZ2V0TmFtZSI+QWx0ZXJuYXRpdmUgdG88L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJMaW5rT2JqZWN0OjpUeXBlIyMjQ29ubmVjdGVkVG8iIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5UaGlzIHNldHRpbmcgZGVmaW5lcyB0aGUgbGluayB0eXBlICdDb25uZWN0ZWRUbycuIElmIHRoZSBzb3VyY2UgbmFtZSBhbmQgdGhlIHRhcmdldCBuYW1lIGNvbnRhaW4gdGhlIHNhbWUgdmFsdWUsIHRoZSByZXN1bHRpbmcgbGluayBpcyBhIG5vbi1kaXJlY3Rpb25hbCBvbmUuIElmIHRoZSB2YWx1ZXMgYXJlIGRpZmZlcmVudCwgdGhlIHJlc3VsdGluZyBsaW5rIGlzIGEgZGlyZWN0aW9uYWwgbGluay48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RGVmaW5pZXJ0IGRlbiBMaW5rdHlwICdDb25uZWN0ZWRUbycuIFdpcmQgYWxzIFNvdXJjZU5hbWUgdW5kIFRhcmdldE5hbWUgZGVyIGdsZWljaGUgSW5oYWx0IGFuZ2VnZWJlbiwgZW50c3RlaHQgZWluIHVuZ2VyaWNodGV0ZXIgTGlua3R5cC4gV2lyZCBhbHMgU291cmNlTmFtZSB1bmQgVGFyZ2V0TmFtZSB2ZXJzY2hpZWRlbmVyIEluaGFsdCBhbmdlZ2ViZW4sIGVudHN0ZWh0IGVpbiBnZXJpY2h0ZXRlciBMaW5rdHlwLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZyYW1ld29yazwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkxpbmtPYmplY3Q8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iU291cmNlTmFtZSI+Q29ubmVjdGVkIHRvPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJUYXJnZXROYW1lIj5Db25uZWN0ZWQgdG88L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJMaW5rT2JqZWN0OjpUeXBlIyMjRGVwZW5kc09uIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+VGhpcyBzZXR0aW5nIGRlZmluZXMgdGhlIGxpbmsgdHlwZSAnRGVwZW5kc09uJy4gSWYgdGhlIHNvdXJjZSBuYW1lIGFuZCB0aGUgdGFyZ2V0IG5hbWUgY29udGFpbiB0aGUgc2FtZSB2YWx1ZSwgdGhlIHJlc3VsdGluZyBsaW5rIGlzIGEgbm9uLWRpcmVjdGlvbmFsIG9uZS4gSWYgdGhlIHZhbHVlcyBhcmUgZGlmZmVyZW50LCB0aGUgcmVzdWx0aW5nIGxpbmsgaXMgYSBkaXJlY3Rpb25hbCBsaW5rLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5EZWZpbmllcnQgZGVuIExpbmt0eXAgJ0RlcGVuZHNPbicuIFdpcmQgYWxzIFNvdXJjZU5hbWUgdW5kIFRhcmdldE5hbWUgZGVyIGdsZWljaGUgSW5oYWx0IGFuZ2VnZWJlbiwgZW50c3RlaHQgZWluIHVuZ2VyaWNodGV0ZXIgTGlua3R5cC4gV2lyZCBhbHMgU291cmNlTmFtZSB1bmQgVGFyZ2V0TmFtZSB2ZXJzY2hpZWRlbmVyIEluaGFsdCBhbmdlZ2ViZW4sIGVudHN0ZWh0IGVpbiBnZXJpY2h0ZXRlciBMaW5rdHlwLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZyYW1ld29yazwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkxpbmtPYmplY3Q8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iU291cmNlTmFtZSI+RGVwZW5kcyBvbjwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iVGFyZ2V0TmFtZSI+UmVxdWlyZWQgZm9yPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iTGlua09iamVjdDo6VHlwZSMjI0luY2x1ZGVzIiBSZXF1aXJlZD0iMSIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+VGhpcyBzZXR0aW5nIGRlZmluZXMgdGhlIGxpbmsgdHlwZSAnSW5jbHVkZXMnLiBJZiB0aGUgc291cmNlIG5hbWUgYW5kIHRoZSB0YXJnZXQgbmFtZSBjb250YWluIHRoZSBzYW1lIHZhbHVlLCB0aGUgcmVzdWx0aW5nIGxpbmsgaXMgYSBub24tZGlyZWN0aW9uYWwgb25lLiBJZiB0aGUgdmFsdWVzIGFyZSBkaWZmZXJlbnQsIHRoZSByZXN1bHRpbmcgbGluayBpcyBhIGRpcmVjdGlvbmFsIGxpbmsuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaWVydCBkZW4gTGlua3R5cCAnSW5jbHVkZXMnLiBXaXJkIGFscyBTb3VyY2VOYW1lIHVuZCBUYXJnZXROYW1lIGRlciBnbGVpY2hlIEluaGFsdCBhbmdlZ2ViZW4sIGVudHN0ZWh0IGVpbiB1bmdlcmljaHRldGVyIExpbmt0eXAuIFdpcmQgYWxzIFNvdXJjZU5hbWUgdW5kIFRhcmdldE5hbWUgdmVyc2NoaWVkZW5lciBJbmhhbHQgYW5nZWdlYmVuLCBlbnRzdGVodCBlaW4gZ2VyaWNodGV0ZXIgTGlua3R5cC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5GcmFtZXdvcms8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Db3JlOjpMaW5rT2JqZWN0PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlNvdXJjZU5hbWUiPkluY2x1ZGVzPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJUYXJnZXROYW1lIj5QYXJ0IG9mPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iTGlua09iamVjdDo6VHlwZSMjI1JlbGV2YW50VG8iIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5UaGlzIHNldHRpbmcgZGVmaW5lcyB0aGUgbGluayB0eXBlICdSZWxldmFudFRvJy4gSWYgdGhlIHNvdXJjZSBuYW1lIGFuZCB0aGUgdGFyZ2V0IG5hbWUgY29udGFpbiB0aGUgc2FtZSB2YWx1ZSwgdGhlIHJlc3VsdGluZyBsaW5rIGlzIGEgbm9uLWRpcmVjdGlvbmFsIG9uZS4gSWYgdGhlIHZhbHVlcyBhcmUgZGlmZmVyZW50LCB0aGUgcmVzdWx0aW5nIGxpbmsgaXMgYSBkaXJlY3Rpb25hbCBsaW5rLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5EZWZpbmllcnQgZGVuIExpbmt0eXAgJ1JlbGV2YW50VG8nLiBXaXJkIGFscyBTb3VyY2VOYW1lIHVuZCBUYXJnZXROYW1lIGRlciBnbGVpY2hlIEluaGFsdCBhbmdlZ2ViZW4sIGVudHN0ZWh0IGVpbiB1bmdlcmljaHRldGVyIExpbmt0eXAuIFdpcmQgYWxzIFNvdXJjZU5hbWUgdW5kIFRhcmdldE5hbWUgdmVyc2NoaWVkZW5lciBJbmhhbHQgYW5nZWdlYmVuLCBlbnRzdGVodCBlaW4gZ2VyaWNodGV0ZXIgTGlua3R5cC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5GcmFtZXdvcms8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Db3JlOjpMaW5rT2JqZWN0PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlNvdXJjZU5hbWUiPlJlbGV2YW50IHRvPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJUYXJnZXROYW1lIj5SZWxldmFudCB0bzwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkxpbmtPYmplY3Q6OlBvc3NpYmxlTGluayMjIzMyMDAiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5UaGlzIHNldHRpbmcgZGVmaW5lcyB0aGF0IGEgJ0lUU01Db25maWdJdGVtJyBvYmplY3QgY2FuIGJlIGxpbmtlZCB3aXRoIG90aGVyICdJVFNNQ29uZmlnSXRlbScgb2JqZWN0cyB1c2luZyB0aGUgJ0FsdGVybmF0aXZlVG8nIGxpbmsgdHlwZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RGVmaW5pZXJ0LCBkYXNzIGVpbiAnSVRTTUNvbmZpZ0l0ZW0nLU9iamVrdCBtaXQgZGVtIExpbmt0eXAgJ0FsdGVybmF0aXZlVG8nIG1pdCBhbmRlcmVuICdJVFNNQ29uZmlnSXRlbSctT2JqZWt0ZW4gdmVybGlua3Qgd2VyZGVuIGthbm4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RnJhbWV3b3JrPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6TGlua09iamVjdDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QxIj5JVFNNQ29uZmlnSXRlbTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MiI+SVRTTUNvbmZpZ0l0ZW08L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlR5cGUiPkFsdGVybmF0aXZlVG88L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJMaW5rT2JqZWN0OjpQb3NzaWJsZUxpbmsjIyMzMjAxIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+VGhpcyBzZXR0aW5nIGRlZmluZXMgdGhhdCBhICdJVFNNQ29uZmlnSXRlbScgb2JqZWN0IGNhbiBiZSBsaW5rZWQgd2l0aCBvdGhlciAnSVRTTUNvbmZpZ0l0ZW0nIG9iamVjdHMgdXNpbmcgdGhlICdDb25uZWN0ZWRUbycgbGluayB0eXBlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5EZWZpbmllcnQsIGRhc3MgZWluICdJVFNNQ29uZmlnSXRlbSctT2JqZWt0IG1pdCBkZW0gTGlua3R5cCAnQ29ubmVjdGVkVG8nIG1pdCBhbmRlcmVuICdJVFNNQ29uZmlnSXRlbSctT2JqZWt0ZW4gdmVybGlua3Qgd2VyZGVuIGthbm4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RnJhbWV3b3JrPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6TGlua09iamVjdDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QxIj5JVFNNQ29uZmlnSXRlbTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MiI+SVRTTUNvbmZpZ0l0ZW08L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlR5cGUiPkNvbm5lY3RlZFRvPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iTGlua09iamVjdDo6UG9zc2libGVMaW5rIyMjMzIwMiIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlRoaXMgc2V0dGluZyBkZWZpbmVzIHRoYXQgYSAnSVRTTUNvbmZpZ0l0ZW0nIG9iamVjdCBjYW4gYmUgbGlua2VkIHdpdGggb3RoZXIgJ0lUU01Db25maWdJdGVtJyBvYmplY3RzIHVzaW5nIHRoZSAnRGVwZW5kc09uJyBsaW5rIHR5cGUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaWVydCwgZGFzcyBlaW4gJ0lUU01Db25maWdJdGVtJy1PYmpla3QgbWl0IGRlbSBMaW5rdHlwICdEZXBlbmRzT24nIG1pdCBhbmRlcmVuICdJVFNNQ29uZmlnSXRlbSctT2JqZWt0ZW4gdmVybGlua3Qgd2VyZGVuIGthbm4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RnJhbWV3b3JrPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6TGlua09iamVjdDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QxIj5JVFNNQ29uZmlnSXRlbTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MiI+SVRTTUNvbmZpZ0l0ZW08L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlR5cGUiPkRlcGVuZHNPbjwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkxpbmtPYmplY3Q6OlBvc3NpYmxlTGluayMjIzMyMDMiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5UaGlzIHNldHRpbmcgZGVmaW5lcyB0aGF0IGEgJ0lUU01Db25maWdJdGVtJyBvYmplY3QgY2FuIGJlIGxpbmtlZCB3aXRoIG90aGVyICdJVFNNQ29uZmlnSXRlbScgb2JqZWN0cyB1c2luZyB0aGUgJ0luY2x1ZGVzJyBsaW5rIHR5cGUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaWVydCwgZGFzcyBlaW4gJ0lUU01Db25maWdJdGVtJy1PYmpla3QgbWl0IGRlbSBMaW5rdHlwICdJbmNsdWRlcycgbWl0IGFuZGVyZW4gJ0lUU01Db25maWdJdGVtJy1PYmpla3RlbiB2ZXJsaW5rdCB3ZXJkZW4ga2Fubi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5GcmFtZXdvcms8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Db3JlOjpMaW5rT2JqZWN0PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik9iamVjdDEiPklUU01Db25maWdJdGVtPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QyIj5JVFNNQ29uZmlnSXRlbTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iVHlwZSI+SW5jbHVkZXM8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJMaW5rT2JqZWN0OjpQb3NzaWJsZUxpbmsjIyMzMjA0IiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+VGhpcyBzZXR0aW5nIGRlZmluZXMgdGhhdCBhICdJVFNNQ29uZmlnSXRlbScgb2JqZWN0IGNhbiBiZSBsaW5rZWQgd2l0aCBvdGhlciAnSVRTTUNvbmZpZ0l0ZW0nIG9iamVjdHMgdXNpbmcgdGhlICdSZWxldmFudFRvJyBsaW5rIHR5cGUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaWVydCwgZGFzcyBlaW4gJ0lUU01Db25maWdJdGVtJy1PYmpla3QgbWl0IGRlbSBMaW5rdHlwICdSZWxldmFudFRvJyBtaXQgYW5kZXJlbiAnSVRTTUNvbmZpZ0l0ZW0nLU9iamVrdGVuIHZlcmxpbmt0IHdlcmRlbiBrYW5uLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZyYW1ld29yazwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkxpbmtPYmplY3Q8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MSI+SVRTTUNvbmZpZ0l0ZW08L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik9iamVjdDIiPklUU01Db25maWdJdGVtPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJUeXBlIj5SZWxldmFudFRvPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iTGlua09iamVjdDo6UG9zc2libGVMaW5rIyMjMzIyMCIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlRoaXMgc2V0dGluZyBkZWZpbmVzIHRoYXQgYSAnSVRTTUNvbmZpZ0l0ZW0nIG9iamVjdCBjYW4gYmUgbGlua2VkIHdpdGggJ1RpY2tldCcgb2JqZWN0cyB1c2luZyB0aGUgJ0FsdGVybmF0aXZlVG8nIGxpbmsgdHlwZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RGVmaW5pZXJ0LCBkYXNzIGVpbiAnSVRTTUNvbmZpZ0l0ZW0nLU9iamVrdCBtaXQgZGVtIExpbmt0eXAgJ0FsdGVybmF0aXZlVG8nIG1pdCAnVGlja2V0Jy1PYmpla3RlbiB2ZXJsaW5rdCB3ZXJkZW4ga2Fubi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5GcmFtZXdvcms8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Db3JlOjpMaW5rT2JqZWN0PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik9iamVjdDEiPklUU01Db25maWdJdGVtPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QyIj5UaWNrZXQ8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlR5cGUiPkFsdGVybmF0aXZlVG88L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJMaW5rT2JqZWN0OjpQb3NzaWJsZUxpbmsjIyMzMjIxIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+VGhpcyBzZXR0aW5nIGRlZmluZXMgdGhhdCBhICdJVFNNQ29uZmlnSXRlbScgb2JqZWN0IGNhbiBiZSBsaW5rZWQgd2l0aCAnVGlja2V0JyBvYmplY3RzIHVzaW5nIHRoZSAnRGVwZW5kc09uJyBsaW5rIHR5cGUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaWVydCwgZGFzcyBlaW4gJ0lUU01Db25maWdJdGVtJy1PYmpla3QgbWl0IGRlbSBMaW5rdHlwICdEZXBlbmRzT24nIG1pdCAnVGlja2V0Jy1PYmpla3RlbiB2ZXJsaW5rdCB3ZXJkZW4ga2Fubi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5GcmFtZXdvcms8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Db3JlOjpMaW5rT2JqZWN0PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik9iamVjdDEiPklUU01Db25maWdJdGVtPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QyIj5UaWNrZXQ8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlR5cGUiPkRlcGVuZHNPbjwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkxpbmtPYmplY3Q6OlBvc3NpYmxlTGluayMjIzMyMjIiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5UaGlzIHNldHRpbmcgZGVmaW5lcyB0aGF0IGEgJ0lUU01Db25maWdJdGVtJyBvYmplY3QgY2FuIGJlIGxpbmtlZCB3aXRoICdUaWNrZXQnIG9iamVjdHMgdXNpbmcgdGhlICdSZWxldmFudFRvJyBsaW5rIHR5cGUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaWVydCwgZGFzcyBlaW4gJ0lUU01Db25maWdJdGVtJy1PYmpla3QgbWl0IGRlbSBMaW5rdHlwICdSZWxldmFudFRvJyBtaXQgJ1RpY2tldCctT2JqZWt0ZW4gdmVybGlua3Qgd2VyZGVuIGthbm4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RnJhbWV3b3JrPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6TGlua09iamVjdDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QxIj5JVFNNQ29uZmlnSXRlbTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MiI+VGlja2V0PC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJUeXBlIj5SZWxldmFudFRvPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iTGlua09iamVjdDo6UG9zc2libGVMaW5rIyMjMzI0MCIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlRoaXMgc2V0dGluZyBkZWZpbmVzIHRoYXQgYSAnSVRTTUNvbmZpZ0l0ZW0nIG9iamVjdCBjYW4gYmUgbGlua2VkIHdpdGggJ1NlcnZpY2UnIG9iamVjdHMgdXNpbmcgdGhlICdBbHRlcm5hdGl2ZVRvJyBsaW5rIHR5cGUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaWVydCwgZGFzcyBlaW4gJ0lUU01Db25maWdJdGVtJy1PYmpla3QgbWl0IGRlbSBMaW5rdHlwICdBbHRlcm5hdGl2ZVRvJyBtaXQgJ1NlcnZpY2UnLU9iamVrdGVuIHZlcmxpbmt0IHdlcmRlbiBrYW5uLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZyYW1ld29yazwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkxpbmtPYmplY3Q8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MSI+SVRTTUNvbmZpZ0l0ZW08L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik9iamVjdDIiPlNlcnZpY2U8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlR5cGUiPkFsdGVybmF0aXZlVG88L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJMaW5rT2JqZWN0OjpQb3NzaWJsZUxpbmsjIyMzMjQxIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+VGhpcyBzZXR0aW5nIGRlZmluZXMgdGhhdCBhICdJVFNNQ29uZmlnSXRlbScgb2JqZWN0IGNhbiBiZSBsaW5rZWQgd2l0aCAnU2VydmljZScgb2JqZWN0cyB1c2luZyB0aGUgJ0RlcGVuZHNPbicgbGluayB0eXBlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5EZWZpbmllcnQsIGRhc3MgZWluICdJVFNNQ29uZmlnSXRlbSctT2JqZWt0IG1pdCBkZW0gTGlua3R5cCAnRGVwZW5kc09uJyBtaXQgJ1NlcnZpY2UnLU9iamVrdGVuIHZlcmxpbmt0IHdlcmRlbiBrYW5uLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZyYW1ld29yazwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkxpbmtPYmplY3Q8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MSI+SVRTTUNvbmZpZ0l0ZW08L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik9iamVjdDIiPlNlcnZpY2U8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlR5cGUiPkRlcGVuZHNPbjwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkxpbmtPYmplY3Q6OlBvc3NpYmxlTGluayMjIzMyNDIiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5UaGlzIHNldHRpbmcgZGVmaW5lcyB0aGF0IGEgJ0lUU01Db25maWdJdGVtJyBvYmplY3QgY2FuIGJlIGxpbmtlZCB3aXRoICdTZXJ2aWNlJyBvYmplY3RzIHVzaW5nIHRoZSAnUmVsZXZhbnRUbycgbGluayB0eXBlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5EZWZpbmllcnQsIGRhc3MgZWluICdJVFNNQ29uZmlnSXRlbSctT2JqZWt0IG1pdCBkZW0gTGlua3R5cCAnUmVsZXZhbnRUbycgbWl0ICdTZXJ2aWNlJy1PYmpla3RlbiB2ZXJsaW5rdCB3ZXJkZW4ga2Fubi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5GcmFtZXdvcms8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Db3JlOjpMaW5rT2JqZWN0PC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik9iamVjdDEiPklUU01Db25maWdJdGVtPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QyIj5TZXJ2aWNlPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJUeXBlIj5SZWxldmFudFRvPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iTGlua09iamVjdDo6UG9zc2libGVMaW5rIyMjMzI2MCIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlRoaXMgc2V0dGluZyBkZWZpbmVzIHRoYXQgYSAnSVRTTUNvbmZpZ0l0ZW0nIG9iamVjdCBjYW4gYmUgbGlua2VkIHdpdGggJ0ZBUScgb2JqZWN0cyB1c2luZyB0aGUgJ05vcm1hbCcgbGluayB0eXBlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5EZWZpbmllcnQsIGRhc3MgZWluICdJVFNNQ29uZmlnSXRlbSctT2JqZWt0IG1pdCBkZW0gTGlua3R5cCAnTm9ybWFsJyBtaXQgJ0ZBUSctT2JqZWt0ZW4gdmVybGlua3Qgd2VyZGVuIGthbm4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RnJhbWV3b3JrPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6TGlua09iamVjdDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QxIj5JVFNNQ29uZmlnSXRlbTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MiI+RkFRPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJUeXBlIj5Ob3JtYWw8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJMaW5rT2JqZWN0OjpQb3NzaWJsZUxpbmsjIyMzMjYxIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+VGhpcyBzZXR0aW5nIGRlZmluZXMgdGhhdCBhICdJVFNNQ29uZmlnSXRlbScgb2JqZWN0IGNhbiBiZSBsaW5rZWQgd2l0aCAnRkFRJyBvYmplY3RzIHVzaW5nIHRoZSAnUGFyZW50Q2hpbGQnIGxpbmsgdHlwZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RGVmaW5pZXJ0LCBkYXNzIGVpbiAnSVRTTUNvbmZpZ0l0ZW0nLU9iamVrdCBtaXQgZGVtIExpbmt0eXAgJ1BhcmVudENoaWxkJyBtaXQgJ0ZBUSctT2JqZWt0ZW4gdmVybGlua3Qgd2VyZGVuIGthbm4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RnJhbWV3b3JrPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6TGlua09iamVjdDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QxIj5JVFNNQ29uZmlnSXRlbTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MiI+RkFRPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJUeXBlIj5QYXJlbnRDaGlsZDwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkxpbmtPYmplY3Q6OlBvc3NpYmxlTGluayMjIzMyNjIiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5UaGlzIHNldHRpbmcgZGVmaW5lcyB0aGF0IGEgJ0lUU01Db25maWdJdGVtJyBvYmplY3QgY2FuIGJlIGxpbmtlZCB3aXRoICdGQVEnIG9iamVjdHMgdXNpbmcgdGhlICdSZWxldmFudFRvJyBsaW5rIHR5cGUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaWVydCwgZGFzcyBlaW4gJ0lUU01Db25maWdJdGVtJy1PYmpla3QgbWl0IGRlbSBMaW5rdHlwICdSZWxldmFudFRvJyBtaXQgJ0ZBUSctT2JqZWt0ZW4gdmVybGlua3Qgd2VyZGVuIGthbm4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RnJhbWV3b3JrPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6TGlua09iamVjdDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QxIj5JVFNNQ29uZmlnSXRlbTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MiI+RkFRPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJUeXBlIj5SZWxldmFudFRvPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iTGlua09iamVjdDo6UG9zc2libGVMaW5rIyMjMzI4MCIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlRoaXMgc2V0dGluZyBkZWZpbmVzIHRoYXQgYSAnU2VydmljZScgb2JqZWN0IGNhbiBiZSBsaW5rZWQgd2l0aCAnRkFRJyBvYmplY3RzIHVzaW5nIHRoZSAnTm9ybWFsJyBsaW5rIHR5cGUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaWVydCwgZGFzcyBlaW4gJ1NlcnZpY2UnLU9iamVrdCBtaXQgZGVtIExpbmt0eXAgJ05vcm1hbCcgbWl0ICdGQVEnLU9iamVrdGVuIHZlcmxpbmt0IHdlcmRlbiBrYW5uLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZyYW1ld29yazwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkxpbmtPYmplY3Q8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MSI+U2VydmljZTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MiI+RkFRPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJUeXBlIj5Ob3JtYWw8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJMaW5rT2JqZWN0OjpQb3NzaWJsZUxpbmsjIyMzMjgxIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+VGhpcyBzZXR0aW5nIGRlZmluZXMgdGhhdCBhICdTZXJ2aWNlJyBvYmplY3QgY2FuIGJlIGxpbmtlZCB3aXRoICdGQVEnIG9iamVjdHMgdXNpbmcgdGhlICdQYXJlbnRDaGlsZCcgbGluayB0eXBlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5EZWZpbmllcnQsIGRhc3MgZWluICdTZXJ2aWNlJy1PYmpla3QgbWl0IGRlbSBMaW5rdHlwICdQYXJlbnRDaGlsZCcgbWl0ICdGQVEnLU9iamVrdGVuIHZlcmxpbmt0IHdlcmRlbiBrYW5uLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZyYW1ld29yazwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkxpbmtPYmplY3Q8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MSI+U2VydmljZTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MiI+RkFRPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJUeXBlIj5QYXJlbnRDaGlsZDwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkxpbmtPYmplY3Q6OlBvc3NpYmxlTGluayMjIzMyODIiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5UaGlzIHNldHRpbmcgZGVmaW5lcyB0aGF0IGEgJ1NlcnZpY2UnIG9iamVjdCBjYW4gYmUgbGlua2VkIHdpdGggJ0ZBUScgb2JqZWN0cyB1c2luZyB0aGUgJ1JlbGV2YW50VG8nIGxpbmsgdHlwZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RGVmaW5pZXJ0LCBkYXNzIGVpbiAnU2VydmljZSctT2JqZWt0IG1pdCBkZW0gTGlua3R5cCAnUmVsZXZhbnRUbycgbWl0ICdGQVEnLU9iamVrdGVuIHZlcmxpbmt0IHdlcmRlbiBrYW5uLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZyYW1ld29yazwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkxpbmtPYmplY3Q8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MSI+U2VydmljZTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MiI+RkFRPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJUeXBlIj5SZWxldmFudFRvPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgoKICAgIDxDb25maWdJdGVtIE5hbWU9IkxpbmtPYmplY3Q6OlBvc3NpYmxlTGluayMjIzM0MDAiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5UaGlzIHNldHRpbmcgZGVmaW5lcyB0aGF0IGEgJ0lUU01Xb3JrT3JkZXInIG9iamVjdCBjYW4gYmUgbGlua2VkIHdpdGggJ1NlcnZpY2UnIG9iamVjdHMgdXNpbmcgdGhlICdOb3JtYWwnIGxpbmsgdHlwZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RGVmaW5pZXJ0LCBkYXNzIGVpbiAnSVRTTVdvcmtPcmRlcictT2JqZWt0IG1pdCBkZW0gTGlua3R5cCAnTm9ybWFsJyBtaXQgJ1NlcnZpY2UnLU9iamVrdGVuIHZlcmxpbmt0IHdlcmRlbiBrYW5uLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZyYW1ld29yazwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkxpbmtPYmplY3Q8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MSI+SVRTTVdvcmtPcmRlcjwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MiI+U2VydmljZTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iVHlwZSI+Tm9ybWFsPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iTGlua09iamVjdDo6UG9zc2libGVMaW5rIyMjMzQwMSIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlRoaXMgc2V0dGluZyBkZWZpbmVzIHRoYXQgYSAnSVRTTVdvcmtPcmRlcicgb2JqZWN0IGNhbiBiZSBsaW5rZWQgd2l0aCAnU2VydmljZScgb2JqZWN0cyB1c2luZyB0aGUgJ0RlcGVuZHNPbicgbGluayB0eXBlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5EZWZpbmllcnQsIGRhc3MgZWluICdJVFNNV29ya09yZGVyJy1PYmpla3QgbWl0IGRlbSBMaW5rdHlwICdEZXBlbmRzT24nIG1pdCAnU2VydmljZSctT2JqZWt0ZW4gdmVybGlua3Qgd2VyZGVuIGthbm4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RnJhbWV3b3JrPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6TGlua09iamVjdDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QxIj5JVFNNV29ya09yZGVyPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QyIj5TZXJ2aWNlPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJUeXBlIj5EZXBlbmRzT248L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJMaW5rT2JqZWN0OjpQb3NzaWJsZUxpbmsjIyMzNDEwIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+VGhpcyBzZXR0aW5nIGRlZmluZXMgdGhhdCBhICdJVFNNV29ya09yZGVyJyBvYmplY3QgY2FuIGJlIGxpbmtlZCB3aXRoICdJVFNNQ29uZmlnSXRlbScgb2JqZWN0cyB1c2luZyB0aGUgJ05vcm1hbCcgbGluayB0eXBlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5EZWZpbmllcnQsIGRhc3MgZWluICdJVFNNV29ya09yZGVyJy1PYmpla3QgbWl0IGRlbSBMaW5rdHlwICdOb3JtYWwnIG1pdCAnSVRTTUNvbmZpZ0l0ZW0nLU9iamVrdGVuIHZlcmxpbmt0IHdlcmRlbiBrYW5uLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZyYW1ld29yazwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkxpbmtPYmplY3Q8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MSI+SVRTTVdvcmtPcmRlcjwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MiI+SVRTTUNvbmZpZ0l0ZW08L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlR5cGUiPk5vcm1hbDwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkxpbmtPYmplY3Q6OlBvc3NpYmxlTGluayMjIzM0MTEiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5UaGlzIHNldHRpbmcgZGVmaW5lcyB0aGF0IGEgJ0lUU01Xb3JrT3JkZXInIG9iamVjdCBjYW4gYmUgbGlua2VkIHdpdGggJ0lUU01Db25maWdJdGVtJyBvYmplY3RzIHVzaW5nIHRoZSAnRGVwZW5kc09uJyBsaW5rIHR5cGUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaWVydCwgZGFzcyBlaW4gJ0lUU01Xb3JrT3JkZXInLU9iamVrdCBtaXQgZGVtIExpbmt0eXAgJ0RlcGVuZHNPbicgbWl0ICdJVFNNQ29uZmlnSXRlbSctT2JqZWt0ZW4gdmVybGlua3Qgd2VyZGVuIGthbm4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RnJhbWV3b3JrPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6TGlua09iamVjdDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QxIj5JVFNNV29ya09yZGVyPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QyIj5JVFNNQ29uZmlnSXRlbTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iVHlwZSI+RGVwZW5kc09uPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iTGlua09iamVjdDo6UG9zc2libGVMaW5rIyMjMzQxMiIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlRoaXMgc2V0dGluZyBkZWZpbmVzIHRoYXQgYSAnSVRTTVdvcmtPcmRlcicgb2JqZWN0IGNhbiBiZSBsaW5rZWQgd2l0aCAnVGlja2V0JyBvYmplY3RzIHVzaW5nIHRoZSAnTm9ybWFsJyBsaW5rIHR5cGUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaWVydCwgZGFzcyBlaW4gJ0lUU01Xb3JrT3JkZXInLU9iamVrdCBtaXQgZGVtIExpbmt0eXAgJ05vcm1hbCcgbWl0ICdUaWNrZXQnLU9iamVrdGVuIHZlcmxpbmt0IHdlcmRlbiBrYW5uLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkZyYW1ld29yazwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OkxpbmtPYmplY3Q8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MSI+SVRTTVdvcmtPcmRlcjwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iT2JqZWN0MiI+VGlja2V0PC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJUeXBlIj5Ob3JtYWw8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJMaW5rT2JqZWN0OjpQb3NzaWJsZUxpbmsjIyMzNDIwIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+VGhpcyBzZXR0aW5nIGRlZmluZXMgdGhhdCBhICdJVFNNQ2hhbmdlJyBvYmplY3QgY2FuIGJlIGxpbmtlZCB3aXRoICdUaWNrZXQnIG9iamVjdHMgdXNpbmcgdGhlICdOb3JtYWwnIGxpbmsgdHlwZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RGVmaW5pZXJ0LCBkYXNzIGVpbiAnSVRTTUNoYW5nZSctT2JqZWt0IG1pdCBkZW0gTGlua3R5cCAnTm9ybWFsJyBtaXQgJ1RpY2tldCctT2JqZWt0ZW4gdmVybGlua3Qgd2VyZGVuIGthbm4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RnJhbWV3b3JrPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6TGlua09iamVjdDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QxIj5JVFNNQ2hhbmdlPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJPYmplY3QyIj5UaWNrZXQ8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlR5cGUiPk5vcm1hbDwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IklUU006OkZyb250ZW5kOjpUZXh0QXJlYSIgUmVxdWlyZWQ9IjEiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPldpZHRoIG9mIElUU00gdGV4dGFyZWFzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5BbnphaGwgZGVyIFplaWNoZW4gcHJvIFplaWxlIGluIElUU00gVGV4dEFyZWFzLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPklUU00gQ29yZTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9Il5bMC05XXsxLDN9JCI+Nzg8L1N0cmluZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJHZW5lcmFsQ2F0YWxvZ1ByZWZlcmVuY2VzIyMjSW5jaWRlbnRTdGF0ZXMiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5QYXJhbWV0ZXJzIGZvciB0aGUgaW5jaWRlbnQgc3RhdGVzIGluIHRoZSBwcmVmZXJlbmNlIHZpZXcuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPlBhcmFtZXRlciBm/HIgZGVuIFZvcmZhbGxzc3RhdHVzIGluIGRlciBBbnNpY2h0IGb8ciBkaWUgRWluc3RlbGx1bmdlbi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5HZW5lcmFsQ2F0YWxvZzwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkdlbmVyYWxDYXRhbG9nOjpQcmVmZXJlbmNlczwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJNb2R1bGUiPktlcm5lbDo6T3V0cHV0OjpIVE1MOjpHZW5lcmFsQ2F0YWxvZ1ByZWZlcmVuY2VzR2VuZXJpYzwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iQ2xhc3MiPklUU006OkNvcmU6OkluY2lkZW50U3RhdGU8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IkxhYmVsIj5JbmNpZGVudCBTdGF0ZSBUeXBlPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJEZXNjIj5JbmNpZGVudCBTdGF0ZSBUeXBlLjwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iRGF0YSI+CiAgICAgICAgICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICAgICAgICAgIDxJdGVtIEtleT0id2FybmluZyI+V2FybmluZzwvSXRlbT4KICAgICAgICAgICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJvcGVyYXRpb25hbCI+T3BlcmF0aW9uYWw8L0l0ZW0+CiAgICAgICAgICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iaW5jaWRlbnQiPkluY2lkZW50PC9JdGVtPgogICAgICAgICAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICAgICAgICAgIDwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iUHJlZktleSI+RnVuY3Rpb25hbGl0eTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iQmxvY2siPk9wdGlvbjwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlZpcnR1YWxGUzo6QmFja2VuZCIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkNvbmZpZ3VyZSBtb2R1bGUgdXNlZCBhcyBiYWNrZW5kIGZvciBWaXJ0dWFsRlMuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPktvbmZpZ3VyYXRpb24gZvxyIGRhcyBCYWNrZW5kLU1vZHVsIGRlcyBWaXJ0dWFsRlMuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+SVRTTSBDb3JlPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6SVRTTUNvcmU8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ0V4PSIiPktlcm5lbDo6U3lzdGVtOjpWaXJ0dWFsRlM6OkRCPC9TdHJpbmc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgo8L290cnNfY29uZmlnPgo=
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSIgPz4KPG90cnNfY29uZmlnIHZlcnNpb249IjEuMCIgaW5pdD0iQ29uZmlnIj4KICAgIDxDVlM+JElkOiBJVFNNU2VydmljZS54bWwsdiAxLjcgMjAwOC8wOC8wOCAxMzowNDoyMyB1YiBFeHAgJDwvQ1ZTPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iRnJvbnRlbmQ6Ok1vZHVsZSMjI0FnZW50SVRTTVNlcnZpY2UiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5Gcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgQWdlbnRJVFNNU2VydmljZSBvYmplY3QgaW4gdGhlIGFnZW50IGludGVyZmFjZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RnJvbnRlbmRtb2R1bC1SZWdpc3RyYXRpb24gZGVzIEFnZW50SVRTTVNlcnZpY2UtT2JqZWt0cyBpbSBBZ2VudC1JbnRlcmZhY2UuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+SVRTTSBDb3JlPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50OjpNb2R1bGVSZWdpc3RyYXRpb248L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8RnJvbnRlbmRNb2R1bGVSZWc+CiAgICAgICAgICAgICAgICA8R3JvdXA+aXRzbS1zZXJ2aWNlPC9Hcm91cD4KICAgICAgICAgICAgICAgIDxHcm91cFJvPml0c20tc2VydmljZTwvR3JvdXBSbz4KICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5JVFNNIFNlcnZpY2UgT3ZlcnZpZXc8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgPE5hdkJhck5hbWU+U2VydmljZTwvTmF2QmFyTmFtZT4KICAgICAgICAgICAgICAgIDxUaXRsZT5TZXJ2aWNlPC9UaXRsZT4KICAgICAgICAgICAgICAgIDxOYXZCYXI+CiAgICAgICAgICAgICAgICAgICAgPERlc2NyaXB0aW9uPlNlcnZpY2UtQXJlYTwvRGVzY3JpcHRpb24+CiAgICAgICAgICAgICAgICAgICAgPE5hbWU+U2VydmljZTwvTmFtZT4KICAgICAgICAgICAgICAgICAgICA8VHlwZT5NZW51PC9UeXBlPgogICAgICAgICAgICAgICAgICAgIDxCbG9jaz5JdGVtQXJlYTwvQmxvY2s+CiAgICAgICAgICAgICAgICAgICAgPEltYWdlPml0c21fc2VydmljZS5wbmc8L0ltYWdlPgogICAgICAgICAgICAgICAgICAgIDxMaW5rPkFjdGlvbj1BZ2VudElUU01TZXJ2aWNlPC9MaW5rPgogICAgICAgICAgICAgICAgICAgIDxOYXZCYXI+U2VydmljZTwvTmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxBY2Nlc3NLZXk+PC9BY2Nlc3NLZXk+CiAgICAgICAgICAgICAgICAgICAgPFByaW8+MzEwMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyPgogICAgICAgICAgICAgICAgPE5hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8RGVzY3JpcHRpb24+U2VydmljZSBPdmVydmlldzwvRGVzY3JpcHRpb24+CiAgICAgICAgICAgICAgICAgICAgPE5hbWU+U2VydmljZTwvTmFtZT4KICAgICAgICAgICAgICAgICAgICA8SW1hZ2U+b3ZlcnZpZXcucG5nPC9JbWFnZT4KICAgICAgICAgICAgICAgICAgICA8TGluaz5BY3Rpb249QWdlbnRJVFNNU2VydmljZTwvTGluaz4KICAgICAgICAgICAgICAgICAgICA8VHlwZT48L1R5cGU+CiAgICAgICAgICAgICAgICAgICAgPEJsb2NrPjwvQmxvY2s+CiAgICAgICAgICAgICAgICAgICAgPE5hdkJhcj5TZXJ2aWNlPC9OYXZCYXI+CiAgICAgICAgICAgICAgICAgICAgPEFjY2Vzc0tleT48L0FjY2Vzc0tleT4KICAgICAgICAgICAgICAgICAgICA8UHJpbz4xMDA8L1ByaW8+CiAgICAgICAgICAgICAgICA8L05hdkJhcj4KICAgICAgICAgICAgPC9Gcm9udGVuZE1vZHVsZVJlZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJGcm9udGVuZDo6TW9kdWxlIyMjQWdlbnRJVFNNU0xBIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RnJvbnRlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIEFnZW50SVRTTVNMQSBvYmplY3QgaW4gdGhlIGFnZW50IGludGVyZmFjZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RnJvbnRlbmRtb2R1bC1SZWdpc3RyYXRpb24gZGVzIEFnZW50SVRTTVNMQS1PYmpla3RzIGltIEFnZW50LUludGVyZmFjZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5JVFNNIENvcmU8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ6Ok1vZHVsZVJlZ2lzdHJhdGlvbjwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxGcm9udGVuZE1vZHVsZVJlZz4KICAgICAgICAgICAgICAgIDxHcm91cFJvPml0c20tc2VydmljZTwvR3JvdXBSbz4KICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5JVFNNIFNMQSBPdmVydmlldzwvRGVzY3JpcHRpb24+CiAgICAgICAgICAgICAgICA8TmF2QmFyTmFtZT5TZXJ2aWNlPC9OYXZCYXJOYW1lPgogICAgICAgICAgICAgICAgPFRpdGxlPlNMQTwvVGl0bGU+CiAgICAgICAgICAgICAgICA8TmF2QmFyPgogICAgICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5TTEEgT3ZlcnZpZXc8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgICAgIDxOYW1lPlNMQTwvTmFtZT4KICAgICAgICAgICAgICAgICAgICA8SW1hZ2U+b3ZlcnZpZXcucG5nPC9JbWFnZT4KICAgICAgICAgICAgICAgICAgICA8TGluaz5BY3Rpb249QWdlbnRJVFNNU0xBPC9MaW5rPgogICAgICAgICAgICAgICAgICAgIDxUeXBlPjwvVHlwZT4KICAgICAgICAgICAgICAgICAgICA8QmxvY2s+PC9CbG9jaz4KICAgICAgICAgICAgICAgICAgICA8TmF2QmFyPlNlcnZpY2U8L05hdkJhcj4KICAgICAgICAgICAgICAgICAgICA8QWNjZXNzS2V5PjwvQWNjZXNzS2V5PgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjIwMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyPgogICAgICAgICAgICA8L0Zyb250ZW5kTW9kdWxlUmVnPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZyb250ZW5kOjpNb2R1bGUjIyNBZ2VudElUU01TZXJ2aWNlWm9vbSIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkZyb250ZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBBZ2VudElUU01TZXJ2aWNlWm9vbSBvYmplY3QgaW4gdGhlIGFnZW50IGludGVyZmFjZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RnJvbnRlbmRtb2R1bC1SZWdpc3RyYXRpb24gZGVzIEFnZW50SVRTTVNlcnZpY2Vab29tLU9iamVrdHMgaW0gQWdlbnQtSW50ZXJmYWNlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPklUU00gQ29yZTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDo6TW9kdWxlUmVnaXN0cmF0aW9uPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEZyb250ZW5kTW9kdWxlUmVnPgogICAgICAgICAgICAgICAgPEdyb3VwUm8+aXRzbS1zZXJ2aWNlPC9Hcm91cFJvPgogICAgICAgICAgICAgICAgPERlc2NyaXB0aW9uPlNlcnZpY2UgWm9vbTwvRGVzY3JpcHRpb24+CiAgICAgICAgICAgICAgICA8VGl0bGU+Wm9vbTwvVGl0bGU+CiAgICAgICAgICAgICAgICA8TmF2QmFyTmFtZT5TZXJ2aWNlPC9OYXZCYXJOYW1lPgogICAgICAgICAgICA8L0Zyb250ZW5kTW9kdWxlUmVnPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZyb250ZW5kOjpNb2R1bGUjIyNBZ2VudElUU01TZXJ2aWNlUHJpbnQiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5Gcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgQWdlbnRJVFNNU2VydmljZVByaW50IG9iamVjdCBpbiB0aGUgYWdlbnQgaW50ZXJmYWNlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5Gcm9udGVuZG1vZHVsLVJlZ2lzdHJhdGlvbiBkZXMgQWdlbnRJVFNNU2VydmljZVByaW50LU9iamVrdHMgaW0gQWdlbnQtSW50ZXJmYWNlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPklUU00gQ29yZTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDo6TW9kdWxlUmVnaXN0cmF0aW9uPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEZyb250ZW5kTW9kdWxlUmVnPgogICAgICAgICAgICAgICAgPEdyb3VwUm8+aXRzbS1zZXJ2aWNlPC9Hcm91cFJvPgogICAgICAgICAgICAgICAgPERlc2NyaXB0aW9uPlNlcnZpY2UgUHJpbnQ8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgPFRpdGxlPlByaW50PC9UaXRsZT4KICAgICAgICAgICAgICAgIDxOYXZCYXJOYW1lPlNlcnZpY2U8L05hdkJhck5hbWU+CiAgICAgICAgICAgIDwvRnJvbnRlbmRNb2R1bGVSZWc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iRnJvbnRlbmQ6Ok1vZHVsZSMjI0FnZW50SVRTTVNMQVpvb20iIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5Gcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgQWdlbnRJVFNNU0xBWm9vbSBvYmplY3QgaW4gdGhlIGFnZW50IGludGVyZmFjZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RnJvbnRlbmRtb2R1bC1SZWdpc3RyYXRpb24gZGVzIEFnZW50SVRTTVNMQVpvb20tT2JqZWt0cyBpbSBBZ2VudC1JbnRlcmZhY2UuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+SVRTTSBDb3JlPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFnZW50OjpNb2R1bGVSZWdpc3RyYXRpb248L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8RnJvbnRlbmRNb2R1bGVSZWc+CiAgICAgICAgICAgICAgICA8R3JvdXBSbz5pdHNtLXNlcnZpY2U8L0dyb3VwUm8+CiAgICAgICAgICAgICAgICA8RGVzY3JpcHRpb24+U0xBIFpvb208L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgPFRpdGxlPlpvb208L1RpdGxlPgogICAgICAgICAgICAgICAgPE5hdkJhck5hbWU+U2VydmljZTwvTmF2QmFyTmFtZT4KICAgICAgICAgICAgPC9Gcm9udGVuZE1vZHVsZVJlZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJGcm9udGVuZDo6TW9kdWxlIyMjQWdlbnRJVFNNU0xBUHJpbnQiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5Gcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgQWdlbnRJVFNNU0xBUHJpbnQgb2JqZWN0IGluIHRoZSBhZ2VudCBpbnRlcmZhY2UuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkZyb250ZW5kbW9kdWwtUmVnaXN0cmF0aW9uIGRlcyBBZ2VudElUU01TTEFQcmludC1PYmpla3RzIGltIEFnZW50LUludGVyZmFjZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5JVFNNIENvcmU8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ6Ok1vZHVsZVJlZ2lzdHJhdGlvbjwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxGcm9udGVuZE1vZHVsZVJlZz4KICAgICAgICAgICAgICAgIDxHcm91cFJvPml0c20tc2VydmljZTwvR3JvdXBSbz4KICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbj5TTEEgUHJpbnQ8L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgPFRpdGxlPlByaW50PC9UaXRsZT4KICAgICAgICAgICAgICAgIDxOYXZCYXJOYW1lPlNlcnZpY2U8L05hdkJhck5hbWU+CiAgICAgICAgICAgIDwvRnJvbnRlbmRNb2R1bGVSZWc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iSVRTTVNlcnZpY2U6OkZyb250ZW5kOjpNZW51TW9kdWxlIyMjMDAwLUJhY2siIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5Nb2R1bGUgdG8gc2hvdyBiYWNrIGxpbmsgaW4gc2VydmljZSBtZW51LjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj7cYmVyIGRpZXNlcyBNb2R1bCB3aXJkIGRlciBadXL8Y2stTGluayBpbiBkZXIgTGlua2xlaXN0ZSBkZXIgU2VydmljZS1BbnNpY2h0IGFuZ2V6ZWlndC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5JVFNNIENvcmU8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ6OklUU01TZXJ2aWNlOjpNZW51TW9kdWxlPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik1vZHVsZSI+S2VybmVsOjpPdXRwdXQ6OkhUTUw6OklUU01TZXJ2aWNlTWVudUdlbmVyaWM8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik5hbWUiPkJhY2s8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IkRlc2NyaXB0aW9uIj5CYWNrPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJBY3Rpb24iPjwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iTGluayI+QWN0aW9uPUFnZW50SVRTTVNlcnZpY2U8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IkxpbmtQYXJhbSI+b25jbGljaz0iamF2YXNjcmlwdDpoaXN0b3J5LmJhY2soKTsgcmV0dXJuIGZhbHNlOyI8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJJVFNNU2VydmljZTo6RnJvbnRlbmQ6Ok1lbnVNb2R1bGUjIyMxMDAtUHJpbnQiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5Nb2R1bGUgdG8gc2hvdyBwcmludCBsaW5rIGluIHNlcnZpY2UgbWVudS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+3GJlciBkaWVzZXMgTW9kdWwgd2lyZCBkZXIgRHJ1Y2tlbi1MaW5rIGluIGRlciBMaW5rbGVpc3RlIGRlciBTZXJ2aWNlLUFuc2ljaHQgYW5nZXplaWd0LjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPklUU00gQ29yZTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDo6SVRTTVNlcnZpY2U6Ok1lbnVNb2R1bGU8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iTW9kdWxlIj5LZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTVNlcnZpY2VNZW51R2VuZXJpYzwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iTmFtZSI+UHJpbnQ8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IkRlc2NyaXB0aW9uIj5QcmludDwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iQWN0aW9uIj5BZ2VudElUU01TZXJ2aWNlUHJpbnQ8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IkxpbmsiPkFjdGlvbj1BZ2VudElUU01TZXJ2aWNlUHJpbnQmYW1wO1NlcnZpY2VJRD0kUURhdGF7IlNlcnZpY2VJRCJ9PC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJMaW5rUGFyYW0iPnRhcmdldD0icHJpbnQiPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iSVRTTVNlcnZpY2U6OkZyb250ZW5kOjpNZW51TW9kdWxlIyMjMjAwLUxpbmsiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5Nb2R1bGUgdG8gc2hvdyB0aGUgbGluayBsaW5rIGluIHNlcnZpY2UgbWVudS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+3GJlciBkaWVzZXMgTW9kdWwgd2lyZCBkZXIgVmVya278cGZlbi1MaW5rIGluIGRlciBMaW5rbGVpc3RlIGRlciBTZXJ2aWNlLUFuc2ljaHQgYW5nZXplaWd0LjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPklUU00gQ29yZTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDo6SVRTTVNlcnZpY2U6Ok1lbnVNb2R1bGU8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iTW9kdWxlIj5LZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTVNlcnZpY2VNZW51TGluazwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iTmFtZSI+TGluazwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iRGVzY3JpcHRpb24iPkxpbms8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IkFjdGlvbiI+QWdlbnRJVFNNU2VydmljZTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iTGluayI+QWN0aW9uPUFnZW50TGlua09iamVjdCZhbXA7U291cmNlT2JqZWN0PVNlcnZpY2UmYW1wO1NvdXJjZUtleT0kUURhdGF7IlNlcnZpY2VJRCJ9PC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iSVRTTVNMQTo6RnJvbnRlbmQ6Ok1lbnVNb2R1bGUjIyMwMDAtQmFjayIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPk1vZHVsZSB0byBzaG93IGJhY2sgbGluayBpbiBzbGEgbWVudS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+3GJlciBkaWVzZXMgTW9kdWwgd2lyZCBkZXIgWnVy/GNrLUxpbmsgaW4gZGVyIExpbmtsZWlzdGUgZGVyIFNMQS1BbnNpY2h0IGFuZ2V6ZWlndC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5JVFNNIENvcmU8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWdlbnQ6OklUU01TTEE6Ok1lbnVNb2R1bGU8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iTW9kdWxlIj5LZXJuZWw6Ok91dHB1dDo6SFRNTDo6SVRTTVNMQU1lbnVHZW5lcmljPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJOYW1lIj5CYWNrPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJEZXNjcmlwdGlvbiI+QmFjazwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iQWN0aW9uIj48L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IkxpbmsiPkFjdGlvbj1BZ2VudElUU01TTEE8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IkxpbmtQYXJhbSI+b25jbGljaz0iamF2YXNjcmlwdDpoaXN0b3J5LmJhY2soKTsgcmV0dXJuIGZhbHNlOyI8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJJVFNNU0xBOjpGcm9udGVuZDo6TWVudU1vZHVsZSMjIzEwMC1QcmludCIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPk1vZHVsZSB0byBzaG93IHByaW50IGxpbmsgaW4gc2xhIG1lbnUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPtxiZXIgZGllc2VzIE1vZHVsIHdpcmQgZGVyIERydWNrZW4tTGluayBpbiBkZXIgTGlua2xlaXN0ZSBkZXIgU0xBLUFuc2ljaHQgYW5nZXplaWd0LjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPklUU00gQ29yZTwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZ2VudDo6SVRTTVNMQTo6TWVudU1vZHVsZTwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJNb2R1bGUiPktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJVFNNU0xBTWVudUdlbmVyaWM8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik5hbWUiPlByaW50PC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJEZXNjcmlwdGlvbiI+UHJpbnQ8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IkFjdGlvbiI+QWdlbnRJVFNNU0xBUHJpbnQ8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IkxpbmsiPkFjdGlvbj1BZ2VudElUU01TTEFQcmludCZhbXA7U0xBSUQ9JFFEYXRheyJTTEFJRCJ9PC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJMaW5rUGFyYW0iPnRhcmdldD0icHJpbnQiPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgo8L290cnNfY29uZmlnPgo=
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSIgPz4KPG90cnNfY29uZmlnIHZlcnNpb249IjEuMCIgaW5pdD0iQ29uZmlnIj4KICAgIDxDVlM+JElkOiBUaWNrZXRJVFNNU2VydmljZS54bWwsdiAxLjEwIDIwMTAvMDMvMDEgMTM6MDE6MTkgdWIgRXhwICQ8L0NWUz4KICAgIDxDVlM+JE9sZElkOiBUaWNrZXQueG1sLHYgMS4yNzAuMi45IDIwMTAvMDIvMjYgMTE6NTE6MTYgbWFydGluIEV4cCAkPC9DVlM+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJUaWNrZXQ6OlNlcnZpY2UiIFJlcXVpcmVkPSIxIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5JZiB0aWNrZXQgc2VydmljZS9TTEEgZmVhdHVyZSBpcyBlbmFibGVkLCB5b3UgY2FuIGRlZmluZSB0aWNrZXQgc2VydmljZXMgYW5kIFNMQXMgZm9yIHRpY2tldHMgKGUuIGcuIGVtYWlsLCBkZXNrdG9wLCBuZXR3b3JrLCAuLi4pLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5XZW5uIGRhcyBUaWNrZXQtU2VydmljZS9TTEEgRmVhdHVyZSBha3RpdmllcnQgaXN0LCBr9m5uZW4gVGlja2V0IFNlcnZpY2VzIHVuZCBTTEFzIHBybyBUaWNrZXQgZ2VzZXR6dCB3ZXJkZW4gKHouIEIuIEVtYWlsLCBBcmJlaXRzcGxhdHosIE5ldHp3ZXJrLCAuLi4pLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPlRpY2tldDwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkNvcmU6OlRpY2tldDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxPcHRpb24gU2VsZWN0ZWRJRD0iMSI+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjAiPk5vPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSIxIj5ZZXM8L0l0ZW0+CiAgICAgICAgICAgIDwvT3B0aW9uPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KPC9vdHJzX2NvbmZpZz4K
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9iZ19JVFNNQ29yZS5wbSAtIHRoZSBidWxnYXJpYW4gdHJhbnNsYXRpb24gb2YgSVRTTUNvcmUKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgQ29weXJpZ2h0IChDKSAyMDA3LTIwMDggTWlsZW4gS291dGV2CiMgLS0KIyAkSWQ6IGJnX0lUU01Db3JlLnBtLHYgMS4xNSAyMDEwLzA2LzAxIDE5OjI1OjIyIG1iIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmJnX0lUU01Db3JlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS4xNSAkKSBbMV07CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgIG15ICRMYW5nID0gJFNlbGYtPntUcmFuc2xhdGlvbn07CgogICAgcmV0dXJuIGlmIHJlZiAkTGFuZyBuZSAnSEFTSCc7CgogICAgJExhbmctPnsnQ3JpdGljYWxpdHknfSAgICAgICAgICAgICAgICAgICAgICAgICA9ICfK8Ojy6Pft7vHyJzsKICAgICRMYW5nLT57J0ltcGFjdCd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnwuvo/+3o5Sc7CiAgICAkTGFuZy0+eydDcml0aWNhbGl0eSA8LT4gSW1wYWN0IDwtPiBQcmlvcml0eSd9ID0gJ8rw6PLo9+3u8fI8LT7C6+j/7ejlPC0+z/Do7vLo8uXyJzsKICAgICRMYW5nLT57J2FsbG9jYXRpb24nfSAgICAgICAgICAgICAgICAgICAgICAgICAgPSAn7u/w5eTl6+XtJzsKICAgICRMYW5nLT57J1ByaW9yaXR5IGFsbG9jYXRpb24nfSAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1JlbGV2YW50IHRvJ30gICAgICAgICAgICAgICAgICAgICAgICAgPSAn0fru8uLl8uXtIPEnOwogICAgJExhbmctPnsnSW5jbHVkZXMnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICfC6uv+9+Xt6Cc7CiAgICAkTGFuZy0+eydQYXJ0IG9mJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ9fg8fIg7vInOwogICAgJExhbmctPnsnRGVwZW5kcyBvbid9ICAgICAgICAgICAgICAgICAgICAgICAgICA9ICfH4OLo8egg7vInOwogICAgJExhbmctPnsnUmVxdWlyZWQgZm9yJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICfN5e7h9e7k6Owg5+AnOwogICAgJExhbmctPnsnQ29ubmVjdGVkIHRvJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICfR4vrw5+DtIPEnOwogICAgJExhbmctPnsnQWx0ZXJuYXRpdmUgdG8nfSAgICAgICAgICAgICAgICAgICAgICA9ICfA6/Ll8O3g8uji5e0g7eAnOwogICAgJExhbmctPnsnSW5jaWRlbnQgU3RhdGUnfSAgICAgICAgICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnQ3VycmVudCBJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnQ3VycmVudCBTdGF0ZSd9ICAgICAgICAgICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnU2VydmljZS1BcmVhJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnTWluaW11bSBUaW1lIEJldHdlZW4gSW5jaWRlbnRzJ30gICAgICA9ICfM6O3o7ODr7e4g4vDl7OUg7OXm5PMg6O326OTl7fLo8uUnOwogICAgJExhbmctPnsnU2VydmljZSBPdmVydmlldyd9ICAgICAgICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnU0xBIE92ZXJ2aWV3J30gICAgICAgICAgICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnQXNzb2NpYXRlZCBTZXJ2aWNlcyd9ICAgICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnQXNzb2NpYXRlZCBTTEFzJ30gICAgICAgICAgICAgICAgICAgICA9ICfR4vrw5+Dt6CBTTEEg5O7j7uLu8OgnOwogICAgJExhbmctPnsnQmFjayBFbmQnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICfO8e3u4u3gIPHo8fLl7OAvQmFja0VuZCc7CiAgICAkTGFuZy0+eydEZW1vbnN0cmF0aW9uJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ8Tl7O7t8fLw4Pbo/yc7CiAgICAkTGFuZy0+eydFbmQgVXNlciBTZXJ2aWNlJ30gICAgICAgICAgICAgICAgICAgID0gJ9Px6/Pj6CDn4CDq8ODp7egg7+7y8OXh6PLl6+gnOwogICAgJExhbmctPnsnRnJvbnQgRW5kJ30gICAgICAgICAgICAgICAgICAgICAgICAgICA9ICfK6+jl7fLx6uAg8ejx8uXs4C9Gcm9udEVuZCc7CiAgICAkTGFuZy0+eydJVCBNYW5hZ2VtZW50J30gICAgICAgICAgICAgICAgICAgICAgID0gJ9Pv8ODi6+Xt6OUg7eAgyNInOwogICAgJExhbmctPnsnSVQgT3BlcmF0aW9uYWwnfSAgICAgICAgICAgICAgICAgICAgICA9ICfI0iDO7+Xw4Pbo6Cc7CiAgICAkTGFuZy0+eydPdGhlcid9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ8Tw8+PoJzsKICAgICRMYW5nLT57J1Byb2plY3QnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnz/Du5eryJzsKICAgICRMYW5nLT57J1JlcG9ydGluZyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnzvL35fLt7vHyJzsKICAgICRMYW5nLT57J1RyYWluaW5nJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnzuHz9+Xt6OUnOwogICAgJExhbmctPnsnVW5kZXJwaW5uaW5nIENvbnRyYWN0J30gICAgICAgICAgICAgICA9ICfO8e3u4uXtIOTu4+7i7vAnOwogICAgJExhbmctPnsnQXZhaWxhYmlsaXR5J30gICAgICAgICAgICAgICAgICAgICAgICA9ICfE7vHy+u/t7vHyJzsKICAgICRMYW5nLT57J0Vycm9ycyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnw/Dl+OroJzsKICAgICRMYW5nLT57J090aGVyJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnxPDz4+gnOwogICAgJExhbmctPnsnUmVjb3ZlcnkgVGltZSd9ICAgICAgICAgICAgICAgICAgICAgICA9ICfC8OXs5SDn4CDi+ufx8uDt7uL/4uDt5Sc7CiAgICAkTGFuZy0+eydSZXNvbHV0aW9uIFJhdGUnfSAgICAgICAgICAgICAgICAgICAgID0gJ8Lw5ezlIOfgIPDg5/Dl+ODi4O3lJzsKICAgICRMYW5nLT57J1Jlc3BvbnNlIFRpbWUnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnwvDl7OUg5+Ag7vLj7uLu8Cc7CiAgICAkTGFuZy0+eydUcmFuc2FjdGlvbnMnfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ9Hk5evq6C/y8ODt5+Dq9ujoJzsKCiAgICByZXR1cm4gMTsKfQoKMTsK
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9jdF9JVFNNQ29yZS5wbSAtIHRoZSBjYXRhbGFuIHRyYW5zbGF0aW9uIG9mIElUU01Db3JlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTAgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIENvcHlyaWdodCAoQykgMjAwOCBTaXN0ZW1lcyBPVElDIChpYnNhbHV0KSAtIEFudG9uaW8gTGluZGUKIyAtLQojICRJZDogY3RfSVRTTUNvcmUucG0sdiAxLjUgMjAxMC8wNi8wMSAxOToyNToyMiBtYiBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpjdF9JVFNNQ29yZTsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuNSAkKSBbMV07CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgIG15ICRMYW5nID0gJFNlbGYtPntUcmFuc2xhdGlvbn07CgogICAgcmV0dXJuIGlmIHJlZiAkTGFuZyBuZSAnSEFTSCc7CgogICAgJExhbmctPnsnQ3JpdGljYWxpdHknfSAgICAgICAgICAgICAgICAgICAgICAgICA9ICdFc3RhdCBjcu10aWMnOwogICAgJExhbmctPnsnSW1wYWN0J30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdJbXBhY3RlJzsKICAgICRMYW5nLT57J0NyaXRpY2FsaXR5IDwtPiBJbXBhY3QgPC0+IFByaW9yaXR5J30gPSAnRXN0YXQgY3LtdGljIDwtPiBJbXBhY3RlIDwtPiBQcmlvcml0YXQnOwogICAgJExhbmctPnsnYWxsb2NhdGlvbid9ICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdhc3NpZ25hcic7CiAgICAkTGFuZy0+eydQcmlvcml0eSBhbGxvY2F0aW9uJ30gICAgICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydSZWxldmFudCB0byd9ICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1JlbGV2YW50IHBlcic7CiAgICAkTGFuZy0+eydJbmNsdWRlcyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0luY2xvdSc7CiAgICAkTGFuZy0+eydQYXJ0IG9mJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1BhcnQgZGUnOwogICAgJExhbmctPnsnRGVwZW5kcyBvbid9ICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdEZXDobiBkZSc7CiAgICAkTGFuZy0+eydSZXF1aXJlZCBmb3InfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ1JlcXVlcml0IHBlcic7CiAgICAkTGFuZy0+eydDb25uZWN0ZWQgdG8nfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ0Nvbm5lY3RhdCBhJzsKICAgICRMYW5nLT57J0FsdGVybmF0aXZlIHRvJ30gICAgICAgICAgICAgICAgICAgICAgPSAnQWx0ZXJuYXRpdmEgYSc7CiAgICAkTGFuZy0+eydJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICAgICAgICAgID0gJ0VzdGF0IGRlIGxcJ2luY2lkZW50JzsKICAgICRMYW5nLT57J0N1cnJlbnQgSW5jaWRlbnQgU3RhdGUnfSAgICAgICAgICAgICAgPSAnRXN0YXQgYWN0dWFsIGRlIGxcJ2luY2lkZW50JzsKICAgICRMYW5nLT57J0N1cnJlbnQgU3RhdGUnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnRXN0YXQgYWN0dWFsJzsKICAgICRMYW5nLT57J1NlcnZpY2UtQXJlYSd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnU2VydmVpLcByZWEnOwogICAgJExhbmctPnsnTWluaW11bSBUaW1lIEJldHdlZW4gSW5jaWRlbnRzJ30gICAgICA9ICdUZW1wcyBt7W5pbSBlbnRyZSBpbmNpZGVudHMnOwogICAgJExhbmctPnsnU2VydmljZSBPdmVydmlldyd9ICAgICAgICAgICAgICAgICAgICA9ICdWaXNp8yBnZW5lcmFsIGRlbCBzZXJ2ZWknOwogICAgJExhbmctPnsnU0xBIE92ZXJ2aWV3J30gICAgICAgICAgICAgICAgICAgICAgICA9ICdWaXNp8yBnZW5lcmFsIGRlIFNMQSc7CiAgICAkTGFuZy0+eydBc3NvY2lhdGVkIFNlcnZpY2VzJ30gICAgICAgICAgICAgICAgID0gJ1NlcnZlaXMgYXNzb2NpYXRzJzsKICAgICRMYW5nLT57J0Fzc29jaWF0ZWQgU0xBcyd9ICAgICAgICAgICAgICAgICAgICAgPSAnU0xBcyBhc3NvY2lhdHMnOwogICAgJExhbmctPnsnQmFjayBFbmQnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdTZXJ2aWRvcic7CiAgICAkTGFuZy0+eydEZW1vbnN0cmF0aW9uJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ0RlbW9zdHJhY2nzJzsKICAgICRMYW5nLT57J0VuZCBVc2VyIFNlcnZpY2UnfSAgICAgICAgICAgICAgICAgICAgPSAnU2VydmVpIHVzdWFyaSBmaW5hbCc7CiAgICAkTGFuZy0+eydGcm9udCBFbmQnfSAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0NsaWVudCc7CiAgICAkTGFuZy0+eydJVCBNYW5hZ2VtZW50J30gICAgICAgICAgICAgICAgICAgICAgID0gJ0dlc3Rp8yBJVCc7CiAgICAkTGFuZy0+eydJVCBPcGVyYXRpb25hbCd9ICAgICAgICAgICAgICAgICAgICAgID0gJ09wZXJhY2nzIElUJzsKICAgICRMYW5nLT57J090aGVyJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnQWx0cmVzJzsKICAgICRMYW5nLT57J1Byb2plY3QnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnUHJvamVjdGUnOwogICAgJExhbmctPnsnUmVwb3J0aW5nJ30gICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdJbmZvcm1lcyc7CiAgICAkTGFuZy0+eydUcmFpbmluZyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0Zvcm1hY2nzJzsKICAgICRMYW5nLT57J1VuZGVycGlubmluZyBDb250cmFjdCd9ICAgICAgICAgICAgICAgPSAnQ29udHJhY3RlIGRlIHN1cG9ydCc7CiAgICAkTGFuZy0+eydBdmFpbGFiaWxpdHknfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ0Rpc3BvbmliaWxpdGF0JzsKICAgICRMYW5nLT57J0Vycm9ycyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnRXJyb3JzJzsKICAgICRMYW5nLT57J090aGVyJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnQWx0cmVzJzsKICAgICRMYW5nLT57J1JlY292ZXJ5IFRpbWUnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnVGVtcHMgZGUgcmVjdXBlcmFjafMnOwogICAgJExhbmctPnsnUmVzb2x1dGlvbiBSYXRlJ30gICAgICAgICAgICAgICAgICAgICA9ICdQZXJjZW50YXRnZSBkZSByZXNvbHVjafMnOwogICAgJExhbmctPnsnUmVzcG9uc2UgVGltZSd9ICAgICAgICAgICAgICAgICAgICAgICA9ICdUZW1wcyBkZSByZXNwb3N0YSc7CiAgICAkTGFuZy0+eydUcmFuc2FjdGlvbnMnfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ1RyYW5zYWNjaW9ucyc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9jel9JVFNNQ29yZS5wbSAtIHRoZSBjemVjaCB0cmFuc2xhdGlvbiBvZiBJVFNNQ29yZQojIENvcHlyaWdodCAoQykgMjAwMS0yMDEwIE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyBDb3B5cmlnaHQgKEMpIDIwMDctMjAwOCBNaWxlbiBLb3V0ZXYKIyBDb3B5cmlnaHQgKEMpIDIwMTAgTzJCUy5jb20sIHMgci5vLiBKYWt1YiBIYW51cwojIC0tCiMgJElkOiBjel9JVFNNQ29yZS5wbSx2IDEuMTYgMjAxMC8wNi8wMSAxOToyNToyMiBtYiBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpjel9JVFNNQ29yZTsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuMTYgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0NyaXRpY2FsaXR5J30gICAgICAgICAgICAgICAgICAgICAgICAgPSAnS3JpdGnobm9zdCc7CiAgICAkTGFuZy0+eydJbXBhY3QnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1ZsaXYnOwogICAgJExhbmctPnsnQ3JpdGljYWxpdHkgPC0+IEltcGFjdCA8LT4gUHJpb3JpdHknfSA9ICdLcml0aehub3N0PC0+VmxpdjwtPlByaW9yaXRhJzsKICAgICRMYW5nLT57J1ByaW9yaXR5IGFsbG9jYXRpb24nfSAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J2FsbG9jYXRpb24nfSAgICAgICAgICAgICAgICAgICAgICAgICAgPSAncPhpZOxsaXQnOwogICAgJExhbmctPnsnUmVsZXZhbnQgdG8nfSAgICAgICAgICAgICAgICAgICAgICAgICA9ICdSZWxldmFudG7tIGsnOwogICAgJExhbmctPnsnSW5jbHVkZXMnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdaYWhybnVqZSc7CiAgICAkTGFuZy0+eydQYXJ0IG9mJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ8jhc3Qgeic7CiAgICAkTGFuZy0+eydEZXBlbmRzIG9uJ30gICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1phbGW+7SBuYSc7CiAgICAkTGFuZy0+eydSZXF1aXJlZCBmb3InfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ1BvvmFkb3Zhbv0gcHJvJzsKICAgICRMYW5nLT57J0Nvbm5lY3RlZCB0byd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnU3BvamVuIHMnOwogICAgJExhbmctPnsnQWx0ZXJuYXRpdmUgdG8nfSAgICAgICAgICAgICAgICAgICAgICA9ICdBbHRlcm5hdGl2bu0gayc7CiAgICAkTGFuZy0+eydJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICAgICAgICAgID0gJ1N0YXYgSW5jaWRlbnR1JzsKICAgICRMYW5nLT57J0N1cnJlbnQgSW5jaWRlbnQgU3RhdGUnfSAgICAgICAgICAgICAgPSAnU2916GFzbv0gU3RhdiBJbmNpZGVudHUnOwogICAgJExhbmctPnsnQ3VycmVudCBTdGF0ZSd9ICAgICAgICAgICAgICAgICAgICAgICA9ICdTb3XoYXNu/SBTdGF2JzsKICAgICRMYW5nLT57J1NlcnZpY2UtQXJlYSd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnUHJvc3RvciDaZHK+YnknOwogICAgJExhbmctPnsnTWluaW11bSBUaW1lIEJldHdlZW4gSW5jaWRlbnRzJ30gICAgICA9ICdNaW5pbeFsbu0g6GFzIG1lemkgaW5jaWRlbnR5JzsKICAgICRMYW5nLT57J1NlcnZpY2UgT3ZlcnZpZXcnfSAgICAgICAgICAgICAgICAgICAgPSAnUPhlaGxlZCBTbHW+YnknOwogICAgJExhbmctPnsnU0xBIE92ZXJ2aWV3J30gICAgICAgICAgICAgICAgICAgICAgICA9ICdTTEEgUPhlaGxlZCc7CiAgICAkTGFuZy0+eydBc3NvY2lhdGVkIFNlcnZpY2VzJ30gICAgICAgICAgICAgICAgID0gJ1D4afhhemVu6SBTbHW+YnknOwogICAgJExhbmctPnsnQXNzb2NpYXRlZCBTTEFzJ30gICAgICAgICAgICAgICAgICAgICA9ICdQ+Gn4YXplbukgU0xBIHNtbG91dnknOwogICAgJExhbmctPnsnQmFjayBFbmQnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICda4WtsYWRu7SByb3pocmFu7S9CYWNrRW5kJzsKICAgICRMYW5nLT57J0RlbW9uc3RyYXRpb24nfSAgICAgICAgICAgICAgICAgICAgICAgPSAnVWvhemthJzsKICAgICRMYW5nLT57J0VuZCBVc2VyIFNlcnZpY2UnfSAgICAgICAgICAgICAgICAgICAgPSAnU2x1vmJ5IGtvbmNvdv1tIHW+aXZhdGVs+W0nOwogICAgJExhbmctPnsnRnJvbnQgRW5kJ30gICAgICAgICAgICAgICAgICAgICAgICAgICA9ICda4Wthem5pY2vpIHJvemhyYW7tL0Zyb250RW5kJzsKICAgICRMYW5nLT57J0lUIE1hbmFnZW1lbnQnfSAgICAgICAgICAgICAgICAgICAgICAgPSAn2O16ZW7tIElUJzsKICAgICRMYW5nLT57J0lUIE9wZXJhdGlvbmFsJ30gICAgICAgICAgICAgICAgICAgICAgPSAnSVQgT3BlcmFjZSc7CiAgICAkTGFuZy0+eydPdGhlcid9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0RhbLntJzsKICAgICRMYW5nLT57J1Byb2plY3QnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnUHJvamVrdCc7CiAgICAkTGFuZy0+eydSZXBvcnRpbmcnfSAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1JlcG9ydGluZyc7CiAgICAkTGFuZy0+eydUcmFpbmluZyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ6lrb2xlbu0nOwogICAgJExhbmctPnsnVW5kZXJwaW5uaW5nIENvbnRyYWN0J30gICAgICAgICAgICAgICA9ICda4WtsYWRu7SBzbWxvdXZhJzsKICAgICRMYW5nLT57J0F2YWlsYWJpbGl0eSd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnRG9zdHVwbm9zdCc7CiAgICAkTGFuZy0+eydFcnJvcnMnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0NoeWJ5JzsKICAgICRMYW5nLT57J090aGVyJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnRGFsue0nOwogICAgJExhbmctPnsnUmVjb3ZlcnkgVGltZSd9ICAgICAgICAgICAgICAgICAgICAgICA9ICfIYXMgT2Jub3Z5JzsKICAgICRMYW5nLT57J1Jlc29sdXRpb24gUmF0ZSd9ICAgICAgICAgICAgICAgICAgICAgPSAnyGFzINhluWVu7Sc7CiAgICAkTGFuZy0+eydSZXNwb25zZSBUaW1lJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ8hhcyBPZHBvduxkaSc7CiAgICAkTGFuZy0+eydUcmFuc2FjdGlvbnMnfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ1RyYW5zYWtjZSc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9kYV9JVFNNQ29yZS5wbSAtIHByb3ZpZGVzIGRhIChEYW5pc2gpIGxhbmd1YWdlIHRyYW5zbGF0aW9uCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTAgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBkYV9JVFNNQ29yZS5wbSx2IDEuMSAyMDEwLzA2LzI1IDA4OjU2OjU3IG1iIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmRhX0lUU01Db3JlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS4xICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAkTGFuZy0+eydDcml0aWNhbGl0eSd9ICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0tyaXRpa2FsaXRldCc7CiAgICAkTGFuZy0+eydJbXBhY3QnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1Dldmlya25pbmcnOwogICAgJExhbmctPnsnQ3JpdGljYWxpdHkgPC0+IEltcGFjdCA8LT4gUHJpb3JpdHknfSA9ICdLcml0aWthbGl0ZXQgPC0+IFDldmlya25pbmcgPC0+IFByaW9yaXRldCc7CiAgICAkTGFuZy0+eydhbGxvY2F0ZSd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ3RpbGRlbGUnOwogICAgJExhbmctPnsnUmVsZXZhbnQgdG8nfSAgICAgICAgICAgICAgICAgICAgICAgICA9ICdSZWxldmFudCBmb3InOwogICAgJExhbmctPnsnSW5jbHVkZXMnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdJbmRrbHVkZXJlJzsKICAgICRMYW5nLT57J1BhcnQgb2YnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnRGVsIGFmJzsKICAgICRMYW5nLT57J0RlcGVuZHMgb24nfSAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnQWZo5m5nZXIgYWYnOwogICAgJExhbmctPnsnUmVxdWlyZWQgZm9yJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICdLcuZ2ZXMgZm9yJzsKICAgICRMYW5nLT57J0Nvbm5lY3RlZCB0byd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnRm9yYnVuZGV0IHRpbCc7CiAgICAkTGFuZy0+eydBbHRlcm5hdGl2ZSB0byd9ICAgICAgICAgICAgICAgICAgICAgID0gJ0FsdGVybmF0aXYgdGlsJzsKICAgICRMYW5nLT57J0luY2lkZW50IFN0YXRlJ30gICAgICAgICAgICAgICAgICAgICAgPSAnSW5jaWRlbnQgdGlsc3RhbmQnOwogICAgJExhbmctPnsnQ3VycmVudCBJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICA9ICdOdXbmcmVuZGUgSW5jaWRlbnQgdGlsc3RhbmQnOwogICAgJExhbmctPnsnQ3VycmVudCBTdGF0ZSd9ICAgICAgICAgICAgICAgICAgICAgICA9ICdOdXbmcmVuZGUgdGlsc3RhbmQnOwogICAgJExhbmctPnsnU2VydmljZS1BcmVhJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICdTZXJ2aWNlIG9tcuVkZSc7CiAgICAkTGFuZy0+eydNaW5pbXVtIFRpbWUgQmV0d2VlbiBJbmNpZGVudHMnfSAgICAgID0gJ01pbmltdW1zdGlkIG1lbGxlbSBJbmNpZGVudHMnOwogICAgJExhbmctPnsnU2VydmljZSBPdmVydmlldyd9ICAgICAgICAgICAgICAgICAgICA9ICdTZXJ2aWNlIG92ZXJzaWd0JzsKICAgICRMYW5nLT57J1NMQSBPdmVydmlldyd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnU0xBIG92ZXJzaWd0JzsKICAgICRMYW5nLT57J0Fzc29jaWF0ZWQgU2VydmljZXMnfSAgICAgICAgICAgICAgICAgPSAnVGlsa255dHRlZGUgc2VydmljZXMnOwogICAgJExhbmctPnsnQXNzb2NpYXRlZCBTTEFzJ30gICAgICAgICAgICAgICAgICAgICA9ICdUaWxrbnl0dGVkZSBTTEFzJzsKICAgICRMYW5nLT57J0JhY2sgRW5kJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnQmFja2VuZCc7CiAgICAkTGFuZy0+eydEZW1vbnN0cmF0aW9uJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ0RlbW9uc3RyYXRpb24nOwogICAgJExhbmctPnsnRW5kIFVzZXIgU2VydmljZSd9ICAgICAgICAgICAgICAgICAgICA9ICdLdW5kZXNlcnZpY2UnOwogICAgJExhbmctPnsnRnJvbnQgRW5kJ30gICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdGcm9udGVuZCc7CiAgICAkTGFuZy0+eydJVCBNYW5hZ2VtZW50J30gICAgICAgICAgICAgICAgICAgICAgID0gJ0lUIE1hbmFnZW1lbnQnOwogICAgJExhbmctPnsnSVQgT3BlcmF0aW9uYWwnfSAgICAgICAgICAgICAgICAgICAgICA9ICdJVCBvcGVyYXRpb25lbCc7CiAgICAkTGFuZy0+eydPdGhlcid9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0FuZHJlJzsKICAgICRMYW5nLT57J1Byb2plY3QnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnUHJvamVrdCc7CiAgICAkTGFuZy0+eydSZXBvcnRpbmcnfSAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1JlcG9ydGVyaW5nJzsKICAgICRMYW5nLT57J1RyYWluaW5nJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnVHLmbmluZyc7CiAgICAkTGFuZy0+eydVbmRlcnBpbm5pbmcgQ29udHJhY3QnfSAgICAgICAgICAgICAgID0gJ1VuZGVybGlnZ2VuZGUga29udHJha3QnOwogICAgJExhbmctPnsnQXZhaWxhYmlsaXR5J30gICAgICAgICAgICAgICAgICAgICAgICA9ICdUaWxn5m5nZWxpZ2hlZCc7CiAgICAkTGFuZy0+eydFcnJvcnMnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0ZlamwnOwogICAgJExhbmctPnsnT3RoZXInfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdBbmRyZSc7CiAgICAkTGFuZy0+eydSZWNvdmVyeSBUaW1lJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ0dlbmV0YWJsZXJpbmdzdGlkJzsKICAgICRMYW5nLT57J1Jlc29sdXRpb24gUmF0ZSd9ICAgICAgICAgICAgICAgICAgICAgPSAnTPhzbmluZ3NyYXRlJzsKICAgICRMYW5nLT57J1Jlc3BvbnNlIFRpbWUnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnUmVha3Rpb25zdGlkJzsKICAgICRMYW5nLT57J1RyYW5zYWN0aW9ucyd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnVHJhbnNha3Rpb25lcic7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9kZV9JVFNNQ29yZS5wbSAtIHRoZSBnZXJtYW4gdHJhbnNsYXRpb24gb2YgSVRTTUNvcmUKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IGRlX0lUU01Db3JlLnBtLHYgMS4xNyAyMDEwLzA2LzAxIDE5OjI1OjIyIG1iIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmRlX0lUU01Db3JlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS4xNyAkKSBbMV07CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgIG15ICRMYW5nID0gJFNlbGYtPntUcmFuc2xhdGlvbn07CgogICAgcmV0dXJuIGlmIHJlZiAkTGFuZyBuZSAnSEFTSCc7CgogICAgJExhbmctPnsnQ3JpdGljYWxpdHknfSAgICAgICAgICAgICAgICAgICAgICAgICA9ICdLcml0aWthbGl05HQnOwogICAgJExhbmctPnsnSW1wYWN0J30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdBdXN3aXJrdW5nJzsKICAgICRMYW5nLT57J0NyaXRpY2FsaXR5IDwtPiBJbXBhY3QgPC0+IFByaW9yaXR5J30gPSAnS3JpdGlrYWxpdOR0IDwtPiBBdXN3aXJrdW5nIDwtPiBQcmlvcml05HQnOwogICAgJExhbmctPnsnYWxsb2NhdGlvbid9ICAgICAgICAgICAgICAgICAgICAgICAgICA9ICd6dW9yZG5lbic7CiAgICAkTGFuZy0+eydQcmlvcml0eSBhbGxvY2F0aW9uJ30gICAgICAgICAgICAgICAgID0gJ1ByaW9yaXTkdCB6dW9yZG5lbic7CiAgICAkTGFuZy0+eydSZWxldmFudCB0byd9ICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1JlbGV2YW50IGb8cic7CiAgICAkTGFuZy0+eydJbmNsdWRlcyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0JlaW5oYWx0ZXQnOwogICAgJExhbmctPnsnUGFydCBvZid9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdUZWlsIHZvbic7CiAgICAkTGFuZy0+eydEZXBlbmRzIG9uJ30gICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0jkbmd0IGFiIHZvbic7CiAgICAkTGFuZy0+eydSZXF1aXJlZCBmb3InfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ0JlbvZ0aWd0IGb8cic7CiAgICAkTGFuZy0+eydDb25uZWN0ZWQgdG8nfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ1ZlcmJ1bmRlbiBtaXQnOwogICAgJExhbmctPnsnQWx0ZXJuYXRpdmUgdG8nfSAgICAgICAgICAgICAgICAgICAgICA9ICdBbHRlcm5hdGl2IHp1JzsKICAgICRMYW5nLT57J0luY2lkZW50IFN0YXRlJ30gICAgICAgICAgICAgICAgICAgICAgPSAnVm9yZmFsbHNzdGF0dXMnOwogICAgJExhbmctPnsnQ3VycmVudCBJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICA9ICdBa3R1ZWxsZXIgVm9yZmFsbHNzdGF0dXMnOwogICAgJExhbmctPnsnQ3VycmVudCBTdGF0ZSd9ICAgICAgICAgICAgICAgICAgICAgICA9ICdBa3R1ZWxsZXIgU3RhdHVzJzsKICAgICRMYW5nLT57J1NlcnZpY2UtQXJlYSd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnU2VydmljZS1CZXJlaWNoJzsKICAgICRMYW5nLT57J01pbmltdW0gVGltZSBCZXR3ZWVuIEluY2lkZW50cyd9ICAgICAgPSAnTWluZGVzdHplaXQgendpc2NoZW4gSW5jaWRlbnRzJzsKICAgICRMYW5nLT57J1NlcnZpY2UgT3ZlcnZpZXcnfSAgICAgICAgICAgICAgICAgICAgPSAnU2VydmljZSDcYmVyc2ljaHQnOwogICAgJExhbmctPnsnU0xBIE92ZXJ2aWV3J30gICAgICAgICAgICAgICAgICAgICAgICA9ICdTTEEg3GJlcnNpY2h0JzsKICAgICRMYW5nLT57J0Fzc29jaWF0ZWQgU2VydmljZXMnfSAgICAgICAgICAgICAgICAgPSAnWnVnZWj2cmlnZSBTZXJ2aWNlcyc7CiAgICAkTGFuZy0+eydBc3NvY2lhdGVkIFNMQXMnfSAgICAgICAgICAgICAgICAgICAgID0gJ1p1Z2Vo9nJpZ2UgU0xBcyc7CiAgICAkTGFuZy0+eydCYWNrIEVuZCd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0JhY2tlbmQnOwogICAgJExhbmctPnsnRGVtb25zdHJhdGlvbid9ICAgICAgICAgICAgICAgICAgICAgICA9ICdEZW1vbnN0cmF0aW9uJzsKICAgICRMYW5nLT57J0VuZCBVc2VyIFNlcnZpY2UnfSAgICAgICAgICAgICAgICAgICAgPSAnQW53ZW5kZXItU2VydmljZSc7CiAgICAkTGFuZy0+eydGcm9udCBFbmQnfSAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0Zyb250ZW5kJzsKICAgICRMYW5nLT57J0lUIE1hbmFnZW1lbnQnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnSVQgTWFuYWdlbWVudCc7CiAgICAkTGFuZy0+eydJVCBPcGVyYXRpb25hbCd9ICAgICAgICAgICAgICAgICAgICAgID0gJ0lUIEJldHJpZWInOwogICAgJExhbmctPnsnT3RoZXInfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdTb25zdGlnZXMnOwogICAgJExhbmctPnsnUHJvamVjdCd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdQcm9qZWt0JzsKICAgICRMYW5nLT57J1JlcG9ydGluZyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnUmVwb3J0aW5nJzsKICAgICRMYW5nLT57J1RyYWluaW5nJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnVHJhaW5pbmcnOwogICAgJExhbmctPnsnVW5kZXJwaW5uaW5nIENvbnRyYWN0J30gICAgICAgICAgICAgICA9ICdVbmRlcnBpbm5pbmcgQ29udHJhY3QnOwogICAgJExhbmctPnsnQXZhaWxhYmlsaXR5J30gICAgICAgICAgICAgICAgICAgICAgICA9ICdWZXJm/GdiYXJrZWl0JzsKICAgICRMYW5nLT57J0Vycm9ycyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnRmVobGVyJzsKICAgICRMYW5nLT57J090aGVyJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnU29uc3RpZ2VzJzsKICAgICRMYW5nLT57J1JlY292ZXJ5IFRpbWUnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnV2llZGVyaGVyc3RlbGx1bmdzemVpdCc7CiAgICAkTGFuZy0+eydSZXNvbHV0aW9uIFJhdGUnfSAgICAgICAgICAgICAgICAgICAgID0gJ0z2c3VuZ3N6ZWl0JzsKICAgICRMYW5nLT57J1Jlc3BvbnNlIFRpbWUnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnUmVha3Rpb25zemVpdCc7CiAgICAkTGFuZy0+eydUcmFuc2FjdGlvbnMnfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ1RyYW5zYWt0aW9uZW4nOwoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9lc19JVFNNQ29yZS5wbSAtIHRoZSBzcGFuaXNoIHRyYW5zbGF0aW9uIG9mIElUU01Db3JlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTAgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIENvcHlyaWdodCAoQykgMjAwOCBBcXVpbGVzIENvaGVuCiMgLS0KIyAkSWQ6IGVzX0lUU01Db3JlLnBtLHYgMS42IDIwMTAvMDYvMDEgMTk6MjU6MjIgbWIgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6ZXNfSVRTTUNvcmU7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjYgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0NyaXRpY2FsaXR5J30gICAgICAgICAgICAgICAgICAgICAgICAgPSAnVXJnZW5jaWEnOwogICAgJExhbmctPnsnSW1wYWN0J30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdJbXBhY3RvJzsKICAgICRMYW5nLT57J0NyaXRpY2FsaXR5IDwtPiBJbXBhY3QgPC0+IFByaW9yaXR5J30gPSAnVXJnZW5jaWEgPC0+IEltcGFjdG8gPC0+IFByaW9yaWRhZCc7CiAgICAkTGFuZy0+eydhbGxvY2F0aW9uJ30gICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0FzaWduYXInOwogICAgJExhbmctPnsnUHJpb3JpdHkgYWxsb2NhdGlvbid9ICAgICAgICAgICAgICAgICA9ICdBc2lnbmFyIHByaW9yaWRhZCc7CiAgICAkTGFuZy0+eydSZWxldmFudCB0byd9ICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1JlbGV2YW50ZSBhJzsKICAgICRMYW5nLT57J0luY2x1ZGVzJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnSW5jbHV5ZSc7CiAgICAkTGFuZy0+eydQYXJ0IG9mJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1BhcnRlIGRlJzsKICAgICRMYW5nLT57J0RlcGVuZHMgb24nfSAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnRGVwZW5kZSBlbic7CiAgICAkTGFuZy0+eydSZXF1aXJlZCBmb3InfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ1JlcXVlcmlkbyBwYXJhJzsKICAgICRMYW5nLT57J0Nvbm5lY3RlZCB0byd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnQ29uZWN0YWRvIGEnOwogICAgJExhbmctPnsnQWx0ZXJuYXRpdmUgdG8nfSAgICAgICAgICAgICAgICAgICAgICA9ICdBbHRlcmFudGl2YSBhJzsKICAgICRMYW5nLT57J0luY2lkZW50IFN0YXRlJ30gICAgICAgICAgICAgICAgICAgICAgPSAnRXN0YWRvIGRlbCBJbmNpZGVudGUnOwogICAgJExhbmctPnsnQ3VycmVudCBJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICA9ICdFc3RhZG8gQWN0dWFsIGRlbCBJbmNpZGVudGUnOwogICAgJExhbmctPnsnQ3VycmVudCBTdGF0ZSd9ICAgICAgICAgICAgICAgICAgICAgICA9ICdFc3RhZG8gQWN0dWFsJzsKICAgICRMYW5nLT57J1NlcnZpY2UtQXJlYSd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnQXJlYS1TZXJ2aWNpbyc7CiAgICAkTGFuZy0+eydNaW5pbXVtIFRpbWUgQmV0d2VlbiBJbmNpZGVudHMnfSAgICAgID0gJ03tbmltbyBUaWVtcG8gZW50cmUgSW5jaWRlbnRlcyc7CiAgICAkTGFuZy0+eydTZXJ2aWNlIE92ZXJ2aWV3J30gICAgICAgICAgICAgICAgICAgID0gJ0Rlc2NyaXBjafNuIGRlIFNlcnZpY2lvcyc7CiAgICAkTGFuZy0+eydTTEEgT3ZlcnZpZXcnfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ0Rlc2NyaXBjafNuIGRlIFNMQSc7CiAgICAkTGFuZy0+eydBc3NvY2lhdGVkIFNlcnZpY2VzJ30gICAgICAgICAgICAgICAgID0gJ1NlcnZpY2lvcyBBc29jaWFkb3MnOwogICAgJExhbmctPnsnQXNzb2NpYXRlZCBTTEFzJ30gICAgICAgICAgICAgICAgICAgICA9ICdTTEFzIEFzb2NpYWRvcyc7CiAgICAkTGFuZy0+eydCYWNrIEVuZCd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydEZW1vbnN0cmF0aW9uJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ0RlbW9zdHJhY2nzbic7CiAgICAkTGFuZy0+eydFbmQgVXNlciBTZXJ2aWNlJ30gICAgICAgICAgICAgICAgICAgID0gJ1NlcnZpY2lvIGRlIFVzdWFyaW8gRmluYWwnOwogICAgJExhbmctPnsnRnJvbnQgRW5kJ30gICAgICAgICAgICAgICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnSVQgTWFuYWdlbWVudCd9ICAgICAgICAgICAgICAgICAgICAgICA9ICdBZG1pbmlzdHJhY2nzbiBkZSBUSSc7CiAgICAkTGFuZy0+eydJVCBPcGVyYXRpb25hbCd9ICAgICAgICAgICAgICAgICAgICAgID0gJ09wZXJhY2nzbiBkZSBUSSc7CiAgICAkTGFuZy0+eydPdGhlcid9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ090cm8nOwogICAgJExhbmctPnsnUHJvamVjdCd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdQcm95ZWN0byc7CiAgICAkTGFuZy0+eydSZXBvcnRpbmcnfSAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0luZm9ybWVzJzsKICAgICRMYW5nLT57J1RyYWluaW5nJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnRW50cmVuYW1pZW50byc7CiAgICAkTGFuZy0+eydVbmRlcnBpbm5pbmcgQ29udHJhY3QnfSAgICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydBdmFpbGFiaWxpdHknfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ0Rpc3BvbmliaWxpZGFkJzsKICAgICRMYW5nLT57J0Vycm9ycyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnRXJyb3Jlcyc7CiAgICAkTGFuZy0+eydPdGhlcid9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ090cm8nOwogICAgJExhbmctPnsnUmVjb3ZlcnkgVGltZSd9ICAgICAgICAgICAgICAgICAgICAgICA9ICdUaWVtcG8gZGUgUmVjdXBlcmFjafNuJzsKICAgICRMYW5nLT57J1Jlc29sdXRpb24gUmF0ZSd9ICAgICAgICAgICAgICAgICAgICAgPSAnVGFzYSBkZSBSZXNvbHVjafNuJzsKICAgICRMYW5nLT57J1Jlc3BvbnNlIFRpbWUnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnVGllbXBvIGRlIFJlc3B1ZXN0YSc7CiAgICAkTGFuZy0+eydUcmFuc2FjdGlvbnMnfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ1RyYW5zYWNjaW9uZXMnOwoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9mYV9JVFNNQ29yZS5wbSAtIHRoZSBwZXJzaWFuIChmYXJzaSkgdHJhbnNsYXRpb24gb2YgSVRTTUNvcmUKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgQ29weXJpZ2h0IChDKSAyMDAzLTIwMDkgQWZzaGFyIE1vaGViYmkgPGFmc2hhci5tb2hlYmJpIGF0IGdtYWlsLmNvbT4KIyAtLS0KIyAkSWQ6IGZhX0lUU01Db3JlLnBtLHYgMS4zIDIwMTAvMDYvMDEgMTk6MjU6MjIgbWIgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6ZmFfSVRTTUNvcmU7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjMgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0NyaXRpY2FsaXR5J30gICAgICAgICAgICAgICAgICAgICAgICAgPSAn2KfZh9mF24zYqic7CiAgICAkTGFuZy0+eydJbXBhY3QnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ9in2KvYsSc7CiAgICAkTGFuZy0+eydDcml0aWNhbGl0eSA8LT4gSW1wYWN0IDwtPiBQcmlvcml0eSd9ID0gJ9in2YfZhduM2KogPC0+INin2KvYsSA8LT4g2KfZiNmE2YjbjNiqJzsKICAgICRMYW5nLT57J2FsbG9jYXRpb24nfSAgICAgICAgICAgICAgICAgICAgICAgICAgPSAn2KfYrtiq2LXYp9i1JzsKICAgICRMYW5nLT57J1ByaW9yaXR5IGFsbG9jYXRpb24nfSAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1JlbGV2YW50IHRvJ30gICAgICAgICAgICAgICAgICAgICAgICAgPSAn2YXYsdiq2KjYtyDYqNinJzsKICAgICRMYW5nLT57J0luY2x1ZGVzJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAn2YXYtNiq2YXZhCDYp9iz2Kog2KjYsSc7CiAgICAkTGFuZy0+eydQYXJ0IG9mJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ9io2K7YtNuMINin2LInOwogICAgJExhbmctPnsnRGVwZW5kcyBvbid9ICAgICAgICAgICAgICAgICAgICAgICAgICA9ICfZiNin2KjYs9iq2Ycg2KfYs9iqINio2YcnOwogICAgJExhbmctPnsnUmVxdWlyZWQgZm9yJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICfZhdmI2LHYryDZhtuM2KfYsiDYp9iz2Kog2KjYsdin24wnOwogICAgJExhbmctPnsnQ29ubmVjdGVkIHRvJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICfZhdiq2LXZhCDYp9iz2Kog2KjZhyc7CiAgICAkTGFuZy0+eydBbHRlcm5hdGl2ZSB0byd9ICAgICAgICAgICAgICAgICAgICAgID0gJ9is2KfbjNqv2LLbjNmG24wg2KjYsdin24wnOwogICAgJExhbmctPnsnSW5jaWRlbnQgU3RhdGUnfSAgICAgICAgICAgICAgICAgICAgICA9ICfZiNi22LnbjNiqINix2K7Yr9in2K8nOwogICAgJExhbmctPnsnQ3VycmVudCBJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICA9ICfZiNi22LnbjNiqINis2KfYsduMINix2K7Yr9in2K8nOwogICAgJExhbmctPnsnQ3VycmVudCBTdGF0ZSd9ICAgICAgICAgICAgICAgICAgICAgICA9ICfZiNi22LnbjNiqINis2KfYsduMJzsKICAgICRMYW5nLT57J1NlcnZpY2UtQXJlYSd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAn2KjYrti0INiz2LHZiNuM2LMnOwogICAgJExhbmctPnsnTWluaW11bSBUaW1lIEJldHdlZW4gSW5jaWRlbnRzJ30gICAgICA9ICfYrdiv2KfZgtmEINiy2YXYp9mGINio24zZhiDYr9mIINix2K7Yr9in2K8nOwogICAgJExhbmctPnsnU2VydmljZSBPdmVydmlldyd9ICAgICAgICAgICAgICAgICAgICA9ICfYrtmE2KfYtdmHINiz2LHZiNuM2LMnOwogICAgJExhbmctPnsnU0xBIE92ZXJ2aWV3J30gICAgICAgICAgICAgICAgICAgICAgICA9ICfYrtmE2KfYtdmHIFNMQSc7CiAgICAkTGFuZy0+eydBc3NvY2lhdGVkIFNlcnZpY2VzJ30gICAgICAgICAgICAgICAgID0gJ9iz2LHZiNuM2LPigIzZh9in24wg2YXYsdiq2KjYtyc7CiAgICAkTGFuZy0+eydBc3NvY2lhdGVkIFNMQXMnfSAgICAgICAgICAgICAgICAgICAgID0gJ1NMQdmH2KfbjCDZhdix2KrYqNi3JzsKICAgICRMYW5nLT57J0JhY2sgRW5kJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAn2b7YtNiqINi12K3ZhtmHJzsKICAgICRMYW5nLT57J0RlbW9uc3RyYXRpb24nfSAgICAgICAgICAgICAgICAgICAgICAgPSAn2YbZhdin24zYtCc7CiAgICAkTGFuZy0+eydFbmQgVXNlciBTZXJ2aWNlJ30gICAgICAgICAgICAgICAgICAgID0gJ9iz2LHZiNuM2LMg2qnYp9ix2KjYsSDZhtmH2KfbjNuMJzsKICAgICRMYW5nLT57J0Zyb250IEVuZCd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAn2KzZhNmIINi12K3ZhtmHJzsKICAgICRMYW5nLT57J0lUIE1hbmFnZW1lbnQnfSAgICAgICAgICAgICAgICAgICAgICAgPSAn2YXYr9uM2LHbjNiqIElUJzsKICAgICRMYW5nLT57J0lUIE9wZXJhdGlvbmFsJ30gICAgICAgICAgICAgICAgICAgICAgPSAn2LnZhdmE24zYp9iqIElUJzsKICAgICRMYW5nLT57J090aGVyJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAn2KjZgtuM2YcnOwogICAgJExhbmctPnsnUHJvamVjdCd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICfZvtix2YjamNmHJzsKICAgICRMYW5nLT57J1JlcG9ydGluZyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAn2q/Ystin2LHYtNuMJzsKICAgICRMYW5nLT57J1RyYWluaW5nJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAn2KLZhdmI2LLYtNuMJzsKICAgICRMYW5nLT57J1VuZGVycGlubmluZyBDb250cmFjdCd9ICAgICAgICAgICAgICAgPSAn2YLYsdin2LHYr9in2K8g2KLZhdin2K/ZhyDahtin2b4nOwogICAgJExhbmctPnsnQXZhaWxhYmlsaXR5J30gICAgICAgICAgICAgICAgICAgICAgICA9ICfZhduM2LLYp9mGINiv2LEg2K/Ys9iq2LHYsyDYqNmI2K/Zhic7CiAgICAkTGFuZy0+eydFcnJvcnMnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ9iu2LfYp9mH2KcnOwogICAgJExhbmctPnsnT3RoZXInfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICfYqNmC24zZhyc7CiAgICAkTGFuZy0+eydSZWNvdmVyeSBUaW1lJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ9iy2YXYp9mGINio2YfYqNmI2K8nOwogICAgJExhbmctPnsnUmVzb2x1dGlvbiBSYXRlJ30gICAgICAgICAgICAgICAgICAgICA9ICfZhtix2K4g2K3ZhCDZhdiz2KbZhNmHJzsKICAgICRMYW5nLT57J1Jlc3BvbnNlIFRpbWUnfSAgICAgICAgICAgICAgICAgICAgICAgPSAn2LLZhdin2YYg2b7Yp9iz2K7ar9mI24zbjCc7CiAgICAkTGFuZy0+eydUcmFuc2FjdGlvbnMnfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ9iq2LHYp9qp2YbYtOKAjNmH2KcnOwoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9mcl9JVFNNQ29yZS5wbSAtIHRoZSBmcmVuY2ggdHJhbnNsYXRpb24gb2YgSVRTTUNvcmUKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPbGl2aWVyIFNhbGxvdSA8b2xpdmllci5zYWxsb3UgYXQgaXJpc2EuZnI+CiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTAgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBmcl9JVFNNQ29yZS5wbSx2IDEuNCAyMDEwLzA2LzAxIDE5OjI1OjIyIG1iIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmZyX0lUU01Db3JlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS40ICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAkTGFuZy0+eydDcml0aWNhbGl0eSd9ICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0NyaXRpY2l06Sc7CiAgICAkTGFuZy0+eydJbXBhY3QnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0ltcGFjdCc7CiAgICAkTGFuZy0+eydDcml0aWNhbGl0eSA8LT4gSW1wYWN0IDwtPiBQcmlvcml0eSd9ID0gJ0NyaXRpY2l06SA8LT4gSW1wYWN0IDwtPiBQcmlvcml06Sc7CiAgICAkTGFuZy0+eydhbGxvY2F0aW9uJ30gICAgICAgICAgICAgICAgICAgICAgICAgID0gJ2FsbG91ZSc7CiAgICAkTGFuZy0+eydQcmlvcml0eSBhbGxvY2F0aW9uJ30gICAgICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydSZWxldmFudCB0byd9ICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0NvcnJlc3BvbmQg4Cc7CiAgICAkTGFuZy0+eydJbmNsdWRlcyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0luY2x1cyc7CiAgICAkTGFuZy0+eydQYXJ0IG9mJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1BhcnQgZGUnOwogICAgJExhbmctPnsnRGVwZW5kcyBvbid9ICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdE6XBlbmQgZGUnOwogICAgJExhbmctPnsnUmVxdWlyZWQgZm9yJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICdSZXF1aXMgcG91cic7CiAgICAkTGFuZy0+eydDb25uZWN0ZWQgdG8nfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ0xp6SDgJzsKICAgICRMYW5nLT57J0FsdGVybmF0aXZlIHRvJ30gICAgICAgICAgICAgICAgICAgICAgPSAnQWx0ZXJuYXRpdmUg4Cc7CiAgICAkTGFuZy0+eydJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICAgICAgICAgID0gJ0V0YXQgZFwnaW5jaWRlbnQnOwogICAgJExhbmctPnsnQ3VycmVudCBJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICA9ICdFdGF0IGFjdHVlbCBkXCdpbmNpZGVudCc7CiAgICAkTGFuZy0+eydDdXJyZW50IFN0YXRlJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ0V0YXQgYWN0dWVsJzsKICAgICRMYW5nLT57J1NlcnZpY2UtQXJlYSd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnWm9uZSBkZSBzZXJ2aWNlJzsKICAgICRMYW5nLT57J01pbmltdW0gVGltZSBCZXR3ZWVuIEluY2lkZW50cyd9ICAgICAgPSAnVGVtcHMgbWluaW1hbCBlbnRyZSBsZXMgaW5jaWRlbnRzJzsKICAgICRMYW5nLT57J1NlcnZpY2UgT3ZlcnZpZXcnfSAgICAgICAgICAgICAgICAgICAgPSAnQXBlcud1IGRlcyBzZXJ2aWNlcyc7CiAgICAkTGFuZy0+eydTTEEgT3ZlcnZpZXcnfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ0FwZXLndSBkZXMgU0xBJzsKICAgICRMYW5nLT57J0Fzc29jaWF0ZWQgU2VydmljZXMnfSAgICAgICAgICAgICAgICAgPSAnU2VydmljZXMgYXNzb2Np6XMnOwogICAgJExhbmctPnsnQXNzb2NpYXRlZCBTTEFzJ30gICAgICAgICAgICAgICAgICAgICA9ICdTTEFzIGFzc29jaellcyc7CiAgICAkTGFuZy0+eydCYWNrIEVuZCd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0JhY2tlbmQnOwogICAgJExhbmctPnsnRGVtb25zdHJhdGlvbid9ICAgICAgICAgICAgICAgICAgICAgICA9ICdE6W1vbnN0cmF0aW9uJzsKICAgICRMYW5nLT57J0VuZCBVc2VyIFNlcnZpY2UnfSAgICAgICAgICAgICAgICAgICAgPSAnU2VydmljZSB1dGlsaXNhdGV1cic7CiAgICAkTGFuZy0+eydGcm9udCBFbmQnfSAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0Zyb250ZW5kJzsKICAgICRMYW5nLT57J0lUIE1hbmFnZW1lbnQnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnR2VzdGlvbiBJVCc7CiAgICAkTGFuZy0+eydJVCBPcGVyYXRpb25hbCd9ICAgICAgICAgICAgICAgICAgICAgID0gJ09w6XJhdGlvbnMgSVQnOwogICAgJExhbmctPnsnT3RoZXInfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdBdXRyZSc7CiAgICAkTGFuZy0+eydQcm9qZWN0J30gICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1Byb2pldCc7CiAgICAkTGFuZy0+eydSZXBvcnRpbmcnfSAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1JhcHBvcnQnOwogICAgJExhbmctPnsnVHJhaW5pbmcnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdGb3JtYXRpb24nOwogICAgJExhbmctPnsnVW5kZXJwaW5uaW5nIENvbnRyYWN0J30gICAgICAgICAgICAgICA9ICdDb250cmF0IGV4dGVybmUnOwogICAgJExhbmctPnsnQXZhaWxhYmlsaXR5J30gICAgICAgICAgICAgICAgICAgICAgICA9ICdEaXNwb25pYmlsaXTpJzsKICAgICRMYW5nLT57J0Vycm9ycyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnRXJyZXVycyc7CiAgICAkTGFuZy0+eydPdGhlcid9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0F1dHJlJzsKICAgICRMYW5nLT57J1JlY292ZXJ5IFRpbWUnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnVGVtcHMgZGUgculwYXJhdGlvbic7CiAgICAkTGFuZy0+eydSZXNvbHV0aW9uIFJhdGUnfSAgICAgICAgICAgICAgICAgICAgID0gJ1RhdXggZGUgculzb2x1dGlvbic7CiAgICAkTGFuZy0+eydSZXNwb25zZSBUaW1lJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ1RlbXBzIGRlIHLpcG9uc2UnOwogICAgJExhbmctPnsnVHJhbnNhY3Rpb25zJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICdUcmFuc2FjdGlvbnMnOwoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9pdF9JVFNNQ29yZS5wbSAtIHRoZSBpdGFsaWFuIHRyYW5zbGF0aW9uIG9mIElUU01Db3JlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTAgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBpdF9JVFNNQ29yZS5wbSx2IDEuMyAyMDEwLzA2LzAxIDE5OjI1OjIyIG1iIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6Oml0X0lUU01Db3JlOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS4zICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAkTGFuZy0+eydDcml0aWNhbGl0eSd9ICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1VyZ2VuemlhJzsKICAgICRMYW5nLT57J0ltcGFjdCd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnSW1wYXR0byc7CiAgICAkTGFuZy0+eydDcml0aWNhbGl0eSA8LT4gSW1wYWN0IDwtPiBQcmlvcml0eSd9ID0gJ1VyZ2VuemEgPC0+IEltcGF0dG8gPC0+IFByaW9yaXTgJzsKICAgICRMYW5nLT57J2FsbG9jYXRpb24nfSAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnYXNzZWduYXJlJzsKICAgICRMYW5nLT57J1ByaW9yaXR5IGFsbG9jYXRpb24nfSAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1JlbGV2YW50IHRvJ30gICAgICAgICAgICAgICAgICAgICAgICAgPSAnUmlsZXZhbnRlIHBlcic7CiAgICAkTGFuZy0+eydJbmNsdWRlcyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0luY2x1ZGUnOwogICAgJExhbmctPnsnUGFydCBvZid9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdQYXJ0ZSBkaSc7CiAgICAkTGFuZy0+eydEZXBlbmRzIG9uJ30gICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0RlcGVuZGUgZGEnOwogICAgJExhbmctPnsnUmVxdWlyZWQgZm9yJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICdSaWNoaWVzdG8gcGVyJzsKICAgICRMYW5nLT57J0Nvbm5lY3RlZCB0byd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnQ29ubmVzc28gYSc7CiAgICAkTGFuZy0+eydBbHRlcm5hdGl2ZSB0byd9ICAgICAgICAgICAgICAgICAgICAgID0gJ0FsdGVybmF0aXZvIGEnOwogICAgJExhbmctPnsnSW5jaWRlbnQgU3RhdGUnfSAgICAgICAgICAgICAgICAgICAgICA9ICdTdGF0byBkZWxsXCdJbmNpZGVudGUnOwogICAgJExhbmctPnsnQ3VycmVudCBJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICA9ICdTdGF0byBBdHR1YWxlIGRlbGxcJ0luY2lkZW50ZSc7CiAgICAkTGFuZy0+eydDdXJyZW50IFN0YXRlJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ1N0YXRvIEF0dHVhbGUnOwogICAgJExhbmctPnsnU2VydmljZS1BcmVhJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICdBcmVhLWRpLVNlcnZpemlvJzsKICAgICRMYW5nLT57J01pbmltdW0gVGltZSBCZXR3ZWVuIEluY2lkZW50cyd9ICAgICAgPSAnTWluaW1vIFRlbXBvIFRyYSBJbmNpZGVudGknOwogICAgJExhbmctPnsnU2VydmljZSBPdmVydmlldyd9ICAgICAgICAgICAgICAgICAgICA9ICdEZXNjcml6aW9uZSBkZWwgU2Vydml6aW8nOwogICAgJExhbmctPnsnU0xBIE92ZXJ2aWV3J30gICAgICAgICAgICAgICAgICAgICAgICA9ICdEZXNjcml6aW9uZSBkZWxsbyBTTEEnOwogICAgJExhbmctPnsnQXNzb2NpYXRlZCBTZXJ2aWNlcyd9ICAgICAgICAgICAgICAgICA9ICdTZXJ2aXppIEFzc29jaWF0aSc7CiAgICAkTGFuZy0+eydBc3NvY2lhdGVkIFNMQXMnfSAgICAgICAgICAgICAgICAgICAgID0gJ1NMQXMgQXNzb2NpYXRpJzsKICAgICRMYW5nLT57J0JhY2sgRW5kJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnQmFjayBFbmQnOwogICAgJExhbmctPnsnRGVtb25zdHJhdGlvbid9ICAgICAgICAgICAgICAgICAgICAgICA9ICdEaW1vc3RyYXppb25lJzsKICAgICRMYW5nLT57J0VuZCBVc2VyIFNlcnZpY2UnfSAgICAgICAgICAgICAgICAgICAgPSAnU2Vydml6aW8gVXRlbnRlIEZpbmFsZSc7CiAgICAkTGFuZy0+eydGcm9udCBFbmQnfSAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0Zyb250IEVuZCc7CiAgICAkTGFuZy0+eydJVCBNYW5hZ2VtZW50J30gICAgICAgICAgICAgICAgICAgICAgID0gJ0lUIE1hbmFnZW1lbnQnOwogICAgJExhbmctPnsnSVQgT3BlcmF0aW9uYWwnfSAgICAgICAgICAgICAgICAgICAgICA9ICdJVCBPcGVyYXRpb25hbCc7CiAgICAkTGFuZy0+eydPdGhlcid9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0FsdHJvJzsKICAgICRMYW5nLT57J1Byb2plY3QnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnUHJvZ2V0dG8nOwogICAgJExhbmctPnsnUmVwb3J0aW5nJ30gICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdSYXBwb3J0aSc7CiAgICAkTGFuZy0+eydUcmFpbmluZyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0Zvcm1hemlvbmUnOwogICAgJExhbmctPnsnVW5kZXJwaW5uaW5nIENvbnRyYWN0J30gICAgICAgICAgICAgICA9ICdVbmRlcnBpbm5pbmcgQ29udHJhY3QnOwogICAgJExhbmctPnsnQXZhaWxhYmlsaXR5J30gICAgICAgICAgICAgICAgICAgICAgICA9ICdEaXNwb25pYmlsaXTgJzsKICAgICRMYW5nLT57J0Vycm9ycyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnRXJyb3JpJzsKICAgICRMYW5nLT57J090aGVyJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnQWx0cm8nOwogICAgJExhbmctPnsnUmVjb3ZlcnkgVGltZSd9ICAgICAgICAgICAgICAgICAgICAgICA9ICdUZW1wbyBkaSBSZWN1cGVybyc7CiAgICAkTGFuZy0+eydSZXNvbHV0aW9uIFJhdGUnfSAgICAgICAgICAgICAgICAgICAgID0gJ1Rhc3NvIGRpIFJpc29sdXppb25lJzsKICAgICRMYW5nLT57J1Jlc3BvbnNlIFRpbWUnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnVGVtcG8gZGkgUmlzcG9zdGEnOwogICAgJExhbmctPnsnVHJhbnNhY3Rpb25zJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICdUcmFuc2F6aW9uaSc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9ubF9JVFNNQ29yZS5wbSAtIHRoZSBEdXRjaCB0cmFuc2xhdGlvbiBvZiBJVFNNQ29yZQojIENvcHlyaWdodCAoQykgMjAwOSBNaWNoaWVsIEJlaWplbiA8bWljaGllbCAnYXQnIGJlZWZyZWVpdC5ubD4KIyAtLQojICRJZDogbmxfSVRTTUNvcmUucG0sdiAxLjMgMjAxMC8wNi8wMSAxOToyNToyMiBtYiBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpubF9JVFNNQ29yZTsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuMyAkKSBbMV07CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgIG15ICRMYW5nID0gJFNlbGYtPntUcmFuc2xhdGlvbn07CgogICAgcmV0dXJuIGlmIHJlZiAkTGFuZyBuZSAnSEFTSCc7CgogICAgJExhbmctPnsnQ3JpdGljYWxpdHknfSAgICAgICAgICAgICAgICAgICAgICAgICA9ICdVcmdlbnRpZSc7CiAgICAkTGFuZy0+eydJbXBhY3QnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0ltcGFjdCc7CiAgICAkTGFuZy0+eydDcml0aWNhbGl0eSA8LT4gSW1wYWN0IDwtPiBQcmlvcml0eSd9ID0gJ1VyZ2VudGllIDwtPiBJbXBhY3QgPC0+IFByaW9yaXRlaXQnOwogICAgJExhbmctPnsnYWxsb2NhdGlvbid9ICAgICAgICAgICAgICAgICAgICAgICAgICA9ICd0b2VrZW5uZW4nOwogICAgJExhbmctPnsnUHJpb3JpdHkgYWxsb2NhdGlvbid9ICAgICAgICAgICAgICAgICA9ICdQcmlvcml0ZWl0IHRvZWtlbm5lbic7CiAgICAkTGFuZy0+eydSZWxldmFudCB0byd9ICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1ZhbiBiZWxhbmcgdm9vcic7CiAgICAkTGFuZy0+eydJbmNsdWRlcyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0JldmF0JzsKICAgICRMYW5nLT57J1BhcnQgb2YnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnT25kZXJkZWVsIHZhbic7CiAgICAkTGFuZy0+eydEZXBlbmRzIG9uJ30gICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0FmaGFua2VsaWprIHZhbic7CiAgICAkTGFuZy0+eydSZXF1aXJlZCBmb3InfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ0Jlbm9kaWdkIHZvb3InOwogICAgJExhbmctPnsnQ29ubmVjdGVkIHRvJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICdWZXJib25kZW4gbWV0JzsKICAgICRMYW5nLT57J0FsdGVybmF0aXZlIHRvJ30gICAgICAgICAgICAgICAgICAgICAgPSAnQWx0ZXJuYXRpZWYgdm9vcic7CiAgICAkTGFuZy0+eydJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICAgICAgICAgID0gJ0luY2lkZW50IHN0YXR1cyc7CiAgICAkTGFuZy0+eydDdXJyZW50IEluY2lkZW50IFN0YXRlJ30gICAgICAgICAgICAgID0gJ0h1aWRpZ2UgaW5jaWRlbnQgc3RhdHVzJzsKICAgICRMYW5nLT57J0N1cnJlbnQgU3RhdGUnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnSHVpZGlnZSBzdGF0dXMnOwogICAgJExhbmctPnsnU2VydmljZS1BcmVhJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICdTZXJ2aWNlLUFyZWEnOwogICAgJExhbmctPnsnTWluaW11bSBUaW1lIEJldHdlZW4gSW5jaWRlbnRzJ30gICAgICA9ICdNaW5pbXVtdGlqZCB0dXNzZW4gaW5jaWRlbnRlbic7CiAgICAkTGFuZy0+eydTZXJ2aWNlIE92ZXJ2aWV3J30gICAgICAgICAgICAgICAgICAgID0gJ1NlcnZpY2Utb3ZlcnppY2h0JzsKICAgICRMYW5nLT57J1NMQSBPdmVydmlldyd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnU0xBLW92ZXJ6aWNodCc7CiAgICAkTGFuZy0+eydBc3NvY2lhdGVkIFNlcnZpY2VzJ30gICAgICAgICAgICAgICAgID0gJ0JpamJlaG9yZW5kZSBTZXJ2aWNlcyc7CiAgICAkTGFuZy0+eydBc3NvY2lhdGVkIFNMQXMnfSAgICAgICAgICAgICAgICAgICAgID0gJ0JpamJlaG9yZW5kZSBTTEFzJzsKICAgICRMYW5nLT57J0JhY2sgRW5kJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnQmFja2VuZCc7CiAgICAkTGFuZy0+eydEZW1vbnN0cmF0aW9uJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ0RlbW9uc3RyYXRpb24nOwogICAgJExhbmctPnsnRW5kIFVzZXIgU2VydmljZSd9ICAgICAgICAgICAgICAgICAgICA9ICdFaW5kZ2VicnVpa2VyIHNlcnZpY2UnOwogICAgJExhbmctPnsnRnJvbnQgRW5kJ30gICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdGcm9udGVuZCc7CiAgICAkTGFuZy0+eydJVCBNYW5hZ2VtZW50J30gICAgICAgICAgICAgICAgICAgICAgID0gJ0lUIE1hbmFnZW1lbnQnOwogICAgJExhbmctPnsnSVQgT3BlcmF0aW9uYWwnfSAgICAgICAgICAgICAgICAgICAgICA9ICdJVCBPcGVyYXRpb25zJzsKICAgICRMYW5nLT57J090aGVyJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnT3ZlcmlnJzsKICAgICRMYW5nLT57J1Byb2plY3QnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnUHJvamVjdCc7CiAgICAkTGFuZy0+eydSZXBvcnRpbmcnfSAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1JhcHBvcnRhZ2UnOwogICAgJExhbmctPnsnVHJhaW5pbmcnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdUcmFpbmluZyc7CiAgICAkTGFuZy0+eydVbmRlcnBpbm5pbmcgQ29udHJhY3QnfSAgICAgICAgICAgICAgID0gJ1VuZGVycGlubmluZyBDb250cmFjdCc7CiAgICAkTGFuZy0+eydBdmFpbGFiaWxpdHknfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ0Jlc2NoaWtiYWFyaGVpZCc7CiAgICAkTGFuZy0+eydFcnJvcnMnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0ZvdXRlbic7CiAgICAkTGFuZy0+eydSZWNvdmVyeSBUaW1lJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ0hlcnN0ZWx0aWpkJzsKICAgICRMYW5nLT57J1Jlc29sdXRpb24gUmF0ZSd9ICAgICAgICAgICAgICAgICAgICAgPSAnT3Bsb3N0aWpkJzsKICAgICRMYW5nLT57J1Jlc3BvbnNlIFRpbWUnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnUmVzcG9uc2lldGlqZCc7CiAgICAkTGFuZy0+eydUcmFuc2FjdGlvbnMnfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ1RyYW5zYWN0aWVzJzsKCiAgICByZXR1cm4gMTsKfQoKMTsK
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9wbF9JVFNNQ29yZS5wbSAtIHRoZSBwb2xpc2ggdHJhbnNsYXRpb24gb2YgSVRTTUNvcmUKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgQ29weXJpZ2h0IChDKSAyMDA4IE1hY2llaiBMb3N6YWpjCiMgLS0KIyAkSWQ6IHBsX0lUU01Db3JlLnBtLHYgMS42IDIwMTAvMDYvMDEgMTk6MjU6MjIgbWIgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6cGxfSVRTTUNvcmU7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjYgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0NyaXRpY2FsaXR5J30gICAgICAgICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0ltcGFjdCd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0NyaXRpY2FsaXR5IDwtPiBJbXBhY3QgPC0+IFByaW9yaXR5J30gPSAnJzsKICAgICRMYW5nLT57J2FsbG9jYXRpb24nfSAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1ByaW9yaXR5IGFsbG9jYXRpb24nfSAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1JlbGV2YW50IHRvJ30gICAgICAgICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0luY2x1ZGVzJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnWmF3aWVyYSc7CiAgICAkTGFuZy0+eydQYXJ0IG9mJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydEZXBlbmRzIG9uJ30gICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1phbGW/bmUgb2QnOwogICAgJExhbmctPnsnUmVxdWlyZWQgZm9yJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICdQb3RyemVibmUgZG8nOwogICAgJExhbmctPnsnQ29ubmVjdGVkIHRvJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICdQb2SzsWN6b25lIGRvJzsKICAgICRMYW5nLT57J0FsdGVybmF0aXZlIHRvJ30gICAgICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0luY2lkZW50IFN0YXRlJ30gICAgICAgICAgICAgICAgICAgICAgPSAnU3RhbiB6ZGFyemVuaWEnOwogICAgJExhbmctPnsnQ3VycmVudCBJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICA9ICdBa3R1YWxueSBzdGFuIHpkYXJ6ZW5pYSc7CiAgICAkTGFuZy0+eydDdXJyZW50IFN0YXRlJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ0FrdHVhbG55IHN0YW4nOwogICAgJExhbmctPnsnU2VydmljZS1BcmVhJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICdTZWtjamEgc2Vyd2lzb3dhJzsKICAgICRMYW5nLT57J01pbmltdW0gVGltZSBCZXR3ZWVuIEluY2lkZW50cyd9ICAgICAgPSAnTWluaW1hbG55IGN6YXMgbWnqZHp5IHpkYXJ6ZW5pYW1pJzsKICAgICRMYW5nLT57J1NlcnZpY2UgT3ZlcnZpZXcnfSAgICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1NMQSBPdmVydmlldyd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0Fzc29jaWF0ZWQgU2VydmljZXMnfSAgICAgICAgICAgICAgICAgPSAnUG+zsWN6b25lIHVzs3VnaSc7CiAgICAkTGFuZy0+eydBc3NvY2lhdGVkIFNMQXMnfSAgICAgICAgICAgICAgICAgICAgID0gJ1Bvs7Fjem9uZSBTTEEnOwogICAgJExhbmctPnsnQmFjayBFbmQnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnRGVtb25zdHJhdGlvbid9ICAgICAgICAgICAgICAgICAgICAgICA9ICdEZW1vbnN0cmFjamEnOwogICAgJExhbmctPnsnRW5kIFVzZXIgU2VydmljZSd9ICAgICAgICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnRnJvbnQgRW5kJ30gICAgICAgICAgICAgICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnSVQgTWFuYWdlbWVudCd9ICAgICAgICAgICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnSVQgT3BlcmF0aW9uYWwnfSAgICAgICAgICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnT3RoZXInfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdJbm5lJzsKICAgICRMYW5nLT57J1Byb2plY3QnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnUHJvamVrdCc7CiAgICAkTGFuZy0+eydSZXBvcnRpbmcnfSAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1JhcG9ydG93YW5pZSc7CiAgICAkTGFuZy0+eydUcmFpbmluZyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1RyZW5pbmcnOwogICAgJExhbmctPnsnVW5kZXJwaW5uaW5nIENvbnRyYWN0J30gICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnQXZhaWxhYmlsaXR5J30gICAgICAgICAgICAgICAgICAgICAgICA9ICdEb3N06nBub7bmJzsKICAgICRMYW5nLT57J0Vycm9ycyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnQrPqZHknOwogICAgJExhbmctPnsnT3RoZXInfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdJbm5lJzsKICAgICRMYW5nLT57J1JlY292ZXJ5IFRpbWUnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnQ3phcyBvZHp5c2thbmlhJzsKICAgICRMYW5nLT57J1Jlc29sdXRpb24gUmF0ZSd9ICAgICAgICAgICAgICAgICAgICAgPSAnQ3phcyByb3p3abF6YW5pYSc7CiAgICAkTGFuZy0+eydSZXNwb25zZSBUaW1lJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ0N6YXMgb2Rwb3dpZWR6aSc7CiAgICAkTGFuZy0+eydUcmFuc2FjdGlvbnMnfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ1RyYW5zYWtjamUnOwoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9wdF9CUl9JVFNNQ29yZS5wbSAtIHRoZSBCcmF6aWxpYW4gdHJhbnNsYXRpb24gb2YgSVRTTUNvcmUKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgQ29weXJpZ2h0IChDKSAyMDEwIENyaXN0aWFubyBLb3JuZPZyZmVyLCBodHRwOi8vd3d3LmRvcmZlci5jb20uYnIvCiMgLS0KIyAkSWQ6IHB0X0JSX0lUU01Db3JlLnBtLHYgMS4yIDIwMTAvMDYvMDEgMTk6MjU6MjIgbWIgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6cHRfQlJfSVRTTUNvcmU7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjIgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0NyaXRpY2FsaXR5J30gICAgICAgICAgICAgICAgICAgICAgICAgPSAnQ3JpdGljYWxpZGFkZSc7CiAgICAkTGFuZy0+eydJbXBhY3QnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0ltcGFjdG8nOwogICAgJExhbmctPnsnQ3JpdGljYWxpdHkgPC0+IEltcGFjdCA8LT4gUHJpb3JpdHknfSA9ICdDcml0aWNhbGlkYWRlIDwtPiBJbXBhY3RvIDwtPiBQcmlvcmlkYWRlJzsKICAgICRMYW5nLT57J2FsbG9jYXRlJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnQWxvY2FyJzsKICAgICRMYW5nLT57J1ByaW9yaXR5IGFsbG9jYXRpb24nfSAgICAgICAgICAgICAgICAgPSAnQWxvY2FyIHByaW9yaWRhZGUnOwogICAgJExhbmctPnsnUmVsZXZhbnQgdG8nfSAgICAgICAgICAgICAgICAgICAgICAgICA9ICdSZWxldmFudGUgYSc7CiAgICAkTGFuZy0+eydJbmNsdWRlcyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ0luY2x1aSc7CiAgICAkTGFuZy0+eydQYXJ0IG9mJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1BhcnRlIGRlJzsKICAgICRMYW5nLT57J0RlcGVuZHMgb24nfSAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnRGVwZW5kZSBkZSc7CiAgICAkTGFuZy0+eydSZXF1aXJlZCBmb3InfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ1JlcXVpc2l0YWRvIHBvcic7CiAgICAkTGFuZy0+eydDb25uZWN0ZWQgdG8nfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ0NvbmVjdGFkbyBhJzsKICAgICRMYW5nLT57J0FsdGVybmF0aXZlIHRvJ30gICAgICAgICAgICAgICAgICAgICAgPSAnQWx0ZXJuYXRpdmEgYSc7CiAgICAkTGFuZy0+eydJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICAgICAgICAgID0gJ1NpdHVh5+NvIGRlIEluY2lkZW50ZXMnOwogICAgJExhbmctPnsnQ3VycmVudCBJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICA9ICdTaXR1YefjbyBBdHVhbCBkZSBJbmNpZGVudGVzJzsKICAgICRMYW5nLT57J0N1cnJlbnQgU3RhdGUnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnU2l0dWHn428gQXR1YWwnOwogICAgJExhbmctPnsnU2VydmljZS1BcmVhJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICdTZXJ2aedvLcFyZWEnOwogICAgJExhbmctPnsnTWluaW11bSBUaW1lIEJldHdlZW4gSW5jaWRlbnRzJ30gICAgICA9ICdUZW1wbyBN7W5pbW8gZW50cmUgSW5jaWRlbnRlcyc7CiAgICAkTGFuZy0+eydTZXJ2aWNlIE92ZXJ2aWV3J30gICAgICAgICAgICAgICAgICAgID0gJ1Jlc3VtbyBkbyBTZXJ2aedvJzsKICAgICRMYW5nLT57J1NMQSBPdmVydmlldyd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnUmVzdW1vIGRhIFNMQSc7CiAgICAkTGFuZy0+eydBc3NvY2lhdGVkIFNlcnZpY2VzJ30gICAgICAgICAgICAgICAgID0gJ1NlcnZp529zIEFzc29jaWFkb3MnOwogICAgJExhbmctPnsnQXNzb2NpYXRlZCBTTEFzJ30gICAgICAgICAgICAgICAgICAgICA9ICdTTEFzIEFzc29jaWFkYXMnOwogICAgJExhbmctPnsnQmFjayBFbmQnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnRGVtb25zdHJhdGlvbid9ICAgICAgICAgICAgICAgICAgICAgICA9ICdEZW1vbnN0cmHn428nOwogICAgJExhbmctPnsnRW5kIFVzZXIgU2VydmljZSd9ICAgICAgICAgICAgICAgICAgICA9ICdTZXJ2aedvIGEgVXN14XJpbyBGaW5hbCc7CiAgICAkTGFuZy0+eydGcm9udCBFbmQnfSAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydJVCBNYW5hZ2VtZW50J30gICAgICAgICAgICAgICAgICAgICAgID0gJ0dlcmVuY2lhbWVudG8gZGUgVEknOwogICAgJExhbmctPnsnSVQgT3BlcmF0aW9uYWwnfSAgICAgICAgICAgICAgICAgICAgICA9ICdPcGVyYef1ZXMgZGUgVEknOwogICAgJExhbmctPnsnT3RoZXInfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdPdXRybyc7CiAgICAkTGFuZy0+eydQcm9qZWN0J30gICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ1Byb2pldG8nOwogICAgJExhbmctPnsnUmVwb3J0aW5nJ30gICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdSZWxhdPNyaW8nOwogICAgJExhbmctPnsnVHJhaW5pbmcnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdUcmVpbmFtZW50byc7CiAgICAkTGFuZy0+eydVbmRlcnBpbm5pbmcgQ29udHJhY3QnfSAgICAgICAgICAgICAgID0gJ0NvbnRyYXRvIGNvbSBUZXJjZWlyb3MgKFVDKSc7CiAgICAkTGFuZy0+eydBdmFpbGFiaWxpdHknfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ0F2YWxpYWJpbGlkYWRlJzsKICAgICRMYW5nLT57J0Vycm9ycyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnRXJyb3MnOwogICAgJExhbmctPnsnT3RoZXInfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICdPdXRybyc7CiAgICAkTGFuZy0+eydSZWNvdmVyeSBUaW1lJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ1RlbXBvIGRlIFJlY3VwZXJh5+NvJzsKICAgICRMYW5nLT57J1Jlc29sdXRpb24gUmF0ZSd9ICAgICAgICAgICAgICAgICAgICAgPSAnVGF4YSBkZSBSZXNvbHXn428nOwogICAgJExhbmctPnsnUmVzcG9uc2UgVGltZSd9ICAgICAgICAgICAgICAgICAgICAgICA9ICdUZW1wbyBkZSBSZXNwb3N0YSc7CiAgICAkTGFuZy0+eydUcmFuc2FjdGlvbnMnfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ1RyYW5zYef1ZXMnOwoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9ydV9JVFNNQ29yZS5wbSAtIHRoZSBydXNzaWFuIHRyYW5zbGF0aW9uIG9mIElUU01Db3JlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTAgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIENvcHlyaWdodCAoQykgMjAwOCBFZ29yIFRzaWxlbmtvIDxiZzhzIGF0IHN5bWxpbmsucnU+CiMgLS0KIyAkSWQ6IHJ1X0lUU01Db3JlLnBtLHYgMS40IDIwMTAvMDYvMDEgMTk6MjU6MjIgbWIgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6cnVfSVRTTUNvcmU7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjQgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0NyaXRpY2FsaXR5J30gICAgICAgICAgICAgICAgICAgICAgICAgPSAnyvDo8uj37e7x8vwnOwogICAgJExhbmctPnsnSW1wYWN0J30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICfC6+j/7ejlJzsKICAgICRMYW5nLT57J0NyaXRpY2FsaXR5IDwtPiBJbXBhY3QgPC0+IFByaW9yaXR5J30gPSAnyvDo8uj37e7x8vwgPC0+IMLr6P/t6OUgPC0+IM/w6O7w6PLl8ic7CiAgICAkTGFuZy0+eydhbGxvY2F0aW9uJ30gICAgICAgICAgICAgICAgICAgICAgICAgID0gJ83g5+3g9+Xt6OUg7/Do7vDo8uXy7uIgJzsKICAgICRMYW5nLT57J1ByaW9yaXR5IGFsbG9jYXRpb24nfSAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1JlbGV2YW50IHRvJ30gICAgICAgICAgICAgICAgICAgICAgICAgPSAnzvLt7vHo8vH/IOonOwogICAgJExhbmctPnsnSW5jbHVkZXMnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICfC6uv+9+Dl8ic7CiAgICAkTGFuZy0+eydQYXJ0IG9mJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ9Hu8fLu6PIg6OcnOwogICAgJExhbmctPnsnRGVwZW5kcyBvbid9ICAgICAgICAgICAgICAgICAgICAgICAgICA9ICfH4OLo8ejyIO7yJzsKICAgICRMYW5nLT57J1JlcXVpcmVkIGZvcid9ICAgICAgICAgICAgICAgICAgICAgICAgPSAn0vDl4fPl8vH/IOTr/yc7CiAgICAkTGFuZy0+eydDb25uZWN0ZWQgdG8nfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ9Hi/+fg7SDxJzsKICAgICRMYW5nLT57J0FsdGVybmF0aXZlIHRvJ30gICAgICAgICAgICAgICAgICAgICAgPSAnx+Ds5e3gIOTr/yc7CiAgICAkTGFuZy0+eydJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICAgICAgICAgID0gJ9Hu8fLu/+3o5SDo7fbo5OXt8uAnOwogICAgJExhbmctPnsnQ3VycmVudCBJbmNpZGVudCBTdGF0ZSd9ICAgICAgICAgICAgICA9ICfS5erz+eXlIPHu8fLu/+3o5SDo7fbo5OXt8uAnOwogICAgJExhbmctPnsnQ3VycmVudCBTdGF0ZSd9ICAgICAgICAgICAgICAgICAgICAgICA9ICfS5erz+eXlIPHu8fLu/+3o5Sc7CiAgICAkTGFuZy0+eydTZXJ2aWNlLUFyZWEnfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ87h5+7wIPHl8OLo8e7iJzsKICAgICRMYW5nLT57J01pbmltdW0gVGltZSBCZXR3ZWVuIEluY2lkZW50cyd9ICAgICAgPSAnzOjt6Ozg6/zt7uUg4vDl7P8g7OXm5PMg6O326OTl7fLg7OgnOwogICAgJExhbmctPnsnU2VydmljZSBPdmVydmlldyd9ICAgICAgICAgICAgICAgICAgICA9ICfO4efu8CDx5fDi6PHu4ic7CiAgICAkTGFuZy0+eydTTEEgT3ZlcnZpZXcnfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ87h5+7wIFNMQSc7CiAgICAkTGFuZy0+eydBc3NvY2lhdGVkIFNlcnZpY2VzJ30gICAgICAgICAgICAgICAgID0gJ9Hi/+fg7e375SDx5fDi6PH7JzsKICAgICRMYW5nLT57J0Fzc29jaWF0ZWQgU0xBcyd9ICAgICAgICAgICAgICAgICAgICAgPSAn0eL/5+Dt7fvlIFNMQSc7CiAgICAkTGFuZy0+eydCYWNrIEVuZCd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ9Hl8OLl8O3g/yD34PHy/Cc7CiAgICAkTGFuZy0+eydEZW1vbnN0cmF0aW9uJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ8Tl7O7t8fLw4Pbo/yc7CiAgICAkTGFuZy0+eydFbmQgVXNlciBTZXJ2aWNlJ30gICAgICAgICAgICAgICAgICAgID0gJ8ru7eX37fvpIPHl8OLo8SDv7uv85+7i4PLl6/8nOwogICAgJExhbmctPnsnRnJvbnQgRW5kJ30gICAgICAgICAgICAgICAgICAgICAgICAgICA9ICfI7fLl8PTl6fHt4P8g9+Dx8vwnOwogICAgJExhbmctPnsnSVQgTWFuYWdlbWVudCd9ICAgICAgICAgICAgICAgICAgICAgICA9ICfT7/Dg4uvl7ejlIMjSJzsKICAgICRMYW5nLT57J0lUIE9wZXJhdGlvbmFsJ30gICAgICAgICAgICAgICAgICAgICAgPSAn3erx7+vz4PLg9uj/IMjSJzsKICAgICRMYW5nLT57J090aGVyJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnxPDz4+7lJzsKICAgICRMYW5nLT57J1Byb2plY3QnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnz+vg7ejw7uLg7ejlJzsKICAgICRMYW5nLT57J1JlcG9ydGluZyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAn0e7x8uDi6+Xt6OUg7vL35fLu4ic7CiAgICAkTGFuZy0+eydUcmFpbmluZyd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ87h8/fl7ejlJzsKICAgICRMYW5nLT57J1VuZGVycGlubmluZyBDb250cmFjdCd9ICAgICAgICAgICAgICAgPSAnyu7t8vDg6vIg7+7k5OXw5uroJzsKICAgICRMYW5nLT57J0F2YWlsYWJpbGl0eSd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnxO7x8vPv7e7x8vwnOwogICAgJExhbmctPnsnRXJyb3JzJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICfO+Ojh6ugnOwogICAgJExhbmctPnsnT3RoZXInfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICfE8PPj7uUnOwogICAgJExhbmctPnsnUmVjb3ZlcnkgVGltZSd9ICAgICAgICAgICAgICAgICAgICAgICA9ICfC8OXs/yDi7vHx8uDt7uLr5e3o/yc7CiAgICAkTGFuZy0+eydSZXNvbHV0aW9uIFJhdGUnfSAgICAgICAgICAgICAgICAgICAgID0gJ87y7e7x6PLl6/zt4P8g8eru8O7x8vwg8OX45e3o/yc7CiAgICAkTGFuZy0+eydSZXNwb25zZSBUaW1lJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ8Lw5ez/IPDl4Or26OgnOwogICAgJExhbmctPnsnVHJhbnNhY3Rpb25zJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICfU6O3g7fHu4vvlIO7v5fDg9ujoJzsKCiAgICByZXR1cm4gMTsKfQoKMTsK
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS96aF9DTl9JVFNNQ29yZS5wbSAtIHRoZSBDaGluZXNlIHNpbXBsZSB0cmFuc2xhdGlvbiBvZiBJVFNNQ29yZQojIENvcHlyaWdodCAoQykgMjAwMS0yMDEwIE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogemhfQ05fSVRTTUNvcmUucG0sdiAxLjQgMjAxMC8wNi8wMSAxOToyNToyMiBtYiBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjp6aF9DTl9JVFNNQ29yZTsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuNCAkKSBbMV07CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgIG15ICRMYW5nID0gJFNlbGYtPntUcmFuc2xhdGlvbn07CgogICAgcmV0dXJuIGlmIHJlZiAkTGFuZyBuZSAnSEFTSCc7CgogICAgJExhbmctPnsnQ3JpdGljYWxpdHknfSAgICAgICAgICAgICAgICAgICAgICAgICA9ICfOo7yxs8y2yCc7CiAgICAkTGFuZy0+eydJbXBhY3QnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ9Owz+y2yCc7CiAgICAkTGFuZy0+eydDcml0aWNhbGl0eSA8LT4gSW1wYWN0IDwtPiBQcmlvcml0eSd9ID0gJ86jvLGzzLbIIDwtPiDTsM/stsggPC0+INPFz8i8trHwJzsKICAgICRMYW5nLT57J2FsbG9jYXRpb24nfSAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnt9bF5Cc7CiAgICAkTGFuZy0+eydQcmlvcml0eSBhbGxvY2F0aW9uJ30gICAgICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydSZWxldmFudCB0byd9ICAgICAgICAgICAgICAgICAgICAgICAgID0gJ8/gudgnOwogICAgJExhbmctPnsnSW5jbHVkZXMnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICew/MCoJzsKICAgICRMYW5nLT57J1BhcnQgb2YnfSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnsr+31tPaJzsKICAgICRMYW5nLT57J0RlcGVuZHMgb24nfSAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnyKG+9tPaJzsKICAgICRMYW5nLT57J1JlcXVpcmVkIGZvcid9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnsdjQ6LXEJzsKICAgICRMYW5nLT57J0Nvbm5lY3RlZCB0byd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnway907W9JzsKICAgICRMYW5nLT57J0FsdGVybmF0aXZlIHRvJ30gICAgICAgICAgICAgICAgICAgICAgPSAn0aHU8bbUz/MnOwogICAgJExhbmctPnsnSW5jaWRlbnQgU3RhdGUnfSAgICAgICAgICAgICAgICAgICAgICA9ICfKwrz+17TMrCc7CiAgICAkTGFuZy0+eydDdXJyZW50IEluY2lkZW50IFN0YXRlJ30gICAgICAgICAgICAgID0gJ7Wxx7DKwrz+17TMrCc7CiAgICAkTGFuZy0+eydDdXJyZW50IFN0YXRlJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ8/W17QnOwogICAgJExhbmctPnsnU2VydmljZS1BcmVhJ30gICAgICAgICAgICAgICAgICAgICAgICA9ICe3/s7xx/gnOwogICAgJExhbmctPnsnTWluaW11bSBUaW1lIEJldHdlZW4gSW5jaWRlbnRzJ30gICAgICA9ICfX7rbMtcTKsbzk0+vKwrz+JzsKICAgICRMYW5nLT57J1NlcnZpY2UgT3ZlcnZpZXcnfSAgICAgICAgICAgICAgICAgICAgPSAnt/7O8bjFyvYnOwogICAgJExhbmctPnsnU0xBIE92ZXJ2aWV3J30gICAgICAgICAgICAgICAgICAgICAgICA9ICdTTEEguMXK9ic7CiAgICAkTGFuZy0+eydBc3NvY2lhdGVkIFNlcnZpY2VzJ30gICAgICAgICAgICAgICAgID0gJ7nYwaq1xLf+zvEnOwogICAgJExhbmctPnsnQXNzb2NpYXRlZCBTTEFzJ30gICAgICAgICAgICAgICAgICAgICA9ICe52MGqtcQgU0xBcyc7CiAgICAkTGFuZy0+eydCYWNrIEVuZCd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ7rztssnOwogICAgJExhbmctPnsnRGVtb25zdHJhdGlvbid9ICAgICAgICAgICAgICAgICAgICAgICA9ICfKvre2JzsKICAgICRMYW5nLT57J0VuZCBVc2VyIFNlcnZpY2UnfSAgICAgICAgICAgICAgICAgICAgPSAn1+7W1dPDu6e3/s7xJzsKICAgICRMYW5nLT57J0Zyb250IEVuZCd9ICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnx7C2yyc7CiAgICAkTGFuZy0+eydJVCBNYW5hZ2VtZW50J30gICAgICAgICAgICAgICAgICAgICAgID0gJ0lUILncwO0nOwogICAgJExhbmctPnsnSVQgT3BlcmF0aW9uYWwnfSAgICAgICAgICAgICAgICAgICAgICA9ICdJVCDUy9OqJzsKICAgICRMYW5nLT57J090aGVyJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnxuTL/Cc7CiAgICAkTGFuZy0+eydQcm9qZWN0J30gICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gJ8/uxL8nOwogICAgJExhbmctPnsnUmVwb3J0aW5nJ30gICAgICAgICAgICAgICAgICAgICAgICAgICA9ICexqLjmJzsKICAgICRMYW5nLT57J1RyYWluaW5nJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAn0bXBtyc7CiAgICAkTGFuZy0+eydVbmRlcnBpbm5pbmcgQ29udHJhY3QnfSAgICAgICAgICAgICAgID0gJ9LAvt26z82sJzsKICAgICRMYW5nLT57J0F2YWlsYWJpbGl0eSd9ICAgICAgICAgICAgICAgICAgICAgICAgPSAnuam79cfpv/YnOwogICAgJExhbmctPnsnRXJyb3JzJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICe07c7zJzsKICAgICRMYW5nLT57J090aGVyJ30gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnxuTL/Cc7CiAgICAkTGFuZy0+eydSZWNvdmVyeSBUaW1lJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ7vWuLTKsbzkJzsKICAgICRMYW5nLT57J1Jlc29sdXRpb24gUmF0ZSd9ICAgICAgICAgICAgICAgICAgICAgPSAnveK+9r34tsgnOwogICAgJExhbmctPnsnUmVzcG9uc2UgVGltZSd9ICAgICAgICAgICAgICAgICAgICAgICA9ICfP7NOmyrG85Cc7CiAgICAkTGFuZy0+eydUcmFuc2FjdGlvbnMnfSAgICAgICAgICAgICAgICAgICAgICAgID0gJ7270tcnOwoKICAgIHJldHVybiAxOwp9CgoxOwo=
# --
# Kernel/Modules/AdminITSMCIPAllocate.pm - admin frontend of criticality, impact and priority
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: AdminITSMCIPAllocate.pm,v 1.11 2009/05/18 09:48:35 mh Exp $
# --
# 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::AdminITSMCIPAllocate;

use strict;
use warnings;

use Kernel::System::GeneralCatalog;
use Kernel::System::ITSMCIPAllocate;
use Kernel::System::Priority;
use Kernel::System::Valid;

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

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

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

    # check needed objects
    for my $Object (qw(ConfigObject ParamObject LogObject LayoutObject)) {
        if ( !$Self->{$Object} ) {
            $Self->{LayoutObject}->FatalError( Message => "Got no $Object!" );
        }
    }
    $Self->{GeneralCatalogObject} = Kernel::System::GeneralCatalog->new(%Param);
    $Self->{CIPAllocateObject}    = Kernel::System::ITSMCIPAllocate->new(%Param);
    $Self->{PriorityObject}       = Kernel::System::Priority->new(%Param);
    $Self->{ValidObject}          = Kernel::System::Valid->new(%Param);

    return $Self;
}

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

    # ------------------------------------------------------------ #
    # criticality, impact and priority allocation
    # ------------------------------------------------------------ #
    if ( $Self->{Subaction} eq 'CIPAllocate' ) {

        # get option lists
        my %ObjectOption;
        $ObjectOption{CriticalityList} = $Self->{GeneralCatalogObject}->ItemList(
            Class => 'ITSM::Core::Criticality',
        );
        $ObjectOption{ImpactList} = $Self->{GeneralCatalogObject}->ItemList(
            Class => 'ITSM::Core::Impact',
        );
        my %OptionPriorityList = $Self->{PriorityObject}->PriorityList(
            UserID => 1,
        );
        $ObjectOption{PriorityList} = \%OptionPriorityList;

        # get all PriorityIDs of the matrix
        my $AllocateData;
        for my $ImpactID ( keys %{ $ObjectOption{ImpactList} } ) {

            CRITICALITYID:
            for my $CriticalityID ( keys %{ $ObjectOption{CriticalityList} } ) {

                # get form param
                my $PriorityID = $Self->{ParamObject}->GetParam(
                    Param => "PriorityID" . $ImpactID . '-' . $CriticalityID
                ) || '';

                next CRITICALITYID if !$PriorityID;

                $AllocateData->{$ImpactID}->{$CriticalityID} = $PriorityID;
            }
        }

        # update allocations
        $Self->{CIPAllocateObject}->AllocateUpdate(
            AllocateData => $AllocateData,
            UserID       => 1,
        );

        return $Self->{LayoutObject}->Redirect( OP => "Action=$Self->{Action}" );
    }

    # ------------------------------------------------------------ #
    # overview
    # ------------------------------------------------------------ #
    else {

        # get option lists
        my %ObjectOption;
        $ObjectOption{CriticalityList} = $Self->{GeneralCatalogObject}->ItemList(
            Class => 'ITSM::Core::Criticality',
        );
        $ObjectOption{ImpactList} = $Self->{GeneralCatalogObject}->ItemList(
            Class => 'ITSM::Core::Impact',
        );
        my %OptionPriorityList = $Self->{PriorityObject}->PriorityList(
            UserID => 1,
        );
        $ObjectOption{PriorityList} = \%OptionPriorityList;

        # get allocation data
        my $AllocateData = $Self->{CIPAllocateObject}->AllocateList(
            UserID => 1,
        );

        my $AllocateMatrix;
        $AllocateMatrix->[0]->[0]->{Class} = 'Description';

        # generate table description (Impact)
        my $Counter1 = 1;
        for my $Impact (
            sort { $ObjectOption{ImpactList}->{$a} cmp $ObjectOption{ImpactList}->{$b} }
            keys %{ $ObjectOption{ImpactList} }
            )
        {
            $AllocateMatrix->[$Counter1]->[0]->{ObjectType}   = 'Impact';
            $AllocateMatrix->[$Counter1]->[0]->{ImpactKey}    = $Impact;
            $AllocateMatrix->[$Counter1]->[0]->{ObjectOption} = $ObjectOption{ImpactList}{$Impact};
            $AllocateMatrix->[$Counter1]->[0]->{Class}        = 'Description';
            $Counter1++;
        }

        # generate table description (Criticality)
        my $Counter2 = 1;
        for my $Criticality (
            sort { $ObjectOption{CriticalityList}->{$a} cmp $ObjectOption{CriticalityList}->{$b} }
            keys %{ $ObjectOption{CriticalityList} }
            )
        {
            $AllocateMatrix->[0]->[$Counter2]->{ObjectType}     = 'Criticality';
            $AllocateMatrix->[0]->[$Counter2]->{CriticalityKey} = $Criticality;
            $AllocateMatrix->[0]->[$Counter2]->{ObjectOption}
                = $ObjectOption{CriticalityList}{$Criticality};
            $AllocateMatrix->[0]->[$Counter2]->{Class} = 'Description';
            $Counter2++;
        }

        # generate content
        for my $Row ( 1 .. ( $Counter1 - 1 ) ) {
            for my $Column ( 1 .. ( $Counter2 - 1 ) ) {

                # extract keys
                my $ImpactKey      = $AllocateMatrix->[$Row]->[0]->{ImpactKey};
                my $CriticalityKey = $AllocateMatrix->[0]->[$Column]->{CriticalityKey};

                # create option string
                my $OptionStrg = $Self->{LayoutObject}->BuildSelection(
                    Name       => 'PriorityID' . $ImpactKey . '-' . $CriticalityKey,
                    Data       => $ObjectOption{PriorityList},
                    SelectedID => $AllocateData->{$ImpactKey}{$CriticalityKey} || '',
                );

                $AllocateMatrix->[$Row]->[$Column]->{OptionStrg} = $OptionStrg;
                $AllocateMatrix->[$Row]->[$Column]->{Class}      = 'Content';
            }
        }

        # output allocation matrix
        for my $RowRef ( @{$AllocateMatrix} ) {
            $Self->{LayoutObject}->Block(
                Name => 'CIPAllocateRow',
            );

            for my $Cell ( @{$RowRef} ) {
                $Self->{LayoutObject}->Block(
                    Name => 'CIPAllocateRowColumn' . $Cell->{Class},
                    Data => $Cell,
                );
            }
        }

        # output header and navbar
        my $Output = $Self->{LayoutObject}->Header();
        $Output .= $Self->{LayoutObject}->NavigationBar();

        # generate output
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AdminITSMCIPAllocate',
            Data         => \%Param,
        );
        $Output .= $Self->{LayoutObject}->Footer();

        return $Output;
    }
}

1;

# --
# Kernel/Modules/AdminService.pm - admin frontend to manage services
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: AdminService.pm,v 1.2 2009/06/30 14:50:24 ub Exp $
# $OldId: AdminService.pm,v 1.17 2009/04/22 14:42:02 mh Exp $
# --
# 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::AdminService;

use strict;
use warnings;

use Kernel::System::Service;
use Kernel::System::Valid;
# ---
# ITSM
# ---
use Kernel::System::GeneralCatalog;
# ---

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

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

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

    # check all needed objects
    for (qw(ParamObject DBObject LayoutObject ConfigObject LogObject)) {
        if ( !$Self->{$_} ) {
            $Self->{LayoutObject}->FatalError( Message => "Got no $_!" );
        }
    }
    $Self->{ServiceObject} = Kernel::System::Service->new(%Param);
    $Self->{ValidObject}   = Kernel::System::Valid->new(%Param);
# ---
# ITSM
# ---
    $Self->{GeneralCatalogObject} = Kernel::System::GeneralCatalog->new(%Param);
# ---

    return $Self;
}

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

    # ------------------------------------------------------------ #
    # service edit
    # ------------------------------------------------------------ #
    if ( $Self->{Subaction} eq 'ServiceEdit' ) {
        my %ServiceData;

        # get params
        $ServiceData{ServiceID} = $Self->{ParamObject}->GetParam( Param => "ServiceID" );
        if ( $ServiceData{ServiceID} ne 'NEW' ) {
            %ServiceData = $Self->{ServiceObject}->ServiceGet(
                ServiceID => $ServiceData{ServiceID},
                UserID    => $Self->{UserID},
            );
        }

        # output header
        my $Output = $Self->{LayoutObject}->Header();
        $Output .= $Self->{LayoutObject}->NavigationBar();

        # output overview
        $Self->{LayoutObject}->Block(
            Name => 'Overview',
            Data => { %Param, },
        );

        # generate ParentOptionStrg
        my $TreeView = 0;
        if ( $Self->{ConfigObject}->Get('Ticket::Frontend::ListType') eq 'tree' ) {
            $TreeView = 1;
        }
        my %ServiceList = $Self->{ServiceObject}->ServiceList(
            Valid  => 0,
            UserID => $Self->{UserID},
        );
        $ServiceData{ParentOptionStrg} = $Self->{LayoutObject}->BuildSelection(
            Data           => \%ServiceList,
            Name           => 'ParentID',
            SelectedID     => $ServiceData{ParentID},
            PossibleNone   => 1,
            TreeView       => $TreeView,
            Sort           => 'TreeView',
            DisabledBranch => $ServiceData{Name},
            Translation    => 0,
            Max            => 200,
        );
# ---
# ITSM
# ---
        # generate TypeOptionStrg
        my $TypeList = $Self->{GeneralCatalogObject}->ItemList(
            Class => 'ITSM::Service::Type',
        );
        $ServiceData{TypeOptionStrg} = $Self->{LayoutObject}->BuildSelection(
            Data => $TypeList,
            Name => 'TypeID',
            SelectedID => $ServiceData{TypeID},
        );
        # generate CriticalityOptionStrg
        my $CriticalityList = $Self->{GeneralCatalogObject}->ItemList(
            Class => 'ITSM::Core::Criticality',
        );
        $ServiceData{CriticalityOptionStrg} = $Self->{LayoutObject}->BuildSelection(
            Data => $CriticalityList,
            Name => 'CriticalityID',
            SelectedID => $ServiceData{CriticalityID},
        );
# ---

        # generate ValidOptionStrg
        my %ValidList = $Self->{ValidObject}->ValidList();
        $ServiceData{ValidOptionStrg} = $Self->{LayoutObject}->BuildSelection(
            Data       => \%ValidList,
            Name       => 'ValidID',
            SelectedID => $ServiceData{ValidID} || 1,
        );

        # output service edit
        $Self->{LayoutObject}->Block(
            Name => 'ServiceEdit',
            Data => { %Param, %ServiceData, },
        );

        # show each preferences setting
        my %Preferences = ();
        if ( $Self->{ConfigObject}->Get('ServicePreferences') ) {
            %Preferences = %{ $Self->{ConfigObject}->Get('ServicePreferences') };
        }
        for my $Item ( sort keys %Preferences ) {
            my $Module = $Preferences{$Item}->{Module}
                || 'Kernel::Output::HTML::ServicePreferencesGeneric';

            # load module
            if ( !$Self->{MainObject}->Require($Module) ) {
                return $Self->{LayoutObject}->FatalError();
            }
            my $Object = $Module->new(
                %{$Self},
                ConfigItem => $Preferences{$Item},
                Debug      => $Self->{Debug},
            );
            my @Params = $Object->Param( ServiceData => \%ServiceData );
            if (@Params) {
                for my $ParamItem (@Params) {
                    $Self->{LayoutObject}->Block(
                        Name => 'Item',
                        Data => { %Param, },
                    );
                    if (
                        ref( $ParamItem->{Data} ) eq 'HASH'
                        || ref( $Preferences{$Item}->{Data} ) eq 'HASH'
                        )
                    {
                        $ParamItem->{'Option'} = $Self->{LayoutObject}->OptionStrgHashRef(
                            %{ $Preferences{$Item} },
                            %{$ParamItem},
                        );
                    }
                    $Self->{LayoutObject}->Block(
                        Name => $ParamItem->{Block} || $Preferences{$Item}->{Block} || 'Option',
                        Data => {
                            %{ $Preferences{$Item} },
                            %{$ParamItem},
                        },
                    );
                }
            }
        }

        # generate output
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AdminService',
            Data         => \%Param,
        );
        $Output .= $Self->{LayoutObject}->Footer();

        return $Output;
    }

    # ------------------------------------------------------------ #
    # service save
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'ServiceSave' ) {

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

        # get params
        my %GetParam;
# ---
# ITSM
# ---
#        for (qw(ServiceID ParentID Name ValidID Comment)) {
        for (qw(ServiceID ParentID Name ValidID Comment TypeID CriticalityID)) {
# ---
            $GetParam{$_} = $Self->{ParamObject}->GetParam( Param => $_ ) || '';
        }

        # save to database
        if ( $GetParam{ServiceID} eq 'NEW' ) {
            $GetParam{ServiceID} = $Self->{ServiceObject}->ServiceAdd(
                %GetParam,
                UserID => $Self->{UserID},
            );
            if ( !$GetParam{ServiceID} ) {
                return $Self->{LayoutObject}->ErrorScreen();
            }

        }
        else {
            my $Success = $Self->{ServiceObject}->ServiceUpdate(
                %GetParam,
                UserID => $Self->{UserID},
            );
            if ( !$Success ) {
                return $Self->{LayoutObject}->ErrorScreen();
            }
        }

        # update preferences
        my %ServiceData = $Self->{ServiceObject}->ServiceGet(
            ServiceID => $GetParam{ServiceID},
            UserID    => $Self->{UserID},
        );
        my %Preferences = ();
        if ( $Self->{ConfigObject}->Get('ServicePreferences') ) {
            %Preferences = %{ $Self->{ConfigObject}->Get('ServicePreferences') };
        }
        for my $Item ( sort keys %Preferences ) {
            my $Module = $Preferences{$Item}->{Module}
                || 'Kernel::Output::HTML::ServicePreferencesGeneric';

            # load module
            if ( !$Self->{MainObject}->Require($Module) ) {
                return $Self->{LayoutObject}->FatalError();
            }

            my $Object = $Module->new(
                %{$Self},
                ConfigItem => $Preferences{$Item},
                Debug      => $Self->{Debug},
            );
            my $Note;
            my @Params = $Object->Param( ServiceData => \%ServiceData );
            if (@Params) {
                my %GetParam = ();
                for my $ParamItem (@Params) {
                    my @Array = $Self->{ParamObject}->GetArray( Param => $ParamItem->{Name} );
                    $GetParam{ $ParamItem->{Name} } = \@Array;
                }
                if ( !$Object->Run( GetParam => \%GetParam, ServiceData => \%ServiceData ) ) {
                    $Note .= $Self->{LayoutObject}->Notify( Info => $Object->Error() );
                }
            }
        }

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

    # ------------------------------------------------------------ #
    # service overview
    # ------------------------------------------------------------ #
    else {

        # output header
        my $Output = $Self->{LayoutObject}->Header();
        $Output .= $Self->{LayoutObject}->NavigationBar();

        # check if service is enabled to use it here
        if ( !$Self->{ConfigObject}->Get('Ticket::Service') ) {
            $Output .= $Self->{LayoutObject}->Notify(
                Priority => 'Error',
                Data     => '$Text{"You need to activate %s first to use it!", "Service"}',
                Link =>
                    '$Env{"Baselink"}Action=AdminSysConfig&Subaction=Edit&SysConfigGroup=Ticket&SysConfigSubGroup=Core::Ticket#Ticket::Service"',
            );
        }

        # output overview
        $Self->{LayoutObject}->Block(
            Name => 'Overview',
            Data => { %Param, },
        );

        # output overview result
        $Self->{LayoutObject}->Block(
            Name => 'OverviewList',
            Data => { %Param, },
        );

        # get service list
        my %ServiceList = $Self->{ServiceObject}->ServiceList(
            Valid  => 0,
            UserID => $Self->{UserID},
        );

        # get valid list
        my %ValidList = $Self->{ValidObject}->ValidList();

        # add suffix for correct sorting
        for ( keys %ServiceList ) {
            $ServiceList{$_} .= '::';
        }
        my $CssClass;
        for my $ServiceID ( sort { $ServiceList{$a} cmp $ServiceList{$b} } keys %ServiceList ) {

            # set output class
            if ( $CssClass && $CssClass eq 'searchactive' ) {
                $CssClass = 'searchpassive';
            }
            else {
                $CssClass = 'searchactive';
            }

            # get service data
            my %ServiceData = $Self->{ServiceObject}->ServiceGet(
                ServiceID => $ServiceID,
                UserID    => $Self->{UserID},
            );

            # output row
            if ( $Self->{ConfigObject}->Get('Ticket::Frontend::ListType') eq 'tree' ) {

                # calculate level space
                my @Fragment   = split '::', $ServiceData{Name};
                my $Level      = scalar @Fragment - 1;
                my $LevelSpace = '&nbsp;&nbsp;&nbsp;&nbsp;' x $Level;

                # output row
                $Self->{LayoutObject}->Block(
                    Name => 'OverviewListRow',
                    Data => {
                        %ServiceData,
                        LevelSpace => $LevelSpace,
                        Name       => $ServiceData{NameShort},
                        CssClass   => $CssClass,
                        Valid      => $ValidList{ $ServiceData{ValidID} },
                    },
                );
            }
            else {

                # output row
                $Self->{LayoutObject}->Block(
                    Name => 'OverviewListRow',
                    Data => {
                        %ServiceData,
                        CssClass => $CssClass,
                        Valid    => $ValidList{ $ServiceData{ValidID} },
                    },
                );
            }
        }

        # generate output
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AdminService',
            Data         => \%Param,
        );
        $Output .= $Self->{LayoutObject}->Footer();

        return $Output;
    }
}

1;

# --
# Kernel/Modules/AdminSLA.pm - admin frontend to manage slas
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: AdminSLA.pm,v 1.5 2009/06/30 14:49:43 ub Exp $
# $OldId: AdminSLA.pm,v 1.20 2009/02/17 23:37:11 martin Exp $
# --
# 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::AdminSLA;

use strict;
use warnings;

use Kernel::System::Service;
use Kernel::System::SLA;
use Kernel::System::Valid;
# ---
# ITSM
# ---
use Kernel::System::GeneralCatalog;
# ---

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

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

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

    # check all needed objects
    for (qw(ParamObject DBObject LayoutObject ConfigObject LogObject)) {
        if ( !$Self->{$_} ) {
            $Self->{LayoutObject}->FatalError( Message => "Got no $_!" );
        }
    }
    $Self->{ServiceObject} = Kernel::System::Service->new(%Param);
    $Self->{SLAObject}     = Kernel::System::SLA->new(%Param);
    $Self->{ValidObject}   = Kernel::System::Valid->new(%Param);
# ---
# ITSM
# ---
    $Self->{GeneralCatalogObject} = Kernel::System::GeneralCatalog->new(%Param);
# ---

    return $Self;
}

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

    # ------------------------------------------------------------ #
    # sla edit
    # ------------------------------------------------------------ #
    if ( $Self->{Subaction} eq 'SLAEdit' ) {

        # get params
        my %SLAData;
        $SLAData{SLAID} = $Self->{ParamObject}->GetParam( Param => 'SLAID' );

        if ( $SLAData{SLAID} ) {

            # get sla data
            %SLAData = $Self->{SLAObject}->SLAGet(
                SLAID  => $SLAData{SLAID},
                UserID => $Self->{UserID},
            );
        }
        else {
            $SLAData{ServiceID} = $Self->{ParamObject}->GetParam( Param => 'ServiceID' );
        }

        # get service list
        my %ServiceList = $Self->{ServiceObject}->ServiceList(
            Valid  => 0,
            UserID => $Self->{UserID},
        );

        # generate ServiceOptionStrg
        my $TreeView = 0;
        if ( $Self->{ConfigObject}->Get('Ticket::Frontend::ListType') eq 'tree' ) {
            $TreeView = 1;
        }
        $Param{ServiceOptionStrg} = $Self->{LayoutObject}->BuildSelection(
            Data        => \%ServiceList,
            Name        => 'ServiceIDs',
            SelectedID  => $SLAData{ServiceIDs} || [],
            Multiple    => 1,
            Size        => 5,
            TreeView    => $TreeView,
            Sort        => 'TreeView',
            Translation => 0,
            Max         => 200,
        );
# ---
# ITSM
# ---
        # generate TypeOptionStrg
        my $TypeList = $Self->{GeneralCatalogObject}->ItemList(
            Class => 'ITSM::SLA::Type',
        );
        $Param{TypeOptionStrg} = $Self->{LayoutObject}->BuildSelection(
            Data => $TypeList,
            Name => 'TypeID',
            SelectedID => $SLAData{TypeID},
        );
# ---

        # generate CalendarOptionStrg
        my %CalendarList;
        for ( '', 1 .. 50 ) {
            if ( $Self->{ConfigObject}->Get("TimeVacationDays::Calendar$_") ) {
                $CalendarList{$_} = "Calendar $_ - "
                    . $Self->{ConfigObject}->Get( "TimeZone::Calendar" . $_ . "Name" );
            }
        }
        $SLAData{CalendarOptionStrg} = $Self->{LayoutObject}->BuildSelection(
            Data         => \%CalendarList,
            Name         => 'Calendar',
            SelectedID   => $SLAData{Calendar},
            PossibleNone => 1,
        );
        my %NotifyLevelList = (
            10 => '10%',
            20 => '20%',
            30 => '30%',
            40 => '40%',
            50 => '50%',
            60 => '60%',
            70 => '70%',
            80 => '80%',
            90 => '90%',
        );
        $SLAData{FirstResponseNotifyOptionStrg} = $Self->{LayoutObject}->BuildSelection(
            Data         => \%NotifyLevelList,
            Name         => 'FirstResponseNotify',
            SelectedID   => $SLAData{FirstResponseNotify},
            PossibleNone => 1,
        );
        $SLAData{UpdateNotifyOptionStrg} = $Self->{LayoutObject}->BuildSelection(
            Data         => \%NotifyLevelList,
            Name         => 'UpdateNotify',
            SelectedID   => $SLAData{UpdateNotify},
            PossibleNone => 1,
        );
        $SLAData{SolutionNotifyOptionStrg} = $Self->{LayoutObject}->BuildSelection(
            Data         => \%NotifyLevelList,
            Name         => 'SolutionNotify',
            SelectedID   => $SLAData{SolutionNotify},
            PossibleNone => 1,
        );

        # generate ValidOptionStrg
        my %ValidList = $Self->{ValidObject}->ValidList();
        $SLAData{ValidOptionStrg} = $Self->{LayoutObject}->BuildSelection(
            Data       => \%ValidList,
            Name       => 'ValidID',
            SelectedID => $SLAData{ValidID} || 1,
        );

        # output sla edit
        $Self->{LayoutObject}->Block(
            Name => 'Overview',
            Data => {
                %Param,
            },
        );
        $Self->{LayoutObject}->Block(
            Name => 'SLAEdit',
            Data => {
                %Param,
                %SLAData,
            },
        );

        # show each preferences setting
        my %Preferences = ();
        if ( $Self->{ConfigObject}->Get('SLAPreferences') ) {
            %Preferences = %{ $Self->{ConfigObject}->Get('SLAPreferences') };
        }
        for my $Item ( sort keys %Preferences ) {
            my $Module = $Preferences{$Item}->{Module}
                || 'Kernel::Output::HTML::SLAPreferencesGeneric';

            # load module
            if ( !$Self->{MainObject}->Require($Module) ) {
                return $Self->{LayoutObject}->FatalError();
            }
            my $Object = $Module->new(
                %{$Self},
                ConfigItem => $Preferences{$Item},
                Debug      => $Self->{Debug},
            );
            my @Params = $Object->Param( SLAData => \%SLAData );
            if (@Params) {
                for my $ParamItem (@Params) {
                    $Self->{LayoutObject}->Block(
                        Name => 'SLAItem',
                        Data => { %Param, },
                    );
                    if (
                        ref( $ParamItem->{Data} ) eq 'HASH'
                        || ref( $Preferences{$Item}->{Data} ) eq 'HASH'
                        )
                    {
                        $ParamItem->{'Option'} = $Self->{LayoutObject}->OptionStrgHashRef(
                            %{ $Preferences{$Item} },
                            %{$ParamItem},
                        );
                    }
                    $Self->{LayoutObject}->Block(
                        Name => $ParamItem->{Block} || $Preferences{$Item}->{Block} || 'Option',
                        Data => {
                            %{ $Preferences{$Item} },
                            %{$ParamItem},
                        },
                    );
                }
            }
        }

        # output overview
        my $Output = $Self->{LayoutObject}->Header();
        $Output .= $Self->{LayoutObject}->NavigationBar();
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AdminSLA',
            Data         => \%Param,
        );
        $Output .= $Self->{LayoutObject}->Footer();

        return $Output;
    }

    # ------------------------------------------------------------ #
    # sla save
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'SLASave' ) {

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

        # get params
        my %GetParam;
        for my $Param (
# ---
# ITSM
# ---
#            qw(SLAID Name Calendar FirstResponseTime FirstResponseNotify SolutionTime SolutionNotify UpdateTime UpdateNotify ValidID Comment)
            qw(SLAID Name Calendar FirstResponseTime FirstResponseNotify SolutionTime SolutionNotify UpdateTime UpdateNotify ValidID Comment TypeID MinTimeBetweenIncidents)
# ---
            )
        {
            $GetParam{$Param} = $Self->{ParamObject}->GetParam( Param => $Param ) || '';
        }

        # get service ids
        my @ServiceIDs = $Self->{ParamObject}->GetArray( Param => 'ServiceIDs' );
        $GetParam{ServiceIDs} = \@ServiceIDs;

        # save to database
        if ( !$GetParam{SLAID} ) {

            # add a new sla
            $GetParam{SLAID} = $Self->{SLAObject}->SLAAdd(
                %GetParam,
                UserID => $Self->{UserID},
            );
            if ( !$GetParam{SLAID} ) {
                return $Self->{LayoutObject}->ErrorScreen();
            }
        }
        else {

            # update the sla
            my $Success = $Self->{SLAObject}->SLAUpdate(
                %GetParam,
                UserID => $Self->{UserID},
            );
            if ( !$Success ) {
                return $Self->{LayoutObject}->ErrorScreen();
            }
        }

        # update preferences
        my %SLAData = $Self->{SLAObject}->SLAGet(
            SLAID  => $GetParam{SLAID},
            UserID => $Self->{UserID},
        );
        my %Preferences = ();
        if ( $Self->{ConfigObject}->Get('SLAPreferences') ) {
            %Preferences = %{ $Self->{ConfigObject}->Get('SLAPreferences') };
        }
        for my $Item ( sort keys %Preferences ) {
            my $Module = $Preferences{$Item}->{Module}
                || 'Kernel::Output::HTML::SLAPreferencesGeneric';

            # load module
            if ( !$Self->{MainObject}->Require($Module) ) {
                return $Self->{LayoutObject}->FatalError();
            }

            my $Object = $Module->new(
                %{$Self},
                ConfigItem => $Preferences{$Item},
                Debug      => $Self->{Debug},
            );
            my $Note;
            my @Params = $Object->Param( SLAData => \%SLAData );
            if (@Params) {
                my %GetParam = ();
                for my $ParamItem (@Params) {
                    my @Array = $Self->{ParamObject}->GetArray( Param => $ParamItem->{Name} );
                    $GetParam{ $ParamItem->{Name} } = \@Array;
                }
                if ( !$Object->Run( GetParam => \%GetParam, SLAData => \%SLAData ) ) {
                    $Note .= $Self->{LayoutObject}->Notify( Info => $Object->Error() );
                }
            }
        }

        return $Self->{LayoutObject}->Redirect( OP => "Action=$Self->{Action}" );
    }

    # ------------------------------------------------------------ #
    # sla overview
    # ------------------------------------------------------------ #
    else {

        # output header
        my $Output = $Self->{LayoutObject}->Header();
        $Output .= $Self->{LayoutObject}->NavigationBar();

        # check if service is enabled to use it here
        if ( !$Self->{ConfigObject}->Get('Ticket::Service') ) {
            $Output .= $Self->{LayoutObject}->Notify(
                Priority => 'Error',
                Data     => '$Text{"You need to activate %s first to use it!", "Service"}',
                Link =>
                    '$Env{"Baselink"}Action=AdminSysConfig&Subaction=Edit&SysConfigGroup=Ticket&SysConfigSubGroup=Core::Ticket#Ticket::Service"',
            );
        }

        # output overview
        $Self->{LayoutObject}->Block(
            Name => 'Overview',
            Data => {
                %Param,
            },
        );

        # output overview result
        $Self->{LayoutObject}->Block(
            Name => 'OverviewList',
            Data => {
                %Param,
            },
        );

        # get service list
        my %ServiceList = $Self->{ServiceObject}->ServiceList(
            Valid  => 0,
            UserID => $Self->{UserID},
        );

        # get valid list
        my %ValidList = $Self->{ValidObject}->ValidList();

        # get sla list
        my %SLAList = $Self->{SLAObject}->SLAList(
            Valid  => 0,
            UserID => $Self->{UserID},
        );

        my $CssClass = '';
        SLAID:
        for my $SLAID ( sort { lc $SLAList{$a} cmp lc $SLAList{$b} } keys %SLAList ) {

            # set output object
            $CssClass = $CssClass eq 'searchactive' ? 'searchpassive' : 'searchactive';

            # get the sla data
            my %SLAData = $Self->{SLAObject}->SLAGet(
                SLAID  => $SLAID,
                UserID => $Self->{UserID},
            );

            # build the service list
            my @ServiceList;
            for my $ServiceID (
                sort { lc $ServiceList{$a} cmp lc $ServiceList{$b} }
                @{ $SLAData{ServiceIDs} }
                )
            {
                push @ServiceList, $ServiceList{$ServiceID} || '-';
            }

            # output overview list row
            $Self->{LayoutObject}->Block(
                Name => 'OverviewListRow',
                Data => {
                    %SLAData,
                    Service => $ServiceList[0] || '-',
                    CssClass => $CssClass,
                    Valid    => $ValidList{ $SLAData{ValidID} },
                },
            );

            next SLAID if scalar @ServiceList <= 1;

            # remove the first service id
            shift @ServiceList;

            for my $ServiceName (@ServiceList) {

                # output overview list row
                $Self->{LayoutObject}->Block(
                    Name => 'OverviewListRow',
                    Data => {
                        Service  => $ServiceName,
                        CssClass => $CssClass,
                    },
                );
            }
        }

        # generate output
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AdminSLA',
            Data         => \%Param,
        );
        $Output .= $Self->{LayoutObject}->Footer();

        return $Output;
    }
}

1;

IyAtLQojIEtlcm5lbC9Nb2R1bGVzL0FnZW50SVRTTVNlcnZpY2UucG0gLSB0aGUgT1RSUzo6SVRTTSBTZXJ2aWNlIG1vZHVsZQojIENvcHlyaWdodCAoQykgMjAwMS0yMDEwIE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogQWdlbnRJVFNNU2VydmljZS5wbSx2IDEuNyAyMDEwLzA0LzEzIDEyOjMwOjIxIHViIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TW9kdWxlczo6QWdlbnRJVFNNU2VydmljZTsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6U2VydmljZTsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuNyAkKSBbMV07CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHslUGFyYW19OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgICMgY2hlY2sgbmVlZGVkIG9iamVjdHMKICAgIGZvciBteSAkT2JqZWN0IChxdyhDb25maWdPYmplY3QgUGFyYW1PYmplY3QgREJPYmplY3QgTGF5b3V0T2JqZWN0IExvZ09iamVjdCkpIHsKICAgICAgICBpZiAoICEkU2VsZi0+eyRPYmplY3R9ICkgewogICAgICAgICAgICAkU2VsZi0+e0xheW91dE9iamVjdH0tPkZhdGFsRXJyb3IoIE1lc3NhZ2UgPT4gIkdvdCBubyAkT2JqZWN0ISIgKTsKICAgICAgICB9CiAgICB9CiAgICAkU2VsZi0+e1NlcnZpY2VPYmplY3R9ID0gS2VybmVsOjpTeXN0ZW06OlNlcnZpY2UtPm5ldyglUGFyYW0pOwoKICAgIHJldHVybiAkU2VsZjsKfQoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgb3V0cHV0IG92ZXJ2aWV3CiAgICAkU2VsZi0+e0xheW91dE9iamVjdH0tPkJsb2NrKAogICAgICAgIE5hbWUgPT4gJ092ZXJ2aWV3JywKICAgICAgICBEYXRhID0+IHslUGFyYW19LAogICAgKTsKCiAgICAjIGdldCBzZXJ2aWNlIGxpc3QKICAgIG15ICVTZXJ2aWNlTGlzdCA9ICRTZWxmLT57U2VydmljZU9iamVjdH0tPlNlcnZpY2VMaXN0KAogICAgICAgIFVzZXJJRCA9PiAkU2VsZi0+e1VzZXJJRH0sCiAgICApOwoKICAgICMgYWRkIHN1ZmZpeCBmb3IgY29ycmVjdCBzb3J0aW5nCiAgICBmb3IgbXkgJFNlcnZpY2UgKCB2YWx1ZXMgJVNlcnZpY2VMaXN0ICkgewogICAgICAgICRTZXJ2aWNlIC49ICc6Oic7CiAgICB9CgogICAgIyBzZXQgaW5jaWRlbnQgc2lnbmFsCiAgICBteSAlSW5jaVNpZ25hbHMgPSAoCiAgICAgICAgb3BlcmF0aW9uYWwgPT4gJ2dyZWVubGVkJywKICAgICAgICB3YXJuaW5nICAgICA9PiAneWVsbG93bGVkJywKICAgICAgICBpbmNpZGVudCAgICA9PiAncmVkbGVkJywKICAgICk7CgogICAgIyBjaGVjayBpZiB0cmVldmlldyBpcyBlbmFibGVkCiAgICBteSAkVHJlZVZpZXcgPSAwOwogICAgaWYgKCAkU2VsZi0+e0NvbmZpZ09iamVjdH0tPkdldCgnVGlja2V0OjpGcm9udGVuZDo6TGlzdFR5cGUnKSBlcSAndHJlZScgKSB7CiAgICAgICAgJFRyZWVWaWV3ID0gMTsKICAgIH0KCiAgICBteSAkQ3NzQ2xhc3MgPSAnJzsKICAgIGZvciBteSAkU2VydmljZUlEICggc29ydCB7ICRTZXJ2aWNlTGlzdHskYX0gY21wICRTZXJ2aWNlTGlzdHskYn0gfSBrZXlzICVTZXJ2aWNlTGlzdCApIHsKCiAgICAgICAgIyBzZXQgb3V0cHV0IG9iamVjdAogICAgICAgICRDc3NDbGFzcyA9ICRDc3NDbGFzcyBlcSAnc2VhcmNocGFzc2l2ZScgPyAnc2VhcmNoYWN0aXZlJyA6ICdzZWFyY2hwYXNzaXZlJzsKCiAgICAgICAgIyBnZXQgc2VydmljZSBkYXRhCiAgICAgICAgbXkgJVNlcnZpY2UgPSAkU2VsZi0+e1NlcnZpY2VPYmplY3R9LT5TZXJ2aWNlR2V0KAogICAgICAgICAgICBTZXJ2aWNlSUQgPT4gJFNlcnZpY2VJRCwKICAgICAgICAgICAgVXNlcklEICAgID0+ICRTZWxmLT57VXNlcklEfSwKICAgICAgICApOwoKICAgICAgICAjIG91dHB1dCByb3cKICAgICAgICBpZiAoJFRyZWVWaWV3KSB7CgogICAgICAgICAgICAjIGNhbGN1bGF0ZSBsZXZlbCBzcGFjZQogICAgICAgICAgICBteSBARnJhZ21lbnQgICA9IHNwbGl0ICc6OicsICRTZXJ2aWNle05hbWV9OwogICAgICAgICAgICBteSAkTGV2ZWwgICAgICA9IHNjYWxhciBARnJhZ21lbnQgLSAxOwogICAgICAgICAgICBteSAkTGV2ZWxTcGFjZSA9ICcmbmJzcDsmbmJzcDsmbmJzcDsmbmJzcDsnIHggJExldmVsOwoKICAgICAgICAgICAgIyBvdXRwdXQgb3ZlcnZpZXcgcm93CiAgICAgICAgICAgICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+QmxvY2soCiAgICAgICAgICAgICAgICBOYW1lID0+ICdPdmVydmlld1JvdycsCiAgICAgICAgICAgICAgICBEYXRhID0+IHsKICAgICAgICAgICAgICAgICAgICAlU2VydmljZSwKICAgICAgICAgICAgICAgICAgICBMZXZlbFNwYWNlICAgID0+ICRMZXZlbFNwYWNlLAogICAgICAgICAgICAgICAgICAgIE5hbWUgICAgICAgICAgPT4gJFNlcnZpY2V7TmFtZVNob3J0fSwKICAgICAgICAgICAgICAgICAgICBDdXJJbmNpU2lnbmFsID0+ICRJbmNpU2lnbmFsc3sgJFNlcnZpY2V7Q3VySW5jaVN0YXRlVHlwZX0gfSwKICAgICAgICAgICAgICAgICAgICBDc3NDbGFzcyAgICAgID0+ICRDc3NDbGFzcywKICAgICAgICAgICAgICAgIH0sCiAgICAgICAgICAgICk7CiAgICAgICAgfQogICAgICAgIGVsc2UgewoKICAgICAgICAgICAgIyBvdXRwdXQgb3ZlcnZpZXcgcm93CiAgICAgICAgICAgICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+QmxvY2soCiAgICAgICAgICAgICAgICBOYW1lID0+ICdPdmVydmlld1JvdycsCiAgICAgICAgICAgICAgICBEYXRhID0+IHsKICAgICAgICAgICAgICAgICAgICAlU2VydmljZSwKICAgICAgICAgICAgICAgICAgICBOYW1lICAgICAgICAgID0+ICRTZXJ2aWNle05hbWV9LAogICAgICAgICAgICAgICAgICAgIEN1ckluY2lTaWduYWwgPT4gJEluY2lTaWduYWxzeyAkU2VydmljZXtDdXJJbmNpU3RhdGVUeXBlfSB9LAogICAgICAgICAgICAgICAgICAgIENzc0NsYXNzICAgICAgPT4gJENzc0NsYXNzLAogICAgICAgICAgICAgICAgfSwKICAgICAgICAgICAgKTsKICAgICAgICB9CiAgICB9CgogICAgIyBpbnZlc3RpZ2F0ZSByZWZyZXNoCiAgICBteSAkUmVmcmVzaCA9ICRTZWxmLT57VXNlclJlZnJlc2hUaW1lfSA/IDYwICogJFNlbGYtPntVc2VyUmVmcmVzaFRpbWV9IDogdW5kZWY7CgogICAgIyBvdXRwdXQgaGVhZGVyCiAgICBteSAkT3V0cHV0ID0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5IZWFkZXIoCiAgICAgICAgVGl0bGUgICA9PiAnT3ZlcnZpZXcnLAogICAgICAgIFJlZnJlc2ggPT4gJFJlZnJlc2gsCiAgICApOwogICAgJE91dHB1dCAuPSAkU2VsZi0+e0xheW91dE9iamVjdH0tPk5hdmlnYXRpb25CYXIoKTsKCiAgICAjIGdlbmVyYXRlIG91dHB1dAogICAgJE91dHB1dCAuPSAkU2VsZi0+e0xheW91dE9iamVjdH0tPk91dHB1dCgKICAgICAgICBUZW1wbGF0ZUZpbGUgPT4gJ0FnZW50SVRTTVNlcnZpY2UnLAogICAgICAgIERhdGEgICAgICAgICA9PiBcJVBhcmFtLAogICAgKTsKICAgICRPdXRwdXQgLj0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5Gb290ZXIoKTsKCiAgICByZXR1cm4gJE91dHB1dDsKfQoKMTsK
# --
# Kernel/Modules/AgentITSMServicePrint.pm - print layout for itsm service agent interface
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: AgentITSMServicePrint.pm,v 1.3 2009/05/18 09:48:35 mh Exp $
# --
# 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::AgentITSMServicePrint;

use strict;
use warnings;

use Kernel::System::PDF;
use Kernel::System::Service;
use Kernel::System::SLA;

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

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

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

    # check needed objects
    for my $Object (qw(ConfigObject ParamObject DBObject LayoutObject LogObject)) {
        if ( !$Self->{$Object} ) {
            $Self->{LayoutObject}->FatalError( Message => "Got no $Object!" );
        }
    }
    $Self->{PDFObject}     = Kernel::System::PDF->new(%Param);
    $Self->{ServiceObject} = Kernel::System::Service->new(%Param);
    $Self->{SLAObject}     = Kernel::System::SLA->new(%Param);

    return $Self;
}

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

    # get params
    my $ServiceID = $Self->{ParamObject}->GetParam( Param => 'ServiceID' );

    # check needed stuff
    if ( !$ServiceID ) {
        return $Self->{LayoutObject}->ErrorScreen(
            Message => 'No ServiceID is given!',
            Comment => 'Please contact the admin.',
        );
    }

    # get service
    my %Service = $Self->{ServiceObject}->ServiceGet(
        ServiceID => $ServiceID,
        UserID    => $Self->{UserID},
    );
    if ( !$Service{ServiceID} ) {
        return $Self->{LayoutObject}->ErrorScreen(
            Message => "ServiceID $ServiceID not found in database!",
            Comment => 'Please contact the admin.',
        );
    }

    # get sla list
    my %SLAList = $Self->{SLAObject}->SLAList(
        ServiceID => $Service{ServiceID},
        UserID    => $Self->{UserID},
    );

    # get user data (create by)
    my %CreateBy = $Self->{UserObject}->GetUserData(
        UserID => $Service{CreateBy},
        Cached => 1,
    );

    # get user data (change by)
    my %ChangeBy = $Self->{UserObject}->GetUserData(
        UserID => $Service{ChangeBy},
        Cached => 1,
    );

    # generate pdf output
    if ( $Self->{PDFObject} ) {
        my %Page;
        my $Url = ' ';
        if ( $ENV{REQUEST_URI} ) {
            $Url
                = $Self->{ConfigObject}->Get('HttpType') . '://'
                . $Self->{ConfigObject}->Get('FQDN')
                . $ENV{REQUEST_URI};
        }

        # get maximum number of pages
        $Page{MaxPages} = $Self->{ConfigObject}->Get('PDF::MaxPages');
        if ( !$Page{MaxPages} || $Page{MaxPages} < 1 || $Page{MaxPages} > 1000 ) {
            $Page{MaxPages} = 100;
        }
        $Page{MarginTop}    = 30;
        $Page{MarginRight}  = 40;
        $Page{MarginBottom} = 40;
        $Page{MarginLeft}   = 40;
        $Page{HeaderRight}  = $Self->{LayoutObject}->{LanguageObject}->Get('Service');
        $Page{HeadlineLeft} = $Service{NameShort};
        $Page{HeadlineRight}
            = $Self->{LayoutObject}->{LanguageObject}->Get('printed by') . ' '
            . $Self->{UserFirstname} . ' '
            . $Self->{UserLastname} . ' ('
            . $Self->{UserEmail} . ') '
            . $Self->{LayoutObject}->Output( Template => '$Env{"Time"}' );
        $Page{FooterLeft} = $Url;
        $Page{PageText}   = $Self->{LayoutObject}->{LanguageObject}->Get('Page');
        $Page{PageCount}  = 1;

        # create new pdf document
        $Self->{PDFObject}->DocumentNew(
            Title  => $Self->{ConfigObject}->Get('Product') . ': ' . $Service{NameShort},
            Encode => $Self->{LayoutObject}->{UserCharset},
        );

        # create first pdf page
        $Self->{PDFObject}->PageNew(
            %Page,
            FooterRight => $Page{PageText} . ' ' . $Page{PageCount},
        );
        $Page{PageCount}++;

        # output general infos
        $Self->_PDFOutputGeneralInfos(
            Page     => \%Page,
            Service  => \%Service,
            CreateBy => \%CreateBy,
            ChangeBy => \%ChangeBy,
        );

        # output associated slas
        if (%SLAList) {
            $Self->_PDFOutputAssociatedSLAs(
                Page    => \%Page,
                SLAList => \%SLAList,
            );
        }

        # output detailed infos
        $Self->_PDFOutputDetailedInfos(
            Page    => \%Page,
            Service => \%Service,
        );

        # create file name
        my $Filename = $Self->{MainObject}->FilenameCleanUp(
            Filename => $Service{NameShort},
            Type     => 'Attachment',
        );
        my ( $s, $m, $h, $D, $M, $Y ) = $Self->{TimeObject}->SystemTime2Date(
            SystemTime => $Self->{TimeObject}->SystemTime(),
        );
        $M = sprintf( "%02d", $M );
        $D = sprintf( "%02d", $D );
        $h = sprintf( "%02d", $h );
        $m = sprintf( "%02d", $m );

        # return the pdf document
        return $Self->{LayoutObject}->Attachment(
            Filename    => 'service_' . $Filename . "_$Y-$M-$D\_$h-$m.pdf",
            ContentType => 'application/pdf',
            Content     => $Self->{PDFObject}->DocumentOutput(),
            Type        => 'attachment',
        );
    }

    # generate html output
    else {

        # output header
        my $Output = $Self->{LayoutObject}->PrintHeader( Value => $Service{NameShort} );

        # output associated slas
        if ( keys %SLAList ) {
            $Self->{LayoutObject}->Block( Name => "AssociatedSLAs" );
            for my $SLAID ( sort keys %SLAList ) {
                $Self->{LayoutObject}->Block(
                    Name => "AssociatedSLAsRow",
                    Data => {
                        Name => $SLAList{$SLAID},
                    },
                );
            }
        }

        # generate output
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AgentITSMServicePrint',
            Data         => {
                CreateByUserLogin     => $CreateBy{UserLogin},
                CreateByUserFirstname => $CreateBy{UserFirstname},
                CreateByUserLastname  => $CreateBy{UserLastname},
                ChangeByUserLogin     => $ChangeBy{UserLogin},
                ChangeByUserFirstname => $ChangeBy{UserFirstname},
                ChangeByUserLastname  => $ChangeBy{UserLastname},
                %Service,
            },
        );

        # add footer
        $Output .= $Self->{LayoutObject}->PrintFooter();

        return $Output;
    }
}

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

    # check needed stuff
    for my $Argument (qw(Page Service CreateBy ChangeBy)) {
        if ( !defined $Param{$Argument} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "Need $Argument!"
            );
            return;
        }
    }

    # create left table
    my $TableLeft = [
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Service') . ':',
            Value => $Param{Service}->{NameShort},
        },
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Current Incident State') . ':',
            Value => $Param{Service}->{CurInciState},
        },
    ];

    # create right table
    my $TableRight = [
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Created') . ':',
            Value => $Self->{LayoutObject}->Output(
                Template => '$TimeLong{"$Data{"CreateTime"}"}',
                Data     => \%{ $Param{Service} },
            ),
        },
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Created by') . ':',
            Value => $Param{CreateBy}->{UserLogin} . ' ('
                . $Param{CreateBy}->{UserFirstname} . ' '
                . $Param{CreateBy}->{UserLastname} . ')',
        },
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Last changed') . ':',
            Value => $Self->{LayoutObject}->Output(
                Template => '$TimeLong{"$Data{"ChangeTime"}"}',
                Data     => \%{ $Param{Service} },
            ),
        },
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Last changed by') . ':',
            Value => $Param{ChangeBy}->{UserLogin} . ' ('
                . $Param{ChangeBy}->{UserFirstname} . ' '
                . $Param{ChangeBy}->{UserLastname} . ')',
        },
    ];

    my $Rows = @{$TableLeft};
    if ( @{$TableRight} > $Rows ) {
        $Rows = @{$TableRight};
    }

    my %TableParam;
    for my $Row ( 1 .. $Rows ) {
        $Row--;
        $TableParam{CellData}[$Row][0]{Content}         = $TableLeft->[$Row]->{Key};
        $TableParam{CellData}[$Row][0]{Font}            = 'ProportionalBold';
        $TableParam{CellData}[$Row][1]{Content}         = $TableLeft->[$Row]->{Value};
        $TableParam{CellData}[$Row][2]{Content}         = ' ';
        $TableParam{CellData}[$Row][2]{BackgroundColor} = '#FFFFFF';
        $TableParam{CellData}[$Row][3]{Content}         = $TableRight->[$Row]->{Key};
        $TableParam{CellData}[$Row][3]{Font}            = 'ProportionalBold';
        $TableParam{CellData}[$Row][4]{Content}         = $TableRight->[$Row]->{Value};
    }
    $TableParam{ColumnData}[0]{Width} = 80;
    $TableParam{ColumnData}[1]{Width} = 170.5;
    $TableParam{ColumnData}[2]{Width} = 4;
    $TableParam{ColumnData}[3]{Width} = 80;
    $TableParam{ColumnData}[4]{Width} = 170.5;
    $TableParam{Type}                 = 'Cut';
    $TableParam{Border}               = 0;
    $TableParam{FontSize}             = 6;
    $TableParam{BackgroundColorEven}  = '#AAAAAA';
    $TableParam{BackgroundColorOdd}   = '#DDDDDD';
    $TableParam{Padding}              = 1;
    $TableParam{PaddingTop}           = 3;
    $TableParam{PaddingBottom}        = 3;

    # output table
    PAGE:
    for ( $Param{Page}->{PageCount} .. $Param{Page}->{MaxPages} ) {

        # output table (or a fragment of it)
        %TableParam = $Self->{PDFObject}->Table(%TableParam);

        # stop output or output next page
        last PAGE if $TableParam{State};

        $Self->{PDFObject}->PageNew(
            %{ $Param{Page} },
            FooterRight => $Param{Page}->{PageText} . ' ' . $Param{Page}->{PageCount}
        );
        $Param{Page}->{PageCount}++;
    }

    return 1;
}

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

    # check needed stuff
    for my $Argument (qw(Page Service)) {
        if ( !defined $Param{$Argument} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "Need $Argument!"
            );
            return;
        }
    }

    # set new position
    $Self->{PDFObject}->PositionSet(
        Move => 'relativ',
        Y    => -15,
    );

    # output headline
    $Self->{PDFObject}->Text(
        Text     => $Self->{LayoutObject}->{LanguageObject}->Get('Service'),
        Height   => 7,
        Type     => 'Cut',
        Font     => 'ProportionalBoldItalic',
        FontSize => 7,
        Color    => '#666666',
    );

    # set new position
    $Self->{PDFObject}->PositionSet(
        Move => 'relativ',
        Y    => -4,
    );

    # create table
    my $Table = [
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Service') . ':',
            Value => $Param{Service}->{Name},
        },
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Type') . ':',
            Value => $Self->{LayoutObject}->{LanguageObject}->Get( $Param{Service}->{Type} ),
        },
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Criticality') . ':',
            Value => $Self->{LayoutObject}->{LanguageObject}->Get( $Param{Service}->{Criticality} ),
        },
    ];
    my %TableParam;
    my $Rows = @{$Table};
    for my $Row ( 1 .. $Rows ) {
        $Row--;
        $TableParam{CellData}[$Row][0]{Content} = $Table->[$Row]->{Key};
        $TableParam{CellData}[$Row][0]{Font}    = 'ProportionalBold';
        $TableParam{CellData}[$Row][1]{Content} = $Table->[$Row]->{Value};
    }
    $TableParam{ColumnData}[0]{Width} = 80;
    $TableParam{ColumnData}[1]{Width} = 431;
    $TableParam{Type}                 = 'Cut';
    $TableParam{Border}               = 0;
    $TableParam{FontSize}             = 6;
    $TableParam{BackgroundColor}      = '#DDDDDD';
    $TableParam{Padding}              = 1;
    $TableParam{PaddingTop}           = 3;
    $TableParam{PaddingBottom}        = 3;

    # output table
    PAGE:
    for ( $Param{Page}->{PageCount} .. $Param{Page}->{MaxPages} ) {

        # output table (or a fragment of it)
        %TableParam = $Self->{PDFObject}->Table(%TableParam);

        # stop output or output next page
        last PAGE if $TableParam{State};

        $Self->{PDFObject}->PageNew(
            %{ $Param{Page} },
            FooterRight => $Param{Page}->{PageText} . ' ' . $Param{Page}->{PageCount}
        );
        $Param{Page}->{PageCount}++;
    }

    return 1;
}

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

    # check needed stuff
    for my $Argument (qw(Page SLAList)) {
        if ( !defined $Param{$Argument} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "Need $Argument!"
            );
            return;
        }
    }

    my %TableParam;
    my $Row = 0;

    # generate table data
    for my $SLAID ( sort keys %{ $Param{SLAList} } ) {
        $TableParam{CellData}[$Row][0]{Content}
            = $Self->{LayoutObject}->{LanguageObject}->Get('SLA') . ':';
        $TableParam{CellData}[$Row][0]{Font}    = 'ProportionalBold';
        $TableParam{CellData}[$Row][1]{Content} = $Param{SLAList}->{$SLAID};
        $Row++;
    }
    $TableParam{ColumnData}[0]{Width} = 80;
    $TableParam{ColumnData}[1]{Width} = 431;

    # set new position
    $Self->{PDFObject}->PositionSet(
        Move => 'relativ',
        Y    => -15,
    );

    # output headline
    $Self->{PDFObject}->Text(
        Text     => $Self->{LayoutObject}->{LanguageObject}->Get('Associated SLAs'),
        Height   => 7,
        Type     => 'Cut',
        Font     => 'ProportionalBoldItalic',
        FontSize => 7,
        Color    => '#666666',
    );

    # set new position
    $Self->{PDFObject}->PositionSet(
        Move => 'relativ',
        Y    => -4,
    );

    # table params
    $TableParam{Type}            = 'Cut';
    $TableParam{Border}          = 0;
    $TableParam{FontSize}        = 6;
    $TableParam{BackgroundColor} = '#DDDDDD';
    $TableParam{Padding}         = 1;
    $TableParam{PaddingTop}      = 3;
    $TableParam{PaddingBottom}   = 3;

    # output table
    PAGE:
    for ( $Param{Page}->{PageCount} .. $Param{Page}->{MaxPages} ) {

        # output table (or a fragment of it)
        %TableParam = $Self->{PDFObject}->Table(%TableParam);

        # stop output or output next page
        last PAGE if $TableParam{State};

        $Self->{PDFObject}->PageNew(
            %{ $Param{Page} },
            FooterRight => $Param{Page}->{PageText} . ' ' . $Param{Page}->{PageCount}
        );
        $Param{Page}->{PageCount}++;
    }

    return 1;
}

1;

# --
# Kernel/Modules/AgentITSMServiceZoom.pm - the OTRS::ITSM Service zoom module
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: AgentITSMServiceZoom.pm,v 1.6 2009/10/06 20:29:33 ub Exp $
# --
# 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::AgentITSMServiceZoom;

use strict;
use warnings;

use Kernel::System::LinkObject;
use Kernel::System::Service;
use Kernel::System::SLA;

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

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

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

    # check needed objects
    for my $Object (qw(ConfigObject ParamObject DBObject LayoutObject LogObject)) {
        if ( !$Self->{$Object} ) {
            $Self->{LayoutObject}->FatalError( Message => "Got no $Object!" );
        }
    }
    $Self->{LinkObject}    = Kernel::System::LinkObject->new(%Param);
    $Self->{ServiceObject} = Kernel::System::Service->new(%Param);
    $Self->{SLAObject}     = Kernel::System::SLA->new(%Param);

    return $Self;
}

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

    # get params
    my $ServiceID = $Self->{ParamObject}->GetParam( Param => 'ServiceID' );

    # check needed stuff
    if ( !$ServiceID ) {
        return $Self->{LayoutObject}->ErrorScreen(
            Message => 'No ServiceID is given!',
            Comment => 'Please contact the admin.',
        );
    }

    # get service
    my %Service = $Self->{ServiceObject}->ServiceGet(
        ServiceID => $ServiceID,
        UserID    => $Self->{UserID},
    );
    if ( !$Service{ServiceID} ) {
        return $Self->{LayoutObject}->ErrorScreen(
            Message => "ServiceID $ServiceID not found in database!",
            Comment => 'Please contact the admin.',
        );
    }

    # run config item menu modules
    if ( ref $Self->{ConfigObject}->Get('ITSMService::Frontend::MenuModule') eq 'HASH' ) {
        my %Menus   = %{ $Self->{ConfigObject}->Get('ITSMService::Frontend::MenuModule') };
        my $Counter = 0;
        for my $Menu ( sort keys %Menus ) {

            # load module
            if ( $Self->{MainObject}->Require( $Menus{$Menu}->{Module} ) ) {
                my $Object = $Menus{$Menu}->{Module}->new(
                    %{$Self},
                    ServiceID => $Self->{ServiceID},
                );

                # run module
                $Counter = $Object->Run(
                    %Param,
                    Service => \%Service,
                    Counter => $Counter,
                    Config  => $Menus{$Menu},
                );
            }
            else {
                return $Self->{LayoutObject}->FatalError();
            }
        }
    }

    my $OutputHorizontalRuler = 0;

    # get sla list
    my %SLAList = $Self->{SLAObject}->SLAList(
        ServiceID => $ServiceID,
        UserID    => $Self->{UserID},
    );
    if (%SLAList) {
        $OutputHorizontalRuler = 1;

        # output row
        $Self->{LayoutObject}->Block(
            Name => 'SLA',
        );

        my $CssClass = '';
        for my $SLAID ( sort { $SLAList{$a} cmp $SLAList{$b} } keys %SLAList ) {

            # set output object
            $CssClass = $CssClass eq 'searchpassive' ? 'searchactive' : 'searchpassive';

            # get service data
            my %SLA = $Self->{SLAObject}->SLAGet(
                SLAID  => $SLAID,
                UserID => $Self->{UserID},
            );

            # output row
            $Self->{LayoutObject}->Block(
                Name => 'SLARow',
                Data => {
                    %SLA,
                    CssClass => $CssClass,
                },
            );
        }
    }

    # get linked objects
    my $LinkListWithData = $Self->{LinkObject}->LinkListWithData(
        Object => 'Service',
        Key    => $ServiceID,
        State  => 'Valid',
        UserID => $Self->{UserID},
    );

    # get link table view mode
    my $LinkTableViewMode = $Self->{ConfigObject}->Get('LinkObject::ViewMode');

    # create the link table
    my $LinkTableStrg = $Self->{LayoutObject}->LinkObjectTableCreate(
        LinkListWithData => $LinkListWithData,
        ViewMode         => $LinkTableViewMode,
    );

    # output the link table
    if ($LinkTableStrg) {
        $Self->{LayoutObject}->Block(
            Name => 'LinkTable' . $LinkTableViewMode,
            Data => {
                LinkTableStrg => $LinkTableStrg,
            },
        );

        $OutputHorizontalRuler = 1;
    }

    # output horizontal ruler
    if ($OutputHorizontalRuler) {
        $Self->{LayoutObject}->Block(
            Name => 'HorizontalRuler',
        );
    }

    # set incident signal
    my %InciSignals = (
        operational => 'greenled',
        warning     => 'yellowled',
        incident    => 'redled',
    );

    # get create user data
    my %CreateUser = $Self->{UserObject}->GetUserData(
        UserID => $Service{CreateBy},
        Cached => 1,
    );
    for my $Postfix (qw(UserLogin UserFirstname UserLastname)) {
        $Service{ 'Create' . $Postfix } = $CreateUser{$Postfix};
    }

    # get change user data
    my %ChangeUser = $Self->{UserObject}->GetUserData(
        UserID => $Service{ChangeBy},
        Cached => 1,
    );
    for my $Postfix (qw(UserLogin UserFirstname UserLastname)) {
        $Service{ 'Change' . $Postfix } = $ChangeUser{$Postfix};
    }

    # store last screen
    $Self->{SessionObject}->UpdateSessionID(
        SessionID => $Self->{SessionID},
        Key       => 'LastScreenView',
        Value     => $Self->{RequestedURL},
    );

    # output header
    my $Output = $Self->{LayoutObject}->Header();
    $Output .= $Self->{LayoutObject}->NavigationBar();

    # generate output
    $Output .= $Self->{LayoutObject}->Output(
        TemplateFile => 'AgentITSMServiceZoom',
        Data         => {
            %Param,
            %Service,
            CurInciSignal => $InciSignals{ $Service{CurInciStateType} },
        },
    );
    $Output .= $Self->{LayoutObject}->Footer();

    return $Output;
}

1;

IyAtLQojIEtlcm5lbC9Nb2R1bGVzL0FnZW50SVRTTVNMQS5wbSAtIHRoZSBPVFJTOjpJVFNNIFNMQSBtb2R1bGUKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IEFnZW50SVRTTVNMQS5wbSx2IDEuNCAyMDA5LzA1LzE4IDA5OjQ4OjM1IG1oIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TW9kdWxlczo6QWdlbnRJVFNNU0xBOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIEtlcm5lbDo6U3lzdGVtOjpTTEE7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjQgJCkgWzFdOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7JVBhcmFtfTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBvYmplY3RzCiAgICBmb3IgbXkgJE9iamVjdCAocXcoQ29uZmlnT2JqZWN0IFBhcmFtT2JqZWN0IERCT2JqZWN0IExheW91dE9iamVjdCBMb2dPYmplY3QpKSB7CiAgICAgICAgaWYgKCAhJFNlbGYtPnskT2JqZWN0fSApIHsKICAgICAgICAgICAgJFNlbGYtPntMYXlvdXRPYmplY3R9LT5GYXRhbEVycm9yKCBNZXNzYWdlID0+ICJHb3Qgbm8gJE9iamVjdCEiICk7CiAgICAgICAgfQogICAgfQogICAgJFNlbGYtPntTTEFPYmplY3R9ID0gS2VybmVsOjpTeXN0ZW06OlNMQS0+bmV3KCVQYXJhbSk7CgogICAgcmV0dXJuICRTZWxmOwp9CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBvdXRwdXQgb3ZlcnZpZXcKICAgICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+QmxvY2soCiAgICAgICAgTmFtZSA9PiAnT3ZlcnZpZXcnLAogICAgICAgIERhdGEgPT4geyVQYXJhbX0sCiAgICApOwoKICAgICMgZ2V0IHNsYSBsaXN0CiAgICBteSAlU0xBTGlzdCA9ICRTZWxmLT57U0xBT2JqZWN0fS0+U0xBTGlzdCgKICAgICAgICBVc2VySUQgPT4gJFNlbGYtPntVc2VySUR9LAogICAgKTsKCiAgICBteSAkQ3NzQ2xhc3MgPSAnJzsKICAgIGZvciBteSAkU0xBSUQgKCBzb3J0IHsgJFNMQUxpc3R7JGF9IGNtcCAkU0xBTGlzdHskYn0gfSBrZXlzICVTTEFMaXN0ICkgewoKICAgICAgICAjIHNldCBvdXRwdXQgb2JqZWN0CiAgICAgICAgJENzc0NsYXNzID0gJENzc0NsYXNzIGVxICdzZWFyY2hwYXNzaXZlJyA/ICdzZWFyY2hhY3RpdmUnIDogJ3NlYXJjaHBhc3NpdmUnOwoKICAgICAgICAjIGdldCBzbGEgZGF0YQogICAgICAgIG15ICVTTEEgPSAkU2VsZi0+e1NMQU9iamVjdH0tPlNMQUdldCgKICAgICAgICAgICAgU0xBSUQgID0+ICRTTEFJRCwKICAgICAgICAgICAgVXNlcklEID0+ICRTZWxmLT57VXNlcklEfSwKICAgICAgICApOwoKICAgICAgICAjIG91dHB1dCBvdmVydmlldyByb3cKICAgICAgICAkU2VsZi0+e0xheW91dE9iamVjdH0tPkJsb2NrKAogICAgICAgICAgICBOYW1lID0+ICdPdmVydmlld1JvdycsCiAgICAgICAgICAgIERhdGEgPT4gewogICAgICAgICAgICAgICAgJVNMQSwKICAgICAgICAgICAgICAgIENzc0NsYXNzID0+ICRDc3NDbGFzcywKICAgICAgICAgICAgfSwKICAgICAgICApOwogICAgfQoKICAgICMgaW52ZXN0aWdhdGUgcmVmcmVzaAogICAgbXkgJFJlZnJlc2ggPSAkU2VsZi0+e1VzZXJSZWZyZXNoVGltZX0gPyA2MCAqICRTZWxmLT57VXNlclJlZnJlc2hUaW1lfSA6IHVuZGVmOwoKICAgICMgb3V0cHV0IGhlYWRlcgogICAgbXkgJE91dHB1dCA9ICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+SGVhZGVyKAogICAgICAgIFRpdGxlICAgPT4gJ092ZXJ2aWV3JywKICAgICAgICBSZWZyZXNoID0+ICRSZWZyZXNoLAogICAgKTsKICAgICRPdXRwdXQgLj0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5OYXZpZ2F0aW9uQmFyKCk7CgogICAgIyBnZW5lcmF0ZSBvdXRwdXQKICAgICRPdXRwdXQgLj0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5PdXRwdXQoCiAgICAgICAgVGVtcGxhdGVGaWxlID0+ICdBZ2VudElUU01TTEEnLAogICAgICAgIERhdGEgICAgICAgICA9PiBcJVBhcmFtLAogICAgKTsKICAgICRPdXRwdXQgLj0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5Gb290ZXIoKTsKCiAgICByZXR1cm4gJE91dHB1dDsKfQoKMTsK
# --
# Kernel/Modules/AgentITSMSLAPrint.pm - print layout for itsm sla agent interface
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: AgentITSMSLAPrint.pm,v 1.3 2009/05/18 09:48:35 mh Exp $
# --
# 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::AgentITSMSLAPrint;

use strict;
use warnings;

use Kernel::System::PDF;
use Kernel::System::SLA;

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

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

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

    # check needed objects
    for my $Object (qw(ConfigObject ParamObject DBObject LayoutObject LogObject)) {
        if ( !$Self->{$Object} ) {
            $Self->{LayoutObject}->FatalError( Message => "Got no $Object!" );
        }
    }
    $Self->{PDFObject} = Kernel::System::PDF->new(%Param);
    $Self->{SLAObject} = Kernel::System::SLA->new(%Param);

    return $Self;
}

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

    # get params
    my $SLAID = $Self->{ParamObject}->GetParam( Param => "SLAID" );

    # check needed stuff
    if ( !$SLAID ) {
        return $Self->{LayoutObject}->ErrorScreen(
            Message => "No SLAID is given!",
            Comment => 'Please contact the admin.',
        );
    }

    # get sla
    my %SLA = $Self->{SLAObject}->SLAGet(
        SLAID  => $SLAID,
        UserID => $Self->{UserID},
    );
    if ( !$SLA{SLAID} ) {
        return $Self->{LayoutObject}->ErrorScreen(
            Message => "SLAID $SLAID not found in database!",
            Comment => 'Please contact the admin.',
        );
    }

    # get calendar name
    if ( $SLA{Calendar} ) {
        $SLA{CalendarName} = "Calendar $SLA{Calendar} - "
            . $Self->{ConfigObject}->Get( "TimeZone::Calendar" . $SLA{Calendar} . "Name" );
    }
    else {
        $SLA{CalendarName} = 'Calendar Default';
    }

    # get user data (create by)
    my %CreateBy = $Self->{UserObject}->GetUserData(
        UserID => $SLA{CreateBy},
        Cached => 1,
    );

    # get user data (change by)
    my %ChangeBy = $Self->{UserObject}->GetUserData(
        UserID => $SLA{ChangeBy},
        Cached => 1,
    );

    # generate pdf output
    if ( $Self->{PDFObject} ) {
        my %Page;
        my $Url = ' ';
        if ( $ENV{REQUEST_URI} ) {
            $Url
                = $Self->{ConfigObject}->Get('HttpType') . '://'
                . $Self->{ConfigObject}->Get('FQDN')
                . $ENV{REQUEST_URI};
        }

        # get maximum number of pages
        $Page{MaxPages} = $Self->{ConfigObject}->Get('PDF::MaxPages');
        if ( !$Page{MaxPages} || $Page{MaxPages} < 1 || $Page{MaxPages} > 1000 ) {
            $Page{MaxPages} = 100;
        }
        $Page{MarginTop}    = 30;
        $Page{MarginRight}  = 40;
        $Page{MarginBottom} = 40;
        $Page{MarginLeft}   = 40;
        $Page{HeaderRight}  = $Self->{LayoutObject}->{LanguageObject}->Get('SLA');
        $Page{HeadlineLeft} = $SLA{Name};
        $Page{HeadlineRight}
            = $Self->{LayoutObject}->{LanguageObject}->Get('printed by') . ' '
            . $Self->{UserFirstname} . ' '
            . $Self->{UserLastname} . ' ('
            . $Self->{UserEmail} . ') '
            . $Self->{LayoutObject}->Output( Template => '$Env{"Time"}' );
        $Page{FooterLeft} = $Url;
        $Page{PageText}   = $Self->{LayoutObject}->{LanguageObject}->Get('Page');
        $Page{PageCount}  = 1;

        # create new pdf document
        $Self->{PDFObject}->DocumentNew(
            Title  => $Self->{ConfigObject}->Get('Product') . ': ' . $SLA{Name},
            Encode => $Self->{LayoutObject}->{UserCharset},
        );

        # create first pdf page
        $Self->{PDFObject}->PageNew(
            %Page,
            FooterRight => $Page{PageText} . ' ' . $Page{PageCount},
        );
        $Page{PageCount}++;

        # output general infos
        $Self->_PDFOutputGeneralInfos(
            Page     => \%Page,
            SLA      => \%SLA,
            CreateBy => \%CreateBy,
            ChangeBy => \%ChangeBy,
        );

        # output detailed infos
        $Self->_PDFOutputDetailedInfos(
            Page => \%Page,
            SLA  => \%SLA,
        );

        # create file name
        my $Filename = $Self->{MainObject}->FilenameCleanUp(
            Filename => $SLA{Name},
            Type     => 'Attachment',
        );
        my ( $s, $m, $h, $D, $M, $Y ) = $Self->{TimeObject}->SystemTime2Date(
            SystemTime => $Self->{TimeObject}->SystemTime(),
        );
        $M = sprintf( "%02d", $M );
        $D = sprintf( "%02d", $D );
        $h = sprintf( "%02d", $h );
        $m = sprintf( "%02d", $m );

        # return the pdf document
        return $Self->{LayoutObject}->Attachment(
            Filename    => 'sla_' . $Filename . "_$Y-$M-$D\_$h-$m.pdf",
            ContentType => 'application/pdf',
            Content     => $Self->{PDFObject}->DocumentOutput(),
            Type        => 'attachment',
        );
    }

    # generate html output
    else {

        # output header
        my $Output = $Self->{LayoutObject}->PrintHeader( Value => $SLA{Name} );

        # generate output
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AgentITSMSLAPrint',
            Data         => {
                CreateByUserLogin     => $CreateBy{UserLogin},
                CreateByUserFirstname => $CreateBy{UserFirstname},
                CreateByUserLastname  => $CreateBy{UserLastname},
                ChangeByUserLogin     => $ChangeBy{UserLogin},
                ChangeByUserFirstname => $ChangeBy{UserFirstname},
                ChangeByUserLastname  => $ChangeBy{UserLastname},
                %SLA,
            },
        );

        # add footer
        $Output .= $Self->{LayoutObject}->PrintFooter();

        # return output
        return $Output;
    }
}

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

    # check needed stuff
    for my $Argument (qw(Page SLA CreateBy ChangeBy)) {
        if ( !defined $Param{$Argument} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "Need $Argument!"
            );
            return;
        }
    }

    # create left table
    my $TableLeft = [
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('SLA') . ':',
            Value => $Param{SLA}->{Name},
        },
    ];

    # create right table
    my $TableRight = [
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Created') . ':',
            Value => $Self->{LayoutObject}->Output(
                Template => '$TimeLong{"$Data{"CreateTime"}"}',
                Data     => \%{ $Param{SLA} },
            ),
        },
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Created by') . ':',
            Value => $Param{CreateBy}->{UserLogin} . ' ('
                . $Param{CreateBy}->{UserFirstname} . ' '
                . $Param{CreateBy}->{UserLastname} . ')',
        },
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Last changed') . ':',
            Value => $Self->{LayoutObject}->Output(
                Template => '$TimeLong{"$Data{"ChangeTime"}"}',
                Data     => \%{ $Param{SLA} },
            ),
        },
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Last changed by') . ':',
            Value => $Param{ChangeBy}->{UserLogin} . ' ('
                . $Param{ChangeBy}->{UserFirstname} . ' '
                . $Param{ChangeBy}->{UserLastname} . ')',
        },
    ];

    my $Rows = @{$TableLeft};
    if ( @{$TableRight} > $Rows ) {
        $Rows = @{$TableRight};
    }

    my %TableParam;
    for my $Row ( 1 .. $Rows ) {
        $Row--;
        $TableParam{CellData}[$Row][0]{Content}         = $TableLeft->[$Row]->{Key};
        $TableParam{CellData}[$Row][0]{Font}            = 'ProportionalBold';
        $TableParam{CellData}[$Row][1]{Content}         = $TableLeft->[$Row]->{Value};
        $TableParam{CellData}[$Row][2]{Content}         = ' ';
        $TableParam{CellData}[$Row][2]{BackgroundColor} = '#FFFFFF';
        $TableParam{CellData}[$Row][3]{Content}         = $TableRight->[$Row]->{Key};
        $TableParam{CellData}[$Row][3]{Font}            = 'ProportionalBold';
        $TableParam{CellData}[$Row][4]{Content}         = $TableRight->[$Row]->{Value};
    }
    $TableParam{ColumnData}[0]{Width} = 50;
    $TableParam{ColumnData}[1]{Width} = 200.5;
    $TableParam{ColumnData}[2]{Width} = 4;
    $TableParam{ColumnData}[3]{Width} = 80;
    $TableParam{ColumnData}[4]{Width} = 170.5;
    $TableParam{Type}                 = 'Cut';
    $TableParam{Border}               = 0;
    $TableParam{FontSize}             = 6;
    $TableParam{BackgroundColorEven}  = '#AAAAAA';
    $TableParam{BackgroundColorOdd}   = '#DDDDDD';
    $TableParam{Padding}              = 1;
    $TableParam{PaddingTop}           = 3;
    $TableParam{PaddingBottom}        = 3;

    # output table
    PAGE:
    for ( $Param{Page}->{PageCount} .. $Param{Page}->{MaxPages} ) {

        # output table (or a fragment of it)
        %TableParam = $Self->{PDFObject}->Table(%TableParam);

        # stop output or output next page
        last PAGE if $TableParam{State};

        $Self->{PDFObject}->PageNew(
            %{ $Param{Page} },
            FooterRight => $Param{Page}->{PageText} . ' ' . $Param{Page}->{PageCount}
        );
        $Param{Page}->{PageCount}++;
    }
    return 1;
}

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

    # check needed stuff
    for my $Argument (qw(Page SLA)) {
        if ( !defined $Param{$Argument} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "Need $Argument!"
            );
            return;
        }
    }

    # set new position
    $Self->{PDFObject}->PositionSet(
        Move => 'relativ',
        Y    => -15,
    );

    # output headline
    $Self->{PDFObject}->Text(
        Text     => $Self->{LayoutObject}->{LanguageObject}->Get('SLA'),
        Height   => 7,
        Type     => 'Cut',
        Font     => 'ProportionalBoldItalic',
        FontSize => 7,
        Color    => '#666666',
    );

    # set new position
    $Self->{PDFObject}->PositionSet(
        Move => 'relativ',
        Y    => -4,
    );

    # create table
    my $Table = [
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('SLA') . ':',
            Value => $Param{SLA}->{Name},
        },
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Type') . ':',
            Value => $Self->{LayoutObject}->{LanguageObject}->Get( $Param{SLA}->{Type} ),
        },
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Calendar') . ':',
            Value => $Self->{LayoutObject}->{LanguageObject}->Get( $Param{SLA}->{CalendarName} ),
        },
        {
            Key => $Self->{LayoutObject}->{LanguageObject}->Get('First Response Time') . ':',
            Value =>
                $Self->{LayoutObject}->{LanguageObject}->Get( $Param{SLA}->{FirstResponseTime} )
                . ' '
                . $Self->{LayoutObject}->{LanguageObject}->Get('minutes'),
        },
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Update Time') . ':',
            Value => $Self->{LayoutObject}->{LanguageObject}->Get( $Param{SLA}->{UpdateTime} ) . ' '
                . $Self->{LayoutObject}->{LanguageObject}->Get('minutes'),
        },
        {
            Key   => $Self->{LayoutObject}->{LanguageObject}->Get('Solution Time') . ':',
            Value => $Self->{LayoutObject}->{LanguageObject}->Get( $Param{SLA}->{SolutionTime} )
                . ' '
                . $Self->{LayoutObject}->{LanguageObject}->Get('minutes'),
        },
        {
            Key => $Self->{LayoutObject}->{LanguageObject}->Get('Minimum Time Between Incidents')
                . ':',
            Value => $Self->{LayoutObject}->{LanguageObject}->Get(
                $Param{SLA}->{MinTimeBetweenIncidents},
                )
                . ' '
                . $Self->{LayoutObject}->{LanguageObject}->Get('minutes'),
        },
    ];
    my %TableParam;
    my $Rows = @{$Table};
    for my $Row ( 1 .. $Rows ) {
        $Row--;
        $TableParam{CellData}[$Row][0]{Content} = $Table->[$Row]->{Key};
        $TableParam{CellData}[$Row][0]{Font}    = 'ProportionalBold';
        $TableParam{CellData}[$Row][1]{Content} = $Table->[$Row]->{Value};
    }
    $TableParam{ColumnData}[0]{Width} = 120;
    $TableParam{ColumnData}[1]{Width} = 391;
    $TableParam{Type}                 = 'Cut';
    $TableParam{Border}               = 0;
    $TableParam{FontSize}             = 6;
    $TableParam{BackgroundColor}      = '#DDDDDD';
    $TableParam{Padding}              = 1;
    $TableParam{PaddingTop}           = 3;
    $TableParam{PaddingBottom}        = 3;

    # output table
    PAGE:
    for ( $Param{Page}->{PageCount} .. $Param{Page}->{MaxPages} ) {

        # output table (or a fragment of it)
        %TableParam = $Self->{PDFObject}->Table(%TableParam);

        # stop output or output next page
        last PAGE if $TableParam{State};

        $Self->{PDFObject}->PageNew(
            %{ $Param{Page} },
            FooterRight => $Param{Page}->{PageText} . ' ' . $Param{Page}->{PageCount}
        );
        $Param{Page}->{PageCount}++;
    }
    return 1;
}

1;

IyAtLQojIEtlcm5lbC9Nb2R1bGVzL0FnZW50SVRTTVNMQVpvb20ucG0gLSB0aGUgT1RSUzo6SVRTTSBTTEEgem9vbSBtb2R1bGUKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IEFnZW50SVRTTVNMQVpvb20ucG0sdiAxLjYgMjAwOS8wNS8xOCAwOTo0ODozNSBtaCBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok1vZHVsZXM6OkFnZW50SVRTTVNMQVpvb207Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgS2VybmVsOjpTeXN0ZW06OlNlcnZpY2U7CnVzZSBLZXJuZWw6OlN5c3RlbTo6U0xBOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS42ICQpIFsxXTsKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0geyVQYXJhbX07CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgIyBjaGVjayBuZWVkZWQgb2JqZWN0cwogICAgZm9yIG15ICRPYmplY3QgKHF3KENvbmZpZ09iamVjdCBQYXJhbU9iamVjdCBEQk9iamVjdCBMYXlvdXRPYmplY3QgTG9nT2JqZWN0KSkgewogICAgICAgIGlmICggISRTZWxmLT57JE9iamVjdH0gKSB7CiAgICAgICAgICAgICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+RmF0YWxFcnJvciggTWVzc2FnZSA9PiAiR290IG5vICRPYmplY3QhIiApOwogICAgICAgIH0KICAgIH0KICAgICRTZWxmLT57U2VydmljZU9iamVjdH0gPSBLZXJuZWw6OlN5c3RlbTo6U2VydmljZS0+bmV3KCVQYXJhbSk7CiAgICAkU2VsZi0+e1NMQU9iamVjdH0gICAgID0gS2VybmVsOjpTeXN0ZW06OlNMQS0+bmV3KCVQYXJhbSk7CgogICAgcmV0dXJuICRTZWxmOwp9CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBnZXQgcGFyYW1zCiAgICBteSAkU0xBSUQgPSAkU2VsZi0+e1BhcmFtT2JqZWN0fS0+R2V0UGFyYW0oIFBhcmFtID0+ICJTTEFJRCIgKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgaWYgKCAhJFNMQUlEICkgewogICAgICAgIHJldHVybiAkU2VsZi0+e0xheW91dE9iamVjdH0tPkVycm9yU2NyZWVuKAogICAgICAgICAgICBNZXNzYWdlID0+ICJObyBTTEFJRCBpcyBnaXZlbiEiLAogICAgICAgICAgICBDb21tZW50ID0+ICdQbGVhc2UgY29udGFjdCB0aGUgYWRtaW4uJywKICAgICAgICApOwogICAgfQoKICAgICMgZ2V0IHNsYQogICAgbXkgJVNMQSA9ICRTZWxmLT57U0xBT2JqZWN0fS0+U0xBR2V0KAogICAgICAgIFNMQUlEICA9PiAkU0xBSUQsCiAgICAgICAgVXNlcklEID0+ICRTZWxmLT57VXNlcklEfSwKICAgICk7CiAgICBpZiAoICEkU0xBe1NMQUlEfSApIHsKICAgICAgICByZXR1cm4gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5FcnJvclNjcmVlbigKICAgICAgICAgICAgTWVzc2FnZSA9PiAiU0xBSUQgJFNMQUlEIG5vdCBmb3VuZCBpbiBkYXRhYmFzZSEiLAogICAgICAgICAgICBDb21tZW50ID0+ICdQbGVhc2UgY29udGFjdCB0aGUgYWRtaW4uJywKICAgICAgICApOwogICAgfQoKICAgICMgZ2V0IGNhbGVuZGFyIG5hbWUKICAgIGlmICggJFNMQXtDYWxlbmRhcn0gKSB7CiAgICAgICAgJFNMQXtDYWxlbmRhck5hbWV9ID0gIkNhbGVuZGFyICRTTEF7Q2FsZW5kYXJ9IC0gIgogICAgICAgICAgICAuICRTZWxmLT57Q29uZmlnT2JqZWN0fS0+R2V0KCAiVGltZVpvbmU6OkNhbGVuZGFyIiAuICRTTEF7Q2FsZW5kYXJ9IC4gIk5hbWUiICk7CiAgICB9CiAgICBlbHNlIHsKICAgICAgICAkU0xBe0NhbGVuZGFyTmFtZX0gPSAnQ2FsZW5kYXIgRGVmYXVsdCc7CiAgICB9CgogICAgIyBydW4gY29uZmlnIGl0ZW0gbWVudSBtb2R1bGVzCiAgICBpZiAoIHJlZiAkU2VsZi0+e0NvbmZpZ09iamVjdH0tPkdldCgnSVRTTVNMQTo6RnJvbnRlbmQ6Ok1lbnVNb2R1bGUnKSBlcSAnSEFTSCcgKSB7CiAgICAgICAgbXkgJU1lbnVzICAgPSAleyAkU2VsZi0+e0NvbmZpZ09iamVjdH0tPkdldCgnSVRTTVNMQTo6RnJvbnRlbmQ6Ok1lbnVNb2R1bGUnKSB9OwogICAgICAgIG15ICRDb3VudGVyID0gMDsKICAgICAgICBmb3IgbXkgJE1lbnUgKCBzb3J0IGtleXMgJU1lbnVzICkgewoKICAgICAgICAgICAgIyBsb2FkIG1vZHVsZQogICAgICAgICAgICBpZiAoICRTZWxmLT57TWFpbk9iamVjdH0tPlJlcXVpcmUoICRNZW51c3skTWVudX0tPntNb2R1bGV9ICkgKSB7CiAgICAgICAgICAgICAgICBteSAkT2JqZWN0ID0gJE1lbnVzeyRNZW51fS0+e01vZHVsZX0tPm5ldygKICAgICAgICAgICAgICAgICAgICAleyRTZWxmfSwKICAgICAgICAgICAgICAgICAgICBTTEFJRCA9PiAkU2VsZi0+e1NMQUlEfSwKICAgICAgICAgICAgICAgICk7CgogICAgICAgICAgICAgICAgIyBydW4gbW9kdWxlCiAgICAgICAgICAgICAgICAkQ291bnRlciA9ICRPYmplY3QtPlJ1bigKICAgICAgICAgICAgICAgICAgICAlUGFyYW0sCiAgICAgICAgICAgICAgICAgICAgU0xBICAgICA9PiBcJVNMQSwKICAgICAgICAgICAgICAgICAgICBDb3VudGVyID0+ICRDb3VudGVyLAogICAgICAgICAgICAgICAgICAgIENvbmZpZyAgPT4gJE1lbnVzeyRNZW51fSwKICAgICAgICAgICAgICAgICk7CiAgICAgICAgICAgIH0KICAgICAgICAgICAgZWxzZSB7CiAgICAgICAgICAgICAgICByZXR1cm4gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5GYXRhbEVycm9yKCk7CiAgICAgICAgICAgIH0KICAgICAgICB9CiAgICB9CgogICAgaWYgKCAkU0xBe1NlcnZpY2VJRHN9ICYmIHJlZiAkU0xBe1NlcnZpY2VJRHN9IGVxICdBUlJBWScgJiYgQHsgJFNMQXtTZXJ2aWNlSURzfSB9ICkgewoKICAgICAgICAjIG91dHB1dCByb3cKICAgICAgICAkU2VsZi0+e0xheW91dE9iamVjdH0tPkJsb2NrKAogICAgICAgICAgICBOYW1lID0+ICdTZXJ2aWNlJywKICAgICAgICApOwoKICAgICAgICAjIGNyZWF0ZSBzZXJ2aWNlIGxpc3QKICAgICAgICBteSAlU2VydmljZUxpc3Q7CiAgICAgICAgZm9yIG15ICRTZXJ2aWNlSUQgKCBAeyAkU0xBe1NlcnZpY2VJRHN9IH0gKSB7CgogICAgICAgICAgICAjIGdldCBzZXJ2aWNlIGRhdGEKICAgICAgICAgICAgbXkgJVNlcnZpY2UgPSAkU2VsZi0+e1NlcnZpY2VPYmplY3R9LT5TZXJ2aWNlR2V0KAogICAgICAgICAgICAgICAgU2VydmljZUlEID0+ICRTZXJ2aWNlSUQsCiAgICAgICAgICAgICAgICBVc2VySUQgICAgPT4gJFNlbGYtPntVc2VySUR9LAogICAgICAgICAgICApOwoKICAgICAgICAgICAgIyBhZGQgc2VydmljZSB0byBoYXNoCiAgICAgICAgICAgICRTZXJ2aWNlTGlzdHskU2VydmljZUlEfSA9IFwlU2VydmljZTsKICAgICAgICB9CgogICAgICAgICMgc2V0IGluY2lkZW50IHNpZ25hbAogICAgICAgIG15ICVJbmNpU2lnbmFscyA9ICgKICAgICAgICAgICAgb3BlcmF0aW9uYWwgPT4gJ2dyZWVubGVkJywKICAgICAgICAgICAgd2FybmluZyAgICAgPT4gJ3llbGxvd2xlZCcsCiAgICAgICAgICAgIGluY2lkZW50ICAgID0+ICdyZWRsZWQnLAogICAgICAgICk7CgogICAgICAgIG15ICRDc3NDbGFzcyA9ICcnOwogICAgICAgIGZvciBteSAkU2VydmljZUlEICgKICAgICAgICAgICAgc29ydCB7ICRTZXJ2aWNlTGlzdHskYX0tPntOYW1lfSBjbXAgJFNlcnZpY2VMaXN0eyRifS0+e05hbWV9IH0KICAgICAgICAgICAga2V5cyAlU2VydmljZUxpc3QKICAgICAgICAgICAgKQogICAgICAgIHsKCiAgICAgICAgICAgICMgc2V0IG91dHB1dCBvYmplY3QKICAgICAgICAgICAgJENzc0NsYXNzID0gJENzc0NsYXNzIGVxICdzZWFyY2hwYXNzaXZlJyA/ICdzZWFyY2hhY3RpdmUnIDogJ3NlYXJjaHBhc3NpdmUnOwoKICAgICAgICAgICAgIyBvdXRwdXQgcm93CiAgICAgICAgICAgICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+QmxvY2soCiAgICAgICAgICAgICAgICBOYW1lID0+ICdTZXJ2aWNlUm93JywKICAgICAgICAgICAgICAgIERhdGEgPT4gewogICAgICAgICAgICAgICAgICAgICV7ICRTZXJ2aWNlTGlzdHskU2VydmljZUlEfSB9LAogICAgICAgICAgICAgICAgICAgIEN1ckluY2lTaWduYWwgPT4gJEluY2lTaWduYWxzeyAkU2VydmljZUxpc3R7JFNlcnZpY2VJRH0tPntDdXJJbmNpU3RhdGVUeXBlfSB9LAogICAgICAgICAgICAgICAgICAgIENzc0NsYXNzICAgICAgPT4gJENzc0NsYXNzLAogICAgICAgICAgICAgICAgfSwKICAgICAgICAgICAgKTsKICAgICAgICB9CiAgICB9CgogICAgIyBnZXQgY3JlYXRlIHVzZXIgZGF0YQogICAgbXkgJUNyZWF0ZVVzZXIgPSAkU2VsZi0+e1VzZXJPYmplY3R9LT5HZXRVc2VyRGF0YSgKICAgICAgICBVc2VySUQgPT4gJFNMQXtDcmVhdGVCeX0sCiAgICAgICAgQ2FjaGVkID0+IDEsCiAgICApOwogICAgZm9yIG15ICRQb3N0Zml4IChxdyhVc2VyTG9naW4gVXNlckZpcnN0bmFtZSBVc2VyTGFzdG5hbWUpKSB7CiAgICAgICAgJFNMQXsgJ0NyZWF0ZScgLiAkUG9zdGZpeCB9ID0gJENyZWF0ZVVzZXJ7JFBvc3RmaXh9OwogICAgfQoKICAgICMgZ2V0IGNoYW5nZSB1c2VyIGRhdGEKICAgIG15ICVDaGFuZ2VVc2VyID0gJFNlbGYtPntVc2VyT2JqZWN0fS0+R2V0VXNlckRhdGEoCiAgICAgICAgVXNlcklEID0+ICRTTEF7Q2hhbmdlQnl9LAogICAgICAgIENhY2hlZCA9PiAxLAogICAgKTsKICAgIGZvciBteSAkUG9zdGZpeCAocXcoVXNlckxvZ2luIFVzZXJGaXJzdG5hbWUgVXNlckxhc3RuYW1lKSkgewogICAgICAgICRTTEF7ICdDaGFuZ2UnIC4gJFBvc3RmaXggfSA9ICRDaGFuZ2VVc2VyeyRQb3N0Zml4fTsKICAgIH0KCiAgICAjIG91dHB1dCBoZWFkZXIKICAgIG15ICRPdXRwdXQgPSAkU2VsZi0+e0xheW91dE9iamVjdH0tPkhlYWRlcigpOwogICAgJE91dHB1dCAuPSAkU2VsZi0+e0xheW91dE9iamVjdH0tPk5hdmlnYXRpb25CYXIoKTsKCiAgICAjIGdlbmVyYXRlIG91dHB1dAogICAgJE91dHB1dCAuPSAkU2VsZi0+e0xheW91dE9iamVjdH0tPk91dHB1dCgKICAgICAgICBUZW1wbGF0ZUZpbGUgPT4gJ0FnZW50SVRTTVNMQVpvb20nLAogICAgICAgIERhdGEgICAgICAgICA9PiB7CiAgICAgICAgICAgICVQYXJhbSwKICAgICAgICAgICAgJVNMQSwKICAgICAgICB9LAogICAgKTsKICAgICRPdXRwdXQgLj0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5Gb290ZXIoKTsKCiAgICByZXR1cm4gJE91dHB1dDsKfQoKMTsK
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JVFNNU2VydmljZU1lbnVHZW5lcmljLnBtCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDkgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBJVFNNU2VydmljZU1lbnVHZW5lcmljLnBtLHYgMS4yIDIwMDkvMDUvMTggMDk6NDg6NDYgbWggRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpPdXRwdXQ6OkhUTUw6OklUU01TZXJ2aWNlTWVudUdlbmVyaWM7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjIgJCkgWzFdOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBvYmplY3RzCiAgICBmb3IgbXkgJE9iamVjdCAocXcoQ29uZmlnT2JqZWN0IExvZ09iamVjdCBEQk9iamVjdCBMYXlvdXRPYmplY3QgU2VydmljZU9iamVjdCBVc2VySUQpKSB7CiAgICAgICAgJFNlbGYtPnskT2JqZWN0fSA9ICRQYXJhbXskT2JqZWN0fSB8fCBkaWUgIkdvdCBubyAkT2JqZWN0ISI7CiAgICB9CgogICAgcmV0dXJuICRTZWxmOwp9CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtTZXJ2aWNlfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZyggUHJpb3JpdHkgPT4gJ2Vycm9yJywgTWVzc2FnZSA9PiAnTmVlZCBTZXJ2aWNlIScgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBnZXQgZ3JvdXBzCiAgICBteSAkR3JvdXBzUm8KICAgICAgICA9ICRTZWxmLT57Q29uZmlnT2JqZWN0fS0+R2V0KCdGcm9udGVuZDo6TW9kdWxlJyktPnsgJFBhcmFte0NvbmZpZ30tPntBY3Rpb259IH0tPntHcm91cFJvfQogICAgICAgIHx8IFtdOwogICAgbXkgJEdyb3Vwc1J3CiAgICAgICAgPSAkU2VsZi0+e0NvbmZpZ09iamVjdH0tPkdldCgnRnJvbnRlbmQ6Ok1vZHVsZScpLT57ICRQYXJhbXtDb25maWd9LT57QWN0aW9ufSB9LT57R3JvdXB9CiAgICAgICAgfHwgW107CgogICAgIyBzZXQgYWNjZXNzCiAgICBteSAkQWNjZXNzID0gMTsKCiAgICAjIGNoZWNrIHBlcm1pc3Npb24KICAgIGlmICggJFBhcmFte0NvbmZpZ30tPntBY3Rpb259ICYmICggQHskR3JvdXBzUm99IHx8IEB7JEdyb3Vwc1J3fSApICkgewoKICAgICAgICAjIHNldCBhY2Nlc3MKICAgICAgICAkQWNjZXNzID0gMDsKCiAgICAgICAgIyBmaW5kIHJlYWQgb25seSBncm91cHMKICAgICAgICBST0dST1VQOgogICAgICAgIGZvciBteSAkUm9Hcm91cCAoIEB7JEdyb3Vwc1JvfSApIHsKCiAgICAgICAgICAgIG5leHQgUk9HUk9VUCBpZiAhJFNlbGYtPntMYXlvdXRPYmplY3R9LT57IlVzZXJJc0dyb3VwUm9bJFJvR3JvdXBdIn07CiAgICAgICAgICAgIG5leHQgUk9HUk9VUCBpZiAkU2VsZi0+e0xheW91dE9iamVjdH0tPnsiVXNlcklzR3JvdXBSb1skUm9Hcm91cF0ifSBuZSAnWWVzJzsKCiAgICAgICAgICAgICMgc2V0IGFjY2VzcwogICAgICAgICAgICAkQWNjZXNzID0gMTsKICAgICAgICAgICAgbGFzdCBST0dST1VQOwogICAgICAgIH0KCiAgICAgICAgIyBmaW5kIHJlYWQgd3JpdGUgZ3JvdXBzCiAgICAgICAgUldHUk9VUDoKICAgICAgICBmb3IgbXkgJFJ3R3JvdXAgKCBAeyRHcm91cHNSd30gKSB7CgogICAgICAgICAgICBuZXh0IFJXR1JPVVAgaWYgISRTZWxmLT57TGF5b3V0T2JqZWN0fS0+eyJVc2VySXNHcm91cFskUndHcm91cF0ifTsKICAgICAgICAgICAgbmV4dCBSV0dST1VQIGlmICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+eyJVc2VySXNHcm91cFskUndHcm91cF0ifSBuZSAnWWVzJzsKCiAgICAgICAgICAgICMgc2V0IGFjY2VzcwogICAgICAgICAgICAkQWNjZXNzID0gMTsKICAgICAgICAgICAgbGFzdCBSV0dST1VQOwogICAgICAgIH0KICAgIH0KCiAgICByZXR1cm4gJFBhcmFte0NvdW50ZXJ9IGlmICEkQWNjZXNzOwoKICAgICMgb3V0cHV0IG1lbnUgYmxvY2sKICAgICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+QmxvY2soIE5hbWUgPT4gJ01lbnUnICk7CgogICAgIyBvdXRwdXQgc2VwZXJhdG9yCiAgICBpZiAoICRQYXJhbXtDb3VudGVyfSApIHsKICAgICAgICAkU2VsZi0+e0xheW91dE9iamVjdH0tPkJsb2NrKCBOYW1lID0+ICdNZW51SXRlbVNwbGl0JyApOwogICAgfQoKICAgICMgb3V0cHV0IG1lbnUgaXRlbQogICAgJFNlbGYtPntMYXlvdXRPYmplY3R9LT5CbG9jaygKICAgICAgICBOYW1lID0+ICdNZW51SXRlbScsCiAgICAgICAgRGF0YSA9PiB7CiAgICAgICAgICAgICVQYXJhbSwKICAgICAgICAgICAgJXsgJFBhcmFte1NlcnZpY2V9IH0sCiAgICAgICAgICAgICV7ICRQYXJhbXtDb25maWd9IH0sCiAgICAgICAgfSwKICAgICk7CiAgICAkUGFyYW17Q291bnRlcn0rKzsKCiAgICByZXR1cm4gJFBhcmFte0NvdW50ZXJ9Owp9CgoxOwo=
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JVFNNU2VydmljZU1lbnVMaW5rLnBtCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDkgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBJVFNNU2VydmljZU1lbnVMaW5rLnBtLHYgMS40IDIwMDkvMDUvMTggMDk6NDg6NDYgbWggRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpPdXRwdXQ6OkhUTUw6OklUU01TZXJ2aWNlTWVudUxpbms7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjQgJCkgWzFdOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBvYmplY3RzCiAgICBmb3IgbXkgJE9iamVjdCAoCiAgICAgICAgcXcoQ29uZmlnT2JqZWN0IExvZ09iamVjdCBEQk9iamVjdCBMYXlvdXRPYmplY3QgU2VydmljZU9iamVjdCBMaW5rT2JqZWN0IFVzZXJJRCkKICAgICAgICApCiAgICB7CiAgICAgICAgJFNlbGYtPnskT2JqZWN0fSA9ICRQYXJhbXskT2JqZWN0fSB8fCBkaWUgIkdvdCBubyAkT2JqZWN0ISI7CiAgICB9CgogICAgcmV0dXJuICRTZWxmOwp9CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtTZXJ2aWNlfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZyggUHJpb3JpdHkgPT4gJ2Vycm9yJywgTWVzc2FnZSA9PiAnTmVlZCBTZXJ2aWNlIScgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBnZXQgZ3JvdXBzCiAgICBteSAkR3JvdXBzUncKICAgICAgICA9ICRTZWxmLT57Q29uZmlnT2JqZWN0fS0+R2V0KCdGcm9udGVuZDo6TW9kdWxlJyktPnsgJFBhcmFte0NvbmZpZ30tPntBY3Rpb259IH0tPntHcm91cH0KICAgICAgICB8fCBbXTsKCiAgICAjIHNldCBhY2Nlc3MKICAgIG15ICRBY2Nlc3MgPSAxOwoKICAgICMgY2hlY2sgcGVybWlzc2lvbgogICAgaWYgKCAkUGFyYW17Q29uZmlnfS0+e0FjdGlvbn0gJiYgQHskR3JvdXBzUnd9ICkgewoKICAgICAgICAjIHNldCBhY2Nlc3MKICAgICAgICAkQWNjZXNzID0gMDsKCiAgICAgICAgIyBmaW5kIHJlYWQgd3JpdGUgZ3JvdXBzCiAgICAgICAgUldHUk9VUDoKICAgICAgICBmb3IgbXkgJFJ3R3JvdXAgKCBAeyRHcm91cHNSd30gKSB7CgogICAgICAgICAgICBuZXh0IFJXR1JPVVAgaWYgISRTZWxmLT57TGF5b3V0T2JqZWN0fS0+eyJVc2VySXNHcm91cFskUndHcm91cF0ifTsKICAgICAgICAgICAgbmV4dCBSV0dST1VQIGlmICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+eyJVc2VySXNHcm91cFskUndHcm91cF0ifSBuZSAnWWVzJzsKCiAgICAgICAgICAgICMgc2V0IGFjY2VzcwogICAgICAgICAgICAkQWNjZXNzID0gMTsKICAgICAgICAgICAgbGFzdCBSV0dST1VQOwogICAgICAgIH0KICAgIH0KCiAgICByZXR1cm4gJFBhcmFte0NvdW50ZXJ9IGlmICEkQWNjZXNzOwoKICAgICMgY2hlY2sgaWYgc2VydmljZXMgY2FuIGJlIGxpbmtlZCB3aXRoIG90aGVyIG9iamVjdHMKICAgIG15ICVQb3NzaWJsZU9iamVjdHMgPSAkU2VsZi0+e0xpbmtPYmplY3R9LT5Qb3NzaWJsZU9iamVjdHNMaXN0KAogICAgICAgIE9iamVjdCA9PiAnU2VydmljZScsCiAgICAgICAgVXNlcklEID0+ICRTZWxmLT57VXNlcklEfSwKICAgICk7CgogICAgIyBkb24ndCBzaG93IGxpbmsgbWVudSBpdGVtIGlmIHRoZXJlIGFyZSBubyBsaW5rYWJsZSBvYmplY3RzCiAgICByZXR1cm4gaWYgISVQb3NzaWJsZU9iamVjdHM7CgogICAgJFNlbGYtPntMYXlvdXRPYmplY3R9LT5CbG9jayggTmFtZSA9PiAnTWVudScgKTsKICAgIGlmICggJFBhcmFte0NvdW50ZXJ9ICkgewogICAgICAgICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+QmxvY2soIE5hbWUgPT4gJ01lbnVJdGVtU3BsaXQnICk7CiAgICB9CgogICAgJFNlbGYtPntMYXlvdXRPYmplY3R9LT5CbG9jaygKICAgICAgICBOYW1lID0+ICdNZW51SXRlbScsCiAgICAgICAgRGF0YSA9PiB7CiAgICAgICAgICAgICVQYXJhbSwKICAgICAgICAgICAgJXsgJFBhcmFte1NlcnZpY2V9IH0sCiAgICAgICAgICAgICV7ICRQYXJhbXtDb25maWd9IH0sCiAgICAgICAgfSwKICAgICk7CgogICAgJFBhcmFte0NvdW50ZXJ9Kys7CgogICAgcmV0dXJuICRQYXJhbXtDb3VudGVyfTsKfQoKMTsK
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JVFNNU0xBTWVudUdlbmVyaWMucG0KIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IElUU01TTEFNZW51R2VuZXJpYy5wbSx2IDEuMiAyMDA5LzA1LzE4IDA5OjQ4OjQ2IG1oIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJVFNNU0xBTWVudUdlbmVyaWM7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjIgJCkgWzFdOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBvYmplY3RzCiAgICBmb3IgbXkgJE9iamVjdCAocXcoQ29uZmlnT2JqZWN0IExvZ09iamVjdCBEQk9iamVjdCBMYXlvdXRPYmplY3QgU0xBT2JqZWN0IFVzZXJJRCkpIHsKICAgICAgICAkU2VsZi0+eyRPYmplY3R9ID0gJFBhcmFteyRPYmplY3R9IHx8IGRpZSAiR290IG5vICRPYmplY3QhIjsKICAgIH0KCiAgICByZXR1cm4gJFNlbGY7Cn0KCnN1YiBSdW4gewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgaWYgKCAhJFBhcmFte1NMQX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coIFByaW9yaXR5ID0+ICdlcnJvcicsIE1lc3NhZ2UgPT4gJ05lZWQgU0xBIScgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBnZXQgZ3JvdXBzCiAgICBteSAkR3JvdXBzUm8KICAgICAgICA9ICRTZWxmLT57Q29uZmlnT2JqZWN0fS0+R2V0KCdGcm9udGVuZDo6TW9kdWxlJyktPnsgJFBhcmFte0NvbmZpZ30tPntBY3Rpb259IH0tPntHcm91cFJvfQogICAgICAgIHx8IFtdOwogICAgbXkgJEdyb3Vwc1J3CiAgICAgICAgPSAkU2VsZi0+e0NvbmZpZ09iamVjdH0tPkdldCgnRnJvbnRlbmQ6Ok1vZHVsZScpLT57ICRQYXJhbXtDb25maWd9LT57QWN0aW9ufSB9LT57R3JvdXB9CiAgICAgICAgfHwgW107CgogICAgIyBzZXQgYWNjZXNzCiAgICBteSAkQWNjZXNzID0gMTsKCiAgICAjIGNoZWNrIHBlcm1pc3Npb24KICAgIGlmICggJFBhcmFte0NvbmZpZ30tPntBY3Rpb259ICYmICggQHskR3JvdXBzUm99IHx8IEB7JEdyb3Vwc1J3fSApICkgewoKICAgICAgICAjIHNldCBhY2Nlc3MKICAgICAgICAkQWNjZXNzID0gMDsKCiAgICAgICAgIyBmaW5kIHJlYWQgb25seSBncm91cHMKICAgICAgICBST0dST1VQOgogICAgICAgIGZvciBteSAkUm9Hcm91cCAoIEB7JEdyb3Vwc1JvfSApIHsKCiAgICAgICAgICAgIG5leHQgUk9HUk9VUCBpZiAhJFNlbGYtPntMYXlvdXRPYmplY3R9LT57IlVzZXJJc0dyb3VwUm9bJFJvR3JvdXBdIn07CiAgICAgICAgICAgIG5leHQgUk9HUk9VUCBpZiAkU2VsZi0+e0xheW91dE9iamVjdH0tPnsiVXNlcklzR3JvdXBSb1skUm9Hcm91cF0ifSBuZSAnWWVzJzsKCiAgICAgICAgICAgICMgc2V0IGFjY2VzcwogICAgICAgICAgICAkQWNjZXNzID0gMTsKICAgICAgICAgICAgbGFzdCBST0dST1VQOwogICAgICAgIH0KCiAgICAgICAgIyBmaW5kIHJlYWQgd3JpdGUgZ3JvdXBzCiAgICAgICAgUldHUk9VUDoKICAgICAgICBmb3IgbXkgJFJ3R3JvdXAgKCBAeyRHcm91cHNSd30gKSB7CgogICAgICAgICAgICBuZXh0IFJXR1JPVVAgaWYgISRTZWxmLT57TGF5b3V0T2JqZWN0fS0+eyJVc2VySXNHcm91cFskUndHcm91cF0ifTsKICAgICAgICAgICAgbmV4dCBSV0dST1VQIGlmICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+eyJVc2VySXNHcm91cFskUndHcm91cF0ifSBuZSAnWWVzJzsKCiAgICAgICAgICAgICMgc2V0IGFjY2VzcwogICAgICAgICAgICAkQWNjZXNzID0gMTsKICAgICAgICAgICAgbGFzdCBSV0dST1VQOwogICAgICAgIH0KICAgIH0KCiAgICByZXR1cm4gJFBhcmFte0NvdW50ZXJ9IGlmICEkQWNjZXNzOwoKICAgICMgb3V0cHV0IG1lbnUgYmxvY2sKICAgICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+QmxvY2soIE5hbWUgPT4gJ01lbnUnICk7CgogICAgIyBvdXRwdXQgc2VwZXJhdG9yCiAgICBpZiAoICRQYXJhbXtDb3VudGVyfSApIHsKICAgICAgICAkU2VsZi0+e0xheW91dE9iamVjdH0tPkJsb2NrKCBOYW1lID0+ICdNZW51SXRlbVNwbGl0JyApOwogICAgfQoKICAgICMgb3V0cHV0IG1lbnUgaXRlbQogICAgJFNlbGYtPntMYXlvdXRPYmplY3R9LT5CbG9jaygKICAgICAgICBOYW1lID0+ICdNZW51SXRlbScsCiAgICAgICAgRGF0YSA9PiB7CiAgICAgICAgICAgICVQYXJhbSwKICAgICAgICAgICAgJXsgJFBhcmFte1NMQX0gfSwKICAgICAgICAgICAgJXsgJFBhcmFte0NvbmZpZ30gfSwKICAgICAgICB9LAogICAgKTsKICAgICRQYXJhbXtDb3VudGVyfSsrOwoKICAgIHJldHVybiAkUGFyYW17Q291bnRlcn07Cn0KCjE7Cg==
# --
# Kernel/Output/HTML/LinkObjectService.pm - layout backend module
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: LinkObjectService.pm,v 1.7 2009/10/14 19:26:30 ub Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Output::HTML::LinkObjectService;

use strict;
use warnings;

use Kernel::Output::HTML::Layout;

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

=head1 NAME

Kernel::Output::HTML::LinkObjectService - layout backend module

=head1 SYNOPSIS

All layout functions of link object (service)

=over 4

=cut

=item new()

create an object

    $BackendObject = Kernel::Output::HTML::LinkObjectService->new(
        %Param,
    );

=cut

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

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

    # check needed objects
    for my $Object (
        qw(ConfigObject LogObject MainObject DBObject UserObject EncodeObject
        QueueObject GroupObject ParamObject TimeObject LanguageObject UserLanguage UserID)
        )
    {
        $Self->{$Object} = $Param{$Object} || die "Got no $Object!";
    }

    $Self->{LayoutObject} = Kernel::Output::HTML::Layout->new( %{$Self} );

    # define needed variables
    $Self->{ObjectData} = {
        Object   => 'Service',
        Realname => 'Service',
    };

    return $Self;
}

=item TableCreateComplex()

return an array with the block data

Return

    @BlockData = (
        Object    => 'Service',
        Blockname => 'Service',
        Headline  => [
            {
                Content => '',
                Width   => 20,
            },
            {
                Content => 'Service',
            },
            {
                Content => 'Type',
                Width   => 100,
            },
            {
                Content => 'Criticality',
                Width   => 100,
            },
            {
                Content => 'Changed',
                Width   => 150,
            },
        ],
        ItemList => [
            [
                {
                    Type             => 'InciSignal',
                    Key              => 123,
                    Content          => 'Operational',
                    CurInciStateType => 'Operational',
                },
                {
                    Type      => 'Link',
                    Content   => 'Service Bla',
                    Link      => 'Action=AgentITSMServiceZoom&ServiceID=123',
                    MaxLength => 70,
                },
                {
                    Type    => 'Text',
                    Content => 'Other',
                    Translate => 1,
                },
                {
                    Type    => 'Text',
                    Content => 'High',
                    Translate => 1,
                },
                {
                    Type    => 'TimeLong',
                    Content => '2008-01-01 12:12:00',
                },
            ],
            [
                {
                    Type             => 'InciSignal',
                    Key              => 321,
                    Content          => 'Operational',
                    CurInciStateType => 'Operational',
                },
                {
                    Type      => 'Link',
                    Content   => 'Service Bla',
                    Link      => 'Action=AgentITSMServiceZoom&ServiceID=321',
                    MaxLength => 70,
                },
                {
                    Type    => 'Text',
                    Content => 'Other',
                    Translate => 1,
                },
                {
                    Type    => 'Text',
                    Content => 'Low',
                    Translate => 1,
                },
                {
                    Type    => 'TimeLong',
                    Content => '2007-02-02 22:12:00',
                },
            ],
        ],
    );

    @BlockData = $LinkObject->TableCreateComplex(
        ObjectLinkListWithData => $ObjectLinkListRef,
    );

=cut

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

    # check needed stuff
    if ( !$Param{ObjectLinkListWithData} || ref $Param{ObjectLinkListWithData} ne 'HASH' ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'Need ObjectLinkListWithData!',
        );
        return;
    }

    # convert the list
    my %LinkList;
    for my $LinkType ( keys %{ $Param{ObjectLinkListWithData} } ) {

        # extract link type List
        my $LinkTypeList = $Param{ObjectLinkListWithData}->{$LinkType};

        for my $Direction ( keys %{$LinkTypeList} ) {

            # extract direction list
            my $DirectionList = $Param{ObjectLinkListWithData}->{$LinkType}->{$Direction};

            for my $ServiceID ( keys %{$DirectionList} ) {

                $LinkList{$ServiceID}->{Data} = $DirectionList->{$ServiceID};
            }
        }
    }

    # create the item list
    my @ItemList;
    for my $ServiceID (
        sort { lc $LinkList{$a}{Data}->{Name} cmp lc $LinkList{$b}{Data}->{Name} }
        keys %LinkList
        )
    {

        # extract service data
        my $Service = $LinkList{$ServiceID}{Data};

        my @ItemColumns = (
            {
                Type             => 'CurInciSignal',
                Key              => $ServiceID,
                Content          => $Service->{CurInciState},
                CurInciStateType => $Service->{CurInciStateType},
            },
            {
                Type      => 'Link',
                Content   => $Service->{Name},
                Link      => '$Env{"Baselink"}Action=AgentITSMServiceZoom&ServiceID=' . $ServiceID,
                MaxLength => 70,
            },
            {
                Type      => 'Text',
                Content   => $Service->{Type},
                Translate => 1,
            },
            {
                Type      => 'Text',
                Content   => $Service->{Criticality},
                Translate => 1,
            },
            {
                Type    => 'TimeLong',
                Content => $Service->{ChangeTime},
            },
        );

        push @ItemList, \@ItemColumns;
    }

    return if !@ItemList;

    # define the block data
    my %Block = (
        Object    => $Self->{ObjectData}->{Object},
        Blockname => $Self->{ObjectData}->{Realname},
        Headline  => [
            {
                Content => '',
                Width   => 20,
            },
            {
                Content => 'Service',
            },
            {
                Content => 'Type',
                Width   => 100,
            },
            {
                Content => 'Criticality',
                Width   => 100,
            },
            {
                Content => 'Changed',
                Width   => 150,
            },
        ],
        ItemList => \@ItemList,
    );

    return ( \%Block );
}

=item TableCreateSimple()

return a hash with the link output data

Return

    %LinkOutputData = (
        Normal::Source => {
            Service => [
                {
                    Type    => 'Link',
                    Content => 'S:The servic[..]',
                    Title   => 'Service: The service name',
                    Css     => 'style="text-decoration: line-through"',
                },
                {
                    Type    => 'Link',
                    Content => 'S:Name of servic[..]',
                    Title   => 'Service: Name of service 2',
                },
            ],
        },
        ParentChild::Target => {
            Service => [
                {
                    Type    => 'Link',
                    Content => 'S:Service nam[..]',
                    Title   => 'Service: Service name',
                },
            ],
        },
    );

    %LinkOutputData = $LinkObject->TableCreateSimple(
        ObjectLinkListWithData => $ObjectLinkListRef,
    );

=cut

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

    # check needed stuff
    if ( !$Param{ObjectLinkListWithData} || ref $Param{ObjectLinkListWithData} ne 'HASH' ) {
        $Self->{LogObject}->Log( Priority => 'error', Message => 'Need ObjectLinkListWithData!' );
        return;
    }

    my %LinkOutputData;
    for my $LinkType ( keys %{ $Param{ObjectLinkListWithData} } ) {

        # extract link type List
        my $LinkTypeList = $Param{ObjectLinkListWithData}->{$LinkType};

        for my $Direction ( keys %{$LinkTypeList} ) {

            # extract direction list
            my $DirectionList = $Param{ObjectLinkListWithData}->{$LinkType}->{$Direction};

            my @ItemList;
            for my $ServiceID (
                sort {
                    lc $DirectionList->{$a}->{NameShort} cmp lc $DirectionList->{$b}->{NameShort}
                } keys %{$DirectionList}
                )
            {

                # extract service data
                my $Service = $DirectionList->{$ServiceID};

                # define item data
                my %Item = (
                    Type    => 'Link',
                    Content => "S:$Service->{NameShort}",
                    Title   => "Service: $Service->{Name}",
                    Link => '$Env{"Baselink"}Action=AgentITSMServiceZoom&ServiceID=' . $ServiceID,
                    MaxLength => 20,
                );

                push @ItemList, \%Item;
            }

            # add item list to link output data
            $LinkOutputData{ $LinkType . '::' . $Direction }->{Service} = \@ItemList;
        }
    }

    return %LinkOutputData;
}

=item ContentStringCreate()

return a output string

    my $String = $LayoutObject->ContentStringCreate(
        ContentData => $HashRef,
    );

=cut

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

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

    # extract content
    my $Content = $Param{ContentData};

    return if $Content->{Type} ne 'CurInciSignal';

    # set incident signal
    my %InciSignals = (
        incident    => 'redled',
        operational => 'greenled',
        unknown     => 'grayled',
        warning     => 'yellowled',
    );

    # investigate current incident signal
    $Content->{CurInciStateType} ||= 'unknown';
    my $CurInciSignal = $InciSignals{ $Content->{CurInciStateType} };
    $CurInciSignal ||= $InciSignals{unknown};

    my $String = $Self->{LayoutObject}->Output(
        Template => '<img border="0" src="$Env{"Images"}$QData{"CurInciSignal"}.png" '
            . 'title="$Text{"$QData{"CurInciState"}"}" alt="$Text{"$QData{"CurInciState"}"}">',
        Data => {
            CurInciSignal => $CurInciSignal,
            CurInciState => $Content->{Content} || '',
        },
    );

    return $String;
}

=item SelectableObjectList()

return an array hash with selectable objects

Return

    @SelectableObjectList = (
        {
            Key   => 'Service',
            Value => 'Service',
        },
    );

    @SelectableObjectList = $LinkObject->SelectableObjectList(
        Selected => $Identifier,  # (optional)
    );

=cut

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

    my $Selected;
    if ( $Param{Selected} && $Param{Selected} eq $Self->{ObjectData}->{Object} ) {
        $Selected = 1;
    }

    # object select list
    my @ObjectSelectList = (
        {
            Key      => $Self->{ObjectData}->{Object},
            Value    => $Self->{ObjectData}->{Realname},
            Selected => $Selected,
        },
    );

    return @ObjectSelectList;
}

=item SearchOptionList()

return an array hash with search options

Return

    @SearchOptionList = (
        {
            Key       => 'Name',
            Name      => 'Service',
            InputStrg => $FormString,
            FormData  => 'Service Name',
        },
    );

    @SearchOptionList = $LinkObject->SearchOptionList();

=cut

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

    # search option list
    my @SearchOptionList = (
        {
            Key  => 'Name',
            Name => 'Service',
            Type => 'Text',
        },
    );

    # add formkey
    for my $Row (@SearchOptionList) {
        $Row->{FormKey} = 'SEARCH::' . $Row->{Key};
    }

    # add form data and input string
    ROW:
    for my $Row (@SearchOptionList) {

        # get form data
        $Row->{FormData} = $Self->{ParamObject}->GetParam( Param => $Row->{FormKey} );

        # parse the input text block
        $Self->{LayoutObject}->Block(
            Name => 'InputText',
            Data => {
                Key => $Row->{FormKey},
                Value => $Row->{FormData} || '',
            },
        );

        # add the input string
        $Row->{InputStrg} = $Self->{LayoutObject}->Output(
            TemplateFile => 'LinkObject',
        );

        next ROW;
    }

    return @SearchOptionList;
}

1;

=back

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (http://otrs.org/).

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

=cut

=head1 VERSION

$Revision: 1.7 $ $Date: 2009/10/14 19:26:30 $

=cut

IyAtLQojIEFkbWluSVRTTUNJUEFsbG9jYXRlLmR0bCAtIHByb3ZpZGVzIEhUTUwgZm9ybSBmb3IgQWRtaW5JVFNNQ0lQQWxsb2NhdGUKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IEFkbWluSVRTTUNJUEFsbG9jYXRlLmR0bCx2IDEuOSAyMDEwLzA2LzAxIDE2OjA0OjM1IG1iIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgo8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iMTAwJSIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIzIj4KICAgIDx0cj4KICAgICAgICA8dGQgY29sc3Bhbj0iMiIgY2xhc3M9Im1haW5oZWFkIj4KICAgICAgICAgICAgJEVudnsiQm94MCJ9JFRleHR7IkNyaXRpY2FsaXR5IDwtPiBJbXBhY3QgPC0+IFByaW9yaXR5In0kRW52eyJCb3gxIn0KICAgICAgICA8L3RkPgogICAgPC90cj4KICAgIDx0cj4KICAgICAgICA8dGQgY2xhc3M9Im1haW5ib2R5IiBhbGlnbj0iY2VudGVyIj4KICAgICAgICAgICAgPGJyPgogICAgICAgICAgICA8Zm9ybSBhY3Rpb249IiRFbnZ7IkNHSUhhbmRsZSJ9IiBtZXRob2Q9InBvc3QiPgogICAgICAgICAgICAgICAgPGlucHV0IHR5cGU9ImhpZGRlbiIgbmFtZT0iQWN0aW9uIiB2YWx1ZT0iJEVudnsiQWN0aW9uIn0iPgogICAgICAgICAgICAgICAgPGlucHV0IHR5cGU9ImhpZGRlbiIgbmFtZT0iU3ViYWN0aW9uIiB2YWx1ZT0iQ0lQQWxsb2NhdGUiPgogICAgICAgICAgICAgICAgPHRhYmxlIHdpZHRoPSI4MDAiIGNlbGxzcGFjaW5nPSIwIiBjZWxscGFkZGluZz0iNCI+CiAgICAgICAgICAgICAgICAgICAgPHRyPgogICAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRoZWFkIj4KICAgICAgICAgICAgICAgICAgICAgICAgICAgICRUZXh0eyJQcmlvcml0eSBhbGxvY2F0aW9uIn06CiAgICAgICAgICAgICAgICAgICAgICAgIDwvdGQ+CiAgICAgICAgICAgICAgICAgICAgPC90cj4KICAgICAgICAgICAgICAgICAgICA8dHI+CiAgICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGJvZHkiPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRhYmxlIGJvcmRlcj0iMCIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIzIiB3aWR0aD0iMTAwJSI+CjwhLS0gZHRsOmJsb2NrOkNJUEFsbG9jYXRlUm93IC0tPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDx0cj4KPCEtLSBkdGw6YmxvY2s6Q0lQQWxsb2NhdGVSb3dDb2x1bW5EZXNjcmlwdGlvbiAtLT4KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRkPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRhYmxlIGJvcmRlcj0iMCIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIwIiB3aWR0aD0iMTAwJSI+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRyPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiIGFsaWduPSJjZW50ZXIiPiRUZXh0eyIkRGF0YXsiT2JqZWN0VHlwZSJ9In08L3RkPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPHRyPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnR2YWx1ZSIgYWxpZ249ImNlbnRlciI+JFRleHR7IiREYXRheyJPYmplY3RPcHRpb24ifSJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPC90YWJsZT4KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPC90ZD4KPCEtLSBkdGw6YmxvY2s6Q0lQQWxsb2NhdGVSb3dDb2x1bW5EZXNjcmlwdGlvbiAtLT4KPCEtLSBkdGw6YmxvY2s6Q0lQQWxsb2NhdGVSb3dDb2x1bW5Db250ZW50IC0tPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnR2YWx1ZSIgYWxpZ249ImNlbnRlciI+JERhdGF7Ik9wdGlvblN0cmcifTwvdGQ+CjwhLS0gZHRsOmJsb2NrOkNJUEFsbG9jYXRlUm93Q29sdW1uQ29udGVudCAtLT4KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8L3RyPgo8IS0tIGR0bDpibG9jazpDSVBBbGxvY2F0ZVJvdyAtLT4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDwvdGFibGU+CiAgICAgICAgICAgICAgICAgICAgICAgIDwvdGQ+CiAgICAgICAgICAgICAgICAgICAgPC90cj4KICAgICAgICAgICAgICAgICAgICA8dHI+CiAgICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGZvb3RlciI+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICA8aW5wdXQgY2xhc3M9ImJ1dHRvbiIgdHlwZT0ic3VibWl0IiB2YWx1ZT0iJFRleHR7IlNhdmUifSI+CiAgICAgICAgICAgICAgICAgICAgICAgIDwvdGQ+CiAgICAgICAgICAgICAgICAgICAgPC90cj4KICAgICAgICAgICAgICAgIDwvdGFibGU+CiAgICAgICAgICAgIDwvZm9ybT4KICAgICAgICAgICAgPGJyPgogICAgICAgIDwvdGQ+CiAgICA8L3RyPgo8L3RhYmxlPgo=
# --
# AdminService.dtl - provides HTML form for AdminService
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: AdminService.dtl,v 1.3 2009/10/09 16:36:28 ub Exp $
# $OldId: AdminService.dtl,v 1.8.2.1 2009/10/01 14:12:45 ub Exp $
# --
# 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.
# --

<!-- dtl:block:Overview -->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
    <tr>
        <td colspan="2" class="mainhead">
            $Env{"Box0"}$Text{"Service Management"}$Env{"Box1"}
        </td>
    </tr>
    <tr>
        <td width="30%" class="mainbody">
            <form action="$Env{"CGIHandle"}" method="get">
                <input type="hidden" name="Action" value="$Env{"Action"}"/>
                <input type="hidden" name="Subaction" value="ServiceEdit"/>
                <input type="hidden" name="ServiceID" value="NEW"/>
                <table width="100%" cellspacing="0" cellpadding="4">
                    <tr>
                        <td class="contenthead">$Text{"Add Service"}:</td>
                    </tr>
                    <tr>
                        <td class="contentbody">$Text{"Add a new Service."}</td>
                    </tr>
                    <tr>
                        <td class="contentfooter">
                            <input class="button" type="submit" value="$Text{"Add"}"/>
                        </td>
                    </tr>
                </table>
            </form>
        </td>
        <td width="70%" class="mainbody">
<!-- dtl:block:OverviewList -->
            <table width="100%" cellspacing="0" cellpadding="4">
                <tr>
                    <td class="contenthead">$Text{"List"}:</td>
                </tr>
                <tr>
                    <td class="contentbody">
                        <table width="100%" border="0" cellspacing="0" cellpadding="3">
                            <tr>
                                <td class="contentkey" width="60%">$Text{"Service"}</td>
                                <td align="center" class="contentkey">$Text{"valid"}/$Text{"invalid"}</td>
                            </tr>
<!-- dtl:block:OverviewListRow -->
                            <tr class="$QData{"CssClass"}">
                                <td>
                                    $Data{"LevelSpace"}
                                    <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=ServiceEdit&ServiceID=$QData{"ServiceID"}">
                                    $QData{"Name"}
                                    </a>
                                </td>
                                <td align="center">
                                    $Text{"$QData{"Valid"}"}
                                </td>
                            </tr>
<!-- dtl:block:OverviewListRow -->
                        </table>
                    </td>
                </tr>
                <tr>
                    <td class="contentfooter">
                        &nbsp;
                    </td>
                </tr>
            </table>
<!-- dtl:block:OverviewList -->
<!-- dtl:block:ServiceEdit -->
            <form action="$Env{"CGIHandle"}" method="get">
                <input type="hidden" name="Action" value="$Env{"Action"}"/>
                <input type="hidden" name="Subaction" value="ServiceSave"/>
                <input type="hidden" name="ServiceID" value="$QData{"ServiceID"}"/>
                <table width="100%" cellspacing="0" cellpadding="4">
                    <tr>
                        <td class="contenthead">$Text{"Edit"}:</td>
                    </tr>
                    <tr>
                        <td class="contentbody">
                            <table border="0" cellspacing="0" cellpadding="3">
                                <tr>
                                    <td class="contentkey">$Text{"Service"}: </td>
                                    <td class="contentvalue">
                                        <input type="text" name="Name" value="$QData{"NameShort"}" size="45" maxlength="200"/>
                                    </td>
                                </tr>
                                <tr>
                                    <td class="contentkey">$Text{"Sub-Service of"}: </td>
                                    <td class="contentvalue">$Data{"ParentOptionStrg"}</td>
                                </tr>
# ---
# ITSM
# ---
                                <tr>
                                    <td class="contentkey">$Text{"Type"}: </td>
                                    <td class="contentvalue">$Data{"TypeOptionStrg"}</td>
                                </tr>
                                <tr>
                                    <td class="contentkey">$Text{"Criticality"}: </td>
                                    <td class="contentvalue">$Data{"CriticalityOptionStrg"}</td>
                                </tr>
# ---
                                <tr>
                                    <td class="contentkey">$Text{"Valid"}: </td>
                                    <td class="contentvalue">$Data{"ValidOptionStrg"}</td>
                                </tr>
                                <tr>
                                    <td class="contentkey">$Text{"Comment"}: </td>
                                    <td class="contentvalue">
                                        <input type="text" name="Comment" value="$QData{"Comment"}" size="60" maxlength="150"/>
                                    </td>
                                </tr>
<!-- dtl:block:Item -->
<!-- dtl:block:InputKey -->
                                <tr>
                                    <td class="contentkey">$Text{"$Data{"Label"}"} $Text{"$QData{"Key"}"}:</td>
                                    <td class="contentvalue"><input type="text" name="$QData{"Name"}" value="$QData{"SelectedID"}" size="30"/></td>
                                </tr>
<!-- dtl:block:InputKey -->
<!-- dtl:block:Input -->
                                <tr>
                                    <td class="contentkey">$Text{"$Data{"Label"}"}:<br/><i class="small">($Text{"$Data{"Desc"}"})</i></td>
                                    <td class="contentvalue">
                                        <input type="text" name="$QData{"Name"}" value="$QData{"SelectedID"}" size="30"/>
                                    </td>
                                </tr>
<!-- dtl:block:Input -->
<!-- dtl:block:TextArea -->
                                <tr>
                                    <td class="contentkey">$Text{"$Data{"Label"}"}:<br/><i class="small">($Text{"$Data{"Desc"}"})</i></td>
                                    <td class="contentvalue">
                                        <textarea name="$QData{"Name"}" rows="$QData{"Rows"}" cols="$QData{"Cols"}">$QData{"SelectedID"}</textarea>
                                    </td>
                                </tr>
<!-- dtl:block:TextArea -->
<!-- dtl:block:Option -->
                                <tr>
                                    <td class="contentkey">$Text{"$Data{"Label"}"} $Text{"$QData{"Key"}"}:<br/>
                                        <i class="small">($Text{"$Data{"Desc"}"})</i>
                                    </td>
                                    <td class="contentvalue">$Data{"Option"}</td>
                                </tr>
<!-- dtl:block:Option -->
<!-- dtl:block:Upload -->
                                <tr>
                                    <td class="contentkey">$Text{"$Data{"Label"}"} $Text{"$QData{"Key"}"}</td>
                                    <td class="contentvalue">
                                        <input name="$QData{"Name"}" type="file" size="30" class="fixed"/><br/>
                                        <a href="">$QData{"Filename"}</a>
                                    </td>
                                </tr>
<!-- dtl:block:Upload -->
<!-- dtl:block:Password -->
                                <tr>
                                    <td class="contentkey">$Text{"$Data{"Label"}"} $Text{"$QData{"Key"}"}:</td>
                                    <td class="contentvalue"><input type="password" name="$QData{"Name"}" value="" size="30"/></td>
                                </tr>
<!-- dtl:block:Password -->
<!-- dtl:block:Item -->
                            </table>
                        </td>
                    </tr>
                    <tr>
                        <td class="contentfooter">
                            <input class="button" type="submit" value="$Text{"Submit"}"/>
                        </td>
                    </tr>
                </table>
            </form>
<!-- dtl:block:ServiceEdit -->
        </td>
    </tr>
</table>
<!-- end form -->
<!-- dtl:block:Overview -->

# --
# AdminSLA.dtl - provides HTML form for AdminSLA
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: AdminSLA.dtl,v 1.5 2009/10/09 16:36:28 ub Exp $
# $OldId: AdminSLA.dtl,v 1.13.2.1 2009/10/01 14:12:44 ub Exp $
# --
# 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.
# --

<!-- dtl:block:Overview -->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
    <tr>
        <td colspan="2" class="mainhead">
            $Env{"Box0"}$Text{"SLA Management"}$Env{"Box1"}
        </td>
    </tr>
    <tr>
        <td width="30%" class="mainbody">
            <form action="$Env{"CGIHandle"}" method="get">
                <input type="hidden" name="Action" value="$Env{"Action"}"/>
                <input type="hidden" name="Subaction" value="SLAEdit"/>
                <table width="100%" cellspacing="0" cellpadding="4">
                    <tr>
                        <td class="contenthead">$Text{"Add SLA"}:</td>
                    </tr>
                    <tr>
                        <td class="contentbody">$Text{"Add a new SLA."}</td>
                    </tr>
                    <tr>
                        <td class="contentfooter">
                            <input class="button" type="submit" value="$Text{"Add"}"/>
                        </td>
                    </tr>
                </table>
            </form>
        </td>
        <td width="70%" class="mainbody">
<!-- dtl:block:OverviewList -->
            <table width="100%" cellspacing="0" cellpadding="4">
                <tr>
                    <td class="contenthead">$Text{"List"}:</td>
                </tr>
                <tr>
                    <td class="contentbody">
                        <table width="100%" border="0" cellspacing="0" cellpadding="3">
                            <tr>
                                <td class="contentkey" width="40%">$Text{"SLA"}</td>
                                <td class="contentkey">$Text{"Service"}</td>
                                <td align="center" class="contentkey">$Text{"valid"}/$Text{"invalid"}</td>
                                <td class="contentkey">$Text{"Changed"}</td>
                                <td class="contentkey">$Text{"Created"}</td>
                            </tr>
<!-- dtl:block:OverviewListRow -->
                            <tr>
                                <td class="$QData{"CssClass"}"><a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=SLAEdit&SLAID=$QData{"SLAID"}">$QData{"Name"}</a></td>
                                <td class="$QData{"CssClass"}">$QData{"Service"}</td>
                                <td class="$QData{"CssClass"}" align="center">$Text{"$QData{"Valid"}"}</td>
                                <td class="$QData{"CssClass"}">$TimeShort{"$QData{"ChangeTime"}"}</td>
                                <td class="$QData{"CssClass"}">$TimeShort{"$QData{"CreateTime"}"}</td>
                            </tr>
<!-- dtl:block:OverviewListRow -->
                        </table>
                    </td>
                </tr>
                <tr>
                    <td class="contentfooter">
                        &nbsp;
                    </td>
                </tr>
            </table>
<!-- dtl:block:OverviewList -->
<!-- dtl:block:SLAEdit -->
            <form action="$Env{"CGIHandle"}" method="get">
                <input type="hidden" name="Action" value="$Env{"Action"}"/>
                <input type="hidden" name="Subaction" value="SLASave"/>
                <input type="hidden" name="SLAID" value="$QData{"SLAID"}"/>
                <table width="100%" cellspacing="0" cellpadding="4">
                    <tr>
                        <td class="contenthead">$Text{"Edit"}:</td>
                    </tr>
                    <tr>
                        <td class="contentbody">
                            <table border="0" cellspacing="0" cellpadding="3">
                                <tr>
                                    <td class="contentkey">$Text{"SLA"}: </td>
                                    <td class="contentvalue">
                                        <input type="text" name="Name" value="$QData{"Name"}" size="45" maxlength="200"/>
                                    </td>
                                </tr>
# ---
# ITSM
# ---
                                <tr>
                                    <td class="contentkey">$Text{"Type"}: </td>
                                    <td class="contentvalue">$Data{"TypeOptionStrg"}</td>
                                </tr>
# ---
                                <tr>
                                    <td class="contentkey">$Text{"Service"}: </td>
                                    <td class="contentvalue">$Data{"ServiceOptionStrg"}</td>
                                </tr>
                                <tr>
                                    <td class="contentkey">$Text{"Calendar"}: </td>
                                    <td class="contentvalue">$Data{"CalendarOptionStrg"}</td>
                                </tr>
                                <tr>
                                    <td class="contentkey">
                                        $Text{"Escalation - First Response Time"}:
                                        <br/>
                                        <i class="small">($Text{"0 = no escalation"} - 24 $Text{"hours"} = 1440 $Text{"minutes"} - $Text{"Only business hours are counted."})</i>
                                    </td>
                                    <td class="contentvalue">
                                        <input type="text" name="FirstResponseTime" value="$QData{"FirstResponseTime"}" size="8" maxlength="10"/> $Text{"minutes"} ( $Text{"Notify by"}  $Data{"FirstResponseNotifyOptionStrg"} )
                                    </td>
                                </tr>
                                <tr>
                                    <td class="contentkey">
                                        $Text{"Escalation - Update Time"}:
                                        <br/>
                                        <i class="small">($Text{"0 = no escalation"} - 24 $Text{"hours"} = 1440 $Text{"minutes"} - $Text{"Only business hours are counted."})</i>
                                    </td>
                                    <td class="contentvalue">
                                        <input type="text" name="UpdateTime" value="$QData{"UpdateTime"}" size="8" maxlength="10"/> $Text{"minutes"} ( $Text{"Notify by"}  $Data{"UpdateNotifyOptionStrg"} )
                                    </td>
                                </tr>
                                <tr>
                                    <td class="contentkey">
                                        $Text{"Escalation - Solution Time"}:
                                        <br/>
                                        <i class="small">($Text{"0 = no escalation"} - 24 $Text{"hours"} = 1440 $Text{"minutes"} - $Text{"Only business hours are counted."})</i>
                                    </td>
                                    <td class="contentvalue">
                                        <input type="text" name="SolutionTime" value="$QData{"SolutionTime"}" size="8" maxlength="10"/> $Text{"minutes"} ( $Text{"Notify by"}  $Data{"SolutionNotifyOptionStrg"} )
                                    </td>
                                </tr>
# ---
# ITSM
# ---
                                <tr>
                                    <td class="contentkey">$Text{"Minimum Time Between Incidents"} ($Text{"minutes"}): </td>
                                    <td class="contentvalue">
                                        <input type="text" name="MinTimeBetweenIncidents" value="$QData{"MinTimeBetweenIncidents"}" size="10" maxlength="15">
                                    </td>
                                </tr>
# ---
                                <tr>
                                    <td class="contentkey">$Text{"Valid"}: </td>
                                    <td class="contentvalue">$Data{"ValidOptionStrg"}</td>
                                </tr>
                                <tr>
                                    <td class="contentkey">$Text{"Comment"}: </td>
                                    <td class="contentvalue">
                                        <input type="text" name="Comment" value="$QData{"Comment"}" size="60" maxlength="150"/>
                                    </td>
                                </tr>
<!-- dtl:block:SLAItem -->
<!-- dtl:block:InputKey -->
                                <tr>
                                    <td class="contentkey">$Text{"$Data{"Label"}"} $Text{"$QData{"Key"}"}:</td>
                                    <td class="contentvalue"><input type="text" name="$QData{"Name"}" value="$QData{"SelectedID"}" size="30"/></td>
                                </tr>
<!-- dtl:block:InputKey -->
<!-- dtl:block:Input -->
                                <tr>
                                    <td class="contentkey">$Text{"$Data{"Label"}"}:<br/><i class="small">($Text{"$Data{"Desc"}"})</i></td>
                                    <td class="contentvalue">
                                        <input type="text" name="$QData{"Name"}" value="$QData{"SelectedID"}" size="30"/>
                                    </td>
                                </tr>
<!-- dtl:block:Input -->
<!-- dtl:block:TextArea -->
                                <tr>
                                    <td class="contentkey">$Text{"$Data{"Label"}"}:<br/><i class="small">($Text{"$Data{"Desc"}"})</i></td>
                                    <td class="contentvalue">
                                        <textarea name="$QData{"Name"}" rows="$QData{"Rows"}" cols="$QData{"Cols"}">$QData{"SelectedID"}</textarea>
                                    </td>
                                </tr>
<!-- dtl:block:TextArea -->
<!-- dtl:block:Option -->
                                <tr>
                                    <td class="contentkey">$Text{"$Data{"Label"}"} $Text{"$QData{"Key"}"}:<br/>
                                        <i class="small">($Text{"$Data{"Desc"}"})</i>
                                    </td>
                                    <td class="contentvalue">$Data{"Option"}</td>
                                </tr>
<!-- dtl:block:Option -->
<!-- dtl:block:Upload -->
                                <tr>
                                    <td class="contentkey">$Text{"$Data{"Label"}"} $Text{"$QData{"Key"}"}</td>
                                    <td class="contentvalue">
                                        <input name="$QData{"Name"}" type="file" size="30" class="fixed"/><br/>
                                        <a href="">$QData{"Filename"}</a>
                                    </td>
                                </tr>
<!-- dtl:block:Upload -->
<!-- dtl:block:Password -->
                                <tr>
                                    <td class="contentkey">$Text{"$Data{"Label"}"} $Text{"$QData{"Key"}"}:</td>
                                    <td class="contentvalue"><input type="password" name="$QData{"Name"}" value="" size="30"/></td>
                                </tr>
<!-- dtl:block:Password -->
<!-- dtl:block:SLAItem -->
                            </table>
                        </td>
                    </tr>
                    <tr>
                        <td class="contentfooter">
                            <input class="button" type="submit" value="$Text{"Submit"}"/>
                        </td>
                    </tr>
                </table>
            </form>
<!-- dtl:block:SLAEdit -->
        </td>
    </tr>
</table>
<!-- end form -->
<!-- dtl:block:Overview -->

IyAtLQojIEFnZW50SVRTTVNlcnZpY2UuZHRsIC0gcHJvdmlkZXMgSFRNTCBmb3JtIGZvciBBZ2VudElUU01TZXJ2aWNlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTAgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBBZ2VudElUU01TZXJ2aWNlLmR0bCx2IDEuNSAyMDEwLzAzLzE4IDE1OjA2OjM5IHViIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgo8IS0tIGR0bDpibG9jazpPdmVydmlldyAtLT4KPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjEwMCUiIGNlbGxzcGFjaW5nPSIwIiBjZWxscGFkZGluZz0iMyI+CiAgICA8dHI+CiAgICAgICAgPHRkIGNsYXNzPSJtYWluaGVhZCI+CiAgICAgICAgICAgICRFbnZ7IkJveDAifSRUZXh0eyJPdmVydmlldyJ9OiAkVGV4dHsiU2VydmljZSJ9JEVudnsiQm94MSJ9CiAgICAgICAgPC90ZD4KICAgIDwvdHI+CiAgICA8dHI+CiAgICAgICAgPHRkIGNsYXNzPSJtYWluYm9keSIgYWxpZ249ImNlbnRlciI+CiAgICAgICAgICAgIDxicj4KICAgICAgICAgICAgPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjgwMCIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIzIj4KICAgICAgICAgICAgICAgIDx0ciBjbGFzcz0iY29udGVudGhlYWQiPgogICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSIgd2lkdGg9IjIwIj4mbmJzcDs8L3RkPgogICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+JFRleHR7IlNlcnZpY2UifTwvdGQ+CiAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50a2V5Ij4kVGV4dHsiVHlwZSJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJDcml0aWNhbGl0eSJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiIHdpZHRoPSIxNTAiPiRUZXh0eyJDaGFuZ2VkIn08L3RkPgogICAgICAgICAgICAgICAgPC90cj4KPCEtLSBkdGw6YmxvY2s6T3ZlcnZpZXdSb3cgLS0+CiAgICAgICAgICAgICAgICA8dHIgY2xhc3M9ImNvbnRlbnRib2R5Ij4KICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9IiRRRGF0YXsiQ3NzQ2xhc3MifSIgYWxpZ249InJpZ2h0Ij4KICAgICAgICAgICAgICAgICAgICAgICAgPGltZyBib3JkZXI9IjAiIHNyYz0iJENvbmZpZ3siRnJvbnRlbmQ6OkltYWdlUGF0aCJ9JFFEYXRheyJDdXJJbmNpU2lnbmFsIn0ucG5nIiB0aXRsZT0iJFF1b3RleyIkVGV4dHsiJERhdGF7IkN1ckluY2lTdGF0ZSJ9In0ifSIgYWx0PSIkUXVvdGV7IiRUZXh0eyIkRGF0YXsiQ3VySW5jaVN0YXRlIn0ifSJ9Ij4KICAgICAgICAgICAgICAgICAgICA8L3RkPgogICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iJFFEYXRheyJDc3NDbGFzcyJ9Ij4KICAgICAgICAgICAgICAgICAgICAgICAgJERhdGF7IkxldmVsU3BhY2UifQogICAgICAgICAgICAgICAgICAgICAgICA8YSBocmVmPSIkRW52eyJCYXNlbGluayJ9QWN0aW9uPUFnZW50SVRTTVNlcnZpY2Vab29tJlNlcnZpY2VJRD0kUURhdGF7IlNlcnZpY2VJRCJ9Ij4KICAgICAgICAgICAgICAgICAgICAgICAgJFFEYXRheyJOYW1lIn0KICAgICAgICAgICAgICAgICAgICAgICAgPC9hPgogICAgICAgICAgICAgICAgICAgIDwvdGQ+CiAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPSIkUURhdGF7IkNzc0NsYXNzIn0iPiRUZXh0eyIkUURhdGF7IlR5cGUifSJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9IiRRRGF0YXsiQ3NzQ2xhc3MifSI+JFRleHR7IiRRRGF0YXsiQ3JpdGljYWxpdHkifSJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9IiRRRGF0YXsiQ3NzQ2xhc3MifSIgd2lkdGg9IjE1MCI+JFRpbWVMb25neyIkUURhdGF7IkNoYW5nZVRpbWUifSJ9PC90ZD4KICAgICAgICAgICAgICAgIDwvdHI+CjwhLS0gZHRsOmJsb2NrOk92ZXJ2aWV3Um93IC0tPgogICAgICAgICAgICAgICAgPHRyIGNsYXNzPSJjb250ZW50Zm9vdGVyIj4KICAgICAgICAgICAgICAgICAgICA8dGQgY29sc3Bhbj0iNSI+Jm5ic3A7PC90ZD4KICAgICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgIDwvdGFibGU+CiAgICAgICAgICAgIDxicj4KICAgICAgICA8L3RkPgogICAgPC90cj4KPC90YWJsZT4KPCEtLSBkdGw6YmxvY2s6T3ZlcnZpZXcgLS0+Cg==
IyAtLQojIEFnZW50SVRTTVNlcnZpY2VQcmludC5kdGwgLSBwcm92aWRlcyBIVE1MIHByaW50IHZpZXcKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IEFnZW50SVRTTVNlcnZpY2VQcmludC5kdGwsdiAxLjIgMjAwOS8wNS8xOCAwOTo0ODo0NiBtaCBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKPHRhYmxlIGJvcmRlcj0iMCIgY2VsbHNwYWNpbmc9IjEiIGNlbGxwYWRkaW5nPSIwIiB3aWR0aD0iMTAwJSI+CiAgPHRyPgogICAgPHRkIGNsYXNzPSJtYWluaGVhZCI+JFRleHR7IlNlcnZpY2UifTwvdGQ+CiAgPC90cj4KPC90YWJsZT4KPHRhYmxlIGJvcmRlcj0iMCIgY2VsbHNwYWNpbmc9IjEiIGNlbGxwYWRkaW5nPSIxMCIgd2lkdGg9IjEwMCUiPgogIDx0cj4KICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+CiAgICAgICRUZXh0eyJTZXJ2aWNlLUluZm8ifToKICAgICAgPHRhYmxlIGJvcmRlcj0iMCIgY2VsbHNwYWNpbmc9IjMiIGNlbGxwYWRkaW5nPSIwIiB3aWR0aD0iMTAwJSI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50a2V5IiB3aWR0aD0iMTAlIj4kVGV4dHsiU2VydmljZSJ9OjwvdGQ+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnR2YWx1ZSIgd2lkdGg9IjQwJSI+JFFEYXRheyJOYW1lU2hvcnQifTwvdGQ+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiIHdpZHRoPSIyMCUiPiRUZXh0eyJDcmVhdGVkIn06PC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIiB3aWR0aD0iMzAlIj4kVGltZUxvbmd7IiREYXRheyJDcmVhdGVUaW1lIn0ifTwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJDdXJyZW50IEluY2lkZW50IFN0YXRlIn06PC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj4kVGV4dHsiJFFEYXRheyJDdXJJbmNpU3RhdGUifSJ9PC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+JFRleHR7IkNyZWF0ZWQgYnkifTo8L3RkPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiPiRRRGF0YXsiQ3JlYXRlQnlVc2VyTG9naW4ifSAoJFFEYXRheyJDcmVhdGVCeVVzZXJGaXJzdG5hbWUifSAkUURhdGF7IkNyZWF0ZUJ5VXNlckxhc3RuYW1lIn0pPC90ZD4KICAgICAgICA8L3RyPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+PC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj48L3RkPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50a2V5Ij4kVGV4dHsiTGFzdCBjaGFuZ2VkIn06PC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj4kVGltZUxvbmd7IiREYXRheyJDaGFuZ2VUaW1lIn0ifTwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiPjwvdGQ+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+PC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+JFRleHR7Ikxhc3QgY2hhbmdlZCBieSJ9OjwvdGQ+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+JFFEYXRheyJDaGFuZ2VCeVVzZXJMb2dpbiJ9ICgkUURhdGF7IkNoYW5nZUJ5VXNlckZpcnN0bmFtZSJ9ICRRRGF0YXsiQ2hhbmdlQnlVc2VyTGFzdG5hbWUifSk8L3RkPgogICAgICAgIDwvdHI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50a2V5Ij4mbmJzcDs8L3RkPgogICAgICAgICAgPHRkIGNvbHNwYW49IjMiIGNsYXNzPSJjb250ZW50dmFsdWUiPiZuYnNwOzwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgPC90YWJsZT4KICAgIDwvdGQ+CiAgPC90cj4KPCEtLSBkdGw6YmxvY2s6QXNzb2NpYXRlZFNMQXMgLS0+CiAgPHRyPgogICAgPHRkIGNsYXNzPSJjb250ZW50a2V5Ij4KICAgICAgJFRleHR7IkFzc29jaWF0ZWQgU0xBcyJ9OgogICAgICA8dGFibGUgYm9yZGVyPSIwIiBjZWxsc3BhY2luZz0iMyIgY2VsbHBhZGRpbmc9IjAiIHdpZHRoPSIxMDAlIj4KPCEtLSBkdGw6YmxvY2s6QXNzb2NpYXRlZFNMQXNSb3cgLS0+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50a2V5IiB3aWR0aD0iMjAlIj4kVGV4dHsiU0xBIn06PC90ZD4KICAgICAgICAgIDx0ZCBjb2xzcGFuPSIzIiBjbGFzcz0iY29udGVudHZhbHVlIj4kUURhdGF7Ik5hbWUifTwvdGQ+CiAgICAgICAgPC90cj4KPCEtLSBkdGw6YmxvY2s6QXNzb2NpYXRlZFNMQXNSb3cgLS0+CiAgICAgIDwvdGFibGU+CiAgICA8L3RkPgogIDwvdHI+CjwvdGFibGU+CjwhLS0gZHRsOmJsb2NrOkFzc29jaWF0ZWRTTEFzIC0tPgo8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iNjQwIiBjZWxscGFkZGluZz0iMCIgY2VsbHNwYWNpbmc9IjAiIGJnY29sb3I9IiMwMDAwMDAiIGNsYXNzPSJjb250YWluZXIiPgogIDx0cj4KICAgIDx0ZD4KICAgICAgPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjY0MCIgY2VsbHBhZGRpbmc9IjEwIiBjZWxsc3BhY2luZz0iMSIgYmdjb2xvcj0iI2ZmZmZmZiI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkPgogICAgICAgICAgICA8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iMTAwJSIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIzIj4KICAgICAgICAgICAgICA8dHIgYmdjb2xvcj0iI2ZmZmZmZiI+CiAgICAgICAgICAgICAgICA8dGQgdmFsaWduPSJ0b3AiIHdpZHRoPSIxMDAlIiBjbGFzcz0ic21hbGwiPgogICAgICAgICAgICAgICAgICA8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iMTAwJSIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIxIj4KICAgICAgICAgICAgICAgICAgICA8dHI+CiAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiIHdpZHRoPSIyMCUiPiRUZXh0eyJTZXJ2aWNlIn06PC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj4kUURhdGF7Ik5hbWUifTwvdGQ+CiAgICAgICAgICAgICAgICAgICAgPC90cj4KICAgICAgICAgICAgICAgICAgICA8dHI+CiAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJUeXBlIn06PC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj4kVGV4dHsiJFFEYXRheyJUeXBlIn0ifTwvdGQ+CiAgICAgICAgICAgICAgICAgICAgPC90cj4KICAgICAgICAgICAgICAgICAgICA8dHI+CiAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJDcml0aWNhbGl0eSJ9OjwvdGQ+CiAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+JFRleHR7IiRRRGF0YXsiQ3JpdGljYWxpdHkifSJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICAgICAgICA8L3RhYmxlPgogICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICA8L3RhYmxlPgogICAgICAgICAgPC90ZD4KICAgICAgICA8L3RyPgogICAgICA8L3RhYmxlPgogICAgPC90ZD4KICA8L3RyPgo8L3RhYmxlPg==
IyAtLQojIEFnZW50SVRTTVNlcnZpY2Vab29tLmR0bCAtIHByb3ZpZGVzIEhUTUwgZm9ybSBmb3IgQWdlbnRJVFNNU2VydmljZVpvb20KIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IEFnZW50SVRTTVNlcnZpY2Vab29tLmR0bCx2IDEuNSAyMDA5LzA4LzE4IDIyOjIwOjUyIG1oIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgo8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iMTAwJSIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIzIj4KICA8dHI+CiAgICA8dGQgY2xhc3M9Im1haW5oZWFkIj4KICAgICAgJEVudnsiQm94MCJ9JFRleHR7Ilpvb20ifTogJFRleHR7IlNlcnZpY2UifSRFbnZ7IkJveDEifQogICAgPC90ZD4KICA8L3RyPgo8L3RhYmxlPgo8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iMTAwJSIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIzIj4KICA8dHI+CiAgICA8dGQgd2lkdGg9Ijc1JSIgY2xhc3M9Im1lbnUiPgo8IS0tIGR0bDpibG9jazpNZW51IC0tPgo8IS0tIGR0bDpibG9jazpNZW51SXRlbVNwbGl0IC0tPgogICAgICAtCjwhLS0gZHRsOmJsb2NrOk1lbnVJdGVtU3BsaXQgLS0+CjwhLS0gZHRsOmJsb2NrOk1lbnVJdGVtIC0tPgogICAgICA8YSBocmVmPSIkRW52eyJCYXNlbGluayJ9JERhdGF7IkxpbmsifSIgJERhdGF7IkxpbmtQYXJhbSJ9IG9ubW91c2VvdmVyPSJ3aW5kb3cuc3RhdHVzPSckSlNUZXh0eyIkUURhdGF7Ik5hbWUifSJ9JzsgcmV0dXJuIHRydWU7IiBvbm1vdXNlb3V0PSJ3aW5kb3cuc3RhdHVzPScnOyIgY2xhc3M9Im1lbnVpdGVtIj4kVGV4dHsiJFFEYXRheyJOYW1lIn0ifTwvYT4KPCEtLSBkdGw6YmxvY2s6TWVudUl0ZW0gLS0+CjwhLS0gZHRsOmJsb2NrOk1lbnUgLS0+CiAgICA8L3RkPgogICAgPHRkIGFsaWduPSJyaWdodCIgd2lkdGg9IjI1JSIgY2xhc3M9Im1lbnUiPjwvdGQ+CiAgPC90cj4KPC90YWJsZT4KPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjEwMCUiIGNlbGxzcGFjaW5nPSIwIiBjZWxscGFkZGluZz0iMyI+CiAgPHRyPgogICAgPHRkIGNsYXNzPSJtYWluYm9keSIgYWxpZ249ImNlbnRlciI+CiAgICAgIDxicj4KICAgICAgPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjEwMCUiIGNlbGxzcGFjaW5nPSIwIiBjZWxscGFkZGluZz0iMSI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkIHdpZHRoPSIyMCUiIGNsYXNzPSJjb250ZW50a2V5Ij4kVGV4dHsiU2VydmljZSJ9OiA8L3RkPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiPiRRRGF0YXsiTmFtZSJ9PC90ZD4KICAgICAgICA8L3RyPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCB3aWR0aD0iMjAlIiBjbGFzcz0iY29udGVudGtleSI+JFRleHR7IlR5cGUifTogPC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj4kVGV4dHsiJFFEYXRheyJUeXBlIn0ifTwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgd2lkdGg9IjIwJSIgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJDcml0aWNhbGl0eSJ9OiA8L3RkPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiPiRUZXh0eyIkUURhdGF7IkNyaXRpY2FsaXR5In0ifTwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgPC90YWJsZT4KPCEtLSBkdGw6YmxvY2s6SG9yaXpvbnRhbFJ1bGVyIC0tPgogICAgICA8aHIgc2l6ZT0iMiI+CjwhLS0gZHRsOmJsb2NrOkhvcml6b250YWxSdWxlciAtLT4KICAgICAgPGJyPgo8IS0tIGR0bDpibG9jazpTTEEgLS0+CiAgICAgIDx0YWJsZSB3aWR0aD0iMTAwJSIgYm9yZGVyPSIwIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjAiPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCBhbGlnbj0iY2VudGVyIj4KICAgICAgICAgICAgPHRhYmxlIHdpZHRoPSI5OCUiIGJvcmRlcj0iMCIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIzIj4KICAgICAgICAgICAgICA8dHI+CiAgICAgICAgICAgICAgICA8dGQgY29sc3Bhbj0iNCI+CiAgICAgICAgICAgICAgICAgIDxiPjxpPiRUZXh0eyJBc3NvY2lhdGVkIFNMQXMifTwvaT48L2I+CiAgICAgICAgICAgICAgICA8L3RkPgogICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgICAgPHRyIGNsYXNzPSJjb250ZW50aGVhZCI+CiAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiIHdpZHRoPSIxMCI+Jm5ic3A7PC90ZD4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+JFRleHR7IlNMQSJ9PC90ZD4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+JFRleHR7IlR5cGUifTwvdGQ+CiAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiIHdpZHRoPSIxNTAiPiRUZXh0eyJDaGFuZ2VkIn08L3RkPgogICAgICAgICAgICAgIDwvdHI+CjwhLS0gZHRsOmJsb2NrOlNMQVJvdyAtLT4KICAgICAgICAgICAgICA8dHIgY2xhc3M9ImNvbnRlbnRib2R5Ij4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iJFFEYXRheyJDc3NDbGFzcyJ9Ij4mbmJzcDs8L3RkPgogICAgICAgICAgICAgICAgPHRkIGNsYXNzPSIkUURhdGF7IkNzc0NsYXNzIn0iPgogICAgICAgICAgICAgICAgICA8YSBocmVmPSIkRW52eyJCYXNlbGluayJ9QWN0aW9uPUFnZW50SVRTTVNMQVpvb20mU0xBSUQ9JFFEYXRheyJTTEFJRCJ9Ij4KICAgICAgICAgICAgICAgICAgJFFEYXRheyJOYW1lIn0KICAgICAgICAgICAgICAgICAgPC9hPgogICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iJFFEYXRheyJDc3NDbGFzcyJ9Ij4kVGV4dHsiJFFEYXRheyJUeXBlIn0ifTwvdGQ+CiAgICAgICAgICAgICAgICA8dGQgY2xhc3M9IiRRRGF0YXsiQ3NzQ2xhc3MifSIgd2lkdGg9IjE1MCI+JFRpbWVMb25neyIkUURhdGF7IkNoYW5nZVRpbWUifSJ9PC90ZD4KICAgICAgICAgICAgICA8L3RyPgo8IS0tIGR0bDpibG9jazpTTEFSb3cgLS0+CiAgICAgICAgICAgICAgPHRyIGNsYXNzPSJjb250ZW50Zm9vdGVyIj4KICAgICAgICAgICAgICAgIDx0ZCBjb2xzcGFuPSI0Ij4mbmJzcDs8L3RkPgogICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgIDwvdGFibGU+CiAgICAgICAgICA8L3RkPgogICAgICAgIDwvdHI+CiAgICAgIDwvdGFibGU+CiAgICAgIDxicj4KPCEtLSBkdGw6YmxvY2s6U0xBIC0tPgo8IS0tIGR0bDpibG9jazpMaW5rVGFibGVDb21wbGV4IC0tPgogICAgICA8dGFibGUgd2lkdGg9Ijk4JSIgYWxpZ249ImNlbnRlciIgYm9yZGVyPSIwIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjAiPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZD4KICAgICAgICAgICAgJERhdGF7IkxpbmtUYWJsZVN0cmcifQogICAgICAgICAgPC90ZD4KICAgICAgICA8L3RyPgogICAgICA8L3RhYmxlPgogICAgICA8YnI+CjwhLS0gZHRsOmJsb2NrOkxpbmtUYWJsZUNvbXBsZXggLS0+CiAgICA8L3RkPgogICAgPHRkIHdpZHRoPSIyNSUiIGNsYXNzPSJtZW51Ij4KICAgICAgPHRhYmxlIGJvcmRlcj0iMCIgY2VsbHNwYWNpbmc9IjEiIGNlbGxwYWRkaW5nPSIwIiB3aWR0aD0iMTAwJSI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkPjxiPiRUZXh0eyJDdXJyZW50IEluY2lkZW50IFN0YXRlIn06IDwvYj48L3RkPgogICAgICAgICAgPHRkPgogICAgICAgICAgICA8dGFibGUgYm9yZGVyPSIwIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjAiPgogICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgIDx0ZD4KICAgICAgICAgICAgICAgICAgPGRpdiB0aXRsZT0iJFRleHR7IiRRRGF0YXsiQ3VySW5jaVN0YXRlIn0ifSI+CiAgICAgICAgICAgICAgICAgICRRdW90ZXsiJFRleHR7IiREYXRheyJDdXJJbmNpU3RhdGUifSJ9IiwiMjUifSZuYnNwOyZuYnNwOwogICAgICAgICAgICAgICAgICA8L2Rpdj4KICAgICAgICAgICAgICAgIDwvdGQ+CiAgICAgICAgICAgICAgICA8dGQ+CiAgICAgICAgICAgICAgICAgIDxpbWcgYm9yZGVyPSIwIiBzcmM9IiRDb25maWd7IkZyb250ZW5kOjpJbWFnZVBhdGgifSRRRGF0YXsiQ3VySW5jaVNpZ25hbCJ9LnBuZyIgdGl0bGU9IiRUZXh0eyIkUURhdGF7IkN1ckluY2lTdGF0ZSJ9In0iIGFsdD0iJFRleHR7IiRRRGF0YXsiQ3VySW5jaVN0YXRlIn0ifSI+CiAgICAgICAgICAgICAgICA8L3RkPgogICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgIDwvdGFibGU+CiAgICAgICAgICA8L3RkPgogICAgICAgIDwvdHI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkIGNvbHNwYW49IjIiPiZuYnNwOzwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY2xhc3M9Im1haW5rZXkiPiRUZXh0eyJDcmVhdGVkIn06PC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0ibWFpbnZhbHVlIj4kVGltZUxvbmd7IiREYXRheyJDcmVhdGVUaW1lIn0ifTwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY2xhc3M9Im1haW5rZXkiPiRUZXh0eyJDcmVhdGVkIGJ5In06PC90ZD4KICAgICAgICAgIDx0ZD4KICAgICAgICAgICAgPGRpdiB0aXRsZT0iJFFEYXRheyJDcmVhdGVVc2VyTG9naW4ifSAoJFF1b3RleyIkRGF0YXsiQ3JlYXRlVXNlckZpcnN0bmFtZSJ9ICREYXRheyJDcmVhdGVVc2VyTGFzdG5hbWUifSJ9KSI+CiAgICAgICAgICAgICRRRGF0YXsiQ3JlYXRlVXNlckxvZ2luIiwiMTUifSAoJFF1b3RleyIkRGF0YXsiQ3JlYXRlVXNlckZpcnN0bmFtZSJ9ICREYXRheyJDcmVhdGVVc2VyTGFzdG5hbWUifSIsIjE1In0pCiAgICAgICAgICAgIDwvZGl2PgogICAgICAgICAgPC90ZD4KICAgICAgICA8L3RyPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCBjbGFzcz0ibWFpbmtleSI+JFRleHR7Ikxhc3QgY2hhbmdlZCJ9OjwvdGQ+CiAgICAgICAgICA8dGQgY2xhc3M9Im1haW52YWx1ZSI+JFRpbWVMb25neyIkRGF0YXsiQ2hhbmdlVGltZSJ9In08L3RkPgogICAgICAgIDwvdHI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkIGNsYXNzPSJtYWlua2V5Ij4kVGV4dHsiTGFzdCBjaGFuZ2VkIGJ5In06PC90ZD4KICAgICAgICAgIDx0ZD4KICAgICAgICAgICAgPGRpdiB0aXRsZT0iJFFEYXRheyJDaGFuZ2VVc2VyTG9naW4ifSAoJFF1b3RleyIkRGF0YXsiQ2hhbmdlVXNlckZpcnN0bmFtZSJ9ICREYXRheyJDaGFuZ2VVc2VyTGFzdG5hbWUifSJ9KSI+CiAgICAgICAgICAgICRRRGF0YXsiQ2hhbmdlVXNlckxvZ2luIiwiMTUifSAoJFF1b3RleyIkRGF0YXsiQ2hhbmdlVXNlckZpcnN0bmFtZSJ9ICREYXRheyJDaGFuZ2VVc2VyTGFzdG5hbWUifSIsIjE1In0pCiAgICAgICAgICAgIDwvZGl2PgogICAgICAgICAgPC90ZD4KICAgICAgICA8L3RyPgo8IS0tIGR0bDpibG9jazpMaW5rVGFibGVTaW1wbGUgLS0+CiAgICAgICAgJERhdGF7IkxpbmtUYWJsZVN0cmcifQo8IS0tIGR0bDpibG9jazpMaW5rVGFibGVTaW1wbGUgLS0+CiAgICAgIDwvdGFibGU+CiAgICA8L3RkPgogIDwvdHI+CjwvdGFibGU+Cg==
IyAtLQojIEFnZW50SVRTTVNMQS5kdGwgLSBwcm92aWRlcyBIVE1MIGZvcm0gZm9yIEFnZW50SVRTTVNMQQojIENvcHlyaWdodCAoQykgMjAwMS0yMDA5IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogQWdlbnRJVFNNU0xBLmR0bCx2IDEuMiAyMDA5LzA1LzE4IDA5OjQ4OjQ2IG1oIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgo8IS0tIGR0bDpibG9jazpPdmVydmlldyAtLT4KPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjEwMCUiIGNlbGxzcGFjaW5nPSIwIiBjZWxscGFkZGluZz0iMyI+CiAgPHRyPgogICAgPHRkIGNsYXNzPSJtYWluaGVhZCI+CiAgICAgICRFbnZ7IkJveDAifSRUZXh0eyJPdmVydmlldyJ9OiAkVGV4dHsiU0xBIn0kRW52eyJCb3gxIn0KICAgIDwvdGQ+CiAgPC90cj4KICA8dHI+CiAgICA8dGQgY2xhc3M9Im1haW5ib2R5IiBhbGlnbj0iY2VudGVyIj4KICAgICAgPGJyPgogICAgICA8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iODAwIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjMiPgogICAgICAgIDx0ciBjbGFzcz0iY29udGVudGhlYWQiPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50a2V5IiB3aWR0aD0iMTAiPiZuYnNwOzwvdGQ+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJTTEEifTwvdGQ+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJUeXBlIn08L3RkPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50a2V5IiB3aWR0aD0iMTUwIj4kVGV4dHsiQ2hhbmdlZCJ9PC90ZD4KICAgICAgICA8L3RyPgo8IS0tIGR0bDpibG9jazpPdmVydmlld1JvdyAtLT4KICAgICAgICA8dHIgY2xhc3M9ImNvbnRlbnRib2R5Ij4KICAgICAgICAgIDx0ZCBjbGFzcz0iJFFEYXRheyJDc3NDbGFzcyJ9Ij4mbmJzcDs8L3RkPgogICAgICAgICAgPHRkIGNsYXNzPSIkUURhdGF7IkNzc0NsYXNzIn0iPgogICAgICAgICAgICA8YSBocmVmPSIkRW52eyJCYXNlbGluayJ9QWN0aW9uPUFnZW50SVRTTVNMQVpvb20mU0xBSUQ9JFFEYXRheyJTTEFJRCJ9Ij4KICAgICAgICAgICAgJFFEYXRheyJOYW1lIn0KICAgICAgICAgICAgPC9hPgogICAgICAgICAgPC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iJFFEYXRheyJDc3NDbGFzcyJ9Ij4kVGV4dHsiJFFEYXRheyJUeXBlIn0ifTwvdGQ+CiAgICAgICAgICA8dGQgY2xhc3M9IiRRRGF0YXsiQ3NzQ2xhc3MifSIgd2lkdGg9IjE1MCI+JFRpbWVMb25neyIkUURhdGF7IkNoYW5nZVRpbWUifSJ9PC90ZD4KICAgICAgICA8L3RyPgo8IS0tIGR0bDpibG9jazpPdmVydmlld1JvdyAtLT4KICAgICAgICA8dHIgY2xhc3M9ImNvbnRlbnRmb290ZXIiPgogICAgICAgICAgPHRkIGNvbHNwYW49IjQiPiZuYnNwOzwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgPC90YWJsZT4KICAgICAgPGJyPgogICAgPC90ZD4KICA8L3RyPgo8L3RhYmxlPgo8IS0tIGR0bDpibG9jazpPdmVydmlldyAtLT4K
IyAtLQojIEFnZW50SVRTTVNMQVByaW50LmR0bCAtIHByb3ZpZGVzIEhUTUwgcHJpbnQgdmlldwojIENvcHlyaWdodCAoQykgMjAwMS0yMDA5IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogQWdlbnRJVFNNU0xBUHJpbnQuZHRsLHYgMS4yIDIwMDkvMDUvMTggMDk6NDg6NDYgbWggRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCjx0YWJsZSBib3JkZXI9IjAiIGNlbGxzcGFjaW5nPSIxIiBjZWxscGFkZGluZz0iMCIgd2lkdGg9IjEwMCUiPgogIDx0cj4KICAgIDx0ZCBjbGFzcz0ibWFpbmhlYWQiPiRUZXh0eyJTTEEifTwvdGQ+CiAgPC90cj4KPC90YWJsZT4KPHRhYmxlIGJvcmRlcj0iMCIgY2VsbHNwYWNpbmc9IjEiIGNlbGxwYWRkaW5nPSIxMCIgd2lkdGg9IjEwMCUiPgogIDx0cj4KICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+CiAgICAgICRUZXh0eyJTTEEtSW5mbyJ9OgogICAgICA8dGFibGUgYm9yZGVyPSIwIiBjZWxsc3BhY2luZz0iMyIgY2VsbHBhZGRpbmc9IjAiIHdpZHRoPSIxMDAlIj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiIHdpZHRoPSIyMCUiPiRUZXh0eyJTTEEifTo8L3RkPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiIHdpZHRoPSIzMCUiPiRRRGF0YXsiTmFtZSJ9PC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSIgd2lkdGg9IjIwJSI+JFRleHR7IkNyZWF0ZWQifTo8L3RkPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiIHdpZHRoPSIzMCUiPiRUaW1lTG9uZ3siJERhdGF7IkNyZWF0ZVRpbWUifSJ9PC90ZD4KICAgICAgICA8L3RyPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+PC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj48L3RkPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50a2V5Ij4kVGV4dHsiQ3JlYXRlZCBieSJ9OjwvdGQ+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+JFFEYXRheyJDcmVhdGVCeVVzZXJMb2dpbiJ9ICgkUURhdGF7IkNyZWF0ZUJ5VXNlckZpcnN0bmFtZSJ9ICRRRGF0YXsiQ3JlYXRlQnlVc2VyTGFzdG5hbWUifSk8L3RkPgogICAgICAgIDwvdHI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50a2V5Ij48L3RkPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiPjwvdGQ+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJMYXN0IGNoYW5nZWQifTo8L3RkPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiPiRUaW1lTG9uZ3siJERhdGF7IkNoYW5nZVRpbWUifSJ9PC90ZD4KICAgICAgICA8L3RyPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+PC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj48L3RkPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50a2V5Ij4kVGV4dHsiTGFzdCBjaGFuZ2VkIGJ5In06PC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj4kUURhdGF7IkNoYW5nZUJ5VXNlckxvZ2luIn0gKCRRRGF0YXsiQ2hhbmdlQnlVc2VyRmlyc3RuYW1lIn0gJFFEYXRheyJDaGFuZ2VCeVVzZXJMYXN0bmFtZSJ9KTwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiPiZuYnNwOzwvdGQ+CiAgICAgICAgICA8dGQgY29sc3Bhbj0iMyIgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+Jm5ic3A7PC90ZD4KICAgICAgICA8L3RyPgogICAgICA8L3RhYmxlPgogICAgPC90ZD4KICA8L3RyPgo8L3RhYmxlPgo8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iNjQwIiBjZWxscGFkZGluZz0iMCIgY2VsbHNwYWNpbmc9IjAiIGJnY29sb3I9IiMwMDAwMDAiIGNsYXNzPSJjb250YWluZXIiPgogIDx0cj4KICAgIDx0ZD4KICAgICAgPHRhYmxlIGJvcmRlcj0iMCIgd2lkdGg9IjY0MCIgY2VsbHBhZGRpbmc9IjEwIiBjZWxsc3BhY2luZz0iMSIgYmdjb2xvcj0iI2ZmZmZmZiI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkPgogICAgICAgICAgICA8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iMTAwJSIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIzIj4KICAgICAgICAgICAgICA8dHIgYmdjb2xvcj0iI2ZmZmZmZiI+CiAgICAgICAgICAgICAgICA8dGQgdmFsaWduPSJ0b3AiIHdpZHRoPSIxMDAlIiBjbGFzcz0ic21hbGwiPgogICAgICAgICAgICAgICAgICA8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iMTAwJSIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIxIj4KICAgICAgICAgICAgICAgICAgICA8dHI+CiAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJTTEEifTo8L3RkPgogICAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiPiRRRGF0YXsiTmFtZSJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+JFRleHR7IlR5cGUifTo8L3RkPgogICAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiPiRUZXh0eyIkUURhdGF7IlR5cGUifSJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+JFRleHR7IkNhbGVuZGFyIn06PC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj4kVGV4dHsiJFFEYXRheyJDYWxlbmRhck5hbWUifSJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+JFRleHR7IkZpcnN0IFJlc3BvbnNlIFRpbWUifTo8L3RkPgogICAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiPiRRRGF0YXsiRmlyc3RSZXNwb25zZVRpbWUifSAkVGV4dHsibWludXRlcyJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+JFRleHR7IlVwZGF0ZSBUaW1lIn06PC90ZD4KICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj4kUURhdGF7IlVwZGF0ZVRpbWUifSAkVGV4dHsibWludXRlcyJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICAgICAgICAgIDx0cj4KICAgICAgICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+JFRleHR7IlNvbHV0aW9uIFRpbWUifTo8L3RkPgogICAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiPiRRRGF0YXsiU29sdXRpb25UaW1lIn0gJFRleHR7Im1pbnV0ZXMifTwvdGQ+CiAgICAgICAgICAgICAgICAgICAgPC90cj4KICAgICAgICAgICAgICAgICAgICA8dHI+CiAgICAgICAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJNaW5pbXVtIFRpbWUgQmV0d2VlbiBJbmNpZGVudHMifTo8L3RkPgogICAgICAgICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiPiRRRGF0YXsiTWluVGltZUJldHdlZW5JbmNpZGVudHMifSAkVGV4dHsibWludXRlcyJ9PC90ZD4KICAgICAgICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICAgICAgICA8L3RhYmxlPgogICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICA8L3RhYmxlPgogICAgICAgICAgPC90ZD4KICAgICAgICA8L3RyPgogICAgICA8L3RhYmxlPgogICAgPC90ZD4KICA8L3RyPgo8L3RhYmxlPg==
IyAtLQojIEFnZW50SVRTTVNMQVpvb20uZHRsIC0gcHJvdmlkZXMgSFRNTCBmb3JtIGZvciBBZ2VudElUU01TTEFab29tCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDkgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBBZ2VudElUU01TTEFab29tLmR0bCx2IDEuNCAyMDA5LzA4LzE4IDIyOjIwOjUyIG1oIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgo8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iMTAwJSIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIzIj4KICA8dHI+CiAgICA8dGQgY2xhc3M9Im1haW5oZWFkIj4KICAgICAgJEVudnsiQm94MCJ9JFRleHR7Ilpvb20ifTogJFRleHR7IlNMQSJ9JEVudnsiQm94MSJ9CiAgICA8L3RkPgogIDwvdHI+CjwvdGFibGU+Cjx0YWJsZSBib3JkZXI9IjAiIHdpZHRoPSIxMDAlIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjMiPgogIDx0cj4KICAgIDx0ZCB3aWR0aD0iNzUlIiBjbGFzcz0ibWVudSI+CjwhLS0gZHRsOmJsb2NrOk1lbnUgLS0+CjwhLS0gZHRsOmJsb2NrOk1lbnVJdGVtU3BsaXQgLS0+CiAgICAgIC0KPCEtLSBkdGw6YmxvY2s6TWVudUl0ZW1TcGxpdCAtLT4KPCEtLSBkdGw6YmxvY2s6TWVudUl0ZW0gLS0+CiAgICAgIDxhIGhyZWY9IiRFbnZ7IkJhc2VsaW5rIn0kRGF0YXsiTGluayJ9IiAkRGF0YXsiTGlua1BhcmFtIn0gb25tb3VzZW92ZXI9IndpbmRvdy5zdGF0dXM9JyRKU1RleHR7IiRRRGF0YXsiTmFtZSJ9In0nOyByZXR1cm4gdHJ1ZTsiIG9ubW91c2VvdXQ9IndpbmRvdy5zdGF0dXM9Jyc7IiBjbGFzcz0ibWVudWl0ZW0iPiRUZXh0eyIkUURhdGF7Ik5hbWUifSJ9PC9hPgo8IS0tIGR0bDpibG9jazpNZW51SXRlbSAtLT4KPCEtLSBkdGw6YmxvY2s6TWVudSAtLT4KICAgIDwvdGQ+CiAgICA8dGQgYWxpZ249InJpZ2h0IiB3aWR0aD0iMjUlIiBjbGFzcz0ibWVudSI+PC90ZD4KICA8L3RyPgo8L3RhYmxlPgo8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iMTAwJSIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIzIj4KICA8dHI+CiAgICA8dGQgY2xhc3M9Im1haW5ib2R5IiBhbGlnbj0iY2VudGVyIj4KICAgICAgPGJyPgogICAgICA8dGFibGUgYm9yZGVyPSIwIiB3aWR0aD0iMTAwJSIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIxIj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgd2lkdGg9IjIwJSIgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJTTEEifTogPC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj4kUURhdGF7Ik5hbWUifTwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgd2lkdGg9IjIwJSIgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJUeXBlIn06IDwvdGQ+CiAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnR2YWx1ZSI+JFRleHR7IiRRRGF0YXsiVHlwZSJ9In08L3RkPgogICAgICAgIDwvdHI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkIHdpZHRoPSIyMCUiIGNsYXNzPSJjb250ZW50a2V5Ij4kVGV4dHsiQ2FsZW5kYXIifTogPC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj4kVGV4dHsiJFFEYXRheyJDYWxlbmRhck5hbWUifSJ9PC90ZD4KICAgICAgICA8L3RyPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCB3aWR0aD0iMjAlIiBjbGFzcz0iY29udGVudGtleSI+JFRleHR7IkZpcnN0IFJlc3BvbnNlIFRpbWUifTogPC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj4kUURhdGF7IkZpcnN0UmVzcG9uc2VUaW1lIn0gJFRleHR7Im1pbnV0ZXMifTwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgd2lkdGg9IjIwJSIgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJVcGRhdGUgVGltZSJ9OiA8L3RkPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiPiRRRGF0YXsiVXBkYXRlVGltZSJ9ICRUZXh0eyJtaW51dGVzIn08L3RkPgogICAgICAgIDwvdHI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkIHdpZHRoPSIyMCUiIGNsYXNzPSJjb250ZW50a2V5Ij4kVGV4dHsiU29sdXRpb24gVGltZSJ9OiA8L3RkPgogICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50dmFsdWUiPiRRRGF0YXsiU29sdXRpb25UaW1lIn0gJFRleHR7Im1pbnV0ZXMifTwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgd2lkdGg9IjIwJSIgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJNaW5pbXVtIFRpbWUgQmV0d2VlbiBJbmNpZGVudHMifTogPC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudHZhbHVlIj4kUURhdGF7Ik1pblRpbWVCZXR3ZWVuSW5jaWRlbnRzIn0gJFRleHR7Im1pbnV0ZXMifTwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgPC90YWJsZT4KPCEtLSBkdGw6YmxvY2s6U2VydmljZSAtLT4KICAgICAgPGhyIHNpemU9IjIiPgogICAgICA8YnI+CiAgICAgIDx0YWJsZSB3aWR0aD0iMTAwJSIgYm9yZGVyPSIwIiBjZWxsc3BhY2luZz0iMCIgY2VsbHBhZGRpbmc9IjAiPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCBhbGlnbj0iY2VudGVyIj4KICAgICAgICAgICAgPHRhYmxlIHdpZHRoPSI5OCUiIGJvcmRlcj0iMCIgY2VsbHNwYWNpbmc9IjAiIGNlbGxwYWRkaW5nPSIzIj4KICAgICAgICAgICAgICA8dHI+CiAgICAgICAgICAgICAgICA8dGQgY29sc3Bhbj0iNCI+CiAgICAgICAgICAgICAgICAgIDxiPjxpPiRUZXh0eyJBc3NvY2lhdGVkIFNlcnZpY2VzIn08L2k+PC9iPgogICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICA8L3RyPgogICAgICAgICAgICAgIDx0ciBjbGFzcz0iY29udGVudGhlYWQiPgogICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50a2V5IiB3aWR0aD0iMjAiPiZuYnNwOzwvdGQ+CiAgICAgICAgICAgICAgICA8dGQgY2xhc3M9ImNvbnRlbnRrZXkiPiRUZXh0eyJTZXJ2aWNlIn08L3RkPgogICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50a2V5Ij4kVGV4dHsiVHlwZSJ9PC90ZD4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iY29udGVudGtleSI+JFRleHR7IkNyaXRpY2FsaXR5In08L3RkPgogICAgICAgICAgICAgICAgPHRkIGNsYXNzPSJjb250ZW50a2V5IiB3aWR0aD0iMTUwIj4kVGV4dHsiQ2hhbmdlZCJ9PC90ZD4KICAgICAgICAgICAgICA8L3RyPgo8IS0tIGR0bDpibG9jazpTZXJ2aWNlUm93IC0tPgogICAgICAgICAgICAgIDx0ciBjbGFzcz0iY29udGVudGJvZHkiPgogICAgICAgICAgICAgICAgPHRkIGNsYXNzPSIkUURhdGF7IkNzc0NsYXNzIn0iIGFsaWduPSJyaWdodCI+CiAgICAgICAgICAgICAgICAgIDxpbWcgYm9yZGVyPSIwIiBzcmM9IiRDb25maWd7IkZyb250ZW5kOjpJbWFnZVBhdGgifSRRRGF0YXsiQ3VySW5jaVNpZ25hbCJ9LnBuZyIgdGl0bGU9IiRRdW90ZXsiJFRleHR7IiREYXRheyJDdXJJbmNpU3RhdGUifSJ9In0iIGFsdD0iJFF1b3RleyIkVGV4dHsiJERhdGF7IkN1ckluY2lTdGF0ZSJ9In0ifSI+CiAgICAgICAgICAgICAgICA8L3RkPgogICAgICAgICAgICAgICAgPHRkIGNsYXNzPSIkUURhdGF7IkNzc0NsYXNzIn0iPgogICAgICAgICAgICAgICAgICA8YSBocmVmPSIkRW52eyJCYXNlbGluayJ9QWN0aW9uPUFnZW50SVRTTVNlcnZpY2Vab29tJlNlcnZpY2VJRD0kUURhdGF7IlNlcnZpY2VJRCJ9Ij4KICAgICAgICAgICAgICAgICAgJFFEYXRheyJOYW1lIn0KICAgICAgICAgICAgICAgICAgPC9hPgogICAgICAgICAgICAgICAgPC90ZD4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iJFFEYXRheyJDc3NDbGFzcyJ9Ij4kVGV4dHsiJFFEYXRheyJUeXBlIn0ifTwvdGQ+CiAgICAgICAgICAgICAgICA8dGQgY2xhc3M9IiRRRGF0YXsiQ3NzQ2xhc3MifSI+JFRleHR7IiRRRGF0YXsiQ3JpdGljYWxpdHkifSJ9PC90ZD4KICAgICAgICAgICAgICAgIDx0ZCBjbGFzcz0iJFFEYXRheyJDc3NDbGFzcyJ9IiB3aWR0aD0iMTUwIj4kVGltZUxvbmd7IiRRRGF0YXsiQ2hhbmdlVGltZSJ9In08L3RkPgogICAgICAgICAgICAgIDwvdHI+CjwhLS0gZHRsOmJsb2NrOlNlcnZpY2VSb3cgLS0+CiAgICAgICAgICAgICAgPHRyIGNsYXNzPSJjb250ZW50Zm9vdGVyIj4KICAgICAgICAgICAgICAgIDx0ZCBjb2xzcGFuPSI1Ij4mbmJzcDs8L3RkPgogICAgICAgICAgICAgIDwvdHI+CiAgICAgICAgICAgIDwvdGFibGU+CiAgICAgICAgICA8L3RkPgogICAgICAgIDwvdHI+CiAgICAgIDwvdGFibGU+CjwhLS0gZHRsOmJsb2NrOlNlcnZpY2UgLS0+CiAgICAgIDxicj4KICAgIDwvdGQ+CiAgICA8dGQgd2lkdGg9IjI1JSIgY2xhc3M9Im1lbnUiPgogICAgICA8dGFibGUgYm9yZGVyPSIwIiBjZWxsc3BhY2luZz0iMSIgY2VsbHBhZGRpbmc9IjAiIHdpZHRoPSIxMDAlIj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY2xhc3M9Im1haW5rZXkiPiRUZXh0eyJDcmVhdGVkIn06PC90ZD4KICAgICAgICAgIDx0ZCBjbGFzcz0ibWFpbnZhbHVlIj4kVGltZUxvbmd7IiREYXRheyJDcmVhdGVUaW1lIn0ifTwvdGQ+CiAgICAgICAgPC90cj4KICAgICAgICA8dHI+CiAgICAgICAgICA8dGQgY2xhc3M9Im1haW5rZXkiPiRUZXh0eyJDcmVhdGVkIGJ5In06PC90ZD4KICAgICAgICAgIDx0ZD4KICAgICAgICAgICAgPGRpdiB0aXRsZT0iJFFEYXRheyJDcmVhdGVVc2VyTG9naW4ifSAoJFF1b3RleyIkRGF0YXsiQ3JlYXRlVXNlckZpcnN0bmFtZSJ9ICREYXRheyJDcmVhdGVVc2VyTGFzdG5hbWUifSJ9KSI+CiAgICAgICAgICAgICRRRGF0YXsiQ3JlYXRlVXNlckxvZ2luIiwiMTUifSAoJFF1b3RleyIkRGF0YXsiQ3JlYXRlVXNlckZpcnN0bmFtZSJ9ICREYXRheyJDcmVhdGVVc2VyTGFzdG5hbWUifSIsIjE1In0pCiAgICAgICAgICAgIDwvZGl2PgogICAgICAgICAgPC90ZD4KICAgICAgICA8L3RyPgogICAgICAgIDx0cj4KICAgICAgICAgIDx0ZCBjbGFzcz0ibWFpbmtleSI+JFRleHR7Ikxhc3QgY2hhbmdlZCJ9OjwvdGQ+CiAgICAgICAgICA8dGQgY2xhc3M9Im1haW52YWx1ZSI+JFRpbWVMb25neyIkRGF0YXsiQ2hhbmdlVGltZSJ9In08L3RkPgogICAgICAgIDwvdHI+CiAgICAgICAgPHRyPgogICAgICAgICAgPHRkIGNsYXNzPSJtYWlua2V5Ij4kVGV4dHsiTGFzdCBjaGFuZ2VkIGJ5In06PC90ZD4KICAgICAgICAgIDx0ZD4KICAgICAgICAgICAgPGRpdiB0aXRsZT0iJFFEYXRheyJDaGFuZ2VVc2VyTG9naW4ifSAoJFF1b3RleyIkRGF0YXsiQ2hhbmdlVXNlckZpcnN0bmFtZSJ9ICREYXRheyJDaGFuZ2VVc2VyTGFzdG5hbWUifSJ9KSI+CiAgICAgICAgICAgICRRRGF0YXsiQ2hhbmdlVXNlckxvZ2luIiwiMTUifSAoJFF1b3RleyIkRGF0YXsiQ2hhbmdlVXNlckZpcnN0bmFtZSJ9ICREYXRheyJDaGFuZ2VVc2VyTGFzdG5hbWUifSIsIjE1In0pCiAgICAgICAgICAgIDwvZGl2PgogICAgICAgICAgPC90ZD4KICAgICAgICA8L3RyPgogICAgICA8L3RhYmxlPgogICAgPC90ZD4KICA8L3RyPgo8L3RhYmxlPgo=
IyAtLQojIFJpY2hUZXh0RWRpdG9ySVRTTS5kdGwgLSBwcm92aWRlcyBSaWNoIFRleHQgRWRpdG9yCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTAgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBSaWNoVGV4dEVkaXRvcklUU00uZHRsLHYgMS40IDIwMTAvMDUvMjYgMTc6MDg6MzIgdWIgRXhwICQKIyAkT2xkSWQ6IFJpY2hUZXh0RWRpdG9yLmR0bCx2IDEuMjAuMi4xMSAyMDEwLzA1LzI2IDEyOjQ2OjMxIHViIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgojIGxvYWQganMKPHNjcmlwdCBzcmM9IiRDb25maWd7IkZyb250ZW5kOjpKYXZhU2NyaXB0UGF0aCJ9ZmNrZWRpdG9yLTIuNi40LjEvZmNrZWRpdG9yLmpzIj48L3NjcmlwdD4KCjxzY3JpcHQgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij4KLy8gdXBkYXRlIGJvZHkgd2l0aCBjdXJyZW50IHJpY2ggdGV4dCBjb250ZW50CiMgLS0tCiMgSVRTTQojIC0tLQojZnVuY3Rpb24gUmljaFRleHRVcGRhdGVMaW5rZWRGaWVsZCgpIHsKIyAgICBGQ0tlZGl0b3JBUEkuR2V0SW5zdGFuY2UoJ1JpY2hUZXh0JykuVXBkYXRlTGlua2VkRmllbGQoKTsKZnVuY3Rpb24gUmljaFRleHRVcGRhdGVMaW5rZWRGaWVsZCggRmllbGRuYW1lICkgewogICAgRkNLZWRpdG9yQVBJLkdldEluc3RhbmNlKCBGaWVsZG5hbWUgKS5VcGRhdGVMaW5rZWRGaWVsZCgpOwojIC0tLQp9CgovLyBzZXQgZm9jdXMgdG8gcmljaCB0ZXh0IGVkaXRvcgojIC0tLQojIElUU00KIyAtLS0KI2Z1bmN0aW9uIFJpY2hUZXh0Rm9jdXMoKSB7CiMgICAgRkNLZWRpdG9yQVBJLkdldEluc3RhbmNlKCdSaWNoVGV4dCcpLkZvY3VzKCk7CmZ1bmN0aW9uIFJpY2hUZXh0Rm9jdXMoIEZpZWxkbmFtZSApIHsKICAgIEZDS2VkaXRvckFQSS5HZXRJbnN0YW5jZSggRmllbGRuYW1lICkuRm9jdXMoKTsKIyAtLS0KfQoKLy8gY2xlYW51cCByZWFsIGJvZHkKLy8gSWYgdGhlIGJvZHkgb25seSBjb250YWlucyB3aGl0ZXNwYWNlIG9yIGh0bWwgdGFncyBidXQgbm8gdGV4dCwKLy8gcmVzZXQgaXQgdG8gJycuIFRoaXMgaXMgbmVlZGVkIGZvciB0aGUgZnJvbnRlbmQgYW5kIGJhY2tlbmQKLy8gY29udGVudCBjaGVja3MgdG8gd29yayBwcm9wZXJseS4KZnVuY3Rpb24gUmljaFRleHRDbGVhbnVwKCkgewojIC0tLQojIElUU00KIyAtLS0KIyAgICB2YXIgRWxlbSA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdSaWNoVGV4dCcpOwogICAgZm9yICggdmFyIGkgPSAxOyBpIDw9IDY7IGkrKyApIHsKICAgICAgICB2YXIgRWxlbSA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdSaWNoVGV4dCcgKyBpKTsKIyAtLS0KICAgIGlmIChFbGVtKSB7CiAgICAgICAgdmFyIFN0cmlwcGVkQ29udGVudCA9IEVsZW0udmFsdWUucmVwbGFjZSgvXHMrfCZuYnNwO3w8XC8/XHcrXHM/XC8/Pi9nLCAnJyk7CiAgICAgICAgaWYgKFN0cmlwcGVkQ29udGVudC5sZW5ndGggPT0gMCkgewogICAgICAgICAgICBFbGVtLnZhbHVlID0gJyc7CiAgICAgICAgfQogICAgfQojIC0tLQojIElUU00KIyAtLS0KICAgIH0KIyAtLS0KfQoKIyAtLS0KIyBJVFNNCiMgLS0tCiNmdW5jdGlvbiBJbml0UmljaFRleHQoKSB7CiMgICAgdmFyIG9GQ0tlZGl0b3IgPSBuZXcgRkNLZWRpdG9yKCAnUmljaFRleHQnICk7CmZ1bmN0aW9uIEluaXRSaWNoVGV4dElUU00oIEZpZWxkbmFtZSApIHsKICAgIHZhciBvRkNLZWRpdG9yID0gbmV3IEZDS2VkaXRvciggRmllbGRuYW1lICk7CiMgLS0tCiAgICBvRkNLZWRpdG9yLkhlaWdodCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnJENvbmZpZ3siRnJvbnRlbmQ6OlJpY2hUZXh0SGVpZ2h0In0nIHx8IDMyMDsKICAgIG9GQ0tlZGl0b3IuV2lkdGggICAgICAgICAgICAgICAgICAgICAgICAgICAgICA9ICckQ29uZmlneyJGcm9udGVuZDo6UmljaFRleHRXaWR0aCJ9JyB8fCA2MjA7CiAgICBvRkNLZWRpdG9yLkJhc2VQYXRoICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAnJENvbmZpZ3siRnJvbnRlbmQ6OkphdmFTY3JpcHRQYXRoIn0vZmNrZWRpdG9yLTIuNi40LjEvJzsKICAgIG9GQ0tlZGl0b3IuQ29uZmlnWyJGb3JjZVBhc3RlQXNQbGFpblRleHQiXSAgICA9IGZhbHNlOwogICAgb0ZDS2VkaXRvci5Db25maWdbIkVkaXRvckFyZWFTdHlsZXMiXSAgICAgICAgID0gJ2JvZHkgeyAkQ29uZmlneyJGcm9udGVuZDo6UmljaFRleHQ6OkRlZmF1bHRDU1MifSB9JzsKICAgIG9GQ0tlZGl0b3IuQ29uZmlnWyJBdXRvRGV0ZWN0TGFuZ3VhZ2UiXSAgICAgICA9IGZhbHNlOwoKICAgIC8vIEZpeCBmb3IgYnVnIzUzODMgLSBsYW5ndWFnZSBzdHJpbmdzIHN1Y2ggYXMgcHRfQlIgc2hvdWxkIGJlIHJlZm9ybWF0dGVkIGFzIHB0LWJyOgogICAgb0ZDS2VkaXRvci5Db25maWdbIkRlZmF1bHRMYW5ndWFnZSJdICAgICAgICAgID0gU3RyaW5nKCckRW52eyJVc2VyTGFuZ3VhZ2UifScpLnRvTG93ZXJDYXNlKCkucmVwbGFjZSgiXyIsICItIik7CgogICAgb0ZDS2VkaXRvci5Db25maWdbIkltYWdlVXBsb2FkVVJMIl0gICAgICAgICAgID0gJyRFbnZ7IkJhc2VsaW5rIn0nOwogICAgb0ZDS2VkaXRvci5Db25maWdbIkZpcmVmb3hTcGVsbENoZWNrZXIiXSAgICAgID0gdHJ1ZTsKICAgIG9GQ0tlZGl0b3IuQ29uZmlnWyJCcm93c2VyQ29udGV4dE1lbnUiXSAgICAgICA9IHRydWU7CiAgICBvRkNLZWRpdG9yLkNvbmZpZ1siU3BlbGxDaGVja2VyIl0gICAgICAgICAgICAgPSAnU3BlbGxlclBhZ2VzJzsKICAgIG9GQ0tlZGl0b3IuQ29uZmlnWyJTcGVsbGVyUGFnZXNTZXJ2ZXJTY3JpcHQiXSA9ICckRW52eyJCYXNlbGluayJ9JzsKICAgIG9GQ0tlZGl0b3IuQ29uZmlnWyJGb250U2l6ZXMiXSAgICAgICAgICAgICAgICA9ICc4cHg7MTBweDsxMnB4OzE2cHg7MThweDsyMHB4OzIycHg7MjRweDsyNnB4OzI4cHg7MzBweDsnOwogICAgb0ZDS2VkaXRvci5Db25maWdbIkVudGVyTW9kZSJdICAgICAgICAgICAgICAgID0gJ2JyJzsKICAgIG9GQ0tlZGl0b3IuQ29uZmlnWyJTaGlmdEVudGVyTW9kZSJdICAgICAgICAgICA9ICdicic7CiAgICBvRkNLZWRpdG9yLkNvbmZpZ1siTGlua1VwbG9hZCJdICAgICAgICAgICAgICAgPSBmYWxzZTsKCiAgICBpZiAoICckRW52eyJUZXh0RGlyZWN0aW9uIn0nICkgewogICAgICAgIG9GQ0tlZGl0b3IuQ29uZmlnWyJDb250ZW50TGFuZ0RpcmVjdGlvbiJdID0gJyRFbnZ7IlRleHREaXJlY3Rpb24ifSc7CiAgICB9CgogICAgdmFyIFRvb2xiYXJTZXQgPSAnJzsKICAgIGlmICggZG9jdW1lbnQuY29tcG9zZSAmJiBkb2N1bWVudC5jb21wb3NlLkZvcm1JRCApIHsKICAgICAgICBvRkNLZWRpdG9yLkNvbmZpZ1siQWN0aW9uIl0gPSAnUGljdHVyZVVwbG9hZCc7CiAgICAgICAgb0ZDS2VkaXRvci5Db25maWdbIkZvcm1JRCJdID0gZG9jdW1lbnQuY29tcG9zZS5Gb3JtSUQudmFsdWU7CiAgICAgICAgVG9vbGJhclNldCA9ICdGdWxsJzsKICAgICAgICBpZiAoICckRW52eyJCcm93c2VyU3BlbGxDaGVja2VySW5saW5lIn0nICkgewogICAgICAgICAgICBUb29sYmFyU2V0ID0gJ0Z1bGxTcGVsbCc7CiAgICAgICAgfQogICAgfQogICAgZWxzZSB7CiAgICAgICAgVG9vbGJhclNldCA9ICdTaW1wbGUnOwogICAgICAgIGlmICggJyRFbnZ7IkJyb3dzZXJTcGVsbENoZWNrZXJJbmxpbmUifScgKSB7CiAgICAgICAgICAgIFRvb2xiYXJTZXQgPSAnU2ltcGxlU3BlbGwnOwogICAgICAgIH0KICAgIH0KICAgIG9GQ0tlZGl0b3IuVG9vbGJhclNldCA9IFRvb2xiYXJTZXQ7CgogICAgb0ZDS2VkaXRvci5SZXBsYWNlVGV4dGFyZWEoKTsKfQoKLy8gcHJlc2VydmUgb2xkIHdpbmRvdyBvbmxvYWQgZXZlbnQKdmFyIE9sZE9uTG9hZCA9IHdpbmRvdy5vbmxvYWQ7CndpbmRvdy5vbmxvYWQgPSBmdW5jdGlvbigpIHsKICAgIGlmICh0eXBlb2YgT2xkT25Mb2FkID09ICdmdW5jdGlvbicpIHsKICAgICAgICBPbGRPbkxvYWQoKTsKICAgIH0KIyAtLS0KIyBJVFNNCiMgLS0tCiMgICAgSW5pdFJpY2hUZXh0KCk7CiAgICBmb3IgKCB2YXIgaSA9IDE7IGkgPD0gNjsgaSsrICkgewogICAgICAgIGlmICggZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ1JpY2hUZXh0JyArIGkpICkgewogICAgICAgICAgICB2YXIgbXlFZGl0b3IgPSBJbml0UmljaFRleHRJVFNNKCdSaWNoVGV4dCcgKyBpKTsKICAgICAgICB9CiAgICB9CiMgLS0tCn0KCmZ1bmN0aW9uIEZDS2VkaXRvcl9PbkNvbXBsZXRlKCBlZGl0b3JJbnN0YW5jZSApIHsKICAgIC8vIHdvcmthcm91bmQgZm9yIGh0dHA6Ly9kZXYuZmNrZWRpdG9yLm5ldC90aWNrZXQvMzA1MwogICAgLy8gc2VlIGFsc28gaHR0cDovL3d3dy5mY2tlZGl0b3IubmV0L2ZvcnVtcy92aWV3dG9waWMucGhwP2Y9NiZ0PTEzMjIyCiAgICB2YXIgZmNrX2lmcmFtZSA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdSaWNoVGV4dF9fX0ZyYW1lJyk7CiAgICBpZiAoIGZja19pZnJhbWUgJiYgZmNrX2lmcmFtZS5jb250ZW50RG9jdW1lbnQgKSB7CiAgICAgICAgdmFyIGZja19lZGl0aW5nX2FyZWEgPSBmY2tfaWZyYW1lLmNvbnRlbnREb2N1bWVudC5nZXRFbGVtZW50QnlJZCgneEVkaXRpbmdBcmVhJyk7CiAgICAgICAgZmNrX2VkaXRpbmdfYXJlYS5zdHlsZS5oZWlnaHQgPSAnMTAwLjElJzsKICAgICAgICBzZXRUaW1lb3V0KCBmdW5jdGlvbigpIHsKICAgICAgICAgICAgZmNrX2VkaXRpbmdfYXJlYS5zdHlsZS5oZWlnaHQgPSAnMTAwJScKICAgICAgICB9LCAxMDAgKTsKICAgIH0KCiAgICAvLyBjbGVhbnVwIGhpZGRlbiBib2R5IGZpZWxkIGFmdGVyIHVwZGF0ZSB3aXRoIGNvbnRlbnQgb2YgcmljaCB0ZXh0CiAgICBlZGl0b3JJbnN0YW5jZS5FdmVudHMuQXR0YWNoRXZlbnQoICdPbkFmdGVyTGlua2VkRmllbGRVcGRhdGUnLCBSaWNoVGV4dENsZWFudXAgKTsKfQo8L3NjcmlwdD4K
# --
# Kernel/System/EventHandler.pm - global object events
# Copyright (C) 2001-2010 OTRS AG, http://otrs.org/
# --
# $Id: EventHandler.pm,v 1.2 2010/02/04 13:19:13 ub Exp $
# $OldId: EventHandler.pm,v 1.4 2009/12/08 16:23:46 bes Exp $
# --
# 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::EventHandler;

use strict;
use warnings;

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

=head1 NAME

Kernel::System::EventHandler - event handler lib

=head1 SYNOPSIS

All event handler functions.

=head1 PUBLIC INTERFACE

=over 4

=cut

=item EventHandlerInit()

use vars qw(@ISA);
use Kernel::System::EventHandler;
push @ISA, 'Kernel::System::EventHandler';

    $Self->EventHandlerInit(

        # name of configured event modules
        Config     => 'Example::EventModule',

        # current object, $Self, used in events as "ExampleObject"
        BaseObject => 'ExampleObject',

        # served default objects in any event backend
        Objects    => {
            UserObject => $UserObject,
            XZY        => $XYZ,
        },
    );

e. g.

    $Self->EventHandlerInit(
        Config     => 'Ticket::EventModule',
        BaseObject => 'TicketObject',
        Objects    => {
            UserObject  => $UserObject,
            GroupObject => $GroupObject,
        },
    );

Example XML config:

    <ConfigItem Name="Example::EventModule###99-EscalationIndex" Required="0" Valid="1">
        <Description Lang="en">Example event module updates the example escalation index.</Description>
        <Description Lang="de">Example Event Modul aktualisiert den Example Eskalations-Index.</Description>
        <Group>Example</Group>
        <SubGroup>Core::Example</SubGroup>
        <Setting>
            <Hash>
                <Item Key="Module">Kernel::System::Example::Event::ExampleEscalationIndex</Item>
                <Item Key="Event">(ExampleSLAUpdate|ExampleQueueUpdate|ExampleStateUpdate|ExampleCreate)</Item>
                <Item Key="SomeOption">Some Option accessable via $Param{Config}->{SomeOption} in Run() of event module.</Item>
                <Item Key="Transaction">(0|1)</Item>
            </Hash>
        </Setting>
    </ConfigItem>

=cut
# ---
# ITSM
# ---
=item EventHandlerInit()

use vars qw(@ISA);
use Kernel::System::EventHandler;
push @ISA, 'Kernel::System::EventHandler';

    $Self->EventHandlerInit(

        # name of configured event modules
        Config     => 'ITSM::EventModule',

        # current object, $Self, used in events as "ExampleObject"
        BaseObject => 'ExampleObject',

        # served default objects in any event backend
        Objects    => {
            UserObject => $UserObject,
            XZY        => $XYZ,
        },
    );

e. g.

    $Self->EventHandlerInit(
        Config     => 'ITSM::EventModule',
        BaseObject => 'ChangeObject',
        Objects    => {},
    );

Example XML config:

    <ConfigItem Name="ITSM::EventModule###01-HistoryAdd" Required="0" Valid="1">
        <Description Lang="en">ITSM event module updates the history for Change and WorkOrder objects..</Description>
        <Description Lang="de">ITSM Event Modul aktualisiert die History von Change und WorkOrder Objekten.</Description>
        <Group>ITSM Change Management</Group>
        <SubGroup>Core::ITSMEvent</SubGroup>
        <Setting>
            <Hash>
                <Item Key="Module">Kernel::System::ITSMChange::Event::HistoryAdd</Item>
                <Item Key="Event">(ChangeUpdate|WorkOrderUpdate|ChangeAdd|WorkOrderAdd)</Item>
                <Item Key="SomeOption">Some Option accessable via $Param{Config}->{SomeOption} in Run() of event module.</Item>
                <Item Key="Transaction">(0|1)</Item>
            </Hash>
        </Setting>
    </ConfigItem>
    <ConfigItem Name="ITSM::EventModule###02-HistoryAdd" Required="0" Valid="1">
        <Description Lang="en">ITSM event module updates the ConfigItem History.</Description>
        <Description Lang="de">ITSM Event Modul aktualisiert ConfigItem History.</Description>
        <Group>ITSM Configuration Management</Group>
        <SubGroup>Core::ITSMEvent</SubGroup>
        <Setting>
            <Hash>
                <Item Key="Module">Kernel::System::ITSMConfigurationManagement::Event::HistoryAdd</Item>
                <Item Key="Event">(ConfigItemUpdate|ConfigItemAdd)</Item>
                <Item Key="SomeOption">Some Option accessable via $Param{Config}->{SomeOption} in Run() of event module.</Item>
                <Item Key="Transaction">(0|1)</Item>
            </Hash>
        </Setting>
    </ConfigItem>

=cut
# ---

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

    $Self->{EventHandlerInit} = \%Param;

    return 1;
}

=item EventHandler()

call event handler, returns true if it's executed successfully

    $EventHandler->EventHandler(
        Event => 'TicketStateUpdate',
        Data  => {
            TicketID => 123,
        },
        UserID => 123,
    );

=cut
# ---
# ITSM
# ---
=item EventHandler()

call event handler, returns true if it's executed successfully

    $EventHandler->EventHandler(
        Event => 'ChangeUpdate',
        Data  => {
            ChangeID => 123,
        },
        UserID => 123,
    );

=cut
# ---

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

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

    # get configured modules
    my $Modules = $Self->{ConfigObject}->Get( $Self->{EventHandlerInit}->{Config} );

    # return if there is no one
    return 1 if !$Modules;

    # remember events only on normal mode
    if ( !$Self->{EventHandlerTransaction} ) {
        push @{ $Self->{EventHandlerPipe} }, \%Param;
    }

    # load modules and execute
    MODULE:
    for my $Module ( sort keys %{$Modules} ) {

        # execute only if configured (regexp in Event of config is possible)
        if ( !$Modules->{$Module}->{Event} || $Param{Event} =~ /$Modules->{$Module}->{Event}/ ) {

            # next if we are not in transaction mode, but module is in transaction
            next MODULE if !$Param{Transaction} && $Modules->{$Module}->{Transaction};

            # next if we are in transaction mode, but module is not in transaction
            next MODULE if $Param{Transaction} && !$Modules->{$Module}->{Transaction};

            # load event module
            next MODULE if !$Self->{MainObject}->Require( $Modules->{$Module}->{Module} );

            # get all default objects if given
            my $ObjectRef = $Self->{EventHandlerInit}->{Objects};
            my %Objects;
            if ($ObjectRef) {
                %Objects = %{$ObjectRef};
            }

            # execute event backend
            my $Generic = $Modules->{$Module}->{Module}->new(
                %Objects,
                $Self->{EventHandlerInit}->{BaseObject} => $Self,
            );

            # compatable to old
            # OTRS 3.x: REMOVE ME
            if ( $Param{Data} ) {
                %Param = ( %Param, %{ $Param{Data} } );
            }

            $Generic->Run(
                %Param,
                Config => $Modules->{$Module},
            );
        }
    }

    return 1;
}

=item EventHandlerTransaction()

call all transaction backends for all triggered events till now

    $EventHandler->EventHandlerTransaction();

usually it's done in DESTROY of ExampleObject (e. g. Kernel::System::ExampleObject)

sub DESTROY {
    my $Self = shift;

    # execute all transaction events
    $Self->EventHandlerTransaction();

    return 1;
};

=cut

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

    # remember, we are on destory mode, do not execute new events
    $Self->{EventHandlerTransaction} = 1;

    # execute events on end of transaction
    if ( $Self->{EventHandlerPipe} ) {
        for my $Params ( @{ $Self->{EventHandlerPipe} } ) {
            $Self->EventHandler(
                %Param,
                %{$Params},
                Transaction => 1,
            );
        }

        # delete event pipe
        $Self->{EventHandlerPipe} = undef;
    }

    # reset transaction mode
    $Self->{EventHandlerTransaction} = 0;

    return 1;
}

1;

=back

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (http://otrs.org/).

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

=cut

=head1 VERSION

$Revision: 1.2 $ $Date: 2010/02/04 13:19:13 $

=cut

# --
# Kernel/System/LinkObject/Service.pm - to link service objects
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: Service.pm,v 1.7 2009/08/18 22:20:52 mh Exp $
# --
# 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::LinkObject::Service;

use strict;
use warnings;

use Kernel::System::Service;

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

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

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

    # check needed objects
    for my $Object (
        qw(DBObject ConfigObject LogObject MainObject EncodeObject TimeObject LinkObject)
        )
    {
        $Self->{$Object} = $Param{$Object} || die "Got no $Object!";
    }

    $Self->{ServiceObject} = Kernel::System::Service->new( %{$Self} );

    return $Self;
}

=item LinkListWithData()

fill up the link list with data

    $Success = $LinkObjectBackend->LinkListWithData(
        LinkList => $HashRef,
        UserID   => 1,
    );

=cut

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

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

    # check link list
    if ( ref $Param{LinkList} ne 'HASH' ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'LinkList must be a hash reference!',
        );
        return;
    }

    for my $LinkType ( keys %{ $Param{LinkList} } ) {

        for my $Direction ( keys %{ $Param{LinkList}->{$LinkType} } ) {

            SERVICEID:
            for my $ServiceID ( keys %{ $Param{LinkList}->{$LinkType}->{$Direction} } ) {

                # get service data
                my %ServiceData = $Self->{ServiceObject}->ServiceGet(
                    ServiceID => $ServiceID,
                    UserID    => $Param{UserID},
                );

                # remove id from hash if service can not get
                if ( !%ServiceData ) {
                    delete $Param{LinkList}->{$LinkType}->{$Direction}->{$ServiceID};
                    next SERVICEID;
                }

                # add service data
                $Param{LinkList}->{$LinkType}->{$Direction}->{$ServiceID} = \%ServiceData;
            }
        }
    }

    return 1;
}

=item ObjectDescriptionGet()

return a hash of object descriptions

Return
    %Description = (
        Normal => "Service: ServiceName",
        Long   => "Service: ParentService::ServiceName",
    );

    %Description = $LinkObject->ObjectDescriptionGet(
        Key     => 123,
        UserID  => 1,
    );

=cut

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

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

    # create description
    my %Description = (
        Normal => 'Service',
        Long   => 'Service',
    );

    return %Description if $Param{Mode} && $Param{Mode} eq 'Temporary';

    # get service
    my %Service = $Self->{ServiceObject}->ServiceGet(
        ServiceID => $Param{Key},
        UserID    => 1,
    );

    return if !%Service;

    # create description
    %Description = (
        Normal => "Service '$Service{NameShort}'",
        Long   => "Service '$Service{Name}'",
    );

    return %Description;
}

=item ObjectSearch()

return a hash list of the search results

Return
    $SearchList = {
        NOTLINKED => {
            Source => {
                12  => $DataOfItem12,
                212 => $DataOfItem212,
                332 => $DataOfItem332,
            },
        },
    };

    $SearchList = $LinkObjectBackend->ObjectSearch(
        SearchParams => $HashRef,  # (optional)
        UserID       => 1,
    );

=cut

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

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

    # set default params
    $Param{SearchParams} ||= {};

    # add wildcards
    my %Search;
    if ( $Param{SearchParams}->{Name} ) {
        $Search{Name} = '*' . $Param{SearchParams}->{Name} . '*';
    }

    # search the services
    my @ServiceIDs = $Self->{ServiceObject}->ServiceSearch(
        %{ $Param{SearchParams} },
        %Search,
        Limit  => 50,
        UserID => $Param{UserID},
    );

    my %SearchList;
    SERVICEID:
    for my $ServiceID (@ServiceIDs) {

        # get service data
        my %ServiceData = $Self->{ServiceObject}->ServiceGet(
            ServiceID => $ServiceID,
            UserID    => $Param{UserID},
        );

        next SERVICEID if !%ServiceData;

        # add service data
        $SearchList{NOTLINKED}->{Source}->{$ServiceID} = \%ServiceData;
    }

    return \%SearchList;
}

=item LinkAddPre()

link add pre event module

    $True = $LinkObject->LinkAddPre(
        Key          => 123,
        SourceObject => 'Service',
        SourceKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

    or

    $True = $LinkObject->LinkAddPre(
        Key          => 123,
        TargetObject => 'Service',
        TargetKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

=cut

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

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

    return 1 if $Param{State} eq 'Temporary';

    return 1;
}

=item LinkAddPost()

link add pre event module

    $True = $LinkObject->LinkAddPost(
        Key          => 123,
        SourceObject => 'Service',
        SourceKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

    or

    $True = $LinkObject->LinkAddPost(
        Key          => 123,
        TargetObject => 'Service',
        TargetKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

=cut

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

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

    return 1 if $Param{State} eq 'Temporary';

    return 1;
}

=item LinkDeletePre()

link delete pre event module

    $True = $LinkObject->LinkDeletePre(
        Key          => 123,
        SourceObject => 'Service',
        SourceKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

    or

    $True = $LinkObject->LinkDeletePre(
        Key          => 123,
        TargetObject => 'Service',
        TargetKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

=cut

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

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

    return 1 if $Param{State} eq 'Temporary';

    return 1;
}

=item LinkDeletePost()

link delete post event module

    $True = $LinkObject->LinkDeletePost(
        Key          => 123,
        SourceObject => 'Service',
        SourceKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

    or

    $True = $LinkObject->LinkDeletePost(
        Key          => 123,
        TargetObject => 'Service',
        TargetKey    => 321,
        Type         => 'Normal',
        State        => 'Valid',
        UserID       => 1,
    );

=cut

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

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

    return 1 if $Param{State} eq 'Temporary';

    return 1;
}

1;

# --
# Kernel/System/ITSMCIPAllocate.pm - all criticality, impact and priority allocation functions
# Copyright (C) 2001-2010 OTRS AG, http://otrs.org/
# --
# $Id: ITSMCIPAllocate.pm,v 1.15 2010/02/18 14:32:59 bes Exp $
# --
# 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::ITSMCIPAllocate;

use strict;
use warnings;

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

=head1 NAME

Kernel::System::ITSMCIPAllocate - criticality, impact and priority allocation lib

=head1 SYNOPSIS

All criticality, impact and priority allocation functions.

=head1 PUBLIC INTERFACE

=over 4

=cut

=item new()

create an object

    use Kernel::Config;
    use Kernel::System::Encode;
    use Kernel::System::Log;
    use Kernel::System::ITSMCIPAllocate;
    use Kernel::System::DB;
    use Kernel::System::Main;

    my $ConfigObject = Kernel::Config->new();
    my $EncodeObject = Kernel::System::Encode->new(
        ConfigObject => $ConfigObject,
    );
    my $LogObject = Kernel::System::Log->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
    );
    my $MainObject = Kernel::System::Main->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        LogObject    => $LogObject,
    );
    my $DBObject = Kernel::System::DB->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        LogObject    => $LogObject,
        MainObject   => $MainObject,
    );
    my $CIPAllocateObject = Kernel::System::ITSMCIPAllocate->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
        DBObject     => $DBObject,
    );

=cut

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

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

    # check needed objects
    for my $Object (qw(DBObject ConfigObject LogObject)) {
        $Self->{$Object} = $Param{$Object} || die "Got no $Object!";
    }

    return $Self;
}

=item AllocateList()

return a 2d hash reference of allocations

    my $ListRef = $CIPAllocateObject->AllocateList(
        UserID => 1,
    );

C<$ListRef> is something like

    $ListRet = {
        '6' => {
            '4' => 2,
            '1' => 1,
            '3' => 2,
        },
        '8' => {
            '4' => 4,
            '1' => 2,
            '3' => 3,
        },
    };

meaning that the CriticalityID '6' and the IncidentID '4' suggest the PriorityID '2'.

=cut

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

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

    # ask database
    $Self->{DBObject}->Prepare(
        SQL => 'SELECT criticality_id, impact_id, priority_id FROM cip_allocate',
    );

    # result list
    my %AllocateData;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        $AllocateData{ $Row[1] }{ $Row[0] } = $Row[2];
    }

    return \%AllocateData;
}

=item AllocateUpdate()

update the allocation of criticality, impact and priority

    my $True = $CIPAllocateObject->AllocateUpdate(
        AllocateData => $DataRef,  # 2D hash reference
        UserID       => 1,
    );

=cut

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

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

    # check if allocate data is a hash reference
    if ( ref $Param{AllocateData} ne 'HASH' ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'AllocateData must be a 2D hash reference!',
        );
        return;
    }

    # check if allocate data is a 2D hash reference
    IMPACTID:
    for my $ImpactID ( keys %{ $Param{AllocateData} } ) {

        next IMPACTID if ref $Param{AllocateData}->{$ImpactID} eq 'HASH';

        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'AllocateData must be a 2D hash reference!',
        );
        return;
    }

    # delete old allocations
    $Self->{DBObject}->Do( SQL => 'DELETE FROM cip_allocate' );

    # insert new allocations
    for my $ImpactID ( keys %{ $Param{AllocateData} } ) {

        for my $CriticalityID ( keys %{ $Param{AllocateData}->{$ImpactID} } ) {

            # extract priority
            my $PriorityID = $Param{AllocateData}->{$ImpactID}->{$CriticalityID};

            # insert new allocation
            $Self->{DBObject}->Do(
                SQL => 'INSERT INTO cip_allocate '
                    . '(criticality_id, impact_id, priority_id, '
                    . 'create_time, create_by, change_time, change_by) VALUES '
                    . '(?, ?, ?, current_timestamp, ?, current_timestamp, ?)',
                Bind => [
                    \$CriticalityID, \$ImpactID, \$PriorityID,
                    \$Param{UserID}, \$Param{UserID},
                ],
            );
        }
    }

    return 1;
}

=item PriorityAllocationGet()

return the priority id of a criticality and an impact

    my $PriorityID = $CIPAllocateObject->PriorityAllocationGet(
        CriticalityID => 321,
        ImpactID      => 123,
    );

=cut

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

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

    # get priority id from db
    $Self->{DBObject}->Prepare(
        SQL => 'SELECT priority_id FROM cip_allocate '
            . 'WHERE criticality_id = ? AND impact_id = ?',
        Bind => [ \$Param{CriticalityID}, \$Param{ImpactID} ],
        Limit => 1,
    );

    # fetch result
    my $PriorityID;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        $PriorityID = $Row[0];
    }

    return $PriorityID;
}

1;

=back

=head1 TERMS AND CONDITIONS

This Software is part of the OTRS project (http://otrs.org/).

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

=cut

=head1 VERSION

$Revision: 1.15 $ $Date: 2010/02/18 14:32:59 $

=cut

# --
# Kernel/System/Service.pm - all service function
# Copyright (C) 2001-2010 OTRS AG, http://otrs.org/
# --
# $Id: Service.pm,v 1.15 2010/04/13 17:40:20 ub Exp $
# $OldId: Service.pm,v 1.39.2.1 2010/04/13 17:31:45 ub Exp $
# --
# 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::Service;

use strict;
use warnings;

use Kernel::System::CheckItem;
use Kernel::System::Valid;
# ---
# ITSM
# ---
use Kernel::System::GeneralCatalog;
use Kernel::System::LinkObject;
use Kernel::System::Time;
# ---

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

=head1 NAME

Kernel::System::Service - service lib

=head1 SYNOPSIS

All service functions.

=head1 PUBLIC INTERFACE

=over 4

=cut

=item new()

create an object

    use Kernel::Config;
    use Kernel::System::Encode;
    use Kernel::System::Log;
    use Kernel::System::Main;
    use Kernel::System::DB;
    use Kernel::System::Service;

    my $ConfigObject = Kernel::Config->new();
    my $EncodeObject = Kernel::System::Encode->new(
        ConfigObject => $ConfigObject,
    );
    my $LogObject = Kernel::System::Log->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
    );
    my $MainObject = Kernel::System::Main->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        LogObject    => $LogObject,
    );
    my $DBObject = Kernel::System::DB->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        LogObject    => $LogObject,
        MainObject   => $MainObject,
    );
    my $ServiceObject = Kernel::System::Service->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        LogObject    => $LogObject,
        DBObject     => $DBObject,
        MainObject   => $MainObject,
    );

=cut

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

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

    # check needed objects
    for my $Object (qw(DBObject ConfigObject LogObject EncodeObject MainObject)) {
        $Self->{$Object} = $Param{$Object} || die "Got no $Object!";
    }
    $Self->{CheckItemObject} = Kernel::System::CheckItem->new( %{$Self} );
    $Self->{ValidObject}     = Kernel::System::Valid->new( %{$Self} );
# ---
# ITSM
# ---
    $Self->{TimeObject}           = Kernel::System::Time->new( %{$Self} );
    $Self->{GeneralCatalogObject} = Kernel::System::GeneralCatalog->new( %{$Self} );
    $Self->{LinkObject}           = Kernel::System::LinkObject->new( %{$Self} );
# ---

    # load generator preferences module
    my $GeneratorModule = $Self->{ConfigObject}->Get('Service::PreferencesModule')
        || 'Kernel::System::Service::PreferencesDB';
    if ( $Self->{MainObject}->Require($GeneratorModule) ) {
        $Self->{PreferencesObject} = $GeneratorModule->new(%Param);
    }

    return $Self;
}

=item ServiceList()

return a hash list of services

    my %ServiceList = $ServiceObject->ServiceList(
        Valid  => 0,   # (optional) default 1 (0|1)
        UserID => 1,
    );

=cut

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

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

    # check valid param
    if ( !defined $Param{Valid} ) {
        $Param{Valid} = 1;
    }

    # ask database
    $Self->{DBObject}->Prepare(
        SQL => 'SELECT id, name, valid_id FROM service',
    );

    # fetch the result
    my %ServiceList;
    my %ServiceValidList;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        $ServiceList{ $Row[0] }      = $Row[1];
        $ServiceValidList{ $Row[0] } = $Row[2];
    }

    return %ServiceList if !$Param{Valid};

    # get valid ids
    my @ValidIDs = $Self->{ValidObject}->ValidIDsGet();

    # duplicate service list
    my %ServiceListTmp = %ServiceList;

    # add suffix for correct sorting
    for my $ServiceID ( keys %ServiceListTmp ) {
        $ServiceListTmp{$ServiceID} .= '::';
    }

    my %ServiceInvalidList;
    SERVICEID:
    for my $ServiceID ( sort { $ServiceListTmp{$a} cmp $ServiceListTmp{$b} } keys %ServiceListTmp )
    {

        my $Valid = scalar grep { $_ eq $ServiceValidList{$ServiceID} } @ValidIDs;

        next SERVICEID if $Valid;

        $ServiceInvalidList{ $ServiceList{$ServiceID} } = 1;
        delete $ServiceList{$ServiceID};
    }

    # delete invalid services and childs
    for my $ServiceID ( keys %ServiceList ) {

        INVALIDNAME:
        for my $InvalidName ( keys %ServiceInvalidList ) {

            if ( $ServiceList{$ServiceID} =~ m{ \A $InvalidName :: }xms ) {
                delete $ServiceList{$ServiceID};
                last INVALIDNAME;
            }
        }
    }

    return %ServiceList;
}

=item ServiceGet()

return a service as hash

Return
    $ServiceData{ServiceID}
    $ServiceData{ParentID}
    $ServiceData{Name}
    $ServiceData{NameShort}
    $ServiceData{ValidID}
    $ServiceData{Comment}
    $ServiceData{CreateTime}
    $ServiceData{CreateBy}
    $ServiceData{ChangeTime}
    $ServiceData{ChangeBy}
# ---
# ITSM
# ---
    $ServiceData{TypeID}
    $ServiceData{Type}
    $ServiceData{CriticalityID}
    $ServiceData{Criticality}
    $ServiceData{CurInciStateID}
    $ServiceData{CurInciState}
    $ServiceData{CurInciStateType}
# ---

    my %ServiceData = $ServiceObject->ServiceGet(
        ServiceID => 123,
        UserID    => 1,
    );

=cut

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

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

    # get service from db
    $Self->{DBObject}->Prepare(
        SQL =>
            'SELECT id, name, valid_id, comments, create_time, create_by, change_time, change_by '
# ---
# ITSM
# ---
            . ", type_id, criticality_id "
# ---
            . 'FROM service WHERE id = ?',
        Bind  => [ \$Param{ServiceID} ],
        Limit => 1,
    );

    # fetch the result
    my %ServiceData;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        $ServiceData{ServiceID}  = $Row[0];
        $ServiceData{Name}       = $Row[1];
        $ServiceData{ValidID}    = $Row[2];
        $ServiceData{Comment}    = $Row[3] || '';
        $ServiceData{CreateTime} = $Row[4];
        $ServiceData{CreateBy}   = $Row[5];
        $ServiceData{ChangeTime} = $Row[6];
        $ServiceData{ChangeBy}   = $Row[7];
# ---
# ITSM
# ---
        $ServiceData{TypeID}        = $Row[8];
        $ServiceData{CriticalityID} = $Row[9];
# ---
    }

    # check service
    if ( !$ServiceData{ServiceID} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "No such ServiceID ($Param{ServiceID})!",
        );
        return;
    }

    # create short name and parentid
    $ServiceData{NameShort} = $ServiceData{Name};
    if ( $ServiceData{Name} =~ /^(.*)::(.+?)$/ ) {
        $ServiceData{NameShort} = $2;

        # lookup parent
        my $ServiceID = $Self->ServiceLookup(
            Name => $1,
        );
        $ServiceData{ParentID} = $ServiceID;
    }

    # get service preferences
    my %Preferences = $Self->ServicePreferencesGet(
        ServiceID => $Param{ServiceID},
    );
# ---
# ITSM
# ---

    # get service type list
    my $ServiceTypeList = $Self->{GeneralCatalogObject}->ItemList(
        Class => 'ITSM::Service::Type',
    );
    $ServiceData{Type} = $ServiceTypeList->{ $ServiceData{TypeID} } || '';

    # get criticality list
    my $CriticalityList = $Self->{GeneralCatalogObject}->ItemList(
        Class => 'ITSM::Core::Criticality',
    );
    $ServiceData{Criticality} = $CriticalityList->{ $ServiceData{CriticalityID} } || '';

    # set default incident state type
    $ServiceData{CurInciStateType} = 'operational';

    # get ITSM module directory
    my $ConfigItemModule = $Self->{ConfigObject}->Get('Home') . '/Kernel/System/ITSMConfigItem.pm';

    # check if ITSMConfigurationManagement package is installed
    if ( -e $ConfigItemModule ) {

        # check if a preference setting for CurInciStateTypeFromCIs exists
        if ( $Preferences{CurInciStateTypeFromCIs} ) {

            # set default incident state type from service preferences 'CurInciStateTypeFromCIs'
            $ServiceData{CurInciStateType} = $Preferences{CurInciStateTypeFromCIs};
        }

        # set the preferences setting for CurInciStateTypeFromCIs
        else {

            # get the incident link type
            my $LinkType = $Self->{ConfigObject}->Get('ITSM::Core::IncidentLinkType');

            # find all linked config items
            my %LinkedConfigItemIDs = $Self->{LinkObject}->LinkKeyListWithData(
                Object1   => 'Service',
                Key1      => $Param{ServiceID},
                Object2   => 'ITSMConfigItem',
                State     => 'Valid',
                Type      => $LinkType,
                UserID    => 1,
            );

            # investigate the current incident state of each config item
            CONFIGITEMID:
            for my $ConfigItemID ( keys %LinkedConfigItemIDs ) {

                # extract config item data
                my $ConfigItemData = $LinkedConfigItemIDs{$ConfigItemID};

                next CONFIGITEMID if $ConfigItemData->{CurDeplStateType} ne 'productive';
                next CONFIGITEMID if $ConfigItemData->{CurInciStateType} eq 'operational';

                # check if service must be set to 'warning'
                if ( $ConfigItemData->{CurInciStateType} eq 'warning' ) {
                    $ServiceData{CurInciStateType} = 'warning';
                    next CONFIGITEMID;
                }

                # check if service must be set to 'incident'
                if ( $ConfigItemData->{CurInciStateType} eq 'incident' ) {
                    $ServiceData{CurInciStateType} = 'incident';
                    last CONFIGITEMID;
                }
            }

            # update the current incident state type from CIs of the service
            $Self->ServicePreferencesSet(
                ServiceID => $Param{ServiceID},
                Key       => 'CurInciStateTypeFromCIs',
                Value     => $ServiceData{CurInciStateType},
                UserID    => 1,
            );
        }
    }

    # investigate the state of all child services
    if ( $ServiceData{CurInciStateType} eq 'operational' ) {

        # create the valid string
        my $ValidIDString = join q{, }, $Self->{ValidObject}->ValidIDsGet();

        # prepare name
        my $Name = $ServiceData{Name};
        $Name = $Self->{DBObject}->Quote( $Name, 'Like' );

        # get list of all valid childs
        $Self->{DBObject}->Prepare(
            SQL => "SELECT id, name FROM service "
                . "WHERE name LIKE '" . $Name . "::%' "
                . "AND valid_id IN (" . $ValidIDString . ")",
        );

        # find length of childs prefix
        my $PrefixLength = length "$ServiceData{Name}::";

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

            # extract child part
            my $ChildPart = substr $Row[1], $PrefixLength;

            next ROW if $ChildPart =~ m{ :: }xms;

            push @ChildIDs, $Row[0];
        }

        SERVICEID:
        for my $ServiceID ( @ChildIDs ) {

            # get data of child service
            my %ChildServiceData = $Self->ServiceGet(
                ServiceID => $ServiceID,
                UserID    => $Param{UserID},
            );

            next SERVICEID if $ChildServiceData{CurInciStateType} eq 'operational';

            $ServiceData{CurInciStateType} = 'warning';
            last SERVICEID;
        }
    }

    # define default incident states
    my %DefaultInciStates = (
        operational => 'Operational',
        warning     => 'Warning',
        incident    => 'Incident',
    );

    # get the incident state list of this type
    my $InciStateList = $Self->{GeneralCatalogObject}->ItemList(
        Class         => 'ITSM::Core::IncidentState',
        Preferences   => {
            Functionality => $ServiceData{CurInciStateType},
        },
    );

    my %ReverseInciStateList = reverse %{ $InciStateList };
    $ServiceData{CurInciStateID}
        = $ReverseInciStateList{ $DefaultInciStates{ $ServiceData{CurInciStateType} } };

    # fallback if the default incident state is deactivated
    if ( !$ServiceData{CurInciStateID} ) {
        my @SortedInciList = sort keys %{ $InciStateList };
        $ServiceData{CurInciStateID} = $SortedInciList[0];
    }

    # get incident state functionality
    my $InciState = $Self->{GeneralCatalogObject}->ItemGet(
        ItemID => $ServiceData{CurInciStateID},
    );

    $ServiceData{CurInciState}     = $InciState->{Name};
    $ServiceData{CurInciStateType} = $InciState->{Functionality};
# ---

    # merge hash
    if (%Preferences) {
        %ServiceData = ( %ServiceData, %Preferences );
    }

    return %ServiceData;
}

=item ServiceLookup()

return a service name and id

    my $ServiceName = $ServiceObject->ServiceLookup(
        ServiceID => 123,
    );

    or

    my $ServiceID = $ServiceObject->ServiceLookup(
        Name => 'Service::SubService',
    );

=cut

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

    # check needed stuff
    if ( !$Param{ServiceID} && !$Param{Name} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'Need ServiceID or Name!',
        );
        return;
    }

    if ( $Param{ServiceID} ) {

        # check cache
        my $CacheKey = 'Cache::ServiceLookup::ID::' . $Param{ServiceID};
        if ( defined $Self->{$CacheKey} ) {
            return $Self->{$CacheKey};
        }

        # lookup
        $Self->{DBObject}->Prepare(
            SQL   => 'SELECT name FROM service WHERE id = ?',
            Bind  => [ \$Param{ServiceID} ],
            Limit => 1,
        );
        while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
            $Self->{$CacheKey} = $Row[0];
        }

        return $Self->{$CacheKey};
    }
    else {

        # check cache
        my $CacheKey = 'Cache::ServiceLookup::Name::' . $Param{Name};
        if ( defined $Self->{$CacheKey} ) {
            return $Self->{$CacheKey};
        }

        # lookup
        $Self->{DBObject}->Prepare(
            SQL   => 'SELECT id FROM service WHERE name = ?',
            Bind  => [ \$Param{Name} ],
            Limit => 1,
        );
        while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
            $Self->{$CacheKey} = $Row[0];
        }

        return $Self->{$CacheKey};
    }
}

=item ServiceAdd()

add a service

    my $ServiceID = $ServiceObject->ServiceAdd(
        Name     => 'Service Name',
        ParentID => 1,           # (optional)
        ValidID  => 1,
        Comment  => 'Comment',    # (optional)
        UserID   => 1,
# ---
# ITSM
# ---
        TypeID        => 2,
        CriticalityID => 1,
# ---
    );

=cut

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

    # check needed stuff
# ---
# ITSM
# ---
#    for my $Argument (qw(Name ValidID UserID)) {
    for my $Argument (qw(Name ValidID UserID TypeID CriticalityID)) {
# ---
        if ( !$Param{$Argument} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    # set comment
    $Param{Comment} ||= '';

    # cleanup given params
    for my $Argument (qw(Name Comment)) {
        $Self->{CheckItemObject}->StringClean(
            StringRef         => \$Param{$Argument},
            RemoveAllNewlines => 1,
            RemoveAllTabs     => 1,
        );
    }

    # check service name
    if ( $Param{Name} =~ /::/ ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't add service! Invalid Service name '$Param{Name}'!",
        );
        return;
    }

    # create full name
    $Param{FullName} = $Param{Name};

    # get parent name
    if ( $Param{ParentID} ) {
        my $ParentName = $Self->ServiceLookup( ServiceID => $Param{ParentID}, );
        if ($ParentName) {
            $Param{FullName} = $ParentName . '::' . $Param{Name};
        }
    }

    # find existing service
    $Self->{DBObject}->Prepare(
        SQL   => 'SELECT id FROM service WHERE name = ?',
        Bind  => [ \$Param{FullName} ],
        Limit => 1,
    );
    my $Exists;
    while ( $Self->{DBObject}->FetchrowArray() ) {
        $Exists = 1;
    }

    # add service to database
    if ($Exists) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'Can\'t add service! Service with same name and parent already exists.'
        );
        return;
    }

    return if !$Self->{DBObject}->Do(
# ---
# ITSM
# ---
#        SQL => 'INSERT INTO service '
#            . '(name, valid_id, comments, create_time, create_by, change_time, change_by) '
#            . 'VALUES (?, ?, ?, current_timestamp, ?, current_timestamp, ?)',
#        Bind => [
#            \$Param{FullName}, \$Param{ValidID}, \$Param{Comment},
#            \$Param{UserID}, \$Param{UserID},
#        ],
        SQL => 'INSERT INTO service '
            . '(name, valid_id, comments, create_time, create_by, change_time, change_by, '
            . 'type_id, criticality_id) '
            . 'VALUES (?, ?, ?, current_timestamp, ?, current_timestamp, ?, ?, ?)',
        Bind => [
            \$Param{FullName}, \$Param{ValidID}, \$Param{Comment},
            \$Param{UserID}, \$Param{UserID}, \$Param{TypeID}, \$Param{CriticalityID},
        ],
# ---
    );

    # get service id
    $Self->{DBObject}->Prepare(
        SQL   => 'SELECT id FROM service WHERE name = ?',
        Bind  => [ \$Param{FullName} ],
        Limit => 1,
    );
    my $ServiceID;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        $ServiceID = $Row[0];
    }

    # reset cache
    delete $Self->{ 'Cache::ServiceLookup::ID::' . $ServiceID };
    delete $Self->{ 'Cache::ServiceLookup::Name::' . $Param{FullName} };

    return $ServiceID;
}

=item ServiceUpdate()

update a existing service

    my $True = $ServiceObject->ServiceUpdate(
        ServiceID => 123,
        ParentID  => 1,           # (optional)
        Name      => 'Service Name',
        ValidID   => 1,
        Comment   => 'Comment',    # (optional)
        UserID    => 1,
# ---
# ITSM
# ---
        TypeID        => 2,
        CriticalityID => 1,
# ---
    );

=cut

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

    # check needed stuff
# ---
# ITSM
# ---
#    for my $Argument (qw(ServiceID Name ValidID UserID)) {
    for my $Argument (qw(ServiceID Name ValidID UserID TypeID CriticalityID)) {
# ---
        if ( !$Param{$Argument} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    # set default comment
    $Param{Comment} ||= '';

    # cleanup given params
    for my $Argument (qw(Name Comment)) {
        $Self->{CheckItemObject}->StringClean(
            StringRef         => \$Param{$Argument},
            RemoveAllNewlines => 1,
            RemoveAllTabs     => 1,
        );
    }

    # check service name
    if ( $Param{Name} =~ /::/ ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't update service! Invalid Service name '$Param{Name}'!",
        );
        return;
    }

    # get old name of service
    my $OldServiceName = $Self->ServiceLookup( ServiceID => $Param{ServiceID}, );

    # reset cache
    delete $Self->{ 'Cache::ServiceLookup::ID::' . $Param{ServiceID} };
    delete $Self->{ 'Cache::ServiceLookup::Name::' . $OldServiceName };

    # create full name
    $Param{FullName} = $Param{Name};

    # get parent name
    if ( $Param{ParentID} ) {

        # lookup service
        my $ParentName = $Self->ServiceLookup(
            ServiceID => $Param{ParentID},
        );

        if ($ParentName) {
            $Param{FullName} = $ParentName . '::' . $Param{Name};
        }

        # check, if selected parent was a child of this service
        if ( $Param{FullName} =~ /^(\Q$OldServiceName\E)::/ ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => 'Can\'t update service! Invalid parent was selected.'
            );
            return;
        }
    }

    # find exists service
    $Self->{DBObject}->Prepare(
        SQL   => 'SELECT id FROM service WHERE name = ?',
        Bind  => [ \$Param{FullName} ],
        Limit => 1,
    );
    my $Exists;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        if ( $Param{ServiceID} ne $Row[0] ) {
            $Exists = 1;
        }
    }

    # update service
    if ($Exists) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'Can\'t update service! Service with same name and parent already exists.'
        );
        return;

    }

    # update service
    return if !$Self->{DBObject}->Do(
# ---
# ITSM
# ---
#        SQL => 'UPDATE service SET name = ?, valid_id = ?, comments = ?, '
#            . ' change_time = current_timestamp, change_by = ? WHERE id = ?',
#        Bind => [
#            \$Param{FullName}, \$Param{ValidID}, \$Param{Comment},
#            \$Param{UserID}, \$Param{ServiceID},
#        ],
        SQL => 'UPDATE service SET name = ?, valid_id = ?, comments = ?, '
            . ' change_time = current_timestamp, change_by = ?, type_id = ?, criticality_id = ?'
            . ' WHERE id = ?',
        Bind => [
            \$Param{FullName}, \$Param{ValidID}, \$Param{Comment},
            \$Param{UserID}, \$Param{TypeID}, \$Param{CriticalityID}, \$Param{ServiceID},
        ],
# ---
    );

    # find all childs
    $Self->{DBObject}->Prepare(
        SQL => "SELECT id, name FROM service WHERE name LIKE '"
            . $Self->{DBObject}->Quote( $OldServiceName, 'Like' )
            . "::%'",
    );
    my @Childs;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        my %Child;
        $Child{ServiceID} = $Row[0];
        $Child{Name}      = $Row[1];
        push @Childs, \%Child;
    }

    # update childs
    for my $Child (@Childs) {
        $Child->{Name} =~ s/^(\Q$OldServiceName\E)::/$Param{FullName}::/;
        $Self->{DBObject}->Do(
            SQL => 'UPDATE service SET name = ? WHERE id = ?',
            Bind => [ \$Child->{Name}, \$Child->{ServiceID} ],
        );
    }
    return 1;
}

=item ServiceSearch()

return service ids as an array

    my @ServiceList = $ServiceObject->ServiceSearch(
        Name   => 'Service Name', # (optional)
        Limit  => 122,            # (optional) default 1000
        UserID => 1,
# ---
# ITSM
# ---
        TypeIDs        => 2,
        CriticalityIDs => 1,
# ---
    );

=cut

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

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

    # set default limit
    $Param{Limit} ||= 1000;

    # create sql query
    my $SQL
        = "SELECT id FROM service WHERE valid_id IN ( ${\(join ', ', $Self->{ValidObject}->ValidIDsGet())} )";

    if ( $Param{Name} ) {

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

        # replace * with % and clean the string
        $Param{Name} =~ s{ \*+ }{%}xmsg;
        $Param{Name} =~ s{ %+ }{%}xmsg;

        $SQL .= " AND name LIKE '$Param{Name}' ";
    }
# ---
# ITSM
# ---
    # add type ids
    if ( $Param{TypeIDs} && ref $Param{TypeIDs} eq 'ARRAY' && @{ $Param{TypeIDs} } ) {
        $SQL .= "AND type_id IN (" . join(', ', @{ $Param{TypeIDs} }) . ") ";
    }
    # add criticality ids
    if ($Param{CriticalityIDs} && ref $Param{CriticalityIDs} eq 'ARRAY' && @{ $Param{CriticalityIDs} } ) {
        $SQL .= "AND criticality_id IN (" . join(', ', @{ $Param{CriticalityIDs} }) . ") ";
    }
# ---

    $SQL .= ' ORDER BY name';

    # search service in db
    $Self->{DBObject}->Prepare( SQL => $SQL );

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

    return @ServiceList;
}

=item CustomerUserServiceMemberList()

returns a list of customeruser/service members

    ServiceID: service id
    CustomerUserLogin: customer user login
    DefaultServices: activate or deactivate default services

    Result: HASH -> returns a hash of key => service id, value => service name
            Name -> returns an array of user names
            ID   -> returns an array of user ids

    Example (get services of customer user):

    $ServiceObject->CustomerUserServiceMemberList(
        CustomerUserLogin => 'Test',
        Result            => 'HASH',
        DefaultServices   => 0,
    );

    Example (get customer user of service):

    $ServiceObject->CustomerUserServiceMemberList(
        ServiceID => $ID,
        Result    => 'HASH',
    );

=cut

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

    # check needed stuff
    if ( !$Param{Result} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'Need Result!',
        );
        return;
    }
    if ( !$Param{ServiceID} && !$Param{CustomerUserLogin} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'Need ServiceID or CustomerUserLogin!',
        );
        return;
    }

    # set default
    if ( !defined $Param{DefaultServices} ) {
        $Param{DefaultServices} = 1;
    }

    # db quote
    for ( keys %Param ) {
        $Param{$_} = $Self->{DBObject}->Quote( $Param{$_} );
    }
    for (qw(ServiceID)) {
        $Param{$_} = $Self->{DBObject}->Quote( $Param{$_}, 'Integer' );
    }

    # create cache key
    my $CacheKey = 'CustomerUserServiceMemberList::' . $Param{Result} . '::';
    if ( $Param{ServiceID} ) {
        $CacheKey .= 'ServiceID::' . $Param{ServiceID};
    }
    elsif ( $Param{CustomerUserLogin} ) {
        $CacheKey .= 'CustomerUserLogin::' . $Param{CustomerUserLogin};
    }

    # check cache
    if ( $Param{ServiceID} || $Param{CustomerUserLogin} ) {
        if ( $Self->{ForceCache} ) {
            $Param{Cached} = $Self->{ForceCache};
        }
        if ( $Param{Cached} && $Self->{$CacheKey} ) {
            if ( ref( $Self->{$CacheKey} ) eq 'ARRAY' ) {
                return @{ $Self->{$CacheKey} };
            }
            elsif ( ref( $Self->{$CacheKey} ) eq 'HASH' ) {
                return %{ $Self->{$CacheKey} };
            }
        }
    }

    # sql
    my %Data;
    my @Name;
    my @ID;
    my $SQL = 'SELECT scu.service_id, scu.customer_user_login, s.name '
        . ' FROM '
        . ' service_customer_user scu, service s'
        . ' WHERE '
        . " s.valid_id IN ( ${\(join ', ', $Self->{ValidObject}->ValidIDsGet())} ) AND "
        . ' s.id = scu.service_id AND ';

    if ( $Param{ServiceID} ) {
        $SQL .= " scu.service_id = $Param{ServiceID}";
    }
    elsif ( $Param{CustomerUserLogin} ) {
        $SQL .= " scu.customer_user_login = '$Param{CustomerUserLogin}'";
    }

    $Self->{DBObject}->Prepare( SQL => $SQL );

    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {

        my $Key   = '';
        my $Value = '';
        if ( $Param{ServiceID} ) {
            $Key   = $Row[1];
            $Value = $Row[0];
        }
        else {
            $Key   = $Row[0];
            $Value = $Row[2];
        }

        # remember permissions
        if ( !defined $Data{$Key} ) {
            $Data{$Key} = $Value;
            push @Name, $Value;
            push @ID,   $Key;
        }
    }
    if ( $Param{CustomerUserLogin} && $Param{DefaultServices} && !keys(%Data) ) {
        %Data = $Self->CustomerUserServiceMemberList(
            CustomerUserLogin => '<DEFAULT>',
            Result            => 'HASH',
            DefaultServices   => 0,
        );
        for my $Key ( keys %Data ) {
            push @Name, $Data{$Key};
            push @ID,   $Key;
        }
    }

    # return result
    if ( $Param{Result} && $Param{Result} eq 'ID' ) {
        if ( $Param{ServiceID} || $Param{CustomerUserLogin} ) {

            # cache result
            $Self->{$CacheKey} = \@ID;
        }
        return @ID;
    }
    if ( $Param{Result} && $Param{Result} eq 'Name' ) {
        if ( $Param{ServiceID} || $Param{CustomerUserLogin} ) {

            # cache result
            $Self->{$CacheKey} = \@Name;
        }
        return @Name;
    }
    else {
        if ( $Param{ServiceID} || $Param{CustomerUserLogin} ) {

            # cache result
            $Self->{$CacheKey} = \%Data;
        }
        return %Data;
    }
}

=item CustomerUserServiceMemberAdd()

to add a member to a service

    $ServiceObject->CustomerUserServiceMemberAdd(
        CustomerUserLogin => 'Test1',
        ServiceID         => 6,
        Active            => 1,
        UserID            => 123,
    );

=cut

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

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

    # delete existing relation
    return if !$Self->{DBObject}->Do(
        SQL => 'DELETE FROM service_customer_user WHERE customer_user_login = ? AND service_id = ?',
        Bind => [ \$Param{CustomerUserLogin}, \$Param{ServiceID} ],
    );

    # return if relation is not active
    return if !$Param{Active};

    # insert new relation
    return $Self->{DBObject}->Do(
        SQL => 'INSERT INTO service_customer_user '
            . '(customer_user_login, service_id, create_time, create_by) '
            . 'VALUES (?, ?, current_timestamp, ?)',
        Bind => [ \$Param{CustomerUserLogin}, \$Param{ServiceID}, \$Param{UserID} ]
    );
}

=item ServicePreferencesSet()

set service preferences

    $ServiceObject->ServicePreferencesSet(
        ServiceID => 123,
        Key       => 'UserComment',
        Value     => 'some comment',
        UserID    => 123,
    );

=cut

sub ServicePreferencesSet {
    my $Self = shift;

    return $Self->{PreferencesObject}->ServicePreferencesSet(@_);
}

=item ServicePreferencesGet()

get service preferences

    my %Preferences = $ServiceObject->ServicePreferencesGet(
        ServiceID => 123,
        UserID    => 123,
    );

=cut

sub ServicePreferencesGet {
    my $Self = shift;

    return $Self->{PreferencesObject}->ServicePreferencesGet(@_);
}

1;

=back

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (http://otrs.org/).

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

=cut

=head1 VERSION

$Revision: 1.15 $ $Date: 2010/04/13 17:40:20 $

=cut

# --
# Kernel/System/SLA.pm - all sla function
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: SLA.pm,v 1.5 2009/07/21 00:12:08 ub Exp $
# $OldId: SLA.pm,v 1.36 2009/07/21 00:09:04 ub Exp $
# --
# 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::SLA;

use strict;
use warnings;

use Kernel::System::CheckItem;
use Kernel::System::Valid;
# ---
# ITSM
# ---
use Kernel::System::GeneralCatalog;
# ---

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

=head1 NAME

Kernel::System::SLA - sla lib

=head1 SYNOPSIS

All sla functions.

=head1 PUBLIC INTERFACE

=over 4

=cut

=item new()

create an object

    use Kernel::Config;
    use Kernel::System::Encode;
    use Kernel::System::Log;
    use Kernel::System::Main;
    use Kernel::System::DB;
    use Kernel::System::SLA;

    my $ConfigObject = Kernel::Config->new();
    my $EncodeObject = Kernel::System::Encode->new(
        ConfigObject => $ConfigObject,
    );
    my $LogObject = Kernel::System::Log->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
    );
    my $MainObject = Kernel::System::Main->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        LogObject    => $LogObject,
    );
    my $DBObject = Kernel::System::DB->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        LogObject    => $LogObject,
        MainObject   => $MainObject,
    );
    my $SLAObject = Kernel::System::SLA->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        LogObject    => $LogObject,
        DBObject     => $DBObject,
        MainObject   => $MainObject,
    );

=cut

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

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

    # check needed objects
    for my $Object (qw(DBObject ConfigObject EncodeObject LogObject MainObject)) {
        $Self->{$Object} = $Param{$Object} || die "Got no $Object!";
    }
    $Self->{CheckItemObject} = Kernel::System::CheckItem->new( %{$Self} );
    $Self->{ValidObject}     = Kernel::System::Valid->new( %{$Self} );
# ---
# ITSM
# ---
    $Self->{GeneralCatalogObject} = Kernel::System::GeneralCatalog->new( %{$Self} );
# ---

    # load generator preferences module
    my $GeneratorModule = $Self->{ConfigObject}->Get('SLA::PreferencesModule')
        || 'Kernel::System::SLA::PreferencesDB';
    if ( $Self->{MainObject}->Require($GeneratorModule) ) {
        $Self->{PreferencesObject} = $GeneratorModule->new(%Param);
    }

    return $Self;
}

=item SLAList()

return a hash list of slas

    my %SLAList = $SLAObject->SLAList(
        ServiceID => 1,  # (optional)
        Valid     => 0,  # (optional) default 1 (0|1)
        UserID    => 1,
    );

=cut

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

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

    # set valid param
    if ( !defined $Param{Valid} ) {
        $Param{Valid} = 1;
    }

    # add ServiceID
    my %SQLTable;
    $SQLTable{sla} = 'sla s';
    my @SQLWhere;
    if ( $Param{ServiceID} ) {

        # quote
        $Param{ServiceID} = $Self->{DBObject}->Quote( $Param{ServiceID}, 'Integer' );

        $SQLTable{service} = 'service_sla r';
        push @SQLWhere, "s.id = r.sla_id AND r.service_id = $Param{ServiceID}";
    }

    # add valid part
    if ( $Param{Valid} ) {

        # create the valid list
        my $ValidIDs = join ', ', $Self->{ValidObject}->ValidIDsGet();

        push @SQLWhere, "s.valid_id IN ( $ValidIDs )";
    }

    # create the table and where strings
    my $TableString = join q{, }, values %SQLTable;
    my $WhereString = @SQLWhere ? ' WHERE ' . join q{ AND }, @SQLWhere : '';

    # ask database
    $Self->{DBObject}->Prepare(
        SQL => "SELECT s.id, s.name FROM $TableString $WhereString",
    );

    # fetch the result
    my %SLAList;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        $SLAList{ $Row[0] } = $Row[1];
    }

    return %SLAList;
}

=item SLAGet()

return a sla as hash

Return
    $SLAData{SLAID}
    $SLAData{ServiceIDs}
    $SLAData{Name}
    $SLAData{Calendar}
    $SLAData{FirstResponseTime}
    $SLAData{FirstResponseNotify}
    $SLAData{UpdateTime}
    $SLAData{UpdateNotify}
    $SLAData{SolutionTime}
    $SLAData{SolutionNotify}
    $SLAData{ValidID}
    $SLAData{Comment}
    $SLAData{CreateTime}
    $SLAData{CreateBy}
    $SLAData{ChangeTime}
    $SLAData{ChangeBy}
# ---
# ITSM
# ---
    $SLAData{TypeID}
    $SLAData{Type}
    $SLAData{MinTimeBetweenIncidents}
# ---

    my %SLAData = $SLAObject->SLAGet(
        SLAID  => 123,
        UserID => 1,
    );

=cut

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

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

    # check if result is already cached
    my $CacheKey = 'Cache::SLAGet::' . $Param{SLAID};
    if ( $Self->{$CacheKey} ) {
        return %{ $Self->{$CacheKey} };
    }

    # get sla from db
    $Self->{DBObject}->Prepare(
        SQL => 'SELECT id, name, calendar_name, first_response_time, first_response_notify, '
            . 'update_time, update_notify, solution_time, solution_notify, '
            . 'valid_id, comments, create_time, create_by, change_time, change_by '
# ---
# ITSM
# ---
            . ", type_id, min_time_bet_incidents "
# ---
            . 'FROM sla WHERE id = ?',
        Bind => [
            \$Param{SLAID},
        ],
        Limit => 1,
    );

    # fetch the result
    my %SLAData;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        $SLAData{SLAID}               = $Row[0];
        $SLAData{Name}                = $Row[1];
        $SLAData{Calendar}            = $Row[2] || '';
        $SLAData{FirstResponseTime}   = $Row[3];
        $SLAData{FirstResponseNotify} = $Row[4];
        $SLAData{UpdateTime}          = $Row[5];
        $SLAData{UpdateNotify}        = $Row[6];
        $SLAData{SolutionTime}        = $Row[7];
        $SLAData{SolutionNotify}      = $Row[8];
        $SLAData{ValidID}             = $Row[9];
        $SLAData{Comment}             = $Row[10] || '';
        $SLAData{CreateTime}          = $Row[11];
        $SLAData{CreateBy}            = $Row[12];
        $SLAData{ChangeTime}          = $Row[13];
        $SLAData{ChangeBy}            = $Row[14];
# ---
# ITSM
# ---
        $SLAData{TypeID}                  = $Row[15];
        $SLAData{MinTimeBetweenIncidents} = $Row[16] || 0;
# ---
    }

    # check sla
    if ( !$SLAData{SLAID} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "No such SLAID ($Param{SLAID})!",
        );
        return;
    }
# ---
# ITSM
# ---
    # get sla type list
    my $SLATypeList = $Self->{GeneralCatalogObject}->ItemList(
        Class => 'ITSM::SLA::Type',
    );
    $SLAData{Type} = $SLATypeList->{ $SLAData{TypeID} } || '';
# ---

    # get all service ids
    $Self->{DBObject}->Prepare(
        SQL  => 'SELECT service_id FROM service_sla WHERE sla_id = ? ORDER BY service_id ASC',
        Bind => [ \$SLAData{SLAID} ],
    );

    # fetch the result
    my @ServiceIDs;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        push @ServiceIDs, $Row[0];
    }

    # add the ids
    $SLAData{ServiceIDs} = \@ServiceIDs;

    # get queue preferences
    my %Preferences = $Self->SLAPreferencesGet( SLAID => $Param{SLAID} );

    # merge hash
    if (%Preferences) {
        %SLAData = ( %SLAData, %Preferences );
    }

    # cache the result
    $Self->{$CacheKey} = \%SLAData;

    return %SLAData;
}

=item SLALookup()

return the name or the sla id

    my $SLAName = $SLAObject->SLALookup(
        SLAID => 123,
    );

    or

    my $SLAID = $SLAObject->SLALookup(
        Name => 'SLA Name',
    );

=cut

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

    # check needed stuff
    if ( !$Param{SLAID} && !$Param{Name} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'Need SLAID or Name!',
        );
        return;
    }

    if ( $Param{SLAID} ) {

        # check cache
        my $CacheKey = 'Cache::SLALookup::ID::' . $Param{SLAID};
        if ( defined $Self->{$CacheKey} ) {
            return $Self->{$CacheKey};
        }

        # lookup
        $Self->{DBObject}->Prepare(
            SQL   => 'SELECT name FROM sla WHERE id = ?',
            Bind  => [ \$Param{SLAID}, ],
            Limit => 1,
        );

        # fetch the result
        my $Name;
        while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
            $Name = $Row[0];
        }

        # cache
        $Self->{$CacheKey} = $Name;

        return $Name;
    }
    else {

        # check cache
        my $CacheKey = 'Cache::SLALookup::Name::' . $Param{Name};
        if ( defined $Self->{$CacheKey} ) {
            return $Self->{$CacheKey};
        }

        # lookup
        $Self->{DBObject}->Prepare(
            SQL   => 'SELECT id FROM sla WHERE name = ?',
            Bind  => [ \$Param{Name} ],
            Limit => 1,
        );

        # fetch the result
        my $SLAID;
        while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
            $SLAID = $Row[0];
        }

        # cache
        $Self->{$CacheKey} = $SLAID;

        return $SLAID;
    }
}

=item SLAAdd()

add a sla

    my $SLAID = $SLAObject->SLAAdd(
        ServiceIDs          => [ 1, 5, 7 ],  # (optional)
        Name                => 'Service Name',
        Calendar            => 'Calendar1',  # (optional)
        FirstResponseTime   => 120,          # (optional)
        FirstResponseNotify => 60,           # (optional) notify agent if first response escalation is 60% reached
        UpdateTime          => 180,          # (optional)
        UpdateNotify        => 80,           # (optional) notify agent if update escalation is 80% reached
        SolutionTime        => 580,          # (optional)
        SolutionNotify      => 80,           # (optional) notify agent if solution escalation is 80% reached
        ValidID             => 1,
        Comment             => 'Comment',    # (optional)
        UserID              => 1,
# ---
# ITSM
# ---
        TypeID                  => 2,
        MinTimeBetweenIncidents => 3443,  # (optional)
# ---
    );

=cut

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

    # check needed stuff
# ---
# ITSM
# ---
#    for my $Argument (qw(Name ValidID UserID)) {
    for my $Argument (qw(Name ValidID UserID TypeID)) {
# ---
        if ( !$Param{$Argument} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    # check service ids
    if ( defined $Param{ServiceIDs} && ref $Param{ServiceIDs} ne 'ARRAY' ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'ServiceIDs need to be an array reference!',
        );
        return;
    }

    # set default values
    $Param{ServiceIDs}          ||= [];
    $Param{Calendar}            ||= '';
    $Param{Comment}             ||= '';
    $Param{FirstResponseTime}   ||= 0;
    $Param{FirstResponseNotify} ||= 0;
    $Param{UpdateTime}          ||= 0;
    $Param{UpdateNotify}        ||= 0;
    $Param{SolutionTime}        ||= 0;
    $Param{SolutionNotify}      ||= 0;
# ---
# ITSM
# ---
    $Param{MinTimeBetweenIncidents} ||= 0;
# ---

    # cleanup given params
    for my $Argument (qw(Name Comment)) {
        $Self->{CheckItemObject}->StringClean(
            StringRef         => \$Param{$Argument},
            RemoveAllNewlines => 1,
            RemoveAllTabs     => 1,
        );
    }

    # find exiting sla's with the same name
    $Self->{DBObject}->Prepare(
        SQL   => 'SELECT id FROM sla WHERE name = ?',
        Bind  => [ \$Param{Name} ],
        Limit => 1,
    );

    # fetch the result
    my $NoAdd;
    while ( $Self->{DBObject}->FetchrowArray() ) {
        $NoAdd = 1;
    }

    # abort insert of new sla, if name already exists
    if ($NoAdd) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't add new SLA! '$Param{Name}' already exists.",
        );
        return;
    }

    # add sla to database
    return if !$Self->{DBObject}->Do(
# ---
# ITSM
# ---
#        SQL => 'INSERT INTO sla '
#            . '(name, calendar_name, first_response_time, first_response_notify, '
#            . 'update_time, update_notify, solution_time, solution_notify, '
#            . 'valid_id, comments, create_time, create_by, change_time, change_by) VALUES '
#            . '(?, ?, ?, ?, ?, ?, ?, ?, ?, ?, current_timestamp, ?, current_timestamp, ?)',
#        Bind => [
#            \$Param{Name},                \$Param{Calendar},   \$Param{FirstResponseTime},
#            \$Param{FirstResponseNotify}, \$Param{UpdateTime}, \$Param{UpdateNotify},
#            \$Param{SolutionTime}, \$Param{SolutionNotify}, \$Param{ValidID}, \$Param{Comment},
#            \$Param{UserID}, \$Param{UserID},
#        ],
        SQL => 'INSERT INTO sla '
            . '(name, calendar_name, first_response_time, first_response_notify, '
            . 'update_time, update_notify, solution_time, solution_notify, '
            . 'valid_id, comments, create_time, create_by, change_time, change_by, '
            . 'type_id, min_time_bet_incidents) VALUES '
            . '(?, ?, ?, ?, ?, ?, ?, ?, ?, ?, current_timestamp, ?, current_timestamp, ?, ?, ?)',
        Bind => [
            \$Param{Name},                \$Param{Calendar},   \$Param{FirstResponseTime},
            \$Param{FirstResponseNotify}, \$Param{UpdateTime}, \$Param{UpdateNotify},
            \$Param{SolutionTime}, \$Param{SolutionNotify}, \$Param{ValidID}, \$Param{Comment},
            \$Param{UserID}, \$Param{UserID}, \$Param{TypeID}, \$Param{MinTimeBetweenIncidents},
        ],
# ---
    );

    # get sla id
    return if !$Self->{DBObject}->Prepare(
        SQL   => 'SELECT id FROM sla WHERE name = ?',
        Bind  => [ \$Param{Name} ],
        Limit => 1,
    );

    # fetch the result
    my $SLAID;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        $SLAID = $Row[0];
    }

    # check sla id
    if ( !$SLAID ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't find SLAID for '$Param{Name}'!",
        );
        return;
    }

    # remove all existing allocations
    $Self->{DBObject}->Do(
        SQL  => 'DELETE FROM service_sla WHERE sla_id = ?',
        Bind => [ \$SLAID ],
    );

    # add the new allocations
    for my $ServiceID ( @{ $Param{ServiceIDs} } ) {

        # add one allocation
        $Self->{DBObject}->Do(
            SQL => 'INSERT INTO service_sla (service_id, sla_id) VALUES (?, ?)',
            Bind => [ \$ServiceID, \$SLAID ],
        );
    }

    return $SLAID;
}

=item SLAUpdate()

update a existing sla

    my $True = $SLAObject->SLAUpdate(
        SLAID               => 2,
        ServiceIDs          => [ 1, 2, 3 ],  # (optional)
        Name                => 'Service Name',
        Calendar            => 'Calendar1',  # (optional)
        FirstResponseTime   => 120,          # (optional)
        FirstResponseNotify => 60,           # (optional) notify agent if first response escalation is 60% reached
        UpdateTime          => 180,          # (optional)
        UpdateNotify        => 80,           # (optional) notify agent if update escalation is 80% reached
        SolutionTime        => 580,          # (optional)
        SolutionNotify      => 80,           # (optional) notify agent if solution escalation is 80% reached
        ValidID             => 1,
        Comment             => 'Comment',    # (optional)
        UserID              => 1,
# ---
# ITSM
# ---
        TypeID                  => 2,
        MinTimeBetweenIncidents => 3443,  # (optional)
# ---
    );

=cut

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

    # check needed stuff
# ---
# ITSM
# ---
#    for my $Argument (qw(SLAID Name ValidID UserID)) {
    for my $Argument (qw(SLAID Name ValidID UserID TypeID)) {
# ---
        if ( !$Param{$Argument} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    # check service ids
    if ( defined $Param{ServiceIDs} && ref $Param{ServiceIDs} ne 'ARRAY' ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'ServiceIDs need to be an array reference!',
        );
        return;
    }

    # set default values
    $Param{ServiceIDs}          ||= [];
    $Param{Calendar}            ||= '';
    $Param{Comment}             ||= '';
    $Param{FirstResponseTime}   ||= 0;
    $Param{FirstResponseNotify} ||= 0;
    $Param{UpdateTime}          ||= 0;
    $Param{UpdateNotify}        ||= 0;
    $Param{SolutionTime}        ||= 0;
    $Param{SolutionNotify}      ||= 0;
# ---
# ITSM
# ---
    $Param{MinTimeBetweenIncidents} ||= 0;
# ---

    # cleanup given params
    for my $Argument (qw(Name Comment)) {
        $Self->{CheckItemObject}->StringClean(
            StringRef         => \$Param{$Argument},
            RemoveAllNewlines => 1,
            RemoveAllTabs     => 1,
        );
    }

    # find exiting sla's with the same name
    return if !$Self->{DBObject}->Prepare(
        SQL   => 'SELECT id FROM sla WHERE name = ?',
        Bind  => [ \$Param{Name} ],
        Limit => 1,
    );

    # fetch the result
    my $Update = 0;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        if ( $Row[0] != $Param{SLAID} ) {
            $Update = $Row[0];
        }
    }

    # abort update of sla, if name already exists
    if ($Update) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't update SLA! '$Param{Name}' already exists.",
        );
        return;
    }

    # reset cache
    delete $Self->{ 'Cache::SLAGet::' . $Param{SLAID} };
    delete $Self->{ 'Cache::SLALookup::Name::' . $Param{Name} };
    delete $Self->{ 'Cache::SLALookup::ID::' . $Param{SLAID} };

    # update service
    return if !$Self->{DBObject}->Do(
# ---
# ITSM
# ---
#        SQL => 'UPDATE sla SET name = ?, calendar_name = ?, '
#            . 'first_response_time = ?, first_response_notify = ?, '
#            . 'update_time = ?, update_notify = ?, solution_time = ?, solution_notify = ?, '
#            . 'valid_id = ?, comments = ?, change_time = current_timestamp, change_by = ? '
#            . 'WHERE id = ?',
#        Bind => [
#            \$Param{Name},                \$Param{Calendar},   \$Param{FirstResponseTime},
#            \$Param{FirstResponseNotify}, \$Param{UpdateTime}, \$Param{UpdateNotify},
#            \$Param{SolutionTime}, \$Param{SolutionNotify}, \$Param{ValidID}, \$Param{Comment},
#            \$Param{UserID}, \$Param{SLAID},
#        ],
        SQL => 'UPDATE sla SET name = ?, calendar_name = ?, '
            . 'first_response_time = ?, first_response_notify = ?, '
            . 'update_time = ?, update_notify = ?, solution_time = ?, solution_notify = ?, '
            . 'valid_id = ?, comments = ?, change_time = current_timestamp, change_by = ?, '
            . 'type_id = ?, min_time_bet_incidents = ? '
            . 'WHERE id = ?',
        Bind => [
            \$Param{Name},                \$Param{Calendar},   \$Param{FirstResponseTime},
            \$Param{FirstResponseNotify}, \$Param{UpdateTime}, \$Param{UpdateNotify},
            \$Param{SolutionTime}, \$Param{SolutionNotify}, \$Param{ValidID}, \$Param{Comment},
            \$Param{UserID}, \$Param{TypeID}, \$Param{MinTimeBetweenIncidents}, \$Param{SLAID},
        ],
# ---
    );

    # remove all existing allocations
    return if !$Self->{DBObject}->Do(
        SQL  => 'DELETE FROM service_sla WHERE sla_id = ?',
        Bind => [ \$Param{SLAID}, ]
    );

    # add the new allocations
    for my $ServiceID ( @{ $Param{ServiceIDs} } ) {

        # add one allocation
        return if !$Self->{DBObject}->Do(
            SQL => 'INSERT INTO service_sla (service_id, sla_id) VALUES (?, ?)',
            Bind => [ \$ServiceID, \$Param{SLAID} ],
        );
    }

    return 1;
}

=item SLAPreferencesSet()

set queue preferences

    $SLAObject->SLAPreferencesSet(
        SLAID => 123,
        Key       => 'UserComment',
        Value     => 'some comment',
        UserID    => 123,
    );

=cut

sub SLAPreferencesSet {
    my $Self = shift;

    return $Self->{PreferencesObject}->SLAPreferencesSet(@_);
}

=item SLAPreferencesGet()

get queue preferences

    my %Preferences = $SLAObject->SLAPreferencesGet(
        SLAID => 123,
        UserID    => 123,
    );

=cut

sub SLAPreferencesGet {
    my $Self = shift;

    return $Self->{PreferencesObject}->SLAPreferencesGet(@_);
}

1;

=back

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (http://otrs.org/).

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

=cut

=head1 VERSION

$Revision: 1.5 $ $Date: 2009/07/21 00:12:08 $

=cut

# --
# Kernel/System/Stats.pm - all stats core functions
# Copyright (C) 2001-2010 OTRS AG, http://otrs.org/
# --
# $Id: Stats.pm,v 1.3 2010/03/01 12:17:23 ub Exp $
# $OldId: Stats.pm,v 1.80.2.1 2009/09/28 13:12:39 ub Exp $
# --
# 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::Stats;

use strict;
use warnings;

use MIME::Base64;
use Date::Pcalc qw(:all);
use Kernel::System::XML;

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

=head1 SYNOPSIS

All stats functions.

=head1 PUBLIC INTERFACE

=over 4

=item new()

create an object

    use Kernel::Config;
    use Kernel::System::Encode;
    use Kernel::System::Log;
    use Kernel::System::Main;
    use Kernel::System::DB;
    use Kernel::System::Group;
    use Kernel::System::Time;
    use Kernel::System::CSV;
    use Kernel::System::User;
    use Kernel::System::Stats;

    my $ConfigObject = Kernel::Config->new();
    my $EncodeObject = Kernel::System::Encode->new(
        ConfigObject => $ConfigObject,
    );
    my $LogObject = Kernel::System::Log->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
    );
    my $MainObject = Kernel::System::Main->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        LogObject    => $LogObject,
    );
    my $DBObject = Kernel::System::DB->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        LogObject    => $LogObject,
        MainObject   => $MainObject,
    );
    my $GroupObject = Kernel::System::Group->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
        DBObject     => $DBObject,
    );
    my $TimeObject = Kernel::System::Time->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
    );
    my $CSVObject = Kernel::System::CSV->new(
        LogObject => $LogObject,
    );
    my $UserObject = Kernel::System::User->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
        MainObject   => $MainObject,
        TimeObject   => $TimeObject,
        DBObject     => $DBObject,
        EncodeObject => $EncodeObject,
    );
    my $StatsObject = Kernel::System::Stats->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        LogObject    => $LogObject,
        DBObject     => $DBObject,
        MainObject   => $MainObject,
        CSVObject    => $CSVObject,
        TimeObject   => $TimeObject,
        GroupObject  => $GroupObject,
        UserObject   => $UserObject,
        UserID       => 123,
    );

=cut

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

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

    # check objects list for completeness
    for my $Object (
        qw(
        ConfigObject LogObject UserID GroupObject UserObject TimeObject MainObject CSVObject
        DBObject EncodeObject
        )
        )
    {
        $Self->{$Object} = $Param{$Object} || die "Got no $Object!";
    }

    # create supplementary objects
    $Self->{XMLObject} = Kernel::System::XML->new(%Param);

    # temporary directory
    $Self->{StatsTempDir} = $Self->{ConfigObject}->Get('Home') . '/var/stats/';

    return $Self;
}

=item StatsAdd()

add new stats

    my $StatID = $StatsObject->StatsAdd();

=cut

sub StatsAdd {
    my $Self = shift;

    my $StatID = 1;

    # get new StatID
    my @Keys = $Self->{XMLObject}->XMLHashSearch( Type => 'Stats', );
    if (@Keys) {
        my @SortKeys = sort { $a <=> $b } @Keys;
        $StatID = $SortKeys[-1] + 1;
    }

    # requesting current time stamp
    my $TimeStamp = $Self->{TimeObject}->SystemTime2TimeStamp(
        SystemTime => $Self->{TimeObject}->SystemTime(),
    );

    # meta tags
    my %MetaData = ();
    $MetaData{Created}[0]{Content}   = $TimeStamp;
    $MetaData{CreatedBy}[0]{Content} = $Self->{UserID};
    $MetaData{Changed}[0]{Content}   = $TimeStamp;
    $MetaData{ChangedBy}[0]{Content} = $Self->{UserID};
    $MetaData{Valid}[0]{Content}     = 1;
    $MetaData{StatNumber}[0]{Content}
        = $StatID + $Self->{ConfigObject}->Get('Stats::StatsStartNumber');

    # start new stats record
    my @XMLHash;    # it's an array but the wording is hash
    $XMLHash[0]{otrs_stats}[0] = \%MetaData;
    if (
        !$Self->{XMLObject}->XMLHashAdd(
            Type    => 'Stats',
            Key     => $StatID,
            XMLHash => \@XMLHash,
        )
        )
    {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'StatsAdd: Can not add a new Stat!',
        );
        return 0;
    }

    return $StatID;
}

=item StatsGet()

get a hash ref of the stats you need

    my $HashRef = $StatsObject->StatsGet(
        StatID             => '123',
        NoObjectAttributes => 1,       # optional
    );

=cut

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

    # check necessary data
    if ( !$Param{StatID} ) {
        $Self->{LogObject}->Log( Priority => 'error', Message => 'StatsGet: Need StatID!' );
    }

    # get hash from storage
    my @XMLHash = $Self->{XMLObject}->XMLHashGet(
        Type => 'Stats',
        Key  => $Param{StatID},
    );

    if ( !$XMLHash[0] ) {
        $Self->{LogObject}->Log( Priority => 'error', Message => "StatsGet: Can\'t get Stat!" );
        return 0;
    }

    my %Stat    = ();
    my $StatXML = $XMLHash[0]{otrs_stats}[1];

    # process all strings
    $Stat{StatID} = $Param{StatID};
    for my $Key (
        qw(Title Object File Description SumRow SumCol StatNumber
        Cache StatType Valid ObjectModule CreatedBy ChangedBy Created Changed
        )
        )
    {
        if ( defined $StatXML->{$Key}[1]{Content} ) {
            $Stat{$Key} = $StatXML->{$Key}[1]{Content};
        }
    }

    # process all arrays
    KEY:
    for my $Key (qw(Permission Format GraphSize)) {
        next KEY if !$StatXML->{$Key}[1]{Content};

        $Stat{$Key} = ();
        for my $Index ( 1 .. $#{ $StatXML->{$Key} } ) {
            push @{ $Stat{$Key} }, $StatXML->{$Key}[$Index]->{Content};
        }
    }

    # get the configuration elements of the dynamic stats
    # %Allowed is used to avoid douple selection in different forms
    my %Allowed     = ();
    my %TimeAllowed = ();
    my $TimeElement = $Self->{ConfigObject}->Get('Stats::TimeElement') || 'Time';
    return \%Stat if !$Stat{Object};

    $Stat{ObjectName} = $Self->GetObjectName(
        ObjectModule => $Stat{ObjectModule},
    );

    return \%Stat if $Param{NoObjectAttributes};

    KEY:
    for my $Key (qw(UseAsXvalue UseAsValueSeries UseAsRestriction)) {

        # @StatAttributesSimplified give you arrays without undef array elements
        my @StatAttributesSimplified = ();

        # get the attributes of the object
        my @ObjectAttributes = $Self->GetStatsObjectAttributes(
            ObjectModule => $Stat{ObjectModule},
            Use          => $Key,
        );

        next KEY if !@ObjectAttributes;

        ATTRIBUTE:
        for my $Attribute (@ObjectAttributes) {
            my $Element = $Attribute->{Element};
            if ( $Attribute->{Block} eq 'Time' ) {
                if ( $Key eq 'UseAsValueSeries' ) {
                    if (
                        $Allowed{$Element}
                        && $Allowed{$Element} == 1
                        )
                    {
                        $Allowed{$Element}     = 0;
                        $TimeAllowed{$Element} = 1;
                    }
                    else {
                        $Allowed{$Element} = 1;
                    }
                }
                elsif ( $Key eq 'UseAsRestriction' ) {
                    if (
                        $TimeAllowed{$Element}
                        && $TimeAllowed{$Element} == 1
                        )
                    {
                        $Allowed{$Element} = 1;
                    }
                    else {
                        $Allowed{$Element} = 0;
                    }
                }
            }
            next ATTRIBUTE if $Allowed{$Element};

            if ( $StatXML->{$Key} ) {
                my @StatAttributes = @{ $StatXML->{$Key} };
                if ( !$StatAttributes[0] ) {
                    shift @StatAttributes;
                }
                REF:
                for my $Ref (@StatAttributes) {
                    if ( !defined $Attribute->{Translation} ) {
                        $Attribute->{Translation} = 1;
                    }

                    next REF
                        if !(
                        $Element
                        && $Ref->{Element}
                        && $Element eq $Ref->{Element}
                        );

                    # if selected elements exit, add the information to the StatAttributes
                    $Attribute->{Selected} = 1;
                    if ( $Ref->{Fixed} ) {
                        $Attribute->{Fixed} = 1;
                    }

                    for my $Index ( 1 .. $#{ $Ref->{SelectedValues} } ) {
                        push(
                            @{ $Attribute->{SelectedValues} },
                            $Ref->{SelectedValues}[$Index]->{Content}
                        );
                    }

                    # stettings for working with time elements
                    for (
                        qw(TimeStop TimeStart TimeRelativeUnit
                        TimeRelativeCount TimeScaleCount
                        )
                        )
                    {
# ---
# ITSM
# ---
#                        if ( $Ref->{$_} ) {
                        if ( $Ref->{$_} && ( !$Attribute->{$_} || $Ref->{Fixed} ) ) {
# ---
                            $Attribute->{$_} = $Ref->{$_};
                        }
                    }
                    $Allowed{$Element} = 1;
                }
            }
            push @StatAttributesSimplified, $Attribute;

        }
        $Stat{$Key} = \@StatAttributesSimplified;
    }

    return \%Stat;
}

=item StatsUpdate()

update a stat

    $StatsObject->StatsUpdate(
        StatID => '123',
        Hash   => \%Hash
    );

=cut

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

    # declaration of the hash
    my %StatXML = ();

    # check necessary data
    if ( !$Param{StatID} ) {
        $Self->{LogObject}->Log( Priority => 'error', Message => 'StatsUpdate: Need StatID!' );
    }

    # requesting stats reference
    my $StatOld = $Self->StatsGet( StatID => $Param{StatID} );
    if ( !$StatOld ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message =>
                "StatsUpddate: Can't get stats, perhaps you have an invalid stats id! (StatsID => $Param{StatID})"
        );
        return 0;
    }

    # declare variable
    my $StatNew = $Param{Hash};

    # a delete function can be the better solution
    for my $Key (qw(UseAsXvalue UseAsValueSeries UseAsRestriction)) {
        for ( @{ $StatOld->{$Key} } ) {
            if ( !$_->{Selected} ) {
                $_ = undef;
            }
        }
    }

    # adopt changes
    for my $Key ( keys %{$StatNew} ) {
        $StatOld->{$Key} = $StatNew->{$Key};
    }

    for my $Key ( keys %{$StatOld} ) {
        if ( $Key eq 'UseAsXvalue' || $Key eq 'UseAsValueSeries' || $Key eq 'UseAsRestriction' ) {
            my $Index = 0;
            REF:
            for my $Ref ( @{ $StatOld->{$Key} } ) {
                next REF if !$Ref;

                $Index++;
                $StatXML{$Key}[$Index]{Element} = $Ref->{Element};
                $StatXML{$Key}[$Index]{Fixed}   = $Ref->{Fixed};
                my $SubIndex = 0;
                for my $Value ( @{ $Ref->{SelectedValues} } ) {
                    $SubIndex++;
                    $StatXML{$Key}[$Index]{SelectedValues}[$SubIndex]{Content} = $Value;
                }

                # stettings for working with time elements
                for (qw(TimeStop TimeStart TimeRelativeUnit TimeRelativeCount TimeScaleCount)) {
                    if ( $Ref->{$_} ) {
                        $StatXML{$Key}[$Index]{$_} = $Ref->{$_};
                    }
                }
            }
        }
        elsif ( ref( $StatOld->{$Key} ) eq 'ARRAY' ) {
            for my $Index ( 0 .. $#{ $StatOld->{$Key} } ) {
                $StatXML{$Key}[$Index]{Content} = $StatOld->{$Key}[$Index];
            }
        }
        else {
            if ( defined $StatOld->{$Key} ) {
                $StatXML{$Key}[1]{Content} = $StatOld->{$Key};
            }
        }
    }

    # meta tags
    my $TimeStamp = $Self->{TimeObject}->SystemTime2TimeStamp(
        SystemTime => $Self->{TimeObject}->SystemTime(),
    );
    $StatXML{Changed}[1]{Content}   = $TimeStamp;
    $StatXML{ChangedBy}[1]{Content} = $Self->{UserID};

    # please don't change the functionality of XMLHashDelete and XMLHashAdd
    # into the new function XMLHashUpdate, there is an incompatibility.
    # Perhaps there are intricacies because of the 'Array[0] = undef' definition

    # delete the old record
    if (
        !$Self->{XMLObject}->XMLHashDelete(
            Type => 'Stats',
            Key  => $Param{StatID},
        )
        )
    {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "StatsUpddate: Can't delete XMLHash!"
        );
        return 0;
    }

    # delete cache
    $Self->_DeleteCache( StatID => $Param{StatID} );

    my @Array = ();
    $Array[0]{otrs_stats}[0] = \%StatXML;

    # add the revised record
    if (
        $Self->{XMLObject}->XMLHashAdd(
            Type    => 'Stats',
            Key     => $Param{StatID},
            XMLHash => \@Array
        )
        )
    {
        return 1;
    }
    $Self->{LogObject}->Log( Priority => 'error', Message => "StatsUpddate: Can't add XMLHash!" );

    return 0;
}

=item StatsDelete()

delete a stats

    $StatsObject->StatsDelete(StatID => '123');

=cut

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

    # check necessary data
    if ( !$Param{StatID} ) {
        $Self->{LogObject}->Log( Priority => 'error', Message => "StatsDelete: Need StatID!" );
    }

    # delete the record
    my $Result = $Self->{XMLObject}->XMLHashDelete(
        Type => 'Stats',
        Key  => $Param{StatID},
    );

    # error handling
    if ( !$Result ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "StatsDelete: Can't delete XMLHash!",
        );
        return 0;
    }

    # delete cache
    $Self->_DeleteCache( StatID => $Param{StatID} );

    # get list of installed stats files
    my @StatsFileList = glob $Self->{StatsTempDir} . '*.xml.installed';

    # delete the .installed file in temp dir
    FILE:
    for my $File ( sort @StatsFileList ) {

        # read file content
        my $StatsIDRef = $Self->{MainObject}->FileRead(
            Location => $File,
        );

        next FILE if !$StatsIDRef;
        next FILE if ref $StatsIDRef ne 'SCALAR';
        next FILE if !${$StatsIDRef};

        next FILE if ${$StatsIDRef} ne $Param{StatID};

        # delete .installed file
        $Self->{MainObject}->FileDelete(
            Location => $File,
        );
    }

    # add log message
    $Self->{LogObject}->Log(
        Priority => 'notice',
        Message  => "Delete stats (StatsID = $Param{StatID})",
    );

    return 1;
}

=item GetStatsList()

lists all stats id's

    my $ArrayRef = $StatsObject->GetStatsList(
        OrderBy   => 'ID' || 'Title' || 'Object', # optional
        Direction => 'ASC' || 'DESC',             # optional
    );

=cut

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

    my @SearchResult = ();
    if ( !( @SearchResult = $Self->{XMLObject}->XMLHashSearch( Type => 'Stats' ) ) ) {
        $Self->_AutomaticSampleImport();
        return () if !( @SearchResult = $Self->{XMLObject}->XMLHashSearch( Type => 'Stats' ) );
    }

    # get user groups
    my @Groups = $Self->{GroupObject}->GroupMemberList(
        UserID => $Self->{UserID},
        Type   => 'ro',
        Result => 'ID',
    );

    $Param{OrderBy} ||= 'ID';

    # a solution with more performance is useful
    my %ResultHash = ();
    for my $StatID (@SearchResult) {
        my $Stat = $Self->StatsGet(
            StatID             => $StatID,
            NoObjectAttributes => 1,
        );
        my $UserPermission = 0;
        if ( $Self->{AccessRw} || $Self->{UserID} == 1 ) {
            $UserPermission = 1;
        }

        # these function is similar like other function in the code perhaps we should
        # merge them
        # permission check
        elsif ( $Stat->{Valid} ) {
            MARKE:
            for my $GroupID ( @{ $Stat->{Permission} } ) {
                for my $UserGroup (@Groups) {
                    if ( $GroupID == $UserGroup ) {
                        $UserPermission = 1;
                        last MARKE;
                    }
                }
            }
        }
        if ( $UserPermission == 1 ) {

            # order by title
            if ( $Param{OrderBy} eq 'Title' ) {
                $ResultHash{$StatID} = $Stat->{Title} || '';
            }

            # order by object
            elsif ( $Param{OrderBy} eq 'Object' ) {
                $ResultHash{$StatID} = $Stat->{Object} || '';
            }

            # order by id
            else {
                $ResultHash{$StatID} = int $StatID;
            }
        }
    }
    my @SortArray = ();
    if ( $Param{OrderBy} eq 'ID' ) {
        @SortArray = sort { $ResultHash{$a} <=> $ResultHash{$b} } keys %ResultHash;
    }
    else {
        @SortArray = sort { $ResultHash{$a} cmp $ResultHash{$b} } keys %ResultHash;
    }
    if ( $Param{Direction} && $Param{Direction} eq 'DESC' ) {
        @SortArray = reverse @SortArray;
    }

    return \@SortArray;
}

=item SumBuild()

build sum in x or/and y axis

    $StatArray = $StatsObject->SumBuild(
        Array => \@Result,
        SumRow => 1,
        SumCol => 0,
    );

=cut

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

    my @Data = @{ $Param{Array} };

    # add sum y
    if ( $Param{SumRow} ) {
        push @{ $Data[1] }, 'Sum';
        for my $Index1 ( 2 .. $#Data ) {
            my $Sum = 0;
            for my $Index2 ( 1 .. $#{ $Data[$Index1] } ) {
                if ( $Data[$Index1][$Index2] =~ m{^-?\d+(\.\d+)?$} ) {
                    $Sum += $Data[$Index1][$Index2];
                }
            }
            push @{ $Data[$Index1] }, $Sum;
        }
    }

    # add sum x
    if ( $Param{SumCol} ) {
        my @SumRow = ();
        $SumRow[0] = 'Sum';
        for my $Index1 ( 2 .. $#Data ) {
            for my $Index2 ( 1 .. $#{ $Data[$Index1] } ) {
                if ( $Data[$Index1][$Index2] =~ m{^-?\d+(\.\d+)?$} ) {
                    $SumRow[$Index2] += $Data[$Index1][$Index2];
                }
            }
        }
        push @Data, \@SumRow;
    }
    return \@Data;
}

#=item _GenerateStaticStats()
#
#    take the stat configuration and get the stat table
#
#    my @StatArray = $StatsObject->_GenerateStaticStats(
#        ObjectModule     => $Stat->{ObjectModule},
#        GetParam         => $Param{GetParam},
#        Title            => $Stat->{Title},
#        StatID           => $Stat->{StatID},
#        Cache            => $Stat->{Cache},
#    );
#
#=cut

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

    # check needed params
    NEED:
    for my $Need (qw(ObjectModule GetParam Title StatID)) {
        next NEED if $Param{$Need};
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "_GenerateStaticStats: Need $Need!"
        );
        return;
    }

    # load static modul
    my $ObjectModule = $Param{ObjectModule};
    $Self->{MainObject}->Require($ObjectModule);
    my $StatObject = $ObjectModule->new( %{$Self} );

    my @Result   = ();
    my %GetParam = %{ $Param{GetParam} };

    # use result cache if configured
    if ( $Param{Cache} ) {
        my $Filename = $Self->_CreateStaticResultCacheFilename(
            GetParam => \%GetParam,
            StatID   => $Param{StatID},
        );

        @Result = $Self->_GetResultCache( Filename => $Filename );
    }

    # try to get data if noting is there
    if ( !@Result ) {

        # run stats function
        @Result = $StatObject->Run(
            %GetParam,

            # these two lines are requirements of me, perhaps this
            # information is needed for former static stats
            Format => $Param{Format}[0],
            Module => $Param{ObjectModule},
        );

        # write cache if configured
        if ( $Param{Cache} ) {
            $Self->_WriteResultCache(
                GetParam => \%GetParam,
                StatID   => $Param{StatID},
                Data     => \@Result,
            );
        }
    }
    $Result[0][0] = $Param{Title} . " " . $Result[0][0];

    return @Result;
}

#=item _GenerateDynamicStats()
#
#    take the stat configuration and get the stat table
#
#    my @StatArray = $StatsObject->_GenerateDynamicStats(
#        ObjectModule     => 'Kernel::System::Stats::Dynamic::Ticket',
#        Object           => 'Ticket',
#        UseAsXvalue      => \UseAsXvalueElements,
#        UseAsValueSeries => \UseAsValueSeriesElements,
#        UseAsRestriction => \UseAsRestrictionElements,
#        Title            => 'TicketStat',
#        StatID           => 123,
#        Cache            => 1,      # optional
#    );
#
#=cut

# search for a better way to cache stats (see lines before StatID and Cache)

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

    my @HeaderLine     = ();
    my $TitleTimeStart = '';
    my $TitleTimeStop  = '';

    NEED:
    for my $Need (qw(ObjectModule UseAsXvalue UseAsValueSeries Title Object StatID)) {
        next NEED if $Param{$Need};
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "_GenerateDynamicStats: Need $Need!"
        );
        return;
    }

    # include the needed dynamic object
    my $ObjectModule = $Param{ObjectModule};
    $Self->{MainObject}->Require($ObjectModule);
    my $StatObject = $ObjectModule->new( %{$Self} );

    # get the selected values
    # perhaps i can split the StatGet function to make this needless
    # Problem, i need the block information
    my %NewParam = ();

    $NewParam{Title}        = $Param{Title};
    $NewParam{Object}       = $Param{Object};
    $NewParam{ObjectModule} = $Param{ObjectModule};

    # search for a better way to cache stats (StatID and Cache)
    $NewParam{StatID} = $Param{StatID};
    $NewParam{Cache}  = $Param{Cache};
    for my $Use (qw(UseAsRestriction UseAsXvalue UseAsValueSeries)) {
        my @Array = @{ $Param{$Use} };
        ELEMENT:
        for my $Element (@Array) {
            next ELEMENT if !$Element->{Selected};

            delete $Element->{Selected};
            delete $Element->{Fixed};
            if ( $Element->{Block} eq 'Time' ) {
                delete $Element->{TimePeriodFormat};
                if ( $Element->{TimeRelativeUnit} ) {
                    my ( $s, $m, $h, $D, $M, $Y )
                        = $Self->{TimeObject}->SystemTime2Date(
                        SystemTime => $Self->{TimeObject}->SystemTime(),
                        );

                    my $Count = $Element->{TimeRelativeCount} ? $Element->{TimeRelativeCount} : 1;

                    # -1 because the current time will be not counted
                    $Count -= 1;

                    if ( $Element->{TimeRelativeUnit} eq 'Year' ) {
                        ( $Y, $M, $D ) = Add_Delta_YMD( $Y, $M, $D, -1, 0, 0 );
                        $Element->{TimeStop}
                            = sprintf( "%04d-%02d-%02d %02d:%02d:%02d", $Y, 12, 31, 23, 59, 59 );
                        ( $Y, $M, $D ) = Add_Delta_YMD( $Y, $M, $D, -$Count, 0, 0 );
                        $Element->{TimeStart}
                            = sprintf( "%04d-%02d-%02d %02d:%02d:%02d", $Y, 1, 1, 0, 0, 0 );
                    }
                    elsif ( $Element->{TimeRelativeUnit} eq 'Month' ) {
                        ( $Y, $M, $D ) = Add_Delta_YMD( $Y, $M, $D, 0, -1, 0 );
                        $Element->{TimeStop} = sprintf(
                            "%04d-%02d-%02d %02d:%02d:%02d",
                            $Y, $M, Days_in_Month( $Y, $M ),
                            23, 59, 59
                        );
                        ( $Y, $M, $D ) = Add_Delta_YMD( $Y, $M, $D, 0, -$Count, 0 );
                        $Element->{TimeStart}
                            = sprintf( "%04d-%02d-%02d %02d:%02d:%02d", $Y, $M, 1, 0, 0, 0 );
                    }
                    elsif ( $Element->{TimeRelativeUnit} eq 'Day' ) {
                        ( $Y, $M, $D ) = Add_Delta_YMD( $Y, $M, $D, 0, 0, -1 );
                        $Element->{TimeStop}
                            = sprintf( "%04d-%02d-%02d %02d:%02d:%02d", $Y, $M, $D, 23, 59, 59 );
                        ( $Y, $M, $D ) = Add_Delta_YMD( $Y, $M, $D, 0, 0, -$Count );
                        $Element->{TimeStart}
                            = sprintf( "%04d-%02d-%02d %02d:%02d:%02d", $Y, $M, $D, 0, 0, 0 );
                    }
                    elsif ( $Element->{TimeRelativeUnit} eq 'Hour' ) {
                        ( $Y, $M, $D, $h, $m, $s )
                            = Add_Delta_DHMS( $Y, $M, $D, $h, $m, $s, 0, -1, 0, 0 );
                        $Element->{TimeStop}
                            = sprintf( "%04d-%02d-%02d %02d:%02d:%02d", $Y, $M, $D, $h, 59, 59 );
                        ( $Y, $M, $D, $h, $m, $s )
                            = Add_Delta_DHMS( $Y, $M, $D, $h, $m, $s, 0, -$Count, 0, 0 );
                        $Element->{TimeStart}
                            = sprintf( "%04d-%02d-%02d %02d:%02d:%02d", $Y, $M, $D, $h, 0, 0 );
                    }
                    elsif ( $Element->{TimeRelativeUnit} eq 'Minute' ) {
                        ( $Y, $M, $D, $h, $m, $s )
                            = Add_Delta_DHMS( $Y, $M, $D, $h, $m, $s, 0, 0, -1, 0 );
                        $Element->{TimeStop}
                            = sprintf( "%04d-%02d-%02d %02d:%02d:%02d", $Y, $M, $D, $h, $m, 59 );
                        ( $Y, $M, $D, $h, $m, $s )
                            = Add_Delta_DHMS( $Y, $M, $D, $h, $m, $s, 0, 0, -$Count, 0 );
                        $Element->{TimeStart}
                            = sprintf( "%04d-%02d-%02d %02d:%02d:%02d", $Y, $M, $h, $m, 0, 0 );
                    }
                    elsif ( $Element->{TimeRelativeUnit} eq 'Second' ) {
                        ( $Y, $M, $D, $h, $m, $s )
                            = Add_Delta_DHMS( $Y, $M, $D, $h, $m, $s, 0, 0, 0, -1 );
                        $Element->{TimeStop}
                            = sprintf( "%04d-%02d-%02d %02d:%02d:%02d", $Y, $M, $D, $h, $m, $s );
                        ( $Y, $M, $D, $h, $m, $s )
                            = Add_Delta_DHMS( $Y, $M, $D, $h, $m, $s, 0, 0, 0, -$Count );
                        $Element->{TimeStart}
                            = sprintf( "%04d-%02d-%02d %02d:%02d:%02d", $Y, $M, $D, $h, $m, $s );
                    }
                    delete $Element->{TimeRelativeUnit};
                    delete $Element->{TimeRelativeCount};
                }
                $TitleTimeStart = $Element->{TimeStart};
                $TitleTimeStop  = $Element->{TimeStop};
            }

            # Select All function needed from mkStats.pl and fixed values of the frontend
            elsif ( !$Element->{SelectedValues}[0] ) {
                my @Values = keys( %{ $Element->{Values} } );
                $Element->{SelectedValues} = \@Values;
            }
            push @{ $NewParam{$Use} }, $Element;
        }
    }

    %Param = %NewParam;

    # get all restrictions for the search
    my %RestrictionAttribute = ();
    for my $RestrictionPart ( @{ $Param{UseAsRestriction} } ) {
        my $Element = $RestrictionPart->{Element};
        if ( $RestrictionPart->{Block} eq 'InputField' ) {
            $RestrictionAttribute{$Element} = $RestrictionPart->{SelectedValues}[0];
        }
        elsif ( $RestrictionPart->{Block} eq 'SelectField' ) {
            $RestrictionAttribute{$Element} = $RestrictionPart->{SelectedValues}[0];
        }
        elsif ( $RestrictionPart->{Block} eq 'Time' ) {
            $RestrictionAttribute{ $RestrictionPart->{Values}{TimeStop} }
                = $RestrictionPart->{TimeStop};
            $RestrictionAttribute{ $RestrictionPart->{Values}{TimeStart} }
                = $RestrictionPart->{TimeStart};
        }
        else {
            $RestrictionAttribute{$Element} = $RestrictionPart->{SelectedValues};
        }
    }

    # get the selected Xvalue
    my $Xvalue = {};
    my (
        $VSYear,     $VSMonth,     $VSDay,     $VSHour,     $VSMinute,     $VSSecond,
        $VSStopYear, $VSStopMonth, $VSStopDay, $VSStopHour, $VSStopMinute, $VSStopSecond
    );
    my $TimeAbsolutStopUnixTime = 0;
    my $Count                   = 0;
    my $MonthArrayRef           = _MonthArray();

    my $Element = $Param{UseAsXvalue}[0];
    if ( $Element->{Block} eq 'Time' ) {
        my (
            $Year,   $Month,   $Day,   $Hour,   $Minute,   $Second,
            $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond
        );
        if ( $Element->{TimeStart} =~ m{^(\d\d\d\d)-(\d\d)-(\d\d)\s(\d\d):(\d\d):(\d\d)$}ix ) {
            $Year   = $VSYear   = $1;
            $Month  = $VSMonth  = int $2;
            $Day    = $VSDay    = int $3;
            $Hour   = $VSHour   = int $4;
            $Minute = $VSMinute = int $5;
            $Second = $VSSecond = int $6;
        }

        $TimeAbsolutStopUnixTime
            = $Self->{TimeObject}->TimeStamp2SystemTime( String => $Element->{'TimeStop'} );
        my $TimeStart = 0;
        my $TimeStop  = 0;

        $Count = $Element->{TimeScaleCount} ? $Element->{TimeScaleCount} : 1;

        # in these constellation $Count > 1 is not useful!!
        if (
            $Param{UseAsValueSeries}[0]{Block}
            && $Param{UseAsValueSeries}[0]{Block} eq 'Time'
            && $Element->{SelectedValues}[0] eq 'Day'
            )
        {
            $Count = 1;
        }

        if ( $Element->{SelectedValues}[0] eq 'Minute' ) {
            $Second = 0;
        }
        elsif ( $Element->{SelectedValues}[0] eq 'Hour' ) {
            $Second = 0;
            $Minute = 0;
        }
        elsif ( $Element->{SelectedValues}[0] eq 'Day' ) {
            $Second = 0;
            $Minute = 0;
            $Hour   = 0;
        }
        elsif ( $Element->{SelectedValues}[0] eq 'Month' ) {
            $Second = 0;
            $Minute = 0;
            $Hour   = 0;
            $Day    = 1;
        }
        elsif ( $Element->{SelectedValues}[0] eq 'Year' ) {
            $Second = 0;
            $Minute = 0;
            $Hour   = 0;
            $Day    = 1;
            $Month  = 1;
        }

        # FIXME Timeheader zusammenbauen
        while (
            !$TimeStop
            || $Self->{TimeObject}->TimeStamp2SystemTime( String => $TimeStop )
            < $TimeAbsolutStopUnixTime
            )
        {
            $TimeStart = sprintf(
                "%04d-%02d-%02d %02d:%02d:%02d",
                $Year, $Month, $Day, $Hour, $Minute, $Second
            );
            if ( $Element->{SelectedValues}[0] eq 'Second' ) {
                ( $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond )
                    = Add_Delta_DHMS(
                    $Year, $Month, $Day, $Hour, $Minute, $Second, 0, 0, 0,
                    $Count - 1
                    );
                push(
                    @HeaderLine,
                    sprintf(
                        "%02d:%02d:%02d-%02d:%02d:%02d",
                        $Hour, $Minute, $Second, $ToHour, $ToMinute, $ToSecond
                        )
                );
            }
            elsif ( $Element->{SelectedValues}[0] eq 'Minute' ) {
                ( $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond )
                    = Add_Delta_DHMS(
                    $Year, $Month, $Day, $Hour, $Minute, $Second, 0, 0, $Count,
                    -1
                    );
                push(
                    @HeaderLine,
                    sprintf(
                        "%02d:%02d:%02d-%02d:%02d:%02d",
                        $Hour, $Minute, $Second, $ToHour, $ToMinute, $ToSecond
                        )
                );
            }
            elsif ( $Element->{SelectedValues}[0] eq 'Hour' ) {
                ( $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond )
                    = Add_Delta_DHMS(
                    $Year, $Month, $Day, $Hour, $Minute, $Second, 0, $Count, 0,
                    -1
                    );
                push(
                    @HeaderLine,
                    sprintf(
                        "%02d:%02d:%02d-%02d:%02d:%02d",
                        $Hour, $Minute, $Second, $ToHour, $ToMinute, $ToSecond
                        )
                );
            }
            elsif ( $Element->{SelectedValues}[0] eq 'Day' ) {
                ( $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond )
                    = Add_Delta_DHMS(
                    $Year, $Month, $Day, $Hour, $Minute, $Second, $Count, 0, 0,
                    -1
                    );
                my $Dow = Day_of_Week( $Year, $Month, $Day );
                $Dow = Day_of_Week_Abbreviation($Dow);
                if ( $ToDay eq $Day ) {
                    push @HeaderLine, "$Dow $Day";
                }
                else {
                    push(
                        @HeaderLine,
                        sprintf(
                            "%02d.%02d.%04d - %02d.%02d.%04d",
                            $Day, $Month, $Year, $ToDay, $ToMonth, $ToYear
                            )
                    );
                }
            }
            elsif ( $Element->{SelectedValues}[0] eq 'Month' ) {
                ( $ToYear, $ToMonth, $ToDay ) = Add_Delta_YMD( $Year, $Month, $Day, 0, $Count, 0 );
                ( $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond )
                    = Add_Delta_DHMS(
                    $ToYear, $ToMonth, $ToDay, $Hour, $Minute, $Second, 0, 0, 0,
                    -1
                    );
                if ( $ToMonth eq $Month ) {
                    push @HeaderLine, "$MonthArrayRef->[$Month] $Month";
                }
                else {
                    push(
                        @HeaderLine,
                        sprintf(
                            "%02d.%02d.%04d - %02d.%02d.%04d",
                            $Day, $Month, $Year, $ToDay, $ToMonth, $ToYear
                            )
                    );
                }
            }
            elsif ( $Element->{SelectedValues}[0] eq 'Year' ) {
                ( $ToYear, $ToMonth, $ToDay ) = Add_Delta_YMD( $Year, $Month, $Day, $Count, 0, 0 );
                ( $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond )
                    = Add_Delta_DHMS(
                    $ToYear, $ToMonth, $ToDay, $Hour, $Minute, $Second, 0, 0, 0,
                    -1
                    );
                if ( $ToYear eq $Year ) {
                    push @HeaderLine, $Year;
                }
                else {
                    push(
                        @HeaderLine,
                        sprintf(
                            "%02d.%02d.%04d - %02d.%02d.%04d",
                            $Day, $Month, $Year, $ToDay, $ToMonth, $ToYear
                            )
                    );
                }
            }
            ( $Year, $Month, $Day, $Hour, $Minute, $Second )
                = Add_Delta_DHMS(
                $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond, 0, 0, 0,
                1
                );
            $TimeStop = sprintf(
                "%04d-%02d-%02d %02d:%02d:%02d",
                $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond
            );
            push(
                @{ $Xvalue->{SelectedValues} },
                { TimeStart => $TimeStart, TimeStop => $TimeStop }
            );
        }

        # FIXME ENDE TimeHeader zusammenbaen

        $Xvalue->{Block}  = 'Time';
        $Xvalue->{Values} = $Element->{Values};
    }

    # if Block equal MultiSelectField, Selectfield
    else {
        $Xvalue = $Element;

        # build the headerline

        for my $Valuename ( @{ $Xvalue->{SelectedValues} } ) {
            push @HeaderLine, $Xvalue->{Values}{$Valuename};
        }
    }

    # get the value series
    my %ValueSeries   = ();
    my @ArraySelected = ();
    my $ColumnName    = '';

    # give me all possilbe elements for Value Series
    REF1:
    for my $Ref1 ( @{ $Param{UseAsValueSeries} } ) {

        # all elements which are shown with multiselectfields
        if ( $Ref1->{Block} ne 'Time' ) {
            my %SelectedValues = ();
            for my $Ref2 ( @{ $Ref1->{SelectedValues} } ) {
                $SelectedValues{$Ref2} = $Ref1->{Values}{$Ref2};
            }
            push(
                @ArraySelected,
                {
                    Values  => \%SelectedValues,
                    Element => $Ref1->{Element},
                    Name    => $Ref1->{Name},
                    Block   => $Ref1->{Block},
                }
            );
            next REF1;
        }

        # timescale elements need a special handling
        @HeaderLine = ();

        # these all makes only sense, if the count of xaxis is 1
        if ( $Ref1->{SelectedValues}[0] eq 'Year' ) {
            if ( $Count == 1 ) {
                for ( 1 .. 12 ) {
                    push @HeaderLine, "$MonthArrayRef->[$_] $_";
                }
            }
            else {
                for ( my $Month = 1; $Month < 12; $Month += $Count ) {
                    push(
                        @HeaderLine,
                        "$MonthArrayRef->[$Month] - $MonthArrayRef->[$Month + $Count - 1]"
                    );
                }
            }
            $VSSecond   = 0;
            $VSMinute   = 0;
            $VSHour     = 0;
            $VSDay      = 1;
            $VSMonth    = 1;
            $ColumnName = 'Year';
        }
        elsif ( $Ref1->{SelectedValues}[0] eq 'Month' ) {

            $Count = 1;
            for ( 1 .. 31 ) {
                push @HeaderLine, $_;
            }

            $VSSecond   = 0;
            $VSMinute   = 0;
            $VSHour     = 0;
            $VSDay      = 1;
            $ColumnName = 'Month';
        }
        elsif ( $Ref1->{SelectedValues}[0] eq 'Day' ) {
            for ( my $Hour = 0; $Hour < 24; $Hour += $Count ) {
                push @HeaderLine, sprintf( "%02d:00:00-%02d:59:59", $Hour, $Hour + $Count - 1 );
            }
            $VSSecond   = 0;
            $VSMinute   = 0;
            $VSHour     = 0;
            $ColumnName = 'Day';
        }
        elsif ( $Ref1->{SelectedValues}[0] eq 'Hour' ) {
            for ( my $Minute = 0; $Minute < 60; $Minute += $Count ) {
                my $Time = 'min ' . $Minute . ' - ' . ( $Minute + $Count );
                push @HeaderLine, $Time;
            }
            $VSSecond   = 0;
            $VSMinute   = 0;
            $ColumnName = 'Hour';
        }
        elsif ( $Ref1->{SelectedValues}[0] eq 'Minute' ) {
            if ( $Count == 1 ) {
                for ( 0 .. 59 ) {
                    my $Time = 'sec ' . $_;
                    push @HeaderLine, $Time;
                }
            }
            else {
                for ( my $Second = 0; $Second < 60; $Second += $Count ) {
                    my $Time = 'sec ' . $Second . '-' . ( $Second + $Count );
                    push @HeaderLine, $Time;
                }
            }
            $VSSecond   = 0;
            $ColumnName = 'Minute';
        }

        my $TimeStart     = 0;
        my $TimeStop      = 0;
        my $MonthArrayRef = _MonthArray();

        $Count = 1;

        # Generate the time value series
        my ( $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond );

        if ( $Ref1->{SelectedValues}[0] eq 'Year' ) {
            while (
                $Self->{TimeObject}->TimeStamp2SystemTime( String => $TimeStop )
                < $TimeAbsolutStopUnixTime
                )
            {
                $TimeStart = sprintf( "%04d-01-01 00:00:00", $VSYear );
                ( $ToYear, $ToMonth, $ToDay )
                    = Add_Delta_YMD( $VSYear, $VSMonth, $VSDay, $Count, 0, 0 );
                ( $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond )
                    = Add_Delta_DHMS(
                    $ToYear, $ToMonth, $ToDay, $VSHour, $VSMinute, $VSSecond, 0,
                    0, 0, -1
                    );
                $TimeStop = sprintf( "%04d-12-31 23:59:59", $ToYear );

                $ValueSeries{$VSYear} = {
                    $Ref1->{Values}{TimeStop}  => $TimeStop,
                    $Ref1->{Values}{TimeStart} => $TimeStart
                };

                ( $VSYear, $VSMonth, $VSDay, $VSHour, $VSMinute, $VSSecond )
                    = Add_Delta_DHMS(
                    $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond, 0,
                    0, 0, 1
                    );
            }
        }
        elsif ( $Ref1->{SelectedValues}[0] eq 'Month' ) {
            while (
                $Self->{TimeObject}->TimeStamp2SystemTime( String => $TimeStop )
                < $TimeAbsolutStopUnixTime
                )
            {
                $TimeStart = sprintf( "%04d-%02d-01 00:00:00", $VSYear, $VSMonth );
                ( $ToYear, $ToMonth, $ToDay )
                    = Add_Delta_YMD( $VSYear, $VSMonth, $VSDay, 0, $Count, 0 );
                ( $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond )
                    = Add_Delta_DHMS(
                    $ToYear, $ToMonth, $ToDay, $VSHour, $VSMinute, $VSSecond, 0,
                    0, 0, -1
                    );
                $TimeStop = sprintf( "%04d-%02d-%02d 23:59:59", $ToYear, $ToMonth, $ToDay );

                #                    if ($Count == 1) {
                $ValueSeries{
                    $VSYear . '-'
                        . sprintf( "%02d", $VSMonth ) . ' '
                        . $MonthArrayRef->[$VSMonth]
                    }
                    = {
                    $Ref1->{Values}{TimeStop}  => $TimeStop,
                    $Ref1->{Values}{TimeStart} => $TimeStart
                    };

                ( $VSYear, $VSMonth, $VSDay, $VSHour, $VSMinute, $VSSecond )
                    = Add_Delta_DHMS(
                    $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond, 0,
                    0, 0, 1
                    );
            }
        }
        elsif ( $Ref1->{SelectedValues}[0] eq 'Day' ) {
            while (
                $Self->{TimeObject}->TimeStamp2SystemTime( String => $TimeStop )
                < $TimeAbsolutStopUnixTime
                )
            {
                $TimeStart = sprintf( "%04d-%02d-%02d 00:00:00", $VSYear, $VSMonth, $VSDay );
                ( $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond )
                    = Add_Delta_DHMS(
                    $VSYear, $VSMonth, $VSDay, $VSHour, $VSMinute, $VSSecond,
                    $Count, 0, 0, -1
                    );
                $TimeStop = sprintf( "%04d-%02d-%02d 23:59:59", $ToYear, $ToMonth, $ToDay );

                #                    if ($Count == 1) {
                $ValueSeries{ sprintf( "%04d-%02d-%02d", $VSYear, $VSMonth, $VSDay ) } = {
                    $Ref1->{Values}{TimeStop}  => $TimeStop,
                    $Ref1->{Values}{TimeStart} => $TimeStart
                };

                ( $VSYear, $VSMonth, $VSDay, $VSHour, $VSMinute, $VSSecond )
                    = Add_Delta_DHMS(
                    $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond, 0,
                    0, 0, 1
                    );
            }
        }
        elsif ( $Ref1->{SelectedValues}[0] eq 'Hour' ) {
            while (
                $Self->{TimeObject}->TimeStamp2SystemTime( String => $TimeStop )
                < $TimeAbsolutStopUnixTime
                )
            {
                $TimeStart
                    = sprintf( "%04d-%02d-%02d %02d:00:00", $VSYear, $VSMonth, $VSDay, $VSHour );
                ( $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond )
                    = Add_Delta_DHMS(
                    $VSYear, $VSMonth, $VSDay, $VSHour, $VSMinute, $VSSecond, 0,
                    $Count, 0, -1
                    );
                $TimeStop
                    = sprintf( "%04d-%02d-%02d %02d:59:59", $ToYear, $ToMonth, $ToDay, $ToHour );
                $ValueSeries{
                    sprintf(
                        "%04d-%02d-%02d %02d:00:00 - %02d:59:59",
                        $VSYear, $VSMonth, $VSDay, $VSHour, $ToHour
                        )
                    }
                    = {
                    $Ref1->{Values}{TimeStop}  => $TimeStop,
                    $Ref1->{Values}{TimeStart} => $TimeStart
                    };
                ( $VSYear, $VSMonth, $VSDay, $VSHour, $VSMinute, $VSSecond )
                    = Add_Delta_DHMS(
                    $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond, 0,
                    0, 0, 1
                    );
            }
        }

        elsif ( $Ref1->{SelectedValues}[0] eq 'Minute' ) {
            while (
                $Self->{TimeObject}->TimeStamp2SystemTime( String => $TimeStop )
                < $TimeAbsolutStopUnixTime
                )
            {
                $TimeStart = sprintf(
                    "%04d-%02d-%02d %02d:%02d:00",
                    $VSYear, $VSMonth, $VSDay, $VSHour, $VSMinute
                );
                ( $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond )
                    = Add_Delta_DHMS(
                    $VSYear, $VSMonth, $VSDay, $VSHour, $VSMinute, $VSSecond, 0,
                    0, $Count, -1
                    );
                $TimeStop = sprintf(
                    "%04d-%02d-%02d %02d:%02d:59",
                    $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute
                );
                $ValueSeries{
                    sprintf(
                        "%04d-%02d-%02d %02d:%02d:00 - %02d:%02d:59",
                        $VSYear, $VSMonth, $VSDay, $VSHour, $VSMinute, $ToHour, $ToMinute
                        )
                    }
                    = {
                    $Ref1->{Values}{TimeStop}  => $TimeStop,
                    $Ref1->{Values}{TimeStart} => $TimeStart
                    };
                ( $VSYear, $VSMonth, $VSDay, $VSHour, $VSMinute, $VSSecond )
                    = Add_Delta_DHMS(
                    $ToYear, $ToMonth, $ToDay, $ToHour, $ToMinute, $ToSecond, 0,
                    0, 0, 1
                    );
            }

        }
    }

    # merge the array if two elements for the valueseries are avialable
    KEY:
    for my $Key ( keys %{ $ArraySelected[0]{Values} } ) {
        my $Value0;
        if ( $ArraySelected[0]{Block} eq 'SelectField' ) {
            $Value0 = $Key;
        }
        elsif ( $ArraySelected[0]{Block} eq 'MultiSelectField' ) {
            $Value0 = [$Key];
        }

        if ( !$ArraySelected[1] ) {
            $ValueSeries{ $ArraySelected[0]{Values}{$Key} }
                = { $ArraySelected[0]{Element} => $Value0 };
            next KEY;
        }

        for my $SubKey ( keys %{ $ArraySelected[1]{Values} } ) {
            my $Value1;
            if ( $ArraySelected[1]{Block} eq 'SelectField' ) {
                $Value1 = $SubKey;
            }
            elsif ( $ArraySelected[1]{Block} eq 'MultiSelectField' ) {
                $Value1 = [$SubKey];
            }
            $ValueSeries{
                $ArraySelected[0]{Values}{$Key} . ' - '
                    . $ArraySelected[1]{Values}{$SubKey}
                }
                = {
                $ArraySelected[0]{Element} => $Value0,
                $ArraySelected[1]{Element} => $Value1
                };
        }
    }

    # Use this if no valueseries available
    if ( !%ValueSeries ) {
        $ValueSeries{ $Param{Object} . 's' } = undef;
    }

    # get the first column name in the headerline
    if ($ColumnName) {
        unshift @HeaderLine, $ColumnName;
    }
    elsif ( $ArraySelected[1] ) {
        unshift( @HeaderLine, $ArraySelected[0]{Name} . ' - ' . $ArraySelected[1]{Name} );
    }
    elsif ( $ArraySelected[0] ) {
        unshift( @HeaderLine, $ArraySelected[0]{Name} || '' );
    }

    # push the first array elements in the StatsArray
    my $Title = $Param{Title};
    if ( $TitleTimeStart && $TitleTimeStop ) {
        $Title .= " $TitleTimeStart-$TitleTimeStop";
    }

    # create the cache string
    my $CacheString = $Self->_GetCacheString(%Param);

    # take the cache value if configured and available
    if ( $Param{Cache} ) {
        my @StatArray = $Self->_GetResultCache(
            Filename => 'Stats' . $Param{StatID} . '-' . $CacheString . '.cache',
        );

        return @StatArray if @StatArray;
    }

    # create the table structure
    my %TableStructure = ();
    for my $Row ( sort keys %ValueSeries ) {
        my @Cells = ();
        for my $Cell ( @{ $Xvalue->{SelectedValues} } ) {    # get each cell
            $ValueSeries{$Row} ||= {};
            my %Attributes = ( %{ $ValueSeries{$Row} }, %RestrictionAttribute );

            # the following is necessary if as x-axis and as value-series time is selected
            if ( $Xvalue->{Block} eq 'Time' ) {
                my $TimeStart = $Xvalue->{Values}{TimeStart};
                my $TimeStop  = $Xvalue->{Values}{TimeStop};
                if ( $ValueSeries{$Row}{$TimeStop} && $ValueSeries{$Row}{$TimeStart} ) {
                    if (
                        $Self->{TimeObject}->TimeStamp2SystemTime( String => $Cell->{TimeStop} )
                        > $Self->{TimeObject}->TimeStamp2SystemTime(
                            String => $ValueSeries{$Row}{$TimeStop}
                        )
                        || $Self->{TimeObject}->TimeStamp2SystemTime( String => $Cell->{TimeStart} )
                        < $Self->{TimeObject}->TimeStamp2SystemTime(
                            String => $ValueSeries{$Row}{$TimeStart}
                        )
                        )
                    {
                        next;
                    }
                }
                $Attributes{$TimeStop}  = $Cell->{TimeStop};
                $Attributes{$TimeStart} = $Cell->{TimeStart};
            }
            elsif ( $Xvalue->{Block} eq 'SelectField' ) {
                $Attributes{ $Xvalue->{Element} } = $Cell;
            }
            else {
                $Attributes{ $Xvalue->{Element} } = [$Cell];
            }
            push @Cells, \%Attributes;
        }
        $TableStructure{$Row} = \@Cells;
    }

    my @DataArray = ();
    if ( $StatObject->can('GetStatTable') ) {

        # get the whole stats table
        @DataArray = $StatObject->GetStatTable(
            ValueSeries    => $Param{UseAsValueSeries},    #\%ValueSeries,
            XValue         => $Xvalue,
            Restrictions   => \%RestrictionAttribute,
            TableStructure => \%TableStructure,
        );
    }
    else {
        for my $Row ( sort keys %TableStructure ) {
            my @ResultRow = ($Row);
            for my $Cell ( @{ $TableStructure{$Row} } ) {
                my $Quantity = $StatObject->GetStatElement( %{$Cell} );
                push @ResultRow, $Quantity;
            }
            push @DataArray, \@ResultRow;
        }
    }

    # fill up empty array elements, e.g month as value series (February has 28 day and Januar 31)
    for my $Row (@DataArray) {
        for my $Index ( 1 .. $#HeaderLine ) {
            if ( !defined $Row->[$Index] ) {
                $Row->[$Index] = '';
            }
        }
    }

    # REMARK: it could be also useful to use the indiviual sort if difined
    # so you don't need this function
    if ( $StatObject->can('GetHeaderLine') ) {
        my $HeaderRef = $StatObject->GetHeaderLine(
            XValue => $Xvalue,
        );

        if ($HeaderRef) {
            @HeaderLine = @{$HeaderRef};
        }
    }

    my @StatArray = ( [$Title], \@HeaderLine, @DataArray );

    return @StatArray if !$Param{Cache};

    # check if we should cache this result
    # get the current time
    if ( !$TitleTimeStart || !$TitleTimeStop ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message =>
                "Can't cache: StatID $Param{StatID} have no time period, so you can't cache the stat!",
        );
        return @StatArray;
    }

    if (
        $Self->{TimeObject}->TimeStamp2SystemTime( String => $TitleTimeStop )
        > $Self->{TimeObject}->SystemTime()
        )
    {

        $Self->{LogObject}->Log(
            Priority => 'error',
            Message =>
                "Can't cache StatID $Param{StatID}: The selected end time is in the future!",
        );
        return @StatArray;
    }

    # write the stats cache
    $Self->_SetResultCache(
        Filename => 'Stats' . $Param{StatID} . '-' . $CacheString . '.cache',
        Result   => \@StatArray,
    );
    return @StatArray;
}

=item GenerateGraph()

make graph from result array

    my $Graph = $StatsObject->GenerateGraph(
        Array        => \@StatArray,
        GraphSize    => '800x600',
        HeadArrayRef => $HeadArrayRef,
        Title        => 'All Tickets of the month',
        Format       => 'GD::Graph::lines',
    );

=cut

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

    # check if need params are available
    for (qw(Array GraphSize HeadArrayRef Title Format)) {
        if ( !$Param{$_} ) {
            $Self->{LogObject}->Log( Priority => 'error', Message => "GenerateGraph: Need $_!" );
            return;
        }
    }

    my @StatArray    = @{ $Param{Array} };
    my $HeadArrayRef = $Param{HeadArrayRef};
    my $GDBackend    = $Param{Format};

    # delete SumCol and SumRow if present
    if ( $StatArray[-1][0] eq 'Sum' ) {
        pop @StatArray;
    }
    if ( $HeadArrayRef->[-1] eq 'Sum' ) {
        pop @{$HeadArrayRef};
        for my $Row (@StatArray) {
            pop @{$Row};
        }
    }

    # load gd modules
    for my $Module ( 'GD', 'GD::Graph', $GDBackend ) {
        if ( !$Self->{MainObject}->Require($Module) ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "GenerateGraph: Need $Module!"
            );
            return;
        }
    }

    # remove first y/x position
    my $XLable = shift @{$HeadArrayRef};

    # get first col for legend
    my @YLine = ();
    for my $Tmp (@StatArray) {
        push @YLine, $Tmp->[0];
        shift @{$Tmp};
    }

    # build plot data
    my @PData = ( $HeadArrayRef, @StatArray );
    my ( $XSize, $YSize ) = split( m{x}x, $Param{GraphSize} );
    my $graph = $GDBackend->new( $XSize || 550, $YSize || 350 );
    $graph->set(
        x_label => $XLable,

        #        y_label => 'YLable',
        title => $Param{Title},

        #        y_max_value => 20,
        #        y_tick_number => 16,
        #        y_label_skip => 4,
        #        x_tick_number => 8,
        t_margin    => $Self->{ConfigObject}->Get("Stats::Graph::t_margin")    || 10,
        b_margin    => $Self->{ConfigObject}->Get("Stats::Graph::b_margin")    || 10,
        l_margin    => $Self->{ConfigObject}->Get("Stats::Graph::l_margin")    || 10,
        r_margin    => $Self->{ConfigObject}->Get("Stats::Graph::r_margin")    || 20,
        bgclr       => $Self->{ConfigObject}->Get("Stats::Graph::bgclr")       || 'white',
        transparent => $Self->{ConfigObject}->Get("Stats::Graph::transparent") || 0,
        interlaced  => 1,
        fgclr       => $Self->{ConfigObject}->Get("Stats::Graph::fgclr")       || 'black',
        boxclr      => $Self->{ConfigObject}->Get("Stats::Graph::boxclr")      || 'white',
        accentclr   => $Self->{ConfigObject}->Get("Stats::Graph::accentclr")   || 'black',
        shadowclr   => $Self->{ConfigObject}->Get("Stats::Graph::shadowclr")   || 'black',
        legendclr   => $Self->{ConfigObject}->Get("Stats::Graph::legendclr")   || 'black',
        textclr     => $Self->{ConfigObject}->Get("Stats::Graph::textclr")     || 'black',
        dclrs       => $Self->{ConfigObject}->Get("Stats::Graph::dclrs")
            || [
            qw(red green blue yellow black purple orange pink marine cyan lgray lblue lyellow lgreen lred lpurple lorange lbrown)
            ],
        x_tick_offset       => 0,
        x_label_position    => 1 / 2,
        y_label_position    => 1 / 2,
        x_labels_vertical   => 31,
        line_width          => $Self->{ConfigObject}->Get("Stats::Graph::line_width") || 1,
        legend_placement    => $Self->{ConfigObject}->Get("Stats::Graph::legend_placement") || 'BC',
        legend_spacing      => $Self->{ConfigObject}->Get("Stats::Graph::legend_spacing") || 4,
        legend_marker_width => $Self->{ConfigObject}->Get("Stats::Graph::legend_marker_width")
            || 12,
        legend_marker_height => $Self->{ConfigObject}->Get("Stats::Graph::legend_marker_height")
            || 8,
    );

    # set legend (y-line)
    if ( $Param{Format} ne 'GD::Graph::pie' ) {
        $graph->set_legend(@YLine);
    }

    # plot graph
    my $Ext = '';
    if ( !$graph->can('png') ) {
        $Ext = 'png';
    }
    else {
        $Ext = $graph->export_format;
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't write png! Write: $Ext",
        );
    }
    my $Content = eval { $graph->plot( \@PData )->$Ext() };
    return $Content;
}

=item CompletenessCheck()

    my @Notify = $StatsObject->CompletenessCheck(
        StatData => \%StatData,
        Section => 'All' || 'Specification' || 'ValueSeries' || 'Restrictions || Xaxis'
    );

=cut

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

    my @Notify         = ();
    my @NotifySelected = ();
    my @IndexArray     = ();

    $Notify[0] = {
        Info     => 'Please fill out the required fields!',
        Priority => 'Error'
    };
    $Notify[1] = {
        Info     => 'Please select a file!',
        Priority => 'Error'
    };
    $Notify[2] = {
        Info     => 'Please select an object!',
        Priority => 'Error'
    };
    $Notify[3] = {
        Info     => 'Please select a graph size!',
        Priority => 'Error'
    };
    $Notify[4] = {
        Info     => 'Please select one element for the X-axis!',
        Priority => 'Error'
    };
    $Notify[6] = {
        Info =>
            'Please select only one element or turn of the button \'Fixed\' where the select field is marked!',
        Priority => 'Error'
    };
    $Notify[7] = {
        Info     => 'If you use a checkbox you have to select some attributes of the select field!',
        Priority => 'Error'
    };
    $Notify[8] = {
        Info =>
            'Please insert a value in the selected input field or turn off the \'Fixed\' checkbox!',
        Priority => 'Error'
    };
    $Notify[9] = {
        Info     => 'The selected end time is before the start time!',
        Priority => 'Error'
    };
    $Notify[10] = {
        Info     => 'You have to select one or more attributes from the select field!',
        Priority => 'Error'
    };
    $Notify[11] = {
        Info     => 'The selected Date isn\'t valid!',
        Priority => 'Error'
    };
    $Notify[12] = {
        Info     => 'Please select only one or two elements via the checkbox!',
        Priority => 'Error'
    };
    $Notify[13] = {
        Info     => 'If you use a time scale element you can only select one element!',
        Priority => 'Error'
    };
    $Notify[14] = {
        Info     => 'You have an error in your time selection!',
        Priority => 'Error'
    };
    $Notify[15] = {
        Info     => 'Your reporting time interval is to small, please use a larger time scale!',
        Priority => 'Error'
    };
    $Notify[16] = {
        Info     => 'There is something wrong with your time scale selection. Please check it!',
        Priority => 'Error'
    };
    $Notify[17] = {
        Info     => 'You have to select a time scale like day or month!',
        Priority => 'Error'
    };

    # check if need params are available
    NEED:
    for my $Need (qw(StatData Section)) {
        next NEED if $Param{$Need};
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "CompletenessCheck: Need $Need"
        );
        return;
    }

    my %StatData = %{ $Param{StatData} };
    if ( $Param{Section} eq 'Specification' || $Param{Section} eq 'All' ) {
        for (qw(Title Description StatType Permission Format ObjectModule)) {
            if ( !$StatData{$_} ) {
                push @IndexArray, 0;
                last;
            }
        }
        if ( $StatData{StatType} && $StatData{StatType} eq 'static' && !$StatData{File} ) {
            push @IndexArray, 1;
        }
        if ( $StatData{StatType} && $StatData{StatType} eq 'dynamic' && !$StatData{Object} ) {
            push @IndexArray, 2;
        }
        if ( !$Param{StatData}{GraphSize} && $Param{StatData}{Format} ) {
            for ( @{ $StatData{Format} } ) {
                if ( $_ =~ m{^GD::Graph\.*}x ) {
                    push @IndexArray, 3;
                    last;
                }
            }
        }
    }

    # for form calls
    if ( $StatData{StatType} && $StatData{StatType} eq 'dynamic' ) {
        if (
            ( $Param{Section} eq 'Xaxis' || $Param{Section} eq 'All' )
            && $StatData{StatType} eq 'dynamic'
            )
        {
            my $Flag = 0;
            XVALUE:
            for my $Xvalue ( @{ $StatData{UseAsXvalue} } ) {
                next XVALUE if !$Xvalue->{Selected};

                if ( $Xvalue->{Block} eq 'Time' ) {
                    if ( $Xvalue->{TimeStart} && $Xvalue->{TimeStop} ) {
                        my $TimeStart = $Self->{TimeObject}->TimeStamp2SystemTime(
                            String => $Xvalue->{TimeStart}
                        );
                        my $TimeStop = $Self->{TimeObject}->TimeStamp2SystemTime(
                            String => $Xvalue->{TimeStop}
                        );
                        if ( !$TimeStart || !$TimeStop ) {
                            push @IndexArray, 11;
                            last XVALUE;
                        }
                        elsif ( $TimeStart > $TimeStop ) {
                            push @IndexArray, 9;
                            last XVALUE;
                        }
                    }
                    elsif ( !$Xvalue->{TimeRelativeUnit} || !$Xvalue->{TimeRelativeCount} ) {
                        push @IndexArray, 9;
                        last XVALUE;
                    }

                    if ( !$Xvalue->{SelectedValues}[0] ) {
                        push @IndexArray, 16;
                    }
                    elsif ( $Xvalue->{Fixed} && $#{ $Xvalue->{SelectedValues} } > 0 ) {
                        push @IndexArray, 16;
                    }
                }
                $Flag = 1;
                last XVALUE;
            }
            if ( !$Flag ) {
                push @IndexArray, 4;
            }
        }
        if (
            ( $Param{Section} eq 'ValueSeries' || $Param{Section} eq 'All' )
            && $StatData{StatType} eq 'dynamic'
            )
        {
            my $Counter = 0;
            my $Flag    = 0;
            VALUESERIES:
            for my $ValueSeries ( @{ $StatData{UseAsValueSeries} } ) {
                next VALUESERIES if !$ValueSeries->{Selected};

                if (
                    $ValueSeries->{Block} eq 'Time'
                    || $ValueSeries->{Block} eq 'TimeExtended'
                    )
                {
                    if ( $ValueSeries->{Fixed} && $#{ $ValueSeries->{SelectedValues} } > 0 ) {
                        push @IndexArray, 6;
                    }
                    elsif ( !$ValueSeries->{SelectedValues}[0] ) {
                        push @IndexArray, 7;
                    }
                    $Flag = 1;
                }

                $Counter++;
            }
            if ( $Counter > 1 && $Flag ) {
                push @IndexArray, 13;
            }
            elsif ( $Counter > 2 ) {
                push @IndexArray, 12;
            }
        }
        if (
            ( $Param{Section} eq 'Restrictions' || $Param{Section} eq 'All' )
            && $StatData{StatType} eq 'dynamic'
            )
        {
            RESTRICTION:
            for my $Restriction ( @{ $StatData{UseAsRestriction} } ) {
                next RESTRICTION if !$Restriction->{Selected};

                if ( $Restriction->{Block} eq 'SelectField' ) {
                    if ( $Restriction->{Fixed} && $#{ $Restriction->{SelectedValues} } > 0 ) {
                        push @IndexArray, 6;
                        last RESTRICTION;
                    }
                    elsif ( !$Restriction->{SelectedValues}[0] ) {
                        push @IndexArray, 7;
                        last RESTRICTION;
                    }
                }
                elsif (
                    $Restriction->{Block} eq 'InputField'
                    && !$Restriction->{SelectedValues}[0]
                    && $Restriction->{Fixed}
                    )
                {
                    push @IndexArray, 8;
                    last RESTRICTION;
                }
                elsif (
                    $Restriction->{Block} eq 'Time'
                    || $Restriction->{Block} eq 'TimeExtended'
                    )
                {
                    if ( $Restriction->{TimeStart} && $Restriction->{TimeStop} ) {
                        my $TimeStart = $Self->{TimeObject}->TimeStamp2SystemTime(
                            String => $Restriction->{TimeStart}
                        );
                        my $TimeStop = $Self->{TimeObject}->TimeStamp2SystemTime(
                            String => $Restriction->{TimeStop}
                        );
                        if ( !$TimeStart || !$TimeStop ) {
                            push @IndexArray, 11;
                            last RESTRICTION;
                        }
                        elsif ( $TimeStart > $TimeStop ) {
                            push @IndexArray, 9;
                            last RESTRICTION;
                        }
                    }
                    elsif (
                        !$Restriction->{TimeRelativeUnit}
                        || !$Restriction->{TimeRelativeCount}
                        )
                    {
                        push @IndexArray, 9;
                        last RESTRICTION;
                    }
                }
            }
        }

        # check if the timeperiod is to big or the time scale to small
        # used only for fixed time values
        # remark time functions should be exportet in external functions (tr)
        if ( $Param{Section} eq 'All' && $StatData{StatType} eq 'dynamic' ) {
            my $Stat = $Self->StatsGet( StatID => $StatData{StatID} );

            XVALUE:
            for my $Xvalue ( @{ $Stat->{UseAsXvalue} } ) {
                next XVALUE
                    if !( $Xvalue->{Selected} && $Xvalue->{Fixed} && $Xvalue->{Block} eq 'Time' );

                my $Flag = 1;
                VALUESERIES:
                for my $ValueSeries ( @{ $Stat->{UseAsValueSeries} } ) {
                    if ( $ValueSeries->{Selected} && $ValueSeries->{Block} eq 'Time' ) {
                        $Flag = 0;
                        last VALUESERIES;
                    }
                }

                last XVALUE if !$Flag;

                my $ScalePeriod = 0;
                my $TimePeriod  = 0;

                my $Count = $Xvalue->{TimeScaleCount} ? $Xvalue->{TimeScaleCount} : 1;

                my %TimeInSeconds = (
                    Year   => 31536000,    # 60 * 60 * 60 * 365
                    Month  => 2592000,     # 60 * 60 * 24 * 30
                    Day    => 86400,       # 60 * 60 * 24
                    Hour   => 3600,        # 60 * 60
                    Minute => 60,
                    Second => 1,
                );

                $ScalePeriod = $TimeInSeconds{ $Xvalue->{SelectedValues}[0] };

                if ( !$ScalePeriod ) {
                    push @IndexArray, 17;
                    last XVALUE;
                }

                if ( $Xvalue->{TimeStop} && $Xvalue->{TimeStart} ) {
                    $TimePeriod
                        = (
                        $Self->{TimeObject}->TimeStamp2SystemTime( String => $Xvalue->{TimeStop} )
                        )
                        - (
                        $Self->{TimeObject}->TimeStamp2SystemTime( String => $Xvalue->{TimeStart} )
                        );
                }
                else {
                    $TimePeriod = $TimeInSeconds{ $Xvalue->{TimeRelativeUnit} }
                        * $Xvalue->{TimeRelativeCount};
                }

                my $MaxAttr = $Self->{ConfigObject}->Get('Stats::MaxXaxisAttributes') || 1000;
                if ( $TimePeriod / ( $ScalePeriod * $Count ) > $MaxAttr ) {
                    push @IndexArray, 15;
                }

                last XVALUE;
            }
        }
    }
    for (@IndexArray) {
        push @NotifySelected, $Notify[$_];
    }

    return @NotifySelected;

}

=item GetStatsObjectAttributes()

Get all attributes from the object in dependence of the use

    my %ObjectAttributes = $StatsObject->GetStatsObjectAttributes(
        ObjectModule => 'Ticket',
        Use          => 'UseAsXvalue' || 'UseAsValueSeries' || 'UseAsRestriction',
    );

=cut

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

    my @ObjectAttributes = ();

    # check needed params
    for (qw(ObjectModule Use)) {
        if ( !$Param{$_} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "GetStatsObjectAttributes: Need $_!"
            );
            return;
        }
    }

    # load module
    my $ObjectModule = $Param{ObjectModule};
    $Self->{MainObject}->Require($ObjectModule);
    my $StatObject = $ObjectModule->new( %{$Self} );

    # load attributes
    my @ObjectAttributesRaw = $StatObject->GetObjectAttributes();

    # build the objectattribute array
    for my $HashRef (@ObjectAttributesRaw) {
        if ( $HashRef->{ $Param{Use} } ) {
            delete $HashRef->{UseAsXvalue};
            delete $HashRef->{UseAsValueSeries};
            delete $HashRef->{UseAsRestriction};

            push @ObjectAttributes, $HashRef;
        }
    }

    return @ObjectAttributes;
}

=item GetStaticFiles()

Get all static files

    my $FileHash = $StatsObject->GetStaticFiles(
        OnlyUnusedFiles => 1 | 0, # optional default 0
    );

=cut

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

    my %Filelist = ();

    my $Directory = $Self->{ConfigObject}->Get('Home');
    if ( $Directory !~ m{^.*\/$}x ) {
        $Directory .= '/';
    }
    $Directory .= 'Kernel/System/Stats/Static/';

    if ( !opendir( DIR, $Directory ) ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can not open Directory: $Directory",
        );
        return ();
    }

    my %StaticFiles = ();

    if ( $Param{OnlyUnusedFiles} ) {

        # get all Stats from the db
        my $Result = $Self->GetStatsList();

        if ( defined $Result ) {
            for my $StatID ( @{$Result} ) {
                my $Data = $Self->StatsGet( StatID => $StatID );

                # check witch one are static statistics
                if ( $Data->{File} && $Data->{StatType} eq 'static' ) {
                    $StaticFiles{ $Data->{File} } = 1;
                }
            }
        }
    }

    # read files
    while ( defined( my $Filename = readdir DIR ) ) {
        next if $Filename eq '.';
        next if $Filename eq '..';
        if ( $Filename =~ m{^(.*)\.pm$}x ) {
            if ( !defined $StaticFiles{$1} ) {
                $Filelist{$1} = $1;
            }
        }
    }
    closedir(DIR);

    return \%Filelist;
}

=item GetDynamicFiles()

Get all static objects

    my $FileHash = $StatsObject->GetDynamicFiles();

=cut

sub GetDynamicFiles {
    my $Self = shift;

    my %Filelist = %{ $Self->{ConfigObject}->Get('Stats::DynamicObjectRegistration') };
    OBJECT:
    for my $Object ( keys %Filelist ) {
        if ( !$Filelist{$Object} ) {
            delete $Filelist{$Object};
            next OBJECT;
        }
        $Filelist{$Object} = $Self->GetObjectName(
            ObjectModule => $Filelist{$Object}{Module},
        );
    }
    return if !%Filelist;

    return \%Filelist;
}

=item GetObjectName()

Get the name of a dynamic object

    my $ObjectName = $StatsObject->GetObjectName(
        ObjectModule => 'Kernel::System::Stats::Dynamic::TicketList',
    );

=cut

sub GetObjectName {
    my ( $Self, %Param ) = @_;
    my $Module = $Param{ObjectModule};

    # check if it is cached
    return $Self->{'Cache::ObjectModule'}{$Module} if $Self->{'Cache::ObjectName'}{$Module};

    # load module, return if module does not exist
    # (this is important when stats are uninstalled, see also bug# 4269)
    return if !$Self->{MainObject}->Require($Module);

    # get name
    my $StatObject = $Module->new( %{$Self} );
    my $Name       = $StatObject->GetObjectName();

    # cache the result
    $Self->{'Cache::ObjectModule'}{$Module} = $Name;

    return $Name;
}

=item ObjectFileCheck()

check readable object file

    my $ObjectFileCheck = $StatsObject->ObjectFileCheck(
        Type => 'static',
        Name => 'NewTickets',
    );

=cut

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

    my $Directory = $Self->{ConfigObject}->Get('Home');
    if ( $Directory !~ m{^.*\/$}x ) {
        $Directory .= '/';
    }
    if ( $Param{Type} eq 'static' ) {
        $Directory .= 'Kernel/System/Stats/Static/' . $Param{Name} . '.pm';
    }
    elsif ( $Param{Type} eq 'dynamic' ) {
        $Directory .= 'Kernel/System/Stats/Dynamic/' . $Param{Name} . '.pm';
    }

    return 1 if -r $Directory;
    return 0;
}

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

    my %GetParam = %{ $Param{GetParam} };

    # check if we should cache this result
    # get the current time
    my ( $s, $m, $h, $D, $M, $Y )
        = $Self->{TimeObject}->SystemTime2Date( SystemTime => $Self->{TimeObject}->SystemTime(), );

    # if get params in future do not cache
    if ( $GetParam{Year} && $GetParam{Month} ) {
        return if $GetParam{Year} > $Y;
        return if $GetParam{Year} == $Y && $GetParam{Month} > $M;
        return
            if $GetParam{Year} == $Y
                && $GetParam{Month} == $M
                && $GetParam{Day}
                && $GetParam{Day} >= $D;
    }

    # write cache file
    my $Filename = $Self->_CreateStaticResultCacheFilename(
        GetParam => $Param{GetParam},
        StatID   => $Param{StatID},
    );

    $Self->_SetResultCache(
        Filename => $Filename,
        Result   => $Param{Data},
    );

    return 1;
}

#=item _CreateStaticResultCacheFilename()
#
#create a filename out of the GetParam information and the stat id
#
#    my $Filename = $StatsObject->_CreateStaticResultCacheFilename(
#        GetParam => {
#            Year  => 2008,
#            Month => 3,
#            Day   => 5
#        },
#        StatID   => $Param{StatID},
#    );
#
#=cut

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

    # check needed params
    for my $NeededParam (qw( StatID GetParam )) {
        if ( !$Param{$NeededParam} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "_CreateStaticResultCacheFilename: Need $NeededParam!"
            );
            return;
        }
    }

    my $GetParamRef = $Param{GetParam};

    # format month and day params
    for (qw(Month Day)) {
        if ( $GetParamRef->{$_} ) {
            $GetParamRef->{$_} = sprintf( "%02d", $GetParamRef->{$_} );
        }
    }

    my $Key = '';
    if ( $GetParamRef->{Year} ) {
        $Key .= $GetParamRef->{Year};
    }
    if ( $GetParamRef->{Month} ) {
        $Key .= "-$GetParamRef->{Month}";
    }
    if ( $GetParamRef->{Day} ) {
        $Key .= "-$GetParamRef->{Day}";
    }

    my $MD5Key = $Self->{MainObject}->FilenameCleanUp(
        Filename => $Key,
        Type     => 'md5',
    );

    return 'Stats' . $Param{StatID} . '-' . $MD5Key . '.cache';
}

#=item _SetResultCache()
#
#write the result array as cache in the filesystem
#
#    $StatsObject->_SetResultCache(
#        Filename => 'Stats' . $Param{StatID} . '-' . $MD5Key . '.cache',
#        Result   => $Param{Data},
#    );
#
#=cut

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

    # check needed params
    for my $NeededParam (qw( Filename Result)) {
        if ( !$Param{$NeededParam} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "_SetResultCache: Need $NeededParam!"
            );
            return;
        }
    }

    # convert the result array into a csv string
    my $CSVString = $Self->{CSVObject}->Array2CSV( Data => $Param{Result}, );
    $Self->{EncodeObject}->EncodeOutput( \$CSVString );

    # write the csv string into the filesystem
    my $Filehandle;
    my $Path = $Self->{ConfigObject}->Get('TempDir');
    if ( !open $Filehandle, '>', "$Path/$Param{Filename}" ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't write: $Path/$Param{Filename}!",
        );
        return;
    }

    binmode $Filehandle;
    print $Filehandle $CSVString;
    close $Filehandle;

    return 1;
}

#=item _GetResultCache()
#
#get the result array cache out of the filesystem
#
#    my @Result = $StatsObject->_GetResultCache(
#        Filename => 'Stats' . $Param{StatID} . '-' . $MD5Key . '.cache',
#    );
#
#=cut

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

    # check needed params
    if ( !$Param{Filename} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => '_GetResultCache: Need Filename!',
        );
        return;
    }

    my $Path      = $Self->{ConfigObject}->Get('TempDir');
    my $CSVString = '';
    if ( open my $Filehandle, '<', "$Path/$Param{Filename}" ) {
        binmode $Filehandle;
        while (<$Filehandle>) {
            $CSVString .= $_;
        }
        close $Filehandle;
        $Self->{EncodeObject}->Encode( \$CSVString );
    }

    my $ResultRef = $Self->{CSVObject}->CSV2Array(
        String    => $CSVString,
        Separator => ';',
        Quote     => '"',
    );

    return @{$ResultRef};
}

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

    my $Path = $Self->{ConfigObject}->Get('TempDir');

    if ( $Path !~ m{^.*\/$}x ) {
        $Path .= '/';
    }

    my @Files = glob $Path . 'Stats' . $Param{StatID} . '-*.cache';

    for my $File (@Files) {
        unlink $File;
    }
    return 1;
}

=item Export()

get content from stats for export

    my $ExportFile = $StatsObject->Export(
        StatID => '123',
        ExportStatNumber => 1 || 0, # optional, only useful move statistics from the test system to the productive system
    );

=cut

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

    my %File = ();

    if ( !$Param{StatID} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Export: Need StatID!"
        );
        return;
    }

    my @XMLHash = $Self->{XMLObject}->XMLHashGet(
        Type => 'Stats',

        #Cache => 0,
        Key => $Param{StatID}
    );

    $File{Filename} = $Self->StringAndTimestamp2Filename(
        String => $XMLHash[0]->{otrs_stats}[1]{Title}[1]{Content},
    );
    $File{Filename} .= '.xml';

    # settings for static files
    if (
        $XMLHash[0]->{otrs_stats}[1]{StatType}[1]{Content}
        && $XMLHash[0]->{otrs_stats}[1]{StatType}[1]{Content} eq 'static'
        )
    {
        my $FileLocation = $XMLHash[0]->{otrs_stats}[1]{ObjectModule}[1]{Content};
        $FileLocation =~ s{::}{\/}xg;
        $FileLocation .= '.pm';
        my $File        = $Self->{ConfigObject}->Get('Home') . "/$FileLocation";
        my $FileContent = '';

        open my $Filehandle, '<', $File or die "Can't open: $File: $!";

        # set bin mode
        binmode $Filehandle;
        while (<$Filehandle>) {
            $FileContent .= $_;
        }
        close $Filehandle;

        $Self->{EncodeObject}->Encode( \$FileContent );
        $XMLHash[0]->{otrs_stats}[1]{File}[1]{File}
            = $XMLHash[0]->{otrs_stats}[1]{File}[1]{Content};
        $XMLHash[0]->{otrs_stats}[1]{File}[1]{Content}    = encode_base64( $FileContent, '' );
        $XMLHash[0]->{otrs_stats}[1]{File}[1]{Location}   = $FileLocation;
        $XMLHash[0]->{otrs_stats}[1]{File}[1]{Permission} = '644';
        $XMLHash[0]->{otrs_stats}[1]{File}[1]{Encode}     = 'Base64';
    }

    # delete create and change data
    delete( $XMLHash[0]->{otrs_stats}[1]{Changed} );
    delete( $XMLHash[0]->{otrs_stats}[1]{ChangedBy} );
    delete( $XMLHash[0]->{otrs_stats}[1]{Created} );
    delete( $XMLHash[0]->{otrs_stats}[1]{CreatedBy} );
    delete( $XMLHash[0]->{otrs_stats}[1]{StatID} );
    if ( !$Param{ExportStatNumber} ) {
        delete( $XMLHash[0]->{otrs_stats}[1]{StatNumber} );
    }

    # wrapper to change ids in used spelling
    # wrap permissions
    for my $ID ( @{ $XMLHash[0]->{otrs_stats}[1]{Permission} } ) {
        if ($ID) {
            my %Group = $Self->{GroupObject}->GroupGet( ID => ( $ID->{Content} ) );
            $ID->{Content} = $Group{Name};
        }
    }

    # wrap object dependend ids
    if ( $XMLHash[0]->{otrs_stats}[1]{Object}[1]{Content} ) {

        # load module
        my $ObjectModule = $XMLHash[0]->{otrs_stats}[1]{ObjectModule}[1]{Content};
        $Self->{MainObject}->Require($ObjectModule);
        my $StatObject = $ObjectModule->new( %{$Self} );

        # load attributes
        $XMLHash[0]->{otrs_stats}[1]
            = $StatObject->ExportWrapper( %{ $XMLHash[0]->{otrs_stats}[1] } );
    }

    # convert hash to string
    $File{Content} = $Self->{XMLObject}->XMLHash2XML(@XMLHash);

    return \%File;
}

=item Import()

import a stats from xml file

    my $StatID = $StatsObject->Import(
        Content => $UploadStuff{Content},
    );

=cut

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

    my $StatID = 1;

    if ( !$Param{Content} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Import: Need Content!"
        );
        return;
    }
    my @XMLHash = $Self->{XMLObject}->XMLParse2XMLHash( String => $Param{Content} );

    if ( !$XMLHash[0] ) {
        shift @XMLHash;
    }

    # Get new StatID
    my @Keys = $Self->{XMLObject}->XMLHashSearch( Type => 'Stats', );

    # check if the required elements are available
    for my $Element (
        qw( Description Format Object ObjectModule Permission StatType SumCol SumRow Title Valid)
        )
    {
        if ( !defined $XMLHash[0]{otrs_stats}[1]{$Element}[1]{Content} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message =>
                    "Import: Can't import Stat, because the required element $Element is not available!"
            );
            return;
        }
    }

    # if-clause if a stat-xml includes a StatNumber
    if ( $XMLHash[0]{otrs_stats}[1]{StatNumber} ) {
        my $XMLStatsID = $XMLHash[0]{otrs_stats}[1]{StatNumber}[1]{Content}
            - $Self->{ConfigObject}->Get("Stats::StatsStartNumber");
        for my $Key (@Keys) {
            if ( $Key eq $XMLStatsID ) {
                $Self->{LogObject}->Log(
                    Priority => 'error',
                    Message =>
                        "Import: Can't import StatNumber $Key, because this StatNumber is already used!"
                );
                return;
            }
        }
        $StatID = $XMLStatsID;
    }

    # if no stats number available use this function
    else {
        my @SortKeys = sort { $a <=> $b } @Keys;
        if (@SortKeys) {
            $StatID = $SortKeys[-1] + 1;
        }
    }

    # get time
    my $TimeStamp = $Self->{TimeObject}->SystemTime2TimeStamp(
        SystemTime => $Self->{TimeObject}->SystemTime(),
    );

    # meta tags
    $XMLHash[0]{otrs_stats}[1]{Created}[1]{Content}   = $TimeStamp;
    $XMLHash[0]{otrs_stats}[1]{CreatedBy}[1]{Content} = $Self->{UserID};
    $XMLHash[0]{otrs_stats}[1]{Changed}[1]{Content}   = $TimeStamp;
    $XMLHash[0]{otrs_stats}[1]{ChangedBy}[1]{Content} = $Self->{UserID};
    $XMLHash[0]{otrs_stats}[1]{StatNumber}[1]{Content}
        = $StatID + $Self->{ConfigObject}->Get("Stats::StatsStartNumber");

    my $DynamicFiles = $Self->GetDynamicFiles();

    # Because some xml-parser insert \n instead of <example><example>
    if ( $XMLHash[0]->{otrs_stats}[1]{Object}[1]{Content} ) {
        $XMLHash[0]->{otrs_stats}[1]{Object}[1]{Content} =~ s{\n}{}x;
    }

    if (
        $XMLHash[0]->{otrs_stats}[1]{Object}[1]{Content}
        && !$DynamicFiles->{ $XMLHash[0]->{otrs_stats}[1]{Object}[1]{Content} }
        )
    {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message =>
                "Import: Object $XMLHash[0]->{otrs_stats}[1]{Object}[1]{Content} doesn't exists!"
        );
        return 0;
    }
    if (
        $XMLHash[0]->{otrs_stats}[1]{StatType}[1]{Content}
        && $XMLHash[0]->{otrs_stats}[1]{StatType}[1]{Content} eq 'static'
        )
    {
        my $FileLocation = $XMLHash[0]->{otrs_stats}[1]{ObjectModule}[1]{Content};
        $FileLocation =~ s{::}{\/}gx;
        $FileLocation = $Self->{ConfigObject}->Get('Home') . '/' . $FileLocation . '.pm';

        # write file
        if ( open my $Filehandle, '>', $FileLocation ) {
            print STDERR
                "Notice: Install $FileLocation ($XMLHash[0]->{otrs_stats}[1]{File}[1]{Permission})!\n";
            if (
                $XMLHash[0]->{otrs_stats}[1]{File}[1]{Encode}
                && $XMLHash[0]->{otrs_stats}[1]{File}[1]{Encode} eq 'Base64'
                )
            {
                $XMLHash[0]->{otrs_stats}[1]{File}[1]{Content}
                    = decode_base64( $XMLHash[0]->{otrs_stats}[1]{File}[1]{Content} );
                $Self->{EncodeObject}->EncodeOutput(
                    \$XMLHash[0]->{otrs_stats}[1]{File}[1]{Content}
                );
            }

            # set bin mode
            binmode $Filehandle;
            print $Filehandle $XMLHash[0]->{otrs_stats}[1]{File}[1]{Content};
            close $Filehandle;

            # set permission
            if ( length( $XMLHash[0]->{otrs_stats}[1]{File}[1]{Permission} ) == 3 ) {
                $XMLHash[0]->{otrs_stats}[1]{File}[1]{Permission}
                    = "0$XMLHash[0]->{otrs_stats}[1]{File}[1]{Permission}";
            }
            chmod( oct( $XMLHash[0]->{otrs_stats}[1]{File}[1]{Permission} ), $FileLocation );
            $XMLHash[0]->{otrs_stats}[1]{File}[1]{Content}
                = $XMLHash[0]->{otrs_stats}[1]{File}[1]{File};
            delete $XMLHash[0]->{otrs_stats}[1]{File}[1]{File};
            delete $XMLHash[0]->{otrs_stats}[1]{File}[1]{Location};
            delete $XMLHash[0]->{otrs_stats}[1]{File}[1]{Permission};
            delete $XMLHash[0]->{otrs_stats}[1]{File}[1]{Encode};
        }
    }

    # wrapper to change used spelling in ids
    # wrap permissions
    my %Groups = $Self->{GroupObject}->GroupList( Valid => 1 );

    NAME:
    for my $Name ( @{ $XMLHash[0]->{otrs_stats}[1]{Permission} } ) {
        next NAME if !$Name;

        my $Flag = 1;
        ID:
        for my $ID ( keys %Groups ) {
            if ( $Groups{$ID} eq $Name->{Content} ) {
                $Name->{Content} = $ID;
                $Flag = 0;
                last ID;
            }
        }
        if ($Flag) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "Import: Can' find the permission (group) $Name->{Content}!"
            );
            $Name = undef;
        }
    }

    # wrap object dependend ids
    if ( $XMLHash[0]->{otrs_stats}[1]{Object}[1]{Content} ) {

        # load module
        my $ObjectModule = $XMLHash[0]->{otrs_stats}[1]{ObjectModule}[1]{Content};
        $Self->{MainObject}->Require($ObjectModule);
        my $StatObject = $ObjectModule->new( %{$Self} );

        # load attributes
        $XMLHash[0]->{otrs_stats}[1]
            = $StatObject->ImportWrapper( %{ $XMLHash[0]->{otrs_stats}[1] } );
    }

    # new
    return 0 if !$Self->{XMLObject}->XMLHashAdd(
        Type    => 'Stats',
        Key     => $StatID,
        XMLHash => \@XMLHash,
    );

    return $StatID;
}

=item GetParams()

    get all edit params from stats for view

    my $Params = $StatsObject->GetParams(StatID => '123');

=cut

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

    my @Params = ();

    if ( !$Param{StatID} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "GetParams: Need StatID!"
        );
        return;
    }

    my $Stat = $Self->StatsGet( StatID => $Param{StatID} );

    # static
    # don't remove this if clause, because is required for mkStats.pl
    if ( $Stat->{StatType} eq 'static' ) {

        # load static modul
        my $ObjectModule = $Stat->{ObjectModule};
        $Self->{MainObject}->Require($ObjectModule);
        my $StatObject = $ObjectModule->new( %{$Self} );

        # get params
        @Params = $StatObject->Param();
    }
    return \@Params;
}

=item StatsRun()

    run a stats...

    my $StatArray = $StatsObject->StatsRun(
        StatID => '123',
        GetParam => \%GetParam,
    );

=cut

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

    # check needed params
    NEED:
    for my $Need (qw(StatID GetParam)) {
        next NEED if $Param{$Need};
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "StatsRun: Need $Need!"
        );
        return;
    }

    # use the mirror db if configured
    if ( $Self->{ConfigObject}->Get('Core::MirrorDB::DSN') ) {
        my $ExtraDatabaseObject = Kernel::System::DB->new(
            LogObject    => $Self->{LogObject},
            ConfigObject => $Self->{ConfigObject},
            MainObject   => $Self->{MainObject},
            EncodeObject => $Self->{EncodeObject},
            DatabaseDSN  => $Self->{ConfigObject}->Get('Core::MirrorDB::DSN'),
            DatabaseUser => $Self->{ConfigObject}->Get('Core::MirrorDB::User'),
            DatabasePw   => $Self->{ConfigObject}->Get('Core::MirrorDB::Password'),
        );
        if ( !$ExtraDatabaseObject ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => 'There is no MirroDB!',
            );
            return;
        }
        $Self->{DBObject} = $ExtraDatabaseObject;
    }

    my $Stat     = $Self->StatsGet( StatID => $Param{StatID} );
    my %GetParam = %{ $Param{GetParam} };
    my @Result   = ();

    # get data if it is a static stats
    if ( $Stat->{StatType} eq 'static' ) {
        @Result = $Self->_GenerateStaticStats(
            ObjectModule => $Stat->{ObjectModule},
            GetParam     => $Param{GetParam},
            Title        => $Stat->{Title},
            StatID       => $Stat->{StatID},
            Cache        => $Stat->{Cache},
        );
    }

    # get data if it is a dynaymic stats
    elsif ( $Stat->{StatType} eq 'dynamic' ) {
        @Result = $Self->_GenerateDynamicStats(
            ObjectModule     => $Stat->{ObjectModule},
            Object           => $Stat->{Object},
            UseAsXvalue      => $GetParam{UseAsXvalue},
            UseAsValueSeries => $GetParam{UseAsValueSeries},
            UseAsRestriction => $GetParam{UseAsRestriction},
            Title            => $Stat->{Title},
            StatID           => $Stat->{StatID},
            Cache            => $Stat->{Cache},
        );
    }

    # build sum in row or col
    if ( ( $Stat->{SumRow} || $Stat->{SumCol} ) && $Stat->{Format} !~ m{^GD::Graph\.*}x ) {
        return $Self->SumBuild(
            Array  => \@Result,
            SumRow => $Stat->{SumRow},
            SumCol => $Stat->{SumCol},
        );
    }
    return \@Result;
}

=item StringAndTimestamp2Filename()

builds a filename with a string and a timestamp.
(space will be replaced with _ and - e.g. Title-of-File_2006-12-31_11-59)

    my $Filename = $StatsObject->StringAndTimestamp2Filename(String => 'Title');

=cut

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

    if ( !$Param{String} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'StringAndTimestamp2Filename: Need String!'
        );
        return;
    }

    my ( $s, $m, $h, $D, $M, $Y ) = $Self->{TimeObject}->SystemTime2Date(
        SystemTime => $Self->{TimeObject}->SystemTime(),
    );
    $M = sprintf( "%02d", $M );
    $D = sprintf( "%02d", $D );
    $h = sprintf( "%02d", $h );
    $m = sprintf( "%02d", $m );

    $Param{String} = $Self->{MainObject}->FilenameCleanUp(
        Filename => $Param{String},
        Type     => 'Attachment',
    );

    my $Filename = $Param{String} . '_' . "$Y-$M-$D" . '_' . "$h-$m";

    return $Filename;
}

sub _MonthArray {
    my @MonthArray
        = (
        '', 'Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec',
        );

    return \@MonthArray;
}

=item StatNumber2StatID()

insert the stat number get the stat id

    my $StatID = $StatsObject->StatNumber2StatID(
        StatNumber => 11212
    );

=cut

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

    if ( !$Param{StatNumber} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'StatNumber2StatID: Need StatNumber!',
        );
        return;
    }

    my @Key = $Self->{XMLObject}->XMLHashSearch(
        Type => 'Stats',
        What => [ { "[%]{'otrs_stats'}[%]{'StatNumber'}[%]{'Content'}" => $Param{StatNumber}, }, ],
    );
    if ( @Key && $#Key < 1 ) {
        return $Key[0];
    }

    $Self->{LogObject}->Log(
        Priority => 'error',
        Message  => 'StatNumber invalid!',
    );
    return 0;
}

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

    my $Language  = $Self->{ConfigObject}->Get('DefaultLanguage');
    my $Directory = $Self->{StatsTempDir};

    if ( !opendir( DIRE, $Directory ) ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can not open Directory: $Directory",
        );
        return;
    }

    # check if stats in the default language available, if not use en
    my $Flag = 0;
    while ( defined( my $Filename = readdir DIRE ) ) {
        if ( $Filename =~ m{^.*\.$Language\.xml$}x ) {
            $Flag = 1;
        }
    }

    rewinddir(DIRE);
    if ( !$Flag ) {
        $Language = 'en';
    }

    while ( defined( my $Filename = readdir DIRE ) ) {
        if ( $Filename =~ m{^.*\.$Language\.xml$}x ) {

            # check filesize
            #            my $Filesize = -s $Directory.$Filename;
            #            if ($Filesize > $MaxFilesize) {
            #                print "File: $Filename to big! max. $MaxFilesize byte allowed.\n";
            #                $CommonObject{LogObject}->Log(
            #                    Priority => 'error',
            #                    Message => "Can't file imported: $Directory.$Filename",
            #                );
            #                next;
            #            }

            # read file
            my $Filehandle;
            if ( !open $Filehandle, '<', $Directory . $Filename ) {
                $Self->{LogObject}->Log(
                    Priority => 'error',
                    Message  => "Can not open File: " . $Directory . $Filename,
                );
                closedir(DIRE);
                return;
            }

            my $Content = '';
            while (<$Filehandle>) {
                $Content .= $_;
            }
            close $Filehandle;

            my $StatID = $Self->Import( Content => $Content, );
        }
    }
    closedir(DIRE);

    return 1;
}

=item StatsInstall()

installs stats

    my $Result = $CodeObject->StatsInstall(
        FilePrefix => 'FAQ',  # (optional)
    );

=cut

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

    # prepare prefix
    $Param{FilePrefix} = $Param{FilePrefix} ? $Param{FilePrefix} . '-' : '';

    # start AutomaticSampleImport if no stats are installed
    $Self->GetStatsList();

    # cleanup stats
    $Self->StatsCleanUp();

    # get list of stats files
    my @StatsFileList = glob $Self->{StatsTempDir} . $Param{FilePrefix} . '*.xml';

    # import the stats
    my $InstalledPostfix = '.installed';
    FILE:
    for my $File ( sort @StatsFileList ) {

        next FILE if !-f $File;
        next FILE if -e $File . $InstalledPostfix;

        # read file content
        my $XMLContentRef = $Self->{MainObject}->FileRead(
            Location => $File,
        );

        # import stat
        my $StatID = $Self->Import(
            Content => ${$XMLContentRef},
        );

        next FILE if !$StatID;

        # write installed file with stat id
        $Self->{MainObject}->FileWrite(
            Content  => \$StatID,
            Location => $File . $InstalledPostfix,
        );
    }

    return 1;
}

=item StatsUninstall()

uninstalls stats

    my $Result = $StatsObject->StatsUninstall(
        FilePrefix => 'FAQ',  # (optional)
    );

=cut

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

    # prepare prefix
    $Param{FilePrefix} = $Param{FilePrefix} ? $Param{FilePrefix} . '-' : '';

    # get list of installed stats files
    my @StatsFileList = glob $Self->{StatsTempDir} . $Param{FilePrefix} . '*.xml.installed';

    # delete the stats
    for my $File ( sort @StatsFileList ) {

        # read file content
        my $StatsIDRef = $Self->{MainObject}->FileRead(
            Location => $File,
        );

        # delete stats
        $Self->StatsDelete(
            StatID => ${$StatsIDRef},
        );
    }

    # cleanup stats
    $Self->StatsCleanUp();

    return 1;
}

=item StatsCleanUp()

removed stats with not existing backend file

    my $Result = $StatsObject->StatsCleanUp();

=cut

sub StatsCleanUp {
    my $Self = shift;

    # get a list of all stats
    my $ListRef = $Self->GetStatsList();

    return if !$ListRef;
    return if ref $ListRef ne 'ARRAY';

    STATSID:
    for my $StatsID ( @{$ListRef} ) {

        # get stats
        my $HashRef = $Self->StatsGet(
            StatID             => $StatsID,
            NoObjectAttributes => 1,
        );

        next STATSID if $HashRef
                && ref $HashRef eq 'HASH'
                && $HashRef->{ObjectModule}
                && $Self->{MainObject}->Require( $HashRef->{ObjectModule} );

        # delete stats
        $Self->StatsDelete( StatID => $StatsID );
    }

    return 1;
}

# =item _GetCacheString()
#
# return the cache string
#
#     my $Result = $StatsObject->_GetCacheString(
#         UseAsXvalue      => $UseAsXvalueRef
#         UseAsValueSeries => $UseAsValueSeriesRef,
#         UseAsRestriction => $UseAsRestrictionRef,
#     );
#
# =cut

sub _GetCacheString {
    my ( $Self, %Param ) = @_;
    my $CacheString = '';

    for my $Use (qw(UseAsXvalue UseAsValueSeries UseAsRestriction)) {
        USEREF:
        for my $UseRef ( @{ $Param{$Use} } ) {
            $CacheString .= '__' . $UseRef->{Name} . '_';
            if ( $UseRef->{SelectedValues} ) {
                $CacheString .= join( '_', sort @{ $UseRef->{SelectedValues} } )
            }
            elsif ( $UseRef->{TimeStart} && $UseRef->{TimeStop} ) {
                $CacheString .= $UseRef->{TimeStart} . '-' . $UseRef->{TimeStop};
            }
        }
    }

    my $MD5Key = $Self->{MainObject}->FilenameCleanUp(
        Filename => $CacheString,
        Type     => 'md5',
    );

    return $MD5Key;
}

1;

=back

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (http://otrs.org/).

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

=head1 VERSION

$Revision: 1.3 $ $Date: 2010/03/01 12:17:23 $

=cut

# --
# Kernel/System/VirtualFS.pm - all virtual fs functions
# Copyright (C) 2001-2010 OTRS AG, http://otrs.org/
# --
# $Id: VirtualFS.pm,v 1.4 2010/02/04 13:17:06 ub Exp $
# $OldId: VirtualFS.pm,v 1.4 2010/02/03 14:51:03 bes Exp $
# --
# 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::VirtualFS;

use strict;
use warnings;

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

=head1 NAME

Kernel::System::VirtualFS - virtual fs lib

=head1 SYNOPSIS

All virtual fs functions.

=head1 PUBLIC INTERFACE

=over 4

=cut

=item new()

create an object

    use Kernel::Config;
    use Kernel::System::Encode;
    use Kernel::System::Log;
    use Kernel::System::Main;
    use Kernel::System::DB;
    use Kernel::System::VirtualFS;

    my $ConfigObject = Kernel::Config->new();
    my $EncodeObject = Kernel::System::Encode->new(
        ConfigObject => $ConfigObject,
    );
    my $LogObject = Kernel::System::Log->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
    );
    my $MainObject = Kernel::System::Main->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        LogObject    => $LogObject,
    );
    my $DBObject = Kernel::System::DB->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        LogObject    => $LogObject,
        MainObject   => $MainObject,
    );
    my $VirtualFSObject = Kernel::System::VirtualFS->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
        DBObject     => $DBObject,
        MainObject   => $MainObject,
        EncodeObject => $EncodeObject,
    );

=cut

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

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

    # check needed objects
    for (qw(DBObject ConfigObject LogObject MainObject EncodeObject)) {
        $Self->{$_} = $Param{$_} || die "Got no $_!";
    }

    # load backend
    $Self->{BackendDefault} = $Self->{ConfigObject}->Get('VirtualFS::Backend')
        || 'Kernel::System::VirtualFS::DB';
    if ( !$Self->{MainObject}->Require( $Self->{BackendDefault} ) ) {
        return;
    }
    $Self->{Backend}->{ $Self->{BackendDefault} } = $Self->{BackendDefault}->new(%Param);

    return $Self;
}

=item Read()

read a file from virtual file system

    my %File = $VirtualFSObject->Read(
        Filename => '/some/name.txt',
        Mode     => 'utf8',

        # optional
        DisableWarnings => 1,
    );

returns

    Content  => $ContentSCALAR,

    # preferences data
    Preferences => {

        # generated automaticaly
        Filesize           => '12.4 KBytes',
        FilesizeRaw        => 12345,

        # optional
        ContentType        => 'text/plain',
        ContentID          => '<some_id@example.com>',
        ContentAlternative => 1,
        SomeCustomParams   => 'with our own value',
    },

=cut

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

    # check needed stuff
    for (qw(Filename Mode)) {
        if ( !$Param{$_} ) {
            $Self->{LogObject}->Log( Priority => 'error', Message => "Need $_!" );
            return;
        }
    }

    # lookup
    my ( $FileID, $BackendKey, $Backend ) = $Self->_FileLookup( $Param{Filename} );
    if ( !$BackendKey ) {
        if ( !$Param{DisableWarnings} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "No such file '$Param{Filename}'!",
            );
        }
        return;
    }

    # get preferences
    my %Preferences;
    return if !$Self->{DBObject}->Prepare(
        SQL => 'SELECT preferences_key, preferences_value FROM '
            . 'virtual_fs_preferences WHERE virtual_fs_id = ?',
        Bind => [ \$FileID ],
    );
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        $Preferences{ $Row[0] } = $Row[1];
    }

    # load backend (if not default)
    if ( !$Self->{Backend}->{$Backend} ) {
        return if !$Self->{MainObject}->Require($Backend);
        $Self->{Backend}->{$Backend} = $Backend->new( %{$Self} );
        return if !$Self->{Backend}->{$Backend};
    }

    # get file
    my $Content = $Self->{Backend}->{$Backend}->Read(
        %Param,
        BackendKey => $BackendKey,
    );
    return if !$Content;

    return (
        Preferences => \%Preferences,
        Content     => $Content,
    );
}

=item Write()

write a file to virtual file system

    my $Success = $VirtualFSObject->Write(
        Content  => \$Content,
        Filename => 'SomeFileName.txt',
        Mode     => 'binary'            # (binary|utf8)

        # optional, preferences data
        Preferences => {
            ContentType        => 'text/plain',
            ContentID          => '<some_id@example.com>',
            ContentAlternative => 1,
            SomeCustomParams   => 'with our own value',
        },
    );

=cut

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

    # check needed stuff
    for (qw(Filename Content Mode)) {
        if ( !$Param{$_} ) {
            $Self->{LogObject}->Log( Priority => 'error', Message => "Need $_!" );
            return;
        }
    }

    # lookup
    my ($FileID) = $Self->_FileLookup( $Param{Filename} );
    if ($FileID) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "File already exists '$Param{Filename}'!",
        );
        return;
    }

    # insert
    return if !$Self->{DBObject}->Do(
        SQL => 'INSERT INTO virtual_fs (filename, backend_key, backend, create_time)'
            . ' VALUES ( ?, \'TMP\', ?, current_timestamp)',
        Bind => [ \$Param{Filename}, \$Self->{BackendDefault} ],
    );
    ($FileID) = $Self->_FileLookup( $Param{Filename} );
    if ( !$FileID ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Unable to store '$Param{Filename}'!",
        );
        return;
    }

    # size calculation
    $Param{Preferences}->{FilesizeRaw} = bytes::length( ${ $Param{Content} } );
    my $Filesize = $Param{Preferences}->{FilesizeRaw};
    if ( $Filesize > ( 1024 * 1024 ) ) {
        $Filesize = sprintf "%.1f MBytes", ( $Filesize / ( 1024 * 1024 ) );
    }
    elsif ( $Filesize > 1024 ) {
        $Filesize = sprintf "%.1f KBytes", ( $Filesize / 1024 );
    }
    else {
        $Filesize = $Filesize . ' Bytes';
    }
    $Param{Preferences}->{Filesize} = $Filesize;

    # insert preferences
    for my $Key ( sort keys %{ $Param{Preferences} } ) {
        return if !$Self->{DBObject}->Do(
            SQL => 'INSERT INTO virtual_fs_preferences '
                . '(virtual_fs_id, preferences_key, preferences_value) VALUES ( ?, ?, ?)',
            Bind => [ \$FileID, \$Key, \$Param{Preferences}->{$Key} ],
        );
    }

    # store file
    my $BackendKey = $Self->{Backend}->{ $Self->{BackendDefault} }->Write(%Param);
    return if !$BackendKey;

    # update backend key
    return if !$Self->{DBObject}->Do(
        SQL => 'UPDATE virtual_fs SET backend_key = ? WHERE id = ?',
        Bind => [ \$BackendKey, \$FileID ],
    );

    return 1;
}

=item Delete()

delete a file from virtual file system

    my $Success = $VirtualFSObject->Delete(
        Filename => 'SomeFileName.txt',

        # optional
        DisableWarnings => 1,
    );

=cut

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

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

    # lookup
    my ( $FileID, $BackendKey, $Backend ) = $Self->_FileLookup( $Param{Filename} );
    if ( !$FileID ) {
        if ( !$Param{DisableWarnings} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "No such file '$Param{Filename}'!",
            );
        }
        return;
    }

    # load backend (if not default)
    if ( !$Self->{Backend}->{$Backend} ) {
        return if !$Self->{MainObject}->Require($Backend);
        $Self->{Backend}->{$Backend} = $Backend->new( %{$Self} );
        return if !$Self->{Backend}->{$Backend};
    }

    # delete preferences
    return if !$Self->{DBObject}->Do(
        SQL  => 'DELETE FROM virtual_fs_preferences WHERE virtual_fs_id = ?',
        Bind => [ \$FileID ],
    );

    # delete
    return if !$Self->{DBObject}->Do(
        SQL  => 'DELETE FROM virtual_fs WHERE id = ?',
        Bind => [ \$FileID ],
    );

    # delete file
    return $Self->{Backend}->{$Backend}->Delete(
        %Param,
        BackendKey => $BackendKey,
    );
}
# ---
# ITSM ChangeManagement
# ---
=item Search()

Search for files in virtual file system. In contrast to Find(), you do
not need to know the filename. You can search for files based on virtual fs
preferences. The search in the preferences is logically OR connected.

String search parameters are queried with wildcard behavior.
The integer search parameter, that us C<FileID>, has no wildcard behavior.

The function returns a hash with the ID as the key and the filename is the value.

    # search for all text/plain files
    my %Files = $VirtualFSObject->Search(
        Preferences => {
            ContentType => 'text/plain',
        }
    );

is

  123 => '/some/file.txt',
  23  => '/my.pdf',

=cut

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

    # at least one criteria must be given
    if ( ! keys %Param ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'At least one search criterion must be given',
        );

        return;
    }

    # all where conditions
    my @Where;
    my @Bind;

    # We have to join the preferences table when the search is
    # based on entries in preferences.
    my $JoinClause = '';
    my $GroupByClause = '';
    if ( $Param{Preferences} ) {
        $JoinClause    = 'INNER JOIN virtual_fs_preferences vfp ON vfp.virtual_fs_id = vf.id ';
        $GroupByClause = 'GROUP BY vf.id, vf.filename';

        # push search criteria to where clause
        my @PreferencesWhere;
        for my $Column ( keys %{ $Param{Preferences} } ) {
            my $LikeColumn = $Column;
            $LikeColumn =~ s/\*/%/g;
            $LikeColumn = $Self->{DBObject}->Quote( $LikeColumn, 'Like' );

            my $LikeValue = $Param{Preferences}->{$Column};
            $LikeValue =~ s/\*/%/g;
            $LikeValue = $Self->{DBObject}->Quote( $LikeValue, 'Like' );

            push @PreferencesWhere,
                "( vfp.preferences_key LIKE '$Column' AND preferences_value LIKE '$LikeValue' )";
        }

        my $PreferencesWhereString = join ' OR ', @PreferencesWhere;
        push @Where, ' ( ' . $PreferencesWhereString . ' ) ';

        # delete preferences
        delete $Param{Preferences};
    }

    # map param keys to column maps
    my %StringParam2Column = (
        Filename => 'filename',
        Backend  => 'backend',
    );

    my %IntegerParam2Column = (
        FileID   => 'id',
    );

    # push all remaining parameters to where clause
    for my $Column ( keys %Param ) {

        # SQL for the string search parameters
        if ( $StringParam2Column{$Column} ) {
            my $Like = $Param{$Column};
            $Like =~ s/\*/%/g;
            $Like = $Self->{DBObject}->Quote( $Like, 'Like' );

            my $ColumnName = $StringParam2Column{$Column};

            push @Where, "vf.$ColumnName LIKE '$Like'";
        }

        # SQL for the integer search parameters
        elsif ( $IntegerParam2Column{$Column} ) {
            my $ColumnName = $IntegerParam2Column{$Column};
            push @Where, "vf.$ColumnName = ?";
            push @Bind, \$Param{$Column};
        }

        # all other search parameters are ignored
        else {
            # do nothing
        }
    }

    # join the where conditions
    my $WhereClause = join ' AND ', @Where;

    # do the search
    return if !$Self->{DBObject}->Prepare(
        SQL => 'SELECT vf.id, vf.filename '
            . "FROM virtual_fs vf $JoinClause "
            . "WHERE $WhereClause $GroupByClause",
        Bind => \@Bind,
    );

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

    return %Files;
}
# ---

=item Find()

find files in virtual file system

    my @List = $VirtualFSObject->Find(
        Filename => 'some_what/*.txt',
    );

=cut

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

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

    # prepare search
    my $Like = $Param{Filename};
    $Like =~ s/\*/%/g;
    $Like = $Self->{DBObject}->Quote( $Like, 'Like' );

    # search
    return if !$Self->{DBObject}->Prepare(
        SQL => "SELECT filename FROM virtual_fs WHERE filename LIKE '$Like'",
    );
    my @List;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        push @List, $Row[0];
    }
    return @List;
}

=begin Internal:

=cut

sub _FileLookup {
    my ( $Self, $Filename ) = @_;

    # lookup
    return if !$Self->{DBObject}->Prepare(
        SQL  => 'SELECT id, backend_key, backend FROM virtual_fs WHERE filename = ?',
        Bind => [ \$Filename ],
    );
    my $FileID;
    my $BackendKey;
    my $Backend;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        $FileID     = $Row[0];
        $BackendKey = $Row[1];
        $Backend    = $Row[2];
    }
    return ( $FileID, $BackendKey, $Backend );
}

=end Internal:

=cut

1;

=back

=head1 TERMS AND CONDITIONS

This software is part of the OTRS project (http://otrs.org/).

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

=cut

=head1 VERSION

$Revision: 1.4 $ $Date: 2010/02/04 13:17:06 $

=cut

IyAtLQojIEtlcm5lbC9TeXN0ZW0vVmlydHVhbEZTL0RCLnBtIC0gYWxsIHZpcnR1YWwgZnMgZnVuY3Rpb25zCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDkgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBEQi5wbSx2IDEuMSAyMDA5LzEyLzE0IDE0OjI3OjQzIHJlYiBFeHAgJAojICRPbGRJZDogREIucG0sdiAxLjIgMjAwOS8xMi8xMCAxMTo1OTo1NCBiZXMgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OlZpcnR1YWxGUzo6REI7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSBNSU1FOjpCYXNlNjQ7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjEgJCkgWzFdOwoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBvYmplY3RzCiAgICBmb3IgKHF3KERCT2JqZWN0IENvbmZpZ09iamVjdCBMb2dPYmplY3QgTWFpbk9iamVjdCBFbmNvZGVPYmplY3QpKSB7CiAgICAgICAgJFNlbGYtPnskX30gPSAkUGFyYW17JF99IHx8IGRpZSAiR290IG5vICRfISI7CiAgICB9CgogICAgIyBjb25maWcgKG5vdCB1c2VkIHJpZ2h0IG5vdykKICAgICRTZWxmLT57Q29tcHJlc3N9ID0gMDsKICAgICRTZWxmLT57Q3J5cHR9ICAgID0gMDsKCiAgICByZXR1cm4gJFNlbGY7Cn0KCnN1YiBSZWFkIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGZvciAocXcoQmFja2VuZEtleSBNb2RlKSkgewogICAgICAgIGlmICggISRQYXJhbXskX30gKSB7CiAgICAgICAgICAgICRTZWxmLT57TG9nT2JqZWN0fS0+TG9nKCBQcmlvcml0eSA9PiAnZXJyb3InLCBNZXNzYWdlID0+ICJOZWVkICRfISIgKTsKICAgICAgICAgICAgcmV0dXJuOwogICAgICAgIH0KICAgIH0KCiAgICBteSAkQXR0cmlidXRlcyA9ICRTZWxmLT5fQmFja2VuZEtleVBhcnNlKCVQYXJhbSk7CgogICAgbXkgJEVuY29kZSA9IDE7CiAgICBpZiAoIGxjICRQYXJhbXtNb2RlfSBlcSAnYmluYXJ5JyApIHsKICAgICAgICAkRW5jb2RlID0gMDsKICAgIH0KICAgIHJldHVybiBpZiAhJFNlbGYtPntEQk9iamVjdH0tPlByZXBhcmUoCiAgICAgICAgU1FMICAgID0+ICdTRUxFQ1QgY29udGVudCBGUk9NIHZpcnR1YWxfZnNfZGIgV0hFUkUgaWQgPSA/JywKICAgICAgICBCaW5kICAgPT4gWyBcJEF0dHJpYnV0ZXMtPntGaWxlSUR9IF0sCiAgICAgICAgRW5jb2RlID0+IFskRW5jb2RlXSwKICAgICk7CiAgICBteSAkQ29udGVudDsKICAgIHdoaWxlICggbXkgQFJvdyA9ICRTZWxmLT57REJPYmplY3R9LT5GZXRjaHJvd0FycmF5KCkgKSB7CgogICAgICAgICMgZGVjb2RlIGF0dGFjaG1lbnQgaWYgaXQncyBlLiBnLiBhIHBvc3RncmVzcWwgYmFja2VuZCEhIQogICAgICAgIGlmICggISRTZWxmLT57REJPYmplY3R9LT5HZXREYXRhYmFzZUZ1bmN0aW9uKCdEaXJlY3RCbG9iJykgKSB7CiAgICAgICAgICAgICRDb250ZW50ID0gZGVjb2RlX2Jhc2U2NCggJFJvd1swXSApOwogICAgICAgICAgICBpZiAoJEVuY29kZSkgewogICAgICAgICAgICAgICAgJFNlbGYtPntFbmNvZGVPYmplY3R9LT5FbmNvZGUoIFwkQ29udGVudCApOwogICAgICAgICAgICB9CiAgICAgICAgfQogICAgICAgIGVsc2UgewogICAgICAgICAgICAkQ29udGVudCA9ICRSb3dbMF07CiAgICAgICAgfQogICAgfQogICAgcmV0dXJuIGlmICEkQ29udGVudDsKCiAgICAjIHVuY29tcHJlc3MgKGluIGNhc2UpCiAgICBpZiAoICRBdHRyaWJ1dGVzLT57Q29tcHJlc3N9ICkgewoKICAgICAgICAjICRDb250ZW50ID0gLi4uCiAgICB9CgogICAgIyBkZWNyeXB0IChpbiBjYXNlKQogICAgaWYgKCAkQXR0cmlidXRlcy0+e0NyeXB0fSApIHsKCiAgICAgICAgIyAkQ29udGVudCA9IC4uLgogICAgfQoKICAgIHJldHVybiBcJENvbnRlbnQ7Cn0KCnN1YiBXcml0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBmb3IgKHF3KENvbnRlbnQgRmlsZW5hbWUgTW9kZSkpIHsKICAgICAgICBpZiAoICEkUGFyYW17JF99ICkgewogICAgICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZyggUHJpb3JpdHkgPT4gJ2Vycm9yJywgTWVzc2FnZSA9PiAiTmVlZCAkXyEiICk7CiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICB9CgogICAgIyBjaGVjayBpZiBhbHJlYWR5IGV4aXN0cwogICAgbXkgJEV4aXN0cyA9ICRTZWxmLT5fRmlsZUxvb2t1cCggJFBhcmFte0ZpbGVuYW1lfSApOwogICAgcmV0dXJuIGlmICRFeGlzdHM7CgogICAgIyBjb21wcmVzcyAoaW4gY2FzZSkKICAgIGlmICggJFNlbGYtPntDb21wcmVzc30gKSB7CgogICAgICAgICMgJFBhcmFte0NvbnRlbnR9ID0gLi4uCiAgICB9CgogICAgIyBjcnlwdCAoaW4gY2FzZSkKICAgIGlmICggJFNlbGYtPntDcnlwdH0gKSB7CgogICAgICAgICMgJFBhcmFte0NvbnRlbnR9ID0gLi4uCiAgICB9CgogICAgIyBlbmNvZGUgYXR0YWNobWVudCBpZiBpdCdzIGEgcG9zdGdyZXNxbCBiYWNrZW5kISEhCiAgICBpZiAoICEkU2VsZi0+e0RCT2JqZWN0fS0+R2V0RGF0YWJhc2VGdW5jdGlvbignRGlyZWN0QmxvYicpICkgewogICAgICAgICRTZWxmLT57RW5jb2RlT2JqZWN0fS0+RW5jb2RlT3V0cHV0KCAkUGFyYW17Q29udGVudH0gKTsKICAgICAgICBteSAkQ29udGVudCA9IGVuY29kZV9iYXNlNjQoICR7ICRQYXJhbXtDb250ZW50fSB9ICk7CiAgICAgICAgJFBhcmFte0NvbnRlbnR9ID0gXCRDb250ZW50OwogICAgfQoKICAgIG15ICRFbmNvZGUgPSAxOwogICAgaWYgKCBsYyAkUGFyYW17TW9kZX0gZXEgJ2JpbmFyeScgKSB7CiAgICAgICAgJEVuY29kZSA9IDA7CiAgICB9CiAgICByZXR1cm4gaWYgISRTZWxmLT57REJPYmplY3R9LT5EbygKICAgICAgICBTUUwgPT4gJ0lOU0VSVCBJTlRPIHZpcnR1YWxfZnNfZGIgKGZpbGVuYW1lLCBjb250ZW50LCBjcmVhdGVfdGltZSkgJwogICAgICAgICAgICAuICdWQUxVRVMgKCA/LCA/LCBjdXJyZW50X3RpbWVzdGFtcCApJywKICAgICAgICBCaW5kID0+IFsgXCRQYXJhbXtGaWxlbmFtZX0sICRQYXJhbXtDb250ZW50fSBdLAogICAgKTsKCiAgICBteSAkRmlsZUlEID0gJFNlbGYtPl9GaWxlTG9va3VwKCAkUGFyYW17RmlsZW5hbWV9ICk7CiAgICByZXR1cm4gaWYgISRGaWxlSUQ7CgogICAgbXkgJEJhY2tlbmRLZXkgPSAkU2VsZi0+X0JhY2tlbmRLZXlHZW5lcmF0ZSgKICAgICAgICBGaWxlSUQgICA9PiAkRmlsZUlELAogICAgICAgIENvbXByZXNzID0+ICRTZWxmLT57Q29tcHJlc3N9LAogICAgICAgIENyeXB0ICAgID0+ICRTZWxmLT57Q3J5cHR9LAogICAgICAgIE1vZGUgICAgID0+ICRQYXJhbXtNb2RlfSwKICAgICAgICBWZXJzaW9uICA9PiAkVkVSU0lPTiwKICAgICk7CgogICAgcmV0dXJuICRCYWNrZW5kS2V5Owp9CgpzdWIgRGVsZXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGZvciAocXcoQmFja2VuZEtleSkpIHsKICAgICAgICBpZiAoICEkUGFyYW17JF99ICkgewogICAgICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZyggUHJpb3JpdHkgPT4gJ2Vycm9yJywgTWVzc2FnZSA9PiAiTmVlZCAkXyEiICk7CiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICB9CgogICAgbXkgJEF0dHJpYnV0ZXMgPSAkU2VsZi0+X0JhY2tlbmRLZXlQYXJzZSglUGFyYW0pOwoKICAgIHJldHVybiBpZiAhJFNlbGYtPntEQk9iamVjdH0tPkRvKAogICAgICAgIFNRTCAgPT4gJ0RFTEVURSBGUk9NIHZpcnR1YWxfZnNfZGIgV0hFUkUgaWQgPSA/JywKICAgICAgICBCaW5kID0+IFsgXCRBdHRyaWJ1dGVzLT57RmlsZUlEfSBdLAogICAgKTsKICAgIHJldHVybiAxOwp9CgpzdWIgX0ZpbGVMb29rdXAgewogICAgbXkgKCAkU2VsZiwgJEZpbGVuYW1lICkgPSBAXzsKCiAgICAjIGxvb2t1cAogICAgcmV0dXJuIGlmICEkU2VsZi0+e0RCT2JqZWN0fS0+UHJlcGFyZSgKICAgICAgICBTUUwgID0+ICdTRUxFQ1QgaWQgRlJPTSB2aXJ0dWFsX2ZzX2RiIFdIRVJFIGZpbGVuYW1lID0gPycsCiAgICAgICAgQmluZCA9PiBbIFwkRmlsZW5hbWUgXSwKICAgICk7CiAgICBteSAkRmlsZUlEOwogICAgd2hpbGUgKCBteSBAUm93ID0gJFNlbGYtPntEQk9iamVjdH0tPkZldGNocm93QXJyYXkoKSApIHsKICAgICAgICAkRmlsZUlEID0gJFJvd1swXTsKICAgIH0KICAgIHJldHVybiAkRmlsZUlEOwp9CgpzdWIgX0JhY2tlbmRLZXlHZW5lcmF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRCYWNrZW5kS2V5ID0gJyc7CiAgICBmb3IgbXkgJEtleSAoIHNvcnQga2V5cyAlUGFyYW0gKSB7CiAgICAgICAgJEJhY2tlbmRLZXkgLj0gIiRLZXk9JFBhcmFteyRLZXl9OyI7CiAgICB9CiAgICByZXR1cm4gJEJhY2tlbmRLZXk7Cn0KCnN1YiBfQmFja2VuZEtleVBhcnNlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGZvciAocXcoQmFja2VuZEtleSkpIHsKICAgICAgICBpZiAoICEkUGFyYW17JF99ICkgewogICAgICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZyggUHJpb3JpdHkgPT4gJ2Vycm9yJywgTWVzc2FnZSA9PiAiTmVlZCAkXyEiICk7CiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICB9CgogICAgbXkgJUF0dHJpYnV0ZXM7CiAgICBteSBAUGFpcnMgPSBzcGxpdCAvOy8sICRQYXJhbXtCYWNrZW5kS2V5fTsKICAgIGZvciBteSAkUGFpciAoQFBhaXJzKSB7CiAgICAgICAgbXkgKCAkS2V5LCAkVmFsdWUgKSA9IHNwbGl0IC89LywgJFBhaXI7CiAgICAgICAgJEF0dHJpYnV0ZXN7JEtleX0gPSAkVmFsdWU7CiAgICB9CiAgICByZXR1cm4gXCVBdHRyaWJ1dGVzOwp9CgoxOwo=
IyAtLQojIEtlcm5lbC9TeXN0ZW0vVmlydHVhbEZTL0ZTLnBtIC0gYWxsIHZpcnR1YWwgZnMgZnVuY3Rpb25zCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDkgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBGUy5wbSx2IDEuMSAyMDA5LzEyLzE0IDE0OjI3OjQzIHJlYiBFeHAgJAojICRPbGRJZDogRlMucG0sdiAxLjIgMjAwOS8xMi8xMCAxMTo1OTo1NCBiZXMgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OlZpcnR1YWxGUzo6RlM7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CnVzZSBGaWxlOjpQYXRoOwp1c2UgRmlsZTo6QmFzZW5hbWU7CgojIHRvIGdldCBpdCB3cml0YWJsZSBmb3IgdGhlIG90cnMgZ3JvdXAgKGp1c3QgaW4gY2FzZSkKdW1hc2sgMDAyOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS4xICQpIFsxXTsKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgIyBjaGVjayBuZWVkZWQgb2JqZWN0cwogICAgZm9yIChxdyhEQk9iamVjdCBDb25maWdPYmplY3QgTG9nT2JqZWN0IE1haW5PYmplY3QgRW5jb2RlT2JqZWN0KSkgewogICAgICAgICRTZWxmLT57JF99ID0gJFBhcmFteyRffSB8fCBkaWUgIkdvdCBubyAkXyEiOwogICAgfQoKICAgICMgZ2V0IGRhdGEgZGlyCiAgICAkU2VsZi0+e0RhdGFEaXJ9ICAgID0gJFNlbGYtPntDb25maWdPYmplY3R9LT5HZXQoJ0hvbWUnKSAuICcvdmFyL3ZpcnR1YWxmcyc7CiAgICAkU2VsZi0+e1Blcm1pc3Npb259ID0gJzY2NCc7CgogICAgIyBjaGVjayBmcyB3cml0ZSBwZXJtaXNzaW9ucyEKICAgIG15ICRQYXRoID0gIiRTZWxmLT57RGF0YURpcn0vY2hlY2tfcGVybWlzc2lvbnMuJCQiOwogICAgaWYgKCAtZCAkUGF0aCApIHsKICAgICAgICBGaWxlOjpQYXRoOjpybXRyZWUoIFskUGF0aF0gKSB8fCBkaWUgIkNhbid0IHJlbW92ZSAkUGF0aDogJCFcbiI7CiAgICB9CiAgICBpZiAoIG1rZGlyKCAiJFNlbGYtPntEYXRhRGlyfS9jaGVja19wZXJtaXNzaW9uc18kJCIsIDAyMiApICkgewogICAgICAgIGlmICggIXJtZGlyKCIkU2VsZi0+e0RhdGFEaXJ9L2NoZWNrX3Blcm1pc3Npb25zXyQkIikgKSB7CiAgICAgICAgICAgIGRpZSAiQ2FuJ3QgcmVtb3ZlICRTZWxmLT57RGF0YURpcn0vY2hlY2tfcGVybWlzc2lvbnNfJCQ6ICQhXG4iOwogICAgICAgIH0KICAgICAgICBpZiAoIEZpbGU6OlBhdGg6Om1rcGF0aCggWyRQYXRoXSwgMCwgMDc3NSApICkgewogICAgICAgICAgICBGaWxlOjpQYXRoOjpybXRyZWUoIFskUGF0aF0gKSB8fCBkaWUgIkNhbid0IHJlbW92ZSAkUGF0aDogJCFcbiI7CiAgICAgICAgfQogICAgfQoKICAgICMgY3JlYXRlIGRhdGEgZGlyCiAgICBpZiAoICEtZCAkU2VsZi0+e0RhdGFEaXJ9ICkgewogICAgICAgIG1rZGlyICRTZWxmLT57RGF0YURpcn0gfHwgZGllICQhOwogICAgfQoKICAgICMgY29uZmlnIChub3QgdXNlZCByaWdodCBub3cpCiAgICAkU2VsZi0+e0NvbXByZXNzfSA9IDA7CiAgICAkU2VsZi0+e0NyeXB0fSAgICA9IDA7CgogICAgcmV0dXJuICRTZWxmOwp9CgpzdWIgUmVhZCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBmb3IgKHF3KEJhY2tlbmRLZXkgTW9kZSkpIHsKICAgICAgICBpZiAoICEkUGFyYW17JF99ICkgewogICAgICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZyggUHJpb3JpdHkgPT4gJ2Vycm9yJywgTWVzc2FnZSA9PiAiTmVlZCAkXyEiICk7CiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICB9CgogICAgbXkgJEF0dHJpYnV0ZXMgPSAkU2VsZi0+X0JhY2tlbmRLZXlQYXJzZSglUGFyYW0pOwoKICAgIG15ICRDb250ZW50ID0gJFNlbGYtPntNYWluT2JqZWN0fS0+RmlsZVJlYWQoCiAgICAgICAgRGlyZWN0b3J5ID0+ICRTZWxmLT57RGF0YURpcn0gLiAkQXR0cmlidXRlcy0+e0RhdGFEaXJ9LAogICAgICAgIEZpbGVuYW1lICA9PiAkQXR0cmlidXRlcy0+e0ZpbGVuYW1lfSwKICAgICAgICBNb2RlICAgICAgPT4gJFBhcmFte01vZGV9LAogICAgKTsKCiAgICAjIHVuY29tcHJlc3MgKGluIGNhc2UpCiAgICBpZiAoICRBdHRyaWJ1dGVzLT57Q29tcHJlc3N9ICkgewoKICAgICAgICAjICRDb250ZW50ID0gLi4uCiAgICB9CgogICAgIyBkZWNyeXB0IChpbiBjYXNlKQogICAgaWYgKCAkQXR0cmlidXRlcy0+e0NyeXB0fSApIHsKCiAgICAgICAgIyAkQ29udGVudCA9IC4uLgogICAgfQoKICAgIHJldHVybiBpZiAhJENvbnRlbnQ7CiAgICByZXR1cm4gJENvbnRlbnQ7Cn0KCnN1YiBXcml0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBmb3IgKHF3KENvbnRlbnQgRmlsZW5hbWUgTW9kZSkpIHsKICAgICAgICBpZiAoICEkUGFyYW17JF99ICkgewogICAgICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZyggUHJpb3JpdHkgPT4gJ2Vycm9yJywgTWVzc2FnZSA9PiAiTmVlZCAkXyEiICk7CiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICB9CgogICAgIyBjb21wcmVzcyAoaW4gY2FzZSkKICAgIGlmICggJFNlbGYtPntDb21wcmVzc30gKSB7CgogICAgICAgICMgJFBhcmFte0NvbnRlbnR9ID0gLi4uCiAgICB9CgogICAgIyBjcnlwdCAoaW4gY2FzZSkKICAgIGlmICggJFNlbGYtPntDcnlwdH0gKSB7CgogICAgICAgICMgJFBhcmFte0NvbnRlbnR9ID0gLi4uCiAgICB9CgogICAgbXkgJE1ENSA9ICRTZWxmLT57TWFpbk9iamVjdH0tPkZpbGVuYW1lQ2xlYW5VcCgKICAgICAgICBGaWxlbmFtZSA9PiAkUGFyYW17RmlsZW5hbWV9LAogICAgICAgIFR5cGUgICAgID0+ICdNRDUnLAogICAgKTsKCiAgICBteSAkRGF0YURpciA9ICcnOwogICAgbXkgQERpcnMgPSAkU2VsZi0+X1NwbGl0RGlyKCBGaWxlbmFtZSA9PiAkTUQ1ICk7CiAgICBmb3IgbXkgJERpciAoQERpcnMpIHsKICAgICAgICAkRGF0YURpciAuPSAnLycgLiAkRGlyOwogICAgICAgIG5leHQgaWYgLWUgJFNlbGYtPntEYXRhRGlyfSAuICREYXRhRGlyOwogICAgICAgIG5leHQgaWYgbWtkaXIgJFNlbGYtPntEYXRhRGlyfSAuICREYXRhRGlyOwogICAgICAgICRTZWxmLT57TG9nT2JqZWN0fS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiQ2FuJ3QgY3JlYXRlICRTZWxmLT57RGF0YURpcn0kRGF0YURpcjogJCEiLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICMgd3JpdGUgYXJ0aWNsZSB0byBmcwogICAgbXkgJEZpbGVuYW1lID0gJFNlbGYtPntNYWluT2JqZWN0fS0+RmlsZVdyaXRlKAogICAgICAgIERpcmVjdG9yeSAgPT4gJFNlbGYtPntEYXRhRGlyfSAuICREYXRhRGlyLAogICAgICAgIEZpbGVuYW1lICAgPT4gJE1ENSwKICAgICAgICBNb2RlICAgICAgID0+ICRQYXJhbXtNb2RlfSwKICAgICAgICBDb250ZW50ICAgID0+ICRQYXJhbXtDb250ZW50fSwKICAgICAgICBQZXJtaXNzaW9uID0+ICRTZWxmLT57UGVybWlzc2lvbn0sCiAgICApOwogICAgcmV0dXJuIGlmICEkRmlsZW5hbWU7CgogICAgbXkgJEJhY2tlbmRLZXkgPSAkU2VsZi0+X0JhY2tlbmRLZXlHZW5lcmF0ZSgKICAgICAgICBGaWxlbmFtZSA9PiAkRmlsZW5hbWUsCiAgICAgICAgRGF0YURpciAgPT4gJERhdGFEaXIsCiAgICAgICAgQ29tcHJlc3MgPT4gJFNlbGYtPntDb21wcmVzc30sCiAgICAgICAgQ3J5cHQgICAgPT4gJFNlbGYtPntDcnlwdH0sCiAgICAgICAgTW9kZSAgICAgPT4gJFBhcmFte01vZGV9LAogICAgICAgIFZlcnNpb24gID0+ICRWRVJTSU9OLAogICAgKTsKCiAgICByZXR1cm4gJEJhY2tlbmRLZXk7Cn0KCnN1YiBEZWxldGUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgZm9yIChxdyhCYWNrZW5kS2V5KSkgewogICAgICAgIGlmICggISRQYXJhbXskX30gKSB7CiAgICAgICAgICAgICRTZWxmLT57TG9nT2JqZWN0fS0+TG9nKCBQcmlvcml0eSA9PiAnZXJyb3InLCBNZXNzYWdlID0+ICJOZWVkICRfISIgKTsKICAgICAgICAgICAgcmV0dXJuOwogICAgICAgIH0KICAgIH0KCiAgICBteSAkQXR0cmlidXRlcyA9ICRTZWxmLT5fQmFja2VuZEtleVBhcnNlKCVQYXJhbSk7CgogICAgcmV0dXJuICRTZWxmLT57TWFpbk9iamVjdH0tPkZpbGVEZWxldGUoCiAgICAgICAgRGlyZWN0b3J5ID0+ICRTZWxmLT57RGF0YURpcn0gLiAkQXR0cmlidXRlcy0+e0RhdGFEaXJ9LAogICAgICAgIEZpbGVuYW1lICA9PiAkQXR0cmlidXRlcy0+e0ZpbGVuYW1lfSwKICAgICk7Cn0KCnN1YiBfQmFja2VuZEtleUdlbmVyYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJEJhY2tlbmRLZXkgPSAnJzsKICAgIGZvciBteSAkS2V5ICggc29ydCBrZXlzICVQYXJhbSApIHsKICAgICAgICAkQmFja2VuZEtleSAuPSAiJEtleT0kUGFyYW17JEtleX07IjsKICAgIH0KICAgIHJldHVybiAkQmFja2VuZEtleTsKfQoKc3ViIF9CYWNrZW5kS2V5UGFyc2UgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgZm9yIChxdyhCYWNrZW5kS2V5KSkgewogICAgICAgIGlmICggISRQYXJhbXskX30gKSB7CiAgICAgICAgICAgICRTZWxmLT57TG9nT2JqZWN0fS0+TG9nKCBQcmlvcml0eSA9PiAnZXJyb3InLCBNZXNzYWdlID0+ICJOZWVkICRfISIgKTsKICAgICAgICAgICAgcmV0dXJuOwogICAgICAgIH0KICAgIH0KCiAgICBteSAlQXR0cmlidXRlczsKICAgIG15IEBQYWlycyA9IHNwbGl0IC87LywgJFBhcmFte0JhY2tlbmRLZXl9OwogICAgZm9yIG15ICRQYWlyIChAUGFpcnMpIHsKICAgICAgICBteSAoICRLZXksICRWYWx1ZSApID0gc3BsaXQgLz0vLCAkUGFpcjsKICAgICAgICAkQXR0cmlidXRlc3skS2V5fSA9ICRWYWx1ZTsKICAgIH0KICAgIHJldHVybiBcJUF0dHJpYnV0ZXM7Cn0KCnN1YiBfU3BsaXREaXIgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgZm9yIChxdyhGaWxlbmFtZSkpIHsKICAgICAgICBpZiAoICEkUGFyYW17JF99ICkgewogICAgICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZyggUHJpb3JpdHkgPT4gJ2Vycm9yJywgTWVzc2FnZSA9PiAiTmVlZCAkXyEiICk7CiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICB9CgogICAgbXkgQERpcjsKICAgICREaXJbMF0gPSBzdWJzdHIgJFBhcmFte0ZpbGVuYW1lfSwgMCwgMjsKICAgICREaXJbMV0gPSBzdWJzdHIgJFBhcmFte0ZpbGVuYW1lfSwgMiwgMjsKICAgIHJldHVybiAoICREaXJbMF0sICREaXJbMV0gKTsKfQoKMTsK
# --
# CustomerUserService.t - CustomerUserService tests
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: CustomerUserService.t,v 1.3 2009/06/30 14:55:02 ub Exp $
# $OldId: CustomerUserService.t,v 1.5 2009/02/16 12:41:12 tr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

use Kernel::System::Service;

$Self->{ServiceObject} = Kernel::System::Service->new( %{$Self} );

# save all original default services
my @OriginalDefaultServices = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => '<DEFAULT>',
    Result            => 'ID',
    DefaultServices   => 0,
);

# delete all default services
for my $ServiceID (@OriginalDefaultServices) {
    $Self->{ServiceObject}->CustomerUserServiceMemberAdd(
        CustomerUserLogin => '<DEFAULT>',
        ServiceID         => $ServiceID,
        Active            => 0,
        UserID            => 1,
    );
}

# add service1
my $ServiceRand1 = 'SomeService' . int( rand(1000000) );
my $ServiceID1   = $Self->{ServiceObject}->ServiceAdd(
    Name    => $ServiceRand1,
    Comment => 'Some Comment',
    ValidID => 1,
    UserID  => 1,
# ---
# ITSM
# ---
    TypeID        => 1,
    CriticalityID => 1,
# ---
);

$Self->True(
    $ServiceID1,
    'ServiceAdd1()',
);

# add service2
my $ServiceRand2 = 'SomeService' . int( rand(1000000) );
my $ServiceID2   = $Self->{ServiceObject}->ServiceAdd(
    Name    => $ServiceRand2,
    Comment => 'Some Comment',
    ValidID => 1,
    UserID  => 1,
# ---
# ITSM
# ---
    TypeID        => 1,
    CriticalityID => 1,
# ---
);

$Self->True(
    $ServiceID2,
    'ServiceAdd2()',
);

my $CustomerUser1 = 'SomeUser' . int( rand(1000000) );
my $CustomerUser2 = 'SomeUser' . int( rand(1000000) );

# allocation test 1
my @Allocation1 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser1,
    Result            => 'ID',
    DefaultServices   => 0,
);

$Self->False(
    scalar @Allocation1,
    'CustomerUserServiceMemberList1()',
);

# allocation test 2
my @Allocation2 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser1,
    Result            => 'ID',
);

$Self->False(
    scalar @Allocation2,
    'CustomerUserServiceMemberList2()',
);

# allocation test 3
my @Allocation3 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser2,
    Result            => 'ID',
    DefaultServices   => 0,
);

$Self->False(
    scalar @Allocation3,
    'CustomerUserServiceMemberList3()',
);

# allocation test 4
my @Allocation4 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser2,
    Result            => 'ID',
);

$Self->False(
    scalar @Allocation4,
    'CustomerUserServiceMemberList4()',
);

# set allocation 1
$Self->{ServiceObject}->CustomerUserServiceMemberAdd(
    CustomerUserLogin => '<DEFAULT>',
    ServiceID         => $ServiceID1,
    Active            => 1,
    UserID            => 1,
);

# allocation test 5
my @Allocation5 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser1,
    Result            => 'ID',
    DefaultServices   => 0,
);

$Self->False(
    scalar @Allocation5,
    'CustomerUserServiceMemberList5()',
);

# allocation test 6
my @Allocation6 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser1,
    Result            => 'ID',
);

my $Allocation6Count = @Allocation6;
my $Allocation6Ok    = 0;
if ( $Allocation6Count eq 1 && $Allocation6[0] eq $ServiceID1 ) {
    $Allocation6Ok = 1;
}

$Self->True(
    $Allocation6Ok,
    'CustomerUserServiceMemberList6()',
);

# allocation test 7
my @Allocation7 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser2,
    Result            => 'ID',
    DefaultServices   => 0,
);

$Self->False(
    scalar @Allocation7,
    'CustomerUserServiceMemberList7()',
);

# allocation test 8
my @Allocation8 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser2,
    Result            => 'ID',
);

my $Allocation8Count = @Allocation8;
my $Allocation8Ok    = 0;
if ( $Allocation8Count eq 1 && $Allocation8[0] eq $ServiceID1 ) {
    $Allocation8Ok = 1;
}

$Self->True(
    $Allocation8Ok,
    'CustomerUserServiceMemberList8()',
);

# set allocation 2
$Self->{ServiceObject}->CustomerUserServiceMemberAdd(
    CustomerUserLogin => $CustomerUser1,
    ServiceID         => $ServiceID2,
    Active            => 1,
    UserID            => 1,
);

# allocation test 9
my @Allocation9 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser1,
    Result            => 'ID',
    DefaultServices   => 0,
);

my $Allocation9Count = @Allocation9;
my $Allocation9Ok    = 0;
if ( $Allocation9Count eq 1 && $Allocation9[0] eq $ServiceID2 ) {
    $Allocation9Ok = 1;
}

$Self->True(
    $Allocation9Ok,
    'CustomerUserServiceMemberList9()',
);

# allocation test 10
my @Allocation10 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser1,
    Result            => 'ID',
);

my $Allocation10Count = @Allocation10;
my $Allocation10Ok    = 0;
if ( $Allocation10Count eq 1 && $Allocation10[0] eq $ServiceID2 ) {
    $Allocation10Ok = 1;
}

$Self->True(
    $Allocation10Ok,
    'CustomerUserServiceMemberList10()',
);

# allocation test 11
my @Allocation11 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser2,
    Result            => 'ID',
    DefaultServices   => 0,
);

$Self->False(
    scalar @Allocation11,
    'CustomerUserServiceMemberList11()',
);

# allocation test 12
my @Allocation12 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser2,
    Result            => 'ID',
);

my $Allocation12Count = @Allocation12;
my $Allocation12Ok    = 0;
if ( $Allocation12Count eq 1 && $Allocation12[0] eq $ServiceID1 ) {
    $Allocation12Ok = 1;
}

$Self->True(
    $Allocation12Ok,
    'CustomerUserServiceMemberList12()',
);

# set allocation 3
$Self->{ServiceObject}->CustomerUserServiceMemberAdd(
    CustomerUserLogin => $CustomerUser2,
    ServiceID         => $ServiceID1,
    Active            => 1,
    UserID            => 1,
);
$Self->{ServiceObject}->CustomerUserServiceMemberAdd(
    CustomerUserLogin => $CustomerUser2,
    ServiceID         => $ServiceID2,
    Active            => 1,
    UserID            => 1,
);

# allocation test 13
my @Allocation13 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser1,
    Result            => 'ID',
    DefaultServices   => 0,
);

my $Allocation13Count = @Allocation13;
my $Allocation13Ok    = 0;
if ( $Allocation13Count eq 1 && $Allocation13[0] eq $ServiceID2 ) {
    $Allocation13Ok = 1;
}

$Self->True(
    $Allocation13Ok,
    'CustomerUserServiceMemberList13()',
);

# allocation test 14
my @Allocation14 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser1,
    Result            => 'ID',
);

my $Allocation14Count = @Allocation14;
my $Allocation14Ok    = 0;
if ( $Allocation14Count eq 1 && $Allocation14[0] eq $ServiceID2 ) {
    $Allocation14Ok = 1;
}

$Self->True(
    $Allocation14Ok,
    'CustomerUserServiceMemberList14()',
);

# allocation test 15
my @Allocation15 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser2,
    Result            => 'ID',
    DefaultServices   => 0,
);

my $Allocation15Count = @Allocation15;
my $Allocation15Ok    = 0;
if (
    $Allocation15Count eq 2 && (
        ( $Allocation15[0] eq $ServiceID1 && $Allocation15[1] eq $ServiceID2 ) ||
        ( $Allocation15[0] eq $ServiceID2 && $Allocation15[1] eq $ServiceID1 )
    )
    )
{
    $Allocation15Ok = 1;
}

$Self->True(
    $Allocation15Ok,
    'CustomerUserServiceMemberList15()',
);

# allocation test 16
my @Allocation16 = $Self->{ServiceObject}->CustomerUserServiceMemberList(
    CustomerUserLogin => $CustomerUser2,
    Result            => 'ID',
);

my $Allocation16Count = @Allocation16;
my $Allocation16Ok    = 0;
if (
    $Allocation16Count eq 2 && (
        ( $Allocation16[0] eq $ServiceID1 && $Allocation16[1] eq $ServiceID2 ) ||
        ( $Allocation16[0] eq $ServiceID2 && $Allocation16[1] eq $ServiceID1 )
    )
    )
{
    $Allocation16Ok = 1;
}

$Self->True(
    $Allocation16Ok,
    'CustomerUserServiceMemberList16()',
);

# delete all test allocations to clean system
$Self->{ServiceObject}->CustomerUserServiceMemberAdd(
    CustomerUserLogin => '<DEFAULT>',
    ServiceID         => $ServiceID1,
    Active            => 0,
    UserID            => 1,
);
$Self->{ServiceObject}->CustomerUserServiceMemberAdd(
    CustomerUserLogin => '<DEFAULT>',
    ServiceID         => $ServiceID2,
    Active            => 0,
    UserID            => 1,
);
$Self->{ServiceObject}->CustomerUserServiceMemberAdd(
    CustomerUserLogin => $CustomerUser1,
    ServiceID         => $ServiceID1,
    Active            => 0,
    UserID            => 1,
);
$Self->{ServiceObject}->CustomerUserServiceMemberAdd(
    CustomerUserLogin => $CustomerUser1,
    ServiceID         => $ServiceID2,
    Active            => 0,
    UserID            => 1,
);
$Self->{ServiceObject}->CustomerUserServiceMemberAdd(
    CustomerUserLogin => $CustomerUser2,
    ServiceID         => $ServiceID1,
    Active            => 0,
    UserID            => 1,
);
$Self->{ServiceObject}->CustomerUserServiceMemberAdd(
    CustomerUserLogin => $CustomerUser2,
    ServiceID         => $ServiceID2,
    Active            => 0,
    UserID            => 1,
);

# restore all original default services
for my $ServiceID (@OriginalDefaultServices) {
    $Self->{ServiceObject}->CustomerUserServiceMemberAdd(
        CustomerUserLogin => '<DEFAULT>',
        ServiceID         => $ServiceID,
        Active            => 1,
        UserID            => 1,
    );
}

# set service1 invalid
my $ServiceUpdate1 = $Self->{ServiceObject}->ServiceUpdate(
    ServiceID => $ServiceID1,
    Name      => $ServiceRand1,
    ValidID   => 2,
    UserID    => 1,
# ---
# ITSM
# ---
    TypeID        => 1,
    CriticalityID => 1,
# ---
);

$Self->True(
    $ServiceUpdate1,
    'ServiceUpdate1()',
);

# set service2 invalid
my $ServiceUpdate2 = $Self->{ServiceObject}->ServiceUpdate(
    ServiceID => $ServiceID2,
    Name      => $ServiceRand2,
    ValidID   => 2,
    UserID    => 1,
# ---
# ITSM
# ---
    TypeID        => 1,
    CriticalityID => 1,
# ---
);

$Self->True(
    $ServiceUpdate2,
    'ServiceUpdate2()',
);

1;

IyAtLQojIElUU01DSVBBbGxvY2F0ZS50IC0gY3JpdGljYWxpdHksIGltcGFjdCBwcmlvcml0eSB0ZXN0cwojIENvcHlyaWdodCAoQykgMjAwMS0yMDEwIE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogSVRTTUNJUEFsbG9jYXRlLnQsdiAxLjkgMjAxMC8wMi8xOCAxNDozMzozNiBiZXMgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSB2YXJzIHF3KCRTZWxmKTsKCnVzZSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNJUEFsbG9jYXRlOwoKJFNlbGYtPntDSVBBbGxvY2F0ZU9iamVjdH0gPSBLZXJuZWw6OlN5c3RlbTo6SVRTTUNJUEFsbG9jYXRlLT5uZXcoICV7JFNlbGZ9ICk7CgojIGdldCBjdXJyZW50IGFsbG9jYXRpb24gbGlzdCAoVXNlcklEIGlzIG5lZWRlZCkKbXkgJEFsbG9jYXRlRGF0YTEgPSAkU2VsZi0+e0NJUEFsbG9jYXRlT2JqZWN0fS0+QWxsb2NhdGVMaXN0KCk7CgojIGNoZWNrIHRoZSByZXN1bHQKJFNlbGYtPkZhbHNlKCAkQWxsb2NhdGVEYXRhMSwgJ0FsbG9jYXRlTGlzdCgpJyApOwoKIyBnZXQgY3VycmVudCBhbGxvY2F0aW9uIGxpc3QKbXkgJEFsbG9jYXRlRGF0YTIgPSAkU2VsZi0+e0NJUEFsbG9jYXRlT2JqZWN0fS0+QWxsb2NhdGVMaXN0KAogICAgVXNlcklEID0+IDEsCik7CgojIGNoZWNrIHRoZSByZXN1bHQKJFNlbGYtPlRydWUoICRBbGxvY2F0ZURhdGEyLCAnQWxsb2NhdGVMaXN0KCknICk7CgojIGNoZWNrIHRoZSBhbGxvY2F0aW9uIGhhc2gKbXkgJEhhc2hPSyA9IDE7CmlmICggcmVmICRBbGxvY2F0ZURhdGEyIG5lICdIQVNIJyApIHsKICAgICRIYXNoT0sgPSAwOwp9CgojIGNoZWNrIHRoZSBhbGxvY2F0aW9uIDJkIGhhc2gKaWYgKCRIYXNoT0spIHsKCiAgICBJTVBBQ1RJRDoKICAgIGZvciBteSAkSW1wYWN0SUQgKCBrZXlzICV7JEFsbG9jYXRlRGF0YTJ9ICkgewoKICAgICAgICBpZiAoIHJlZiAkQWxsb2NhdGVEYXRhMi0+eyRJbXBhY3RJRH0gbmUgJ0hBU0gnICkgewogICAgICAgICAgICAkSGFzaE9LID0gMDsKICAgICAgICAgICAgbGFzdCBJTVBBQ1RJRDsKICAgICAgICB9CgogICAgICAgIENSSVRJQ0FMSVRZSUQ6CiAgICAgICAgZm9yIG15ICRDcml0aWNhbGl0eUlEICgga2V5cyAleyAkQWxsb2NhdGVEYXRhMi0+eyRJbXBhY3RJRH0gfSApIHsKCiAgICAgICAgICAgIGlmICggISRDcml0aWNhbGl0eUlEIHx8ICEkQWxsb2NhdGVEYXRhMi0+eyRJbXBhY3RJRH0tPnskQ3JpdGljYWxpdHlJRH0gKSB7CiAgICAgICAgICAgICAgICAkSGFzaE9LID0gMDsKICAgICAgICAgICAgICAgIGxhc3QgSU1QQUNUSUQ7CiAgICAgICAgICAgIH0KICAgICAgICB9CiAgICB9Cn0KCiMgY2hlY2sgSGFzaE9LCiRTZWxmLT5UcnVlKCAkSGFzaE9LLCAnQWxsb2NhdGVMaXN0KCknICk7CgojIGNhbGwgUHJpb3JpdHlBbGxvY2F0aW9uR2V0KCkgZm9yIG9uZSBDcml0aWNhbGl0eS9JbXBhY3QgcGFpcgppZiAoJEhhc2hPSykgewoKICAgIG15ICgkSW1wYWN0SUQpID0ga2V5cyAleyRBbGxvY2F0ZURhdGEyfTsKCiAgICBpZiAoICRBbGxvY2F0ZURhdGEyLT57JEltcGFjdElEfSApIHsKICAgICAgICBteSAoJENyaXRpY2FsaXR5SUQpID0ga2V5cyAleyAkQWxsb2NhdGVEYXRhMi0+eyRJbXBhY3RJRH0gfTsKCiAgICAgICAgbXkgJEV4cGVjdGVkUHJpb3JpdHlJRCA9ICRBbGxvY2F0ZURhdGEyLT57JEltcGFjdElEfS0+eyRDcml0aWNhbGl0eUlEfTsKICAgICAgICBteSAkUHJpb3JpdHlJRCAgICAgICAgID0gJFNlbGYtPntDSVBBbGxvY2F0ZU9iamVjdH0tPlByaW9yaXR5QWxsb2NhdGlvbkdldCgKICAgICAgICAgICAgQ3JpdGljYWxpdHlJRCA9PiAkQ3JpdGljYWxpdHlJRCwKICAgICAgICAgICAgSW1wYWN0SUQgICAgICA9PiAkSW1wYWN0SUQsCiAgICAgICAgKTsKICAgICAgICAkU2VsZi0+SXMoCiAgICAgICAgICAgICRQcmlvcml0eUlELAogICAgICAgICAgICAkRXhwZWN0ZWRQcmlvcml0eUlELAogICAgICAgICAgICAnUHJpb3JpdHlBbGxvY2F0aW9uR2V0KCknLAogICAgICAgICk7CiAgICB9Cn0KCiMgdXBkYXRlIHRoZSBhbGxvY2F0aW9uIGhhc2ggKG5vdCBhbGwgbmVlZGVkIGFyZ3VtZW50cyBnaXZlbikKbXkgJFN1Y2Nlc3MxID0gJFNlbGYtPntDSVBBbGxvY2F0ZU9iamVjdH0tPkFsbG9jYXRlVXBkYXRlKAogICAgVXNlcklEID0+IDEsCik7CgojIGNoZWNrIHRoZSByZXN1bHQKJFNlbGYtPkZhbHNlKCAkU3VjY2VzczEsICdBbGxvY2F0ZVVwZGF0ZSgpJyApOwoKIyB1cGRhdGUgdGhlIGFsbG9jYXRpb24gaGFzaCAobm90IGFsbCBuZWVkZWQgYXJndW1lbnRzIGdpdmVuKQpteSAkU3VjY2VzczIgPSAkU2VsZi0+e0NJUEFsbG9jYXRlT2JqZWN0fS0+QWxsb2NhdGVVcGRhdGUoCiAgICBBbGxvY2F0ZURhdGEgPT4gJEFsbG9jYXRlRGF0YTIsCik7CgojIGNoZWNrIHRoZSByZXN1bHQKJFNlbGYtPkZhbHNlKCAkU3VjY2VzczIsICdBbGxvY2F0ZVVwZGF0ZSgpJyApOwoKIyB1cGRhdGUgdGhlIGFsbG9jYXRpb24gaGFzaCAoYWxsb2NhdGlvbiBoYXNoKQpteSAkU3VjY2VzczMgPSAkU2VsZi0+e0NJUEFsbG9jYXRlT2JqZWN0fS0+QWxsb2NhdGVVcGRhdGUoCiAgICBBbGxvY2F0ZURhdGEgPT4gewogICAgICAgIFRlc3QgID0+ICdhYWEnLAogICAgICAgIFRlc3QyID0+ICdiYmInLAogICAgfSwKICAgIFVzZXJJRCA9PiAxLAopOwoKIyBjaGVjayB0aGUgcmVzdWx0CiRTZWxmLT5GYWxzZSggJFN1Y2Nlc3MzLCAnQWxsb2NhdGVVcGRhdGUoKScgKTsKCiMgdXBkYXRlIHRoZSBhbGxvY2F0aW9uIGhhc2gKbXkgJFN1Y2Nlc3M0ID0gJFNlbGYtPntDSVBBbGxvY2F0ZU9iamVjdH0tPkFsbG9jYXRlVXBkYXRlKAogICAgQWxsb2NhdGVEYXRhID0+ICRBbGxvY2F0ZURhdGEyLAogICAgVXNlcklEICAgICAgID0+IDEsCik7CgojIGNoZWNrIHRoZSByZXN1bHQKJFNlbGYtPlRydWUoICRTdWNjZXNzNCwgJ0FsbG9jYXRlVXBkYXRlKCknICk7CgoxOwo=
# --
# Service.t - Service tests
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: Service.t,v 1.2 2009/06/30 14:55:47 ub Exp $
# $OldId: Service.t,v 1.10 2009/02/16 12:40:23 tr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

use Kernel::System::Service;
use Kernel::System::User;

$Self->{ServiceObject} = Kernel::System::Service->new( %{$Self} );
$Self->{UserObject}    = Kernel::System::User->new( %{$Self} );

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# create needed users
my @UserIDs;
{

    # disable email checks to create new user
    my $CheckEmailAddressesOrg = $Self->{ConfigObject}->Get('CheckEmailAddresses') || 1;
    $Self->{ConfigObject}->Set(
        Key   => 'CheckEmailAddresses',
        Value => 0,
    );

    for my $Counter ( 1 .. 2 ) {

        # create new users for the tests
        my $UserID = $Self->{UserObject}->UserAdd(
            UserFirstname => 'Service' . $Counter,
            UserLastname  => 'UnitTest',
            UserLogin     => 'UnitTest-Service-' . $Counter . int rand 1_000_000,
            UserEmail     => 'UnitTest-Service-' . $Counter . '@localhost',
            ValidID       => 1,
            ChangeUserID  => 1,
        );

        push @UserIDs, $UserID;
    }

    # restore original email check param
    $Self->{ConfigObject}->Set(
        Key   => 'CheckEmailAddresses',
        Value => $CheckEmailAddressesOrg,
    );
}

# create needed random service names
my @ServiceName;
for my $Counter ( 1 .. 11 ) {
    push @ServiceName, 'UnitTest' . int rand 1_000_000;
}

# get original service list for later checks
my %ServiceListOriginal = $Self->{ServiceObject}->ServiceList(
    Valid  => 0,
    UserID => 1,
);

# ------------------------------------------------------------ #
# define general tests
# ------------------------------------------------------------ #

my $ItemData = [

    # this service is NOT complete and must not be added
    {
        Add => {
            ValidID => 1,
            UserID  => 1,
        },
    },

    # this service is NOT complete and must not be added
    {
        Add => {
            Name   => $ServiceName[0],
            UserID => 1,
        },
    },

    # this service is NOT complete and must not be added
    {
        Add => {
            Name    => $ServiceName[0],
            ValidID => 1,
        },
    },
# ---
# ITSM
# ---

    # this service is NOT complete and must not be added
    {
        Add => {
            Name    => $ServiceName[0],
            ValidID => 1,
            UserID  => 1,
            TypeID  => 1,
        },
    },

    # this service is NOT complete and must not be added
    {
        Add => {
            Name          => $ServiceName[0],
            ValidID       => 1,
            UserID        => 1,
            CriticalityID => 1,
        },
    },
# ---

    # this service must be inserted sucessfully
    {
        Add => {
            Name    => $ServiceName[0],
            ValidID => 1,
            UserID  => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
        AddGet => {
            ParentID  => '',
            Name      => $ServiceName[0],
            NameShort => $ServiceName[0],
            ValidID   => 1,
            Comment   => '',
            CreateBy  => 1,
            ChangeBy  => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
    },

    # this service have the same name as one test before and must not be added
    {
        Add => {
            Name    => $ServiceName[0],
            ValidID => 1,
            UserID  => 1,
        },
    },

    # the service one add-test before must be NOT updated (service is NOT complete)
    {
        Update => {
            ValidID => 1,
            UserID  => 1,
        },
    },

    # the service one add-test before must be NOT updated (service is NOT complete)
    {
        Update => {
            Name   => $ServiceName[0] . 'UPDATE1',
            UserID => 1,
        },
    },

    # the service one add-test before must be NOT updated (service is NOT complete)
    {
        Update => {
            Name    => $ServiceName[0] . 'UPDATE1',
            ValidID => 1,
        },
    },
# ---
# ITSM
# ---

    # the service one add-test before must be NOT updated (service is NOT complete)
    {
        Update => {
            Name    => $ServiceName[0] . 'UPDATE1',
            ValidID => 1,
            UserID  => 1,
            TypeID  => 1,
        },
    },

    # the service one add-test before must be NOT updated (service is NOT complete)
    {
        Update => {
            Name          => $ServiceName[0] . 'UPDATE1',
            ValidID       => 1,
            UserID        => 1,
            CriticalityID => 1,
        },
    },
# ---

    # this service must be inserted sucessfully
    {
        Add => {
            Name    => $ServiceName[1],
            ValidID => 1,
            Comment => 'TestComment2',
            UserID  => 1,
# ---
# ITSM
# ---
            TypeID        => 2,
            CriticalityID => 3,
# ---
        },
        AddGet => {
            ParentID  => '',
            Name      => $ServiceName[1],
            NameShort => $ServiceName[1],
            ValidID   => 1,
            Comment   => 'TestComment2',
            CreateBy  => 1,
            ChangeBy  => 1,
# ---
# ITSM
# ---
            TypeID        => 2,
            CriticalityID => 3,
# ---
        },
    },

    # the service one add-test before must be NOT updated (service update arguments NOT complete)
    {
        Update => {
            ValidID => 1,
            UserID  => 1,
        },
    },

    # the service one add-test before must be NOT updated (service update arguments NOT complete)
    {
        Update => {
            Name   => $ServiceName[1] . 'UPDATE2',
            UserID => 1,
        },
    },

    # the service one add-test before must be NOT updated (service update arguments NOT complete)
    {
        Update => {
            Name    => $ServiceName[1] . 'UPDATE2',
            ValidID => 1,
        },
    },

    # the service one add-test before must be updated (service update arguments are complete)
    {
        Update => {
            Name    => $ServiceName[1] . 'UPDATE2',
            ValidID => 2,
            Comment => 'TestComment2UPDATE2',
            UserID  => $UserIDs[0],
# ---
# ITSM
# ---
            TypeID        => 4,
            CriticalityID => 5,
# ---
        },
        UpdateGet => {
            ParentID  => '',
            Name      => $ServiceName[1] . 'UPDATE2',
            NameShort => $ServiceName[1] . 'UPDATE2',
            ValidID   => 2,
            Comment   => 'TestComment2UPDATE2',
            CreateBy  => 1,
            ChangeBy  => $UserIDs[0],
# ---
# ITSM
# ---
            TypeID        => 4,
            CriticalityID => 5,
# ---
        },
    },

    # the service one add-test before must be updated (service update arguments are complete)
    {
        Update => {
            Name    => $ServiceName[1] . 'UPDATE3',
            ValidID => 1,
            Comment => 'TestComment2UPDATE3',
            UserID  => $UserIDs[1],
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
        UpdateGet => {
            ParentID  => '',
            Name      => $ServiceName[1] . 'UPDATE3',
            NameShort => $ServiceName[1] . 'UPDATE3',
            ValidID   => 1,
            Comment   => 'TestComment2UPDATE3',
            CreateBy  => 1,
            ChangeBy  => $UserIDs[1],
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
    },

    # this service has an invalid name and must be NOT inserted
    {
        Update => {
            Name    => $ServiceName[1] . '::UPDATE4',
            ValidID => 1,
            UserID  => 1,
        },
    },

    # this service has an invalid name and must be NOT inserted
    {
        Update => {
            Name    => $ServiceName[1] . '::Test::UPDATE4',
            ValidID => 1,
            UserID  => 1,
        },
    },

    # this service has an invalid name and must be NOT inserted
    {
        Add => {
            Name    => $ServiceName[2] . '::Test',
            ValidID => 1,
            UserID  => 1,
        },
    },

    # this service has an invalid name and must be NOT inserted
    {
        Add => {
            Name    => '::Test' . $ServiceName[2],
            ValidID => 1,
            UserID  => 1,
        },
    },

    # this service has an invalid name and must be NOT inserted
    {
        Add => {
            Name    => $ServiceName[2] . '::Test::Test',
            ValidID => 1,
            UserID  => 1,
        },
    },

    # this service has an invalid name and must be NOT inserted
    {
        Add => {
            Name    => $ServiceName[2] . 'Test::',
            ValidID => 1,
            UserID  => 1,
        },
    },

    # this service must be inserted sucessfully (check string cleaner function)
    {
        Add => {
            Name    => " \t \n \r " . $ServiceName[3] . " \t \n \r ",
            ValidID => 1,
            Comment => " \t \n \r Test Comment \t \n \r ",
            UserID  => 1,
# ---
# ITSM
# ---
            TypeID        => 2,
            CriticalityID => 2,
# ---
        },
        AddGet => {
            ParentID  => '',
            Name      => $ServiceName[3],
            NameShort => $ServiceName[3],
            ValidID   => 1,
            Comment   => 'Test Comment',
            CreateBy  => 1,
            ChangeBy  => 1,
# ---
# ITSM
# ---
            TypeID        => 2,
            CriticalityID => 2,
# ---
        },
    },

    # the service one add-test before must be updated sucessfully (check string cleaner function)
    {
        Update => {
            Name    => " \t \n \r " . $ServiceName[3] . " UPDATE1 \t \n \r ",
            ValidID => 2,
            Comment => " \t \n \r Test Comment \t \n \r ",
            UserID  => $UserIDs[1],
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
        UpdateGet => {
            ParentID  => '',
            Name      => $ServiceName[3] . ' UPDATE1',
            NameShort => $ServiceName[3] . ' UPDATE1',
            ValidID   => 2,
            Comment   => 'Test Comment',
            CreateBy  => 1,
            ChangeBy  => $UserIDs[1],
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
    },

    # this service must be inserted sucessfully (unicode checks)
    {
        Add => {
            Name    => $ServiceName[4] . ' ϒ ϡ Ʃ Ϟ ',
            ValidID => 1,
            Comment => ' Ѡ Ѥ TestComment5 Ϡ Ω ',
            UserID  => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
        AddGet => {
            ParentID  => '',
            Name      => $ServiceName[4] . ' ϒ ϡ Ʃ Ϟ',
            NameShort => $ServiceName[4] . ' ϒ ϡ Ʃ Ϟ',
            ValidID   => 1,
            Comment   => 'Ѡ Ѥ TestComment5 Ϡ Ω',
            CreateBy  => 1,
            ChangeBy  => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
    },

    # the service one add-test before must be updated sucessfully (unicode checks)
    {
        Update => {
            Name    => $ServiceName[4] . ' ϒ ϡ Ʃ Ϟ UPDATE1',
            ValidID => 2,
            Comment => ' Ѡ Ѥ TestComment5 Ϡ Ω UPDATE1',
            UserID  => $UserIDs[0],
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
        UpdateGet => {
            ParentID  => '',
            Name      => $ServiceName[4] . ' ϒ ϡ Ʃ Ϟ UPDATE1',
            NameShort => $ServiceName[4] . ' ϒ ϡ Ʃ Ϟ UPDATE1',
            ValidID   => 2,
            Comment   => 'Ѡ Ѥ TestComment5 Ϡ Ω UPDATE1',
            CreateBy  => 1,
            ChangeBy  => $UserIDs[0],
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
    },

    # this service must be inserted sucessfully (special character checks)
    {
        Add => {
            Name    => ' [test]%*\\ ' . $ServiceName[8] . ' [test]%*\\ ',
            ValidID => 1,
            Comment => ' [test]%*\\ Test Comment [test]%*\\ ',
            UserID  => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
        AddGet => {
            ParentID  => '',
            Name      => '[test]%*\\ ' . $ServiceName[8] . ' [test]%*\\',
            NameShort => '[test]%*\\ ' . $ServiceName[8] . ' [test]%*\\',
            ValidID   => 1,
            Comment   => '[test]%*\\ Test Comment [test]%*\\',
            CreateBy  => 1,
            ChangeBy  => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
    },

    # the service one add-test before must be updated sucessfully (special character checks)
    {
        Update => {
            Name    => ' [test]%*\\ ' . $ServiceName[8] . ' UPDATE1 [test]%*\\ ',
            ValidID => 2,
            Comment => ' [test]%*\\ Test Comment UPDATE1 [test]%*\\ ',
            UserID  => $UserIDs[1],
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
        UpdateGet => {
            ParentID  => '',
            Name      => '[test]%*\\ ' . $ServiceName[8] . ' UPDATE1 [test]%*\\',
            NameShort => '[test]%*\\ ' . $ServiceName[8] . ' UPDATE1 [test]%*\\',
            ValidID   => 2,
            Comment   => '[test]%*\\ Test Comment UPDATE1 [test]%*\\',
            CreateBy  => 1,
            ChangeBy  => $UserIDs[1],
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
    },

    # this service must be inserted sucessfully (used for the following tests)
    {
        Add => {
            Name    => $ServiceName[5],
            ValidID => 1,
            UserID  => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
        AddGet => {
            ParentID  => '',
            Name      => $ServiceName[5],
            NameShort => $ServiceName[5],
            ValidID   => 1,
            CreateBy  => 1,
            ChangeBy  => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
    },

    # this service must be inserted sucessfully (parent service check)
    {
        Add => {
            ParentID => 'LASTADDID',
            Name     => $ServiceName[6],
            ValidID  => 1,
            UserID   => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
        AddGet => {
            ParentID  => 'LASTADDID',
            Name      => $ServiceName[5] . '::' . $ServiceName[6],
            NameShort => $ServiceName[6],
            ValidID   => 1,
            CreateBy  => 1,
            ChangeBy  => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
    },

    # this service must be inserted sucessfully (parent service check)
    {
        Add => {
            ParentID => 'LASTADDID',
            Name     => " \n \t " . $ServiceName[7] . " \n \t ",
            ValidID  => 1,
            UserID   => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
        AddGet => {
            ParentID  => 'LASTADDID',
            Name      => $ServiceName[5] . '::' . $ServiceName[6] . '::' . $ServiceName[7],
            NameShort => $ServiceName[7],
            ValidID   => 1,
            CreateBy  => 1,
            ChangeBy  => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
    },

    # the service must be NOT updated (parent service id and parent id are identical)
    {
        Update => {
            ParentID => 'LASTADDID',
            Name     => $ServiceName[7] . 'UPDATE1',
            ValidID  => 1,
            UserID   => 1,
        },
    },

    # this service must be updated sucessfully (move service to the higherst level)
    {
        Update => {
            ParentID => '',
            Name     => $ServiceName[7] . ' UPDATE1',
            ValidID  => 1,
            UserID   => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
        UpdateGet => {
            ParentID  => '',
            Name      => $ServiceName[7] . ' UPDATE1',
            NameShort => $ServiceName[7] . ' UPDATE1',
            ValidID   => 1,
            CreateBy  => 1,
            ChangeBy  => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
    },

    # this service must be updated sucessfully (move service back with the old parent service)
    {
        Update => {
            ParentID => 'LASTLASTADDID',
            Name     => $ServiceName[7] . ' UPDATE(2)',
            ValidID  => 1,
            UserID   => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
        UpdateGet => {
            ParentID => 'LASTLASTADDID',
            Name     => $ServiceName[5] . '::'
                . $ServiceName[6] . '::'
                . $ServiceName[7]
                . ' UPDATE(2)',
            NameShort => $ServiceName[7] . ' UPDATE(2)',
            ValidID   => 1,
            CreateBy  => 1,
            ChangeBy  => 1,
# ---
# ITSM
# ---
            TypeID        => 1,
            CriticalityID => 1,
# ---
        },
    },
];

# ------------------------------------------------------------ #
# run general tests
# ------------------------------------------------------------ #

my $TestCount = 1;
my $LastAddedServiceID;
my $LastLastAddedServiceID;
my $AddedCounter = 0;

for my $Item ( @{$ItemData} ) {

    if ( $Item->{Add} ) {

        # prepare parent id
        if ( $Item->{Add}->{ParentID} && $Item->{Add}->{ParentID} eq 'LASTADDID' ) {
            $Item->{Add}->{ParentID} = $LastAddedServiceID;
        }
        elsif ( $Item->{Add}->{ParentID} && $Item->{Add}->{ParentID} eq 'LASTLASTADDID' ) {
            $Item->{Add}->{ParentID} = $LastLastAddedServiceID;
        }
        else {
            delete $Item->{Add}->{ParentID};
        }

        # add new service
        my $ServiceID = $Self->{ServiceObject}->ServiceAdd(
            %{ $Item->{Add} },
        );

        # check if service was added successfully or not
        if ( $Item->{AddGet} ) {

            # prepare parent id
            if ( $Item->{AddGet}->{ParentID} && $Item->{AddGet}->{ParentID} eq 'LASTADDID' ) {
                $Item->{AddGet}->{ParentID} = $LastAddedServiceID;
            }
            elsif ( $Item->{AddGet}->{ParentID} && $Item->{AddGet}->{ParentID} eq 'LASTLASTADDID' )
            {
                $Item->{AddGet}->{ParentID} = $LastLastAddedServiceID;
            }

            $Self->True(
                $ServiceID,
                "Test $TestCount: ServiceAdd() - ServiceID: $ServiceID",
            );

            if ($ServiceID) {

                # lookup service name
                my $ServiceName = $Self->{ServiceObject}->ServiceLookup(
                    ServiceID => $ServiceID,
                );

                # lookup test
                $Self->Is(
                    $ServiceName || '',
                    $Item->{AddGet}->{Name} || '',
                    "Test $TestCount: ServiceLookup() - lookup",
                );

                # reverse lookup the service id
                my $ServiceIDNew = $Self->{ServiceObject}->ServiceLookup(
                    Name => $ServiceName || '',
                );

                # reverse lookup test
                $Self->Is(
                    $ServiceIDNew || '',
                    $ServiceID    || '',
                    "Test $TestCount: ServiceLookup() - reverse lookup",
                );

                # set last service id variable
                $LastLastAddedServiceID = $LastAddedServiceID;
                $LastAddedServiceID     = $ServiceID;

                # increment the added counter
                $AddedCounter++;
            }
        }
        else {
            $Self->False(
                $ServiceID,
                "Test $TestCount: ServiceAdd()",
            );
        }

        # get service data to check the values after creation of the service
        my %ServiceGet = $Self->{ServiceObject}->ServiceGet(
            ServiceID => $ServiceID,
            UserID    => $Item->{Add}->{UserID},
        );

        # check service data after creation of the service
        for my $ServiceAttribute ( keys %{ $Item->{AddGet} } ) {
            $Self->Is(
                $ServiceGet{$ServiceAttribute} || '',
                $Item->{AddGet}->{$ServiceAttribute} || '',
                "Test $TestCount: ServiceGet() - $ServiceAttribute",
            );
        }
    }

    if ( $Item->{Update} ) {

        # check last service id varaible
        if ( !$LastAddedServiceID ) {
            $Self->False(
                1,
                "Test $TestCount: NO LAST SERVICE ID GIVEN",
            );
        }

        # prepare parent id
        if ( $Item->{Update}->{ParentID} && $Item->{Update}->{ParentID} eq 'LASTADDID' ) {
            $Item->{Update}->{ParentID} = $LastAddedServiceID;
        }
        elsif ( $Item->{Update}->{ParentID} && $Item->{Update}->{ParentID} eq 'LASTLASTADDID' ) {
            $Item->{Update}->{ParentID} = $LastLastAddedServiceID;
        }
        else {
            delete $Item->{Update}->{ParentID};
        }

        # update the service
        my $UpdateSucess = $Self->{ServiceObject}->ServiceUpdate(
            %{ $Item->{Update} },
            ServiceID => $LastAddedServiceID,
        );

        # check if service was updated successfully or not
        if ( $Item->{UpdateGet} ) {
            $Self->True(
                $UpdateSucess,
                "Test $TestCount: ServiceUpdate() - ServiceID: $LastAddedServiceID",
            );
        }
        else {
            $Self->False(
                $UpdateSucess,
                "Test $TestCount: ServiceUpdate()",
            );
        }

        # prepare parent id
        if ( $Item->{UpdateGet}->{ParentID} && $Item->{UpdateGet}->{ParentID} eq 'LASTADDID' ) {
            $Item->{UpdateGet}->{ParentID} = $LastAddedServiceID;
        }
        elsif (
            $Item->{UpdateGet}->{ParentID}
            && $Item->{UpdateGet}->{ParentID} eq 'LASTLASTADDID'
            )
        {
            $Item->{UpdateGet}->{ParentID} = $LastLastAddedServiceID;
        }

        # get service data to check the values after the update
        my %ServiceGet2 = $Self->{ServiceObject}->ServiceGet(
            ServiceID => $LastAddedServiceID,
            UserID    => $Item->{Update}->{UserID},
        );

        # check service data after update
        for my $ServiceAttribute ( keys %{ $Item->{UpdateGet} } ) {
            $Self->Is(
                $ServiceGet2{$ServiceAttribute} || '',
                $Item->{UpdateGet}->{$ServiceAttribute} || '',
                "Test $TestCount: ServiceGet() - $ServiceAttribute",
            );
        }

        # lookup service name
        my $ServiceName = $Self->{ServiceObject}->ServiceLookup(
            ServiceID => $ServiceGet2{ServiceID},
        );

        # lookup test
        $Self->Is(
            $ServiceName || '',
            $ServiceGet2{Name} || '',
            "Test $TestCount: ServiceLookup() - lookup",
        );

        # reverse lookup the service id
        my $ServiceIDNew = $Self->{ServiceObject}->ServiceLookup(
            Name => $ServiceName || '',
        );

        # reverse lookup test
        $Self->Is(
            $ServiceIDNew || '',
            $ServiceGet2{ServiceID} || '',
            "Test $TestCount: ServiceLookup() - reverse lookup",
        );
    }

    $TestCount++;
}

# ------------------------------------------------------------ #
# ServiceList test 1 (check general functionality)
# ------------------------------------------------------------ #

my %ServiceList1 = $Self->{ServiceObject}->ServiceList(
    Valid  => 0,
    UserID => 1,
);
my %ServiceList1Org = %ServiceListOriginal;

SERVICEID:
for my $ServiceID ( keys %ServiceList1Org ) {

    if ( $ServiceList1{$ServiceID} && $ServiceList1Org{$ServiceID} eq $ServiceList1{$ServiceID} ) {
        delete $ServiceList1{$ServiceID};
    }
    else {
        $ServiceList1{Dummy} = 1;
    }
}

my $ServiceList1Count = scalar keys %ServiceList1;

$Self->Is(
    $ServiceList1Count || '',
    $AddedCounter      || '',
    "Test $TestCount: ServiceList()",
);

$TestCount++;

# ------------------------------------------------------------ #
# ServiceList test 2 (check cache)
# ------------------------------------------------------------ #

my %ServiceList2 = $Self->{ServiceObject}->ServiceList(
    Valid  => 0,
    UserID => 1,
);

my $ServiceList2ServiceID = $Self->{ServiceObject}->ServiceAdd(
    Name    => $ServiceName[9],
    ValidID => 1,
    UserID  => 1,
# ---
# ITSM
# ---
    TypeID        => 1,
    CriticalityID => 1,
# ---
);

my %ServiceList2b = $Self->{ServiceObject}->ServiceList(
    Valid  => 0,
    UserID => 1,
);

SERVICEID:
for my $ServiceID ( keys %ServiceList2 ) {

    if ( $ServiceList2b{$ServiceID} && $ServiceList2{$ServiceID} eq $ServiceList2b{$ServiceID} ) {
        delete $ServiceList2b{$ServiceID};
    }
    else {
        $ServiceList2b{Dummy} = 1;
    }
}

my @ServiceList2IDs   = keys %ServiceList2b;
my $ServiceList2Count = scalar @ServiceList2IDs;

$Self->Is(
    $ServiceList2Count || '',
    1,
    "Test $TestCount: ServiceList() - check number of services",
);

$Self->Is(
    $ServiceList2IDs[0] || '',
    $ServiceList2ServiceID || '',
    "Test $TestCount: ServiceList() - check id of last service",
);

$TestCount++;

# ------------------------------------------------------------ #
# ServiceSearch test 1 (check general functionality)
# ------------------------------------------------------------ #

my @ServiceSearch1Search = $Self->{ServiceObject}->ServiceSearch(
    UserID => 1,
);

my %ServiceSearch1List = $Self->{ServiceObject}->ServiceList(
    UserID => 1,
);

SERVICEID:
for my $ServiceID (@ServiceSearch1Search) {

    if ( $ServiceSearch1List{$ServiceID} ) {
        delete $ServiceSearch1List{$ServiceID};
    }
    else {
        $ServiceSearch1List{Dummy} = 1;
    }
}

my $ServiceSearch1Count = scalar keys %ServiceSearch1List;

$Self->Is(
    $ServiceSearch1Count,
    0,
    "Test $TestCount: ServiceSearch()",
);

$TestCount++;

# ------------------------------------------------------------ #
# make preparations for later tests
# ------------------------------------------------------------ #

# add some needed services for later tests
my @ServiceNames = ( $ServiceName[10] . 'Normal', $ServiceName[10] . 'Ԉ Ӵ Ϫ Ͼ' );
my %ServiceSearch2ServiceID;

my $Counter1 = 0;
for my $ServiceName (@ServiceNames) {

    $ServiceSearch2ServiceID{$Counter1} = $Self->{ServiceObject}->ServiceAdd(
        Name    => $ServiceName,
        ValidID => 1,
        UserID  => 1,
# ---
# ITSM
# ---
        TypeID        => 1,
        CriticalityID => 1,
# ---
    );

    $Counter1++;
}

# ------------------------------------------------------------ #
# ServiceSearch test 2 (general name checks)
# ------------------------------------------------------------ #

my $Counter2 = 0;
for my $ServiceName (@ServiceNames) {

    my @PreparedNames = (
        $ServiceName,
        '*' . $ServiceName,
        $ServiceName . '*',
        '*' . $ServiceName . '*',
        '**' . $ServiceName,
        $ServiceName . '**',
        '**' . $ServiceName . '**',
    );

    for my $PreparedName (@PreparedNames) {

        my @ServiceList = $Self->{ServiceObject}->ServiceSearch(
            Name   => $ServiceName,
            UserID => 1,
        );

        $Self->Is(
            $ServiceList[0] || '',
            $ServiceSearch2ServiceID{$Counter2} || '',
            "Test $TestCount: ServiceSearch() - general name check",
        );

        $TestCount++;
    }

    $Counter2++;
}

1;

# --
# SLA.t - SLA tests
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: SLA.t,v 1.3 2009/06/30 14:55:23 ub Exp $
# $OldId: SLA.t,v 1.10 2009/02/16 12:41:12 tr Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

use Data::Dumper;
use Kernel::System::Service;
use Kernel::System::SLA;
use Kernel::System::User;

$Self->{ServiceObject} = Kernel::System::Service->new( %{$Self} );
$Self->{SLAObject}     = Kernel::System::SLA->new( %{$Self} );
$Self->{UserObject}    = Kernel::System::User->new( %{$Self} );

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# create needed users
my @UserIDs;
{

    # disable email checks to create new user
    my $CheckEmailAddressesOrg = $Self->{ConfigObject}->Get('CheckEmailAddresses') || 1;
    $Self->{ConfigObject}->Set(
        Key   => 'CheckEmailAddresses',
        Value => 0,
    );

    for my $Counter ( 1 .. 2 ) {

        # create new users for the tests
        my $UserID = $Self->{UserObject}->UserAdd(
            UserFirstname => 'SLA' . $Counter,
            UserLastname  => 'UnitTest',
            UserLogin     => 'UnitTest-SLA-' . $Counter . int rand 1_000_000,
            UserEmail     => 'UnitTest-SLA-' . $Counter . '@localhost',
            ValidID       => 1,
            ChangeUserID  => 1,
        );

        push @UserIDs, $UserID;
    }

    # restore original email check param
    $Self->{ConfigObject}->Set(
        Key   => 'CheckEmailAddresses',
        Value => $CheckEmailAddressesOrg,
    );
}

# create needed random service names
my @SLAName;
for my $Counter ( 1 .. 10 ) {
    push @SLAName, 'UnitTest' . int rand 1_000_000;
}

# create some test services
my @ServiceIDs;
for my $Counter ( 1 .. 3 ) {

    # add a service
    my $ServiceID = $Self->{ServiceObject}->ServiceAdd(
        Name    => 'UnitTest-SLA' . int rand 1_000_000,
        ValidID => 1,
        UserID  => 1,
# ---
# ITSM
# ---
        TypeID        => 1,
        CriticalityID => 1,
# ---
    );

    push @ServiceIDs, $ServiceID;
}

# get original sla list for later checks
my %SLAListOriginal = $Self->{SLAObject}->SLAList(
    Valid  => 0,
    UserID => 1,
);

# ------------------------------------------------------------ #
# define general tests
# ------------------------------------------------------------ #

my $ItemData = [

    # this sla is NOT complete and must not be added
    {
        Add => {
            ValidID => 1,
            UserID  => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # this sla is NOT complete and must not be added
    {
        Add => {
            Name   => $SLAName[0],
            UserID => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # this sla is NOT complete and must not be added
    {
        Add => {
            Name    => $SLAName[0],
            ValidID => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },
# ---
# ITSM
# ---
    # this sla is NOT complete and must not be added
    {
        Add => {
            Name    => $SLAName[0],
            ValidID => 1,
            UserID  => 1,
        },
    },
# ---

    # service ids must be an array reference (check return false)
    {
        Add => {
            ServiceIDs => \do {'Dummy'},
            Name       => $SLAName[0],
            ValidID    => 1,
            UserID     => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # service ids must be an array reference (check return false)
    {
        Add => {
            ServiceIDs => '',
            Name       => $SLAName[0],
            ValidID    => 1,
            UserID     => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # service ids must be an array reference (check return false)
    {
        Add => {
            ServiceIDs => {},
            Name       => $SLAName[0],
            ValidID    => 1,
            UserID     => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # this sla must be inserted sucessfully
    {
        Add => {
            Name    => $SLAName[0],
            ValidID => 1,
            UserID  => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
        AddGet => {
            ServiceIDs          => [],
            Name                => $SLAName[0],
            Calendar            => '',
            FirstResponseTime   => 0,
            FirstResponseNotify => 0,
            UpdateTime          => 0,
            UpdateNotify        => 0,
            SolutionTime        => 0,
            SolutionNotify      => 0,
            ValidID             => 1,
            Comment             => '',
            CreateBy            => 1,
            ChangeBy            => 1,
# ---
# ITSM
# ---
            TypeID              => 1,
# ---
        },
    },

    # this sla have the same name as one test before and must not be added
    {
        Add => {
            Name    => $SLAName[0],
            ValidID => 1,
            UserID  => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # the sla one add-test before must be NOT updated (sla is NOT complete)
    {
        Update => {
            ValidID => 1,
            UserID  => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # the sla one add-test before must be NOT updated (sla is NOT complete)
    {
        Update => {
            Name   => $SLAName[0] . 'UPDATE1',
            UserID => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # the sla one add-test before must be NOT updated (sla is NOT complete)
    {
        Update => {
            Name    => $SLAName[0] . 'UPDATE1',
            ValidID => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # the sla one add-test before must be NOT updated (service ids must be an array reference)
    {
        Update => {
            ServiceIDs => \do {'Dummy'},
            Name       => $SLAName[0] . 'UPDATE1',
            ValidID    => 1,
            UserID     => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # the sla one add-test before must be NOT updated (service ids must be an array reference)
    {
        Update => {
            ServiceIDs => '',
            Name       => $SLAName[0] . 'UPDATE1',
            ValidID    => 1,
            UserID     => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # the sla one add-test before must be NOT updated (service ids must be an array reference)
    {
        Update => {
            ServiceIDs => {},
            Name       => $SLAName[0] . 'UPDATE1',
            ValidID    => 1,
            UserID     => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # this sla must be inserted sucessfully (check the returned service id array)
    {
        Add => {
            ServiceIDs => [ $ServiceIDs[0] ],
            Name       => $SLAName[1],
            ValidID    => 1,
            UserID     => 1,
# ---
# ITSM
# ---
            TypeID => 2,
# ---
        },
        AddGet => {
            ServiceIDs          => [ $ServiceIDs[0] ],
            Name                => $SLAName[1],
            Calendar            => '',
            FirstResponseTime   => 0,
            FirstResponseNotify => 0,
            UpdateTime          => 0,
            UpdateNotify        => 0,
            SolutionTime        => 0,
            SolutionNotify      => 0,
            ValidID             => 1,
            Comment             => '',
            CreateBy            => 1,
            ChangeBy            => 1,
# ---
# ITSM
# ---
            TypeID => 2,
# ---
        },
    },

    # this sla must be inserted sucessfully (check the sorting of the returned service id array)
    {
        Add => {
            ServiceIDs => [ $ServiceIDs[1], $ServiceIDs[0] ],
            Name       => $SLAName[2],
            ValidID    => 1,
            UserID     => 1,
# ---
# ITSM
# ---
            TypeID => 2,
# ---
        },
        AddGet => {
            ServiceIDs => [ $ServiceIDs[0], $ServiceIDs[1] ],
            Name       => $SLAName[2],
            Calendar   => '',
            FirstResponseTime   => 0,
            FirstResponseNotify => 0,
            UpdateTime          => 0,
            UpdateNotify        => 0,
            SolutionTime        => 0,
            SolutionNotify      => 0,
            ValidID             => 1,
            Comment             => '',
            CreateBy            => 1,
            ChangeBy            => 1,
# ---
# ITSM
# ---
            TypeID => 2,
# ---
        },
    },

    # the same name already exists (check return false)
    {
        Update => {
            Name    => $SLAName[1],
            ValidID => 1,
            UserID  => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # this sla must be inserted sucessfully
    {
        Add => {
            ServiceIDs => [ $ServiceIDs[1], $ServiceIDs[2], $ServiceIDs[0] ],
            Name       => $SLAName[3],
            Calendar   => '1',
            FirstResponseTime   => 10,
            FirstResponseNotify => 20,
            UpdateTime          => 30,
            UpdateNotify        => 40,
            SolutionTime        => 50,
            SolutionNotify      => 60,
            ValidID             => 1,
            Comment             => 'TestComment2',
            UserID              => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
        AddGet => {
            ServiceIDs => [ $ServiceIDs[0], $ServiceIDs[1], $ServiceIDs[2] ],
            Name       => $SLAName[3],
            Calendar   => '1',
            FirstResponseTime   => 10,
            FirstResponseNotify => 20,
            UpdateTime          => 30,
            UpdateNotify        => 40,
            SolutionTime        => 50,
            SolutionNotify      => 60,
            ValidID             => 1,
            Comment             => 'TestComment2',
            CreateBy            => 1,
            ChangeBy            => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # the sla one add-test before must be NOT updated (sla update arguments NOT complete)
    {
        Update => {
            ValidID => 1,
            UserID  => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # the sla one add-test before must be NOT updated (sla update arguments NOT complete)
    {
        Update => {
            Name   => $SLAName[3] . 'UPDATE1',
            UserID => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # the sla one add-test before must be NOT updated (sla update arguments NOT complete)
    {
        Update => {
            Name    => $SLAName[3] . 'UPDATE1',
            ValidID => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # the sla one add-test before must be updated (sla update arguments are complete)
    {
        Update => {
            ServiceIDs          => [],
            Name                => $SLAName[3] . 'UPDATE2',
            Calendar            => '1',
            FirstResponseTime   => 20,
            FirstResponseNotify => 30,
            UpdateTime          => 40,
            UpdateNotify        => 50,
            SolutionTime        => 60,
            SolutionNotify      => 70,
            ValidID             => 1,
            Comment             => 'TestComment2UPDATE2',
            UserID              => $UserIDs[0],
# ---
# ITSM
# ---
            TypeID => 2,
# ---
        },
        UpdateGet => {
            ServiceIDs          => [],
            Name                => $SLAName[3] . 'UPDATE2',
            Calendar            => '1',
            FirstResponseTime   => 20,
            FirstResponseNotify => 30,
            UpdateTime          => 40,
            UpdateNotify        => 50,
            SolutionTime        => 60,
            SolutionNotify      => 70,
            ValidID             => 1,
            Comment             => 'TestComment2UPDATE2',
            CreateBy            => 1,
            ChangeBy            => $UserIDs[0],
# ---
# ITSM
# ---
            TypeID => 2,
# ---
        },
    },

    # the sla one add-test before must be updated (sla update arguments are complete)
    {
        Update => {
            ServiceIDs          => [ $ServiceIDs[2] ],
            Name                => $SLAName[3] . 'UPDATE3',
            Calendar            => '2',
            FirstResponseTime   => 30,
            FirstResponseNotify => 40,
            UpdateTime          => 50,
            UpdateNotify        => 60,
            SolutionTime        => 70,
            SolutionNotify      => 80,
            ValidID             => 2,
            Comment             => 'TestComment2UPDATE3',
            UserID              => $UserIDs[1],
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
        UpdateGet => {
            ServiceIDs          => [ $ServiceIDs[2] ],
            Name                => $SLAName[3] . 'UPDATE3',
            Calendar            => '2',
            FirstResponseTime   => 30,
            FirstResponseNotify => 40,
            UpdateTime          => 50,
            UpdateNotify        => 60,
            SolutionTime        => 70,
            SolutionNotify      => 80,
            ValidID             => 2,
            Comment             => 'TestComment2UPDATE3',
            CreateBy            => 1,
            ChangeBy            => $UserIDs[1],
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # this sla must be inserted sucessfully (check string cleaner function)
    {
        Add => {
            ServiceIDs => [ $ServiceIDs[0] ],
            Name       => " \t \n \r " . $SLAName[4] . " \t \n \r ",
            ValidID    => 1,
            Comment    => " \t \n \r Test Comment \t \n \r ",
            UserID     => 1,
# ---
# ITSM
# ---
            TypeID => 2,
# ---
        },
        AddGet => {
            ServiceIDs          => [ $ServiceIDs[0] ],
            Name                => $SLAName[4],
            Calendar            => '',
            FirstResponseTime   => 0,
            FirstResponseNotify => 0,
            UpdateTime          => 0,
            UpdateNotify        => 0,
            SolutionTime        => 0,
            SolutionNotify      => 0,
            ValidID             => 1,
            Comment             => 'Test Comment',
            CreateBy            => 1,
            ChangeBy            => 1,
# ---
# ITSM
# ---
            TypeID => 2,
# ---
        },
    },

    # the sla one add-test before must be updated sucessfully (check string cleaner function)
    {
        Update => {
            ServiceIDs => [ $ServiceIDs[1] ],
            Name       => " \t \n \r " . $SLAName[4] . " UPDATE1 \t \n \r ",
            ValidID    => 2,
            Comment    => " \t \n \r Test Comment UPDATE1 \t \n \r ",
            UserID     => $UserIDs[1],
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
        UpdateGet => {
            ServiceIDs          => [ $ServiceIDs[1] ],
            Name                => $SLAName[4] . ' UPDATE1',
            Calendar            => '',
            FirstResponseTime   => 0,
            FirstResponseNotify => 0,
            UpdateTime          => 0,
            UpdateNotify        => 0,
            SolutionTime        => 0,
            SolutionNotify      => 0,
            ValidID             => 2,
            Comment             => 'Test Comment UPDATE1',
            CreateBy            => 1,
            ChangeBy            => $UserIDs[1],
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # this sla must be inserted sucessfully (unicode checks)
    {
        Add => {
            Name    => $SLAName[5] . ' ϒ ϡ Ʃ Ϟ ',
            ValidID => 1,
            Comment => ' Ѡ Ѥ TestComment5 Ϡ Ω ',
            UserID  => 1,
# ---
# ITSM
# ---
            TypeID => 3,
# ---
        },
        AddGet => {
            ServiceIDs          => [],
            Name                => $SLAName[5] . ' ϒ ϡ Ʃ Ϟ',
            Calendar            => '',
            FirstResponseTime   => 0,
            FirstResponseNotify => 0,
            UpdateTime          => 0,
            UpdateNotify        => 0,
            SolutionTime        => 0,
            SolutionNotify      => 0,
            ValidID             => 1,
            Comment             => 'Ѡ Ѥ TestComment5 Ϡ Ω',
            CreateBy            => 1,
            ChangeBy            => 1,
# ---
# ITSM
# ---
            TypeID => 3,
# ---
        },
    },

    # the sla one add-test before must be updated sucessfully (unicode checks)
    {
        Update => {
            Name    => $SLAName[5] . ' ϒ ϡ Ʃ Ϟ UPDATE1',
            ValidID => 2,
            Comment => ' Ѡ Ѥ TestComment5 Ϡ Ω UPDATE1',
            UserID  => $UserIDs[0],
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
        UpdateGet => {
            ServiceIDs          => [],
            Name                => $SLAName[5] . ' ϒ ϡ Ʃ Ϟ UPDATE1',
            Calendar            => '',
            FirstResponseTime   => 0,
            FirstResponseNotify => 0,
            UpdateTime          => 0,
            UpdateNotify        => 0,
            SolutionTime        => 0,
            SolutionNotify      => 0,
            ValidID             => 2,
            Comment             => 'Ѡ Ѥ TestComment5 Ϡ Ω UPDATE1',
            CreateBy            => 1,
            ChangeBy            => $UserIDs[0],
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # this sla must be inserted sucessfully (special character checks)
    {
        Add => {
            ServiceIDs => [],
            Name       => ' [test]%*\\ ' . $SLAName[6] . ' [test]%*\\ ',
            ValidID    => 1,
            Comment    => ' [test]%*\\ Test Comment [test]%*\\ ',
            UserID     => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
        AddGet => {
            ServiceIDs          => [],
            Name                => '[test]%*\\ ' . $SLAName[6] . ' [test]%*\\',
            Calendar            => '',
            FirstResponseTime   => 0,
            FirstResponseNotify => 0,
            UpdateTime          => 0,
            UpdateNotify        => 0,
            SolutionTime        => 0,
            SolutionNotify      => 0,
            ValidID             => 1,
            Comment             => '[test]%*\\ Test Comment [test]%*\\',
            CreateBy            => 1,
            ChangeBy            => 1,
# ---
# ITSM
# ---
            TypeID => 1,
# ---
        },
    },

    # the sla one add-test before must be updated sucessfully (special character checks)
    {
        Update => {
            ServiceIDs => [],
            Name       => ' [test]%*\\ ' . $SLAName[6] . ' UPDATE1 [test]%*\\ ',
            ValidID    => 2,
            Comment    => ' [test]%*\\ Test Comment UPDATE1 [test]%*\\ ',
            UserID     => $UserIDs[1],
# ---
# ITSM
# ---
            TypeID => 2,
# ---
        },
        UpdateGet => {
            ServiceIDs          => [],
            Name                => '[test]%*\\ ' . $SLAName[6] . ' UPDATE1 [test]%*\\',
            Calendar            => '',
            FirstResponseTime   => 0,
            FirstResponseNotify => 0,
            UpdateTime          => 0,
            UpdateNotify        => 0,
            SolutionTime        => 0,
            SolutionNotify      => 0,
            ValidID             => 2,
            Comment             => '[test]%*\\ Test Comment UPDATE1 [test]%*\\',
            CreateBy            => 1,
            ChangeBy            => $UserIDs[1],
# ---
# ITSM
# ---
            TypeID => 2,
# ---
        },
    },
];

# ------------------------------------------------------------ #
# run general tests
# ------------------------------------------------------------ #

my $TestCount = 1;
my $LastAddedSLAID;
my $AddedCounter = 0;

for my $Item ( @{$ItemData} ) {

    if ( $Item->{Add} ) {

        # add new sla
        my $SLAID = $Self->{SLAObject}->SLAAdd(
            %{ $Item->{Add} },
        );

        # check if sla was added successfully or not
        if ( $Item->{AddGet} ) {

            $Self->True(
                $SLAID,
                "Test $TestCount: SLAAdd() - SLAID: $SLAID",
            );

            if ($SLAID) {

                # lookup sla name
                my $SLAName = $Self->{SLAObject}->SLALookup(
                    SLAID => $SLAID,
                );

                # lookup test
                $Self->Is(
                    $SLAName || '',
                    $Item->{AddGet}->{Name} || '',
                    "Test $TestCount: SLALookup() - lookup",
                );

                # reverse lookup the sla id
                my $SLAIDNew = $Self->{SLAObject}->SLALookup(
                    Name => $SLAName || '',
                );

                # reverse lookup test
                $Self->Is(
                    $SLAIDNew || '',
                    $SLAID    || '',
                    "Test $TestCount: SLALookup() - reverse lookup",
                );

                # set last sla id variable
                $LastAddedSLAID = $SLAID;

                # increment the added counter
                $AddedCounter++;
            }
        }
        else {
            $Self->False(
                $SLAID,
                "Test $TestCount: SLAAdd()",
            );
        }

        # get sla data to check the values after creation of the sla
        my %SLAGet = $Self->{SLAObject}->SLAGet(
            SLAID  => $SLAID,
            UserID => $Item->{Add}->{UserID},
            Cache  => 1,
        );

        # turn off all pretty print
        $Data::Dumper::Indent = 0;

        # check sla data after creation of the sla
        for my $SLAAttribute ( keys %{ $Item->{AddGet} } ) {

            # dump the given attribute
            if ( ref $SLAGet{$SLAAttribute} ) {
                $SLAGet{$SLAAttribute} = Data::Dumper::Dumper( $SLAGet{$SLAAttribute} );
            }

            # dump the reference string
            if ( ref $Item->{AddGet}->{$SLAAttribute} ) {
                $Item->{AddGet}->{$SLAAttribute} = Data::Dumper::Dumper(
                    $Item->{AddGet}->{$SLAAttribute},
                );
            }

            $Self->Is(
                $SLAGet{$SLAAttribute} || '',
                $Item->{AddGet}->{$SLAAttribute} || '',
                "Test $TestCount: SLAGet() - $SLAAttribute",
            );
        }
    }

    if ( $Item->{Update} ) {

        # check last sla id varaible
        if ( !$LastAddedSLAID ) {
            $Self->False(
                1,
                "Test $TestCount: NO LAST SERVICE ID GIVEN",
            );
        }

        # update the sla
        my $UpdateSucess = $Self->{SLAObject}->SLAUpdate(
            %{ $Item->{Update} },
            SLAID => $LastAddedSLAID,
        );

        # check if sla was updated successfully or not
        if ( $Item->{UpdateGet} ) {
            $Self->True(
                $UpdateSucess,
                "Test $TestCount: SLAUpdate() - SLAID: $LastAddedSLAID",
            );
        }
        else {
            $Self->False(
                $UpdateSucess,
                "Test $TestCount: SLAUpdate()",
            );
        }

        # get sla data to check the values after the update
        my %SLAGet2 = $Self->{SLAObject}->SLAGet(
            SLAID  => $LastAddedSLAID,
            UserID => $Item->{Update}->{UserID},
        );

        # check sla data after update
        for my $SLAAttribute ( keys %{ $Item->{UpdateGet} } ) {

            # dump the given attribute
            if ( ref $SLAGet2{$SLAAttribute} ) {
                $SLAGet2{$SLAAttribute} = Data::Dumper::Dumper( $SLAGet2{$SLAAttribute} );
            }

            # dump the reference string
            if ( ref $Item->{UpdateGet}->{$SLAAttribute} ) {
                $Item->{UpdateGet}->{$SLAAttribute} = Data::Dumper::Dumper(
                    $Item->{UpdateGet}->{$SLAAttribute},
                );
            }

            $Self->Is(
                $SLAGet2{$SLAAttribute} || '',
                $Item->{UpdateGet}->{$SLAAttribute} || '',
                "Test $TestCount: SLAGet() - $SLAAttribute",
            );
        }

        # lookup sla name
        my $SLAName = $Self->{SLAObject}->SLALookup(
            SLAID => $SLAGet2{SLAID},
        );

        # lookup test
        $Self->Is(
            $SLAName || '',
            $SLAGet2{Name} || '',
            "Test $TestCount: SLALookup() - lookup",
        );

        # reverse lookup the sla id
        my $SLAIDNew = $Self->{SLAObject}->SLALookup(
            Name => $SLAName || '',
        );

        # reverse lookup test
        $Self->Is(
            $SLAIDNew || '',
            $SLAGet2{SLAID} || '',
            "Test $TestCount: SLALookup() - reverse lookup",
        );
    }

    $TestCount++;
}

# ------------------------------------------------------------ #
# SLAList test 1 (check general functionality)
# ------------------------------------------------------------ #

my %SLAList1 = $Self->{SLAObject}->SLAList(
    Valid  => 0,
    UserID => 1,
);
my %SLAList1Org = %SLAListOriginal;

SERVICEID:
for my $SLAID ( keys %SLAList1Org ) {

    if ( $SLAList1{$SLAID} && $SLAList1Org{$SLAID} eq $SLAList1{$SLAID} ) {
        delete $SLAList1{$SLAID};
    }
    else {
        $SLAList1{Dummy} = 1;
    }
}

my $SLAList1Count = scalar keys %SLAList1;

$Self->Is(
    $SLAList1Count || '',
    $AddedCounter  || '',
    "Test $TestCount: SLAList()",
);

1;

iVBORw0KGgoAAAANSUhEUgAAABYAAAAWCAYAAADEtGw7AAAACXBIWXMAAAsTAAALEwEAmpwYAAAABGdBTUEAALGOfPtRkwAAACBjSFJNAAB6JQAAgIMAAPn/AACA6QAAdTAAAOpgAAA6mAAAF2+SX8VGAAAFEElEQVR42qyVe2xTZRjGn9Oe9vS+tuvaddeuG2MUt0E22IAxMExBN3UzwGSMzEsIzhg0RjHBRI2XP9QEVGRAohiFMCKMWyI4hpvIZZNsMjd62YVu67a2o7uWXs9pz/EPI3LRAYm//758yfO+eb7ne18e7qKyciMEAgEAEBaLbUdV1cZlAFD78lakpBjwoJC3H9TqWFV5efk7HE9hFkukxenpc54vLl4p4wkTSpYV5UtHRh1NDsdgE0EQ4DhuVmE+ACgUCixcmIe8gifrt7/92osmk2l5KBwtDIVZwpiemScQ8Be/ULNu5enGCxGXc/jknIwsuMec9xfW6xPit29/v8o95skjSHmafcgtTklJJEiSRDBEQx2rEYTDUbiczsCi/IIIx7Jys6XLcV8r5ucsrS1eUfJuStp89PQNIs2QBKGAhEQiQjhMIxAIYWDIhfWVVct8fnqZrdd+BMDFWTsmCAK9PV2XxLL4V/gCqSROo0KEYTE84kZb6+WpUCgqoCgxjyRJDDrcaG5pZusPfLWeYejxWYWlUhm/pmZLWVq6qVJISQVSiRh+P409uz/59uB3Xz7b0d7mzFlYtEYuk8HhcEOl0hA6rTYw6nT87rs5E/wvYZ5cHrOgZHXlcb3eIAFHgGU5KOQS0OHpHw2GVHcoNHNUKhH4GDoKuUwCDgQy5i56SyZTPn3LT5IERVF3dry6bFPB2cbjDkokz47XJ1NiSghljBxFRUU5hQWLx8rK1m7LzHwkl+NYsBwwODSIUye/3zU/p0Bss7S3sCyLNIMRScmpcLv/SQrZcLjuhEKh/LW8vLpaRJEKESWAiBJAFzc3a06G6Qg4Dv5AAOEwBZksCqFQiiG7reFK69nzO3d+gbq63WCYCHg83p1WAEBSckbFxKQ3wWa10HK5CFIJBZpmEAwGEQyFIBQKEBsrh31ggLVauzmVJrEUAEwmk4kkSQnLsmBZ9t4cj4+7HJcvnj7R12uxJ6ZmlyQn66DVKEFRfIjFQkRZFm3tZny+4+P6ftvFD58qfVwnFFIzmzfXXhgc9uizMtOFsTpDdefVtmaCIAAAxO1V4rSJJeuq3mzqtbbbjEajNsuUq/Z6Z6ItP5+2JKSYsi+1HNuWlZnUve/r+jNO1xgrogQ8hVzG6fU6Yu+eXT8crj9UabVdg9/vuysiPL5OpY6vBRAPIOuJZ7Y4CpeXnwdAyWSq6gMHG8xnz10ZvXDZwk1Phzma5jifL8KNjE5zZsuQ92hDi3nevOwVt6z4G47j/KGgrx2AD8D46HCfwznSdyYaZfppOtRlNM4tzl2wdIkxVY+evmH81Njsn5oOkBKplKAZUB0dnXT9oX0fAQjwZ/s9ESZsjUaZQQAor9hQUbyy9CWJVKaMRgl8s39/6wfvbV1iNlvZ/EWPFrlcN0AzrFCtjmOCAZ+HfND5Oj4+kT817U8Simbg9zPIz8slBx5bFc7JKWT9vgAmJr3wen3U9f6eNM/4mJLAQ7Cp5vVja0o3VExOeKCP18BoSIh4Jmb43dd6CUoshr2/Z3TnZ2/kANwk74E3AimIoZnovM6rrbRWq8TwiAuN51rJjqtmQq1WIOCbYuz2PmmsJqHgnsebDZZlGXP3lcNjrtGbKk3yKqMxETqdGgq5GNNeP35pafrj1LG9S4KBmzYAkYey4q9toy6OUerWejwjnRXrX/30el9Xh/Xaby0ikcTrueGow/9BjFL7HCWS5P/b3Z8DADulJ5doF3IEAAAAAElFTkSuQmCC
# --
# ITSMCore.pm - code to excecute during package installation
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: ITSMCore.pm,v 1.16 2009/10/07 13:27:33 reb Exp $
# --
# 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::ITSMCore;

use strict;
use warnings;

use Kernel::System::GeneralCatalog;
use Kernel::System::Group;
use Kernel::System::ITSMCIPAllocate;
use Kernel::System::Priority;
use Kernel::System::Valid;

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

=head1 NAME

ITSMCore.pm - code to excecute during package installation

=head1 SYNOPSIS

All functions

=head1 PUBLIC INTERFACE

=over 4

=cut

=item new()

create an object

    use Kernel::Config;
    use Kernel::System::Encode;
    use Kernel::System::Log;
    use Kernel::System::Main;
    use Kernel::System::Time;
    use Kernel::System::DB;
    use Kernel::System::XML;
    use var::packagesetup::ITSMCore;

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

=cut

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

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

    # check needed objects
    for my $Object (
        qw(ConfigObject LogObject EncodeObject MainObject TimeObject DBObject XMLObject)
        )
    {
        $Self->{$Object} = $Param{$Object} || die "Got no $Object!";
    }
    $Self->{GeneralCatalogObject} = Kernel::System::GeneralCatalog->new( %{$Self} );
    $Self->{GroupObject}          = Kernel::System::Group->new( %{$Self} );
    $Self->{CIPAllocateObject}    = Kernel::System::ITSMCIPAllocate->new( %{$Self} );
    $Self->{PriorityObject}       = Kernel::System::Priority->new( %{$Self} );
    $Self->{ValidObject}          = Kernel::System::Valid->new( %{$Self} );

    return $Self;
}

=item CodeInstall()

run the code install part

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

=cut

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

    # change background color to the ITSM blue
    $Self->_BackgroundColorChange(
        OldColor => 'bbddff',
        NewColor => '003399',
    );

    # set default CIP matrix
    $Self->_CIPDefaultMatrixSet();

    # add the group itsm-service
    $Self->_GroupAdd(
        Name        => 'itsm-service',
        Description => 'Group for ITSM Service mask access in the agent interface.',
    );

    # fillup empty type_id rows in service table
    $Self->_FillupEmptyServiceTypeID();

    # fillup empty criticality_id rows in service table
    $Self->_FillupEmptyServiceCriticalityID();

    # fillup empty type_id rows in sla table
    $Self->_FillupEmptySLATypeID();

    # set preferences for some GeneralCatalog entries
    $Self->_SetPreferences();

    return 1;
}

=item CodeReinstall()

run the code reinstall part

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

=cut

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

    # change background color to the ITSM blue
    $Self->_BackgroundColorChange(
        OldColor => 'bbddff',
        NewColor => '003399',
    );

    # set default CIP matrix
    $Self->_CIPDefaultMatrixSet();

    # add the group itsm-service
    $Self->_GroupAdd(
        Name        => 'itsm-service',
        Description => 'Group for ITSM Service mask access in the agent interface.',
    );

    # fillup empty type_id rows in service table
    $Self->_FillupEmptyServiceTypeID();

    # fillup empty criticality_id rows in service table
    $Self->_FillupEmptyServiceCriticalityID();

    # fillup empty type_id rows in sla table
    $Self->_FillupEmptySLATypeID();

    return 1;
}

=item CodeUpgrade()

run the code upgrade part

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

=cut

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

    # set default CIP matrix
    $Self->_CIPDefaultMatrixSet();

    # fillup empty type_id rows in service table
    $Self->_FillupEmptyServiceTypeID();

    # fillup empty criticality_id rows in service table
    $Self->_FillupEmptyServiceCriticalityID();

    # fillup empty type_id rows in sla table
    $Self->_FillupEmptySLATypeID();

    return 1;
}

=item CodeUninstall()

run the code uninstall part

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

=cut

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

    # restore the original background
    $Self->_BackgroundColorChange(
        OldColor => '003399',
        NewColor => 'bbddff',
    );

    # deactivate the group itsm-service
    $Self->_GroupDeactivate(
        Name => 'itsm-service',
    );

    return 1;
}

=item _SetPreferences()

    my $Result = $CodeObject->_SetPreferences()

=cut

sub _SetPreferences {
    my $Self = shift;

    my %Map = (
        Operational => 'operational',
        Warning     => 'warning',
        Incident    => 'incident',
    );

    NAME:
    for my $Name ( keys %Map ) {
        my $Item = $Self->{GeneralCatalogObject}->ItemGet(
            Name  => $Name,
            Class => 'ITSM::Core::IncidentState',
        );

        next NAME if !$Item;

        $Self->{GeneralCatalogObject}->GeneralCatalogPreferencesSet(
            ItemID => $Item->{ItemID},
            Key    => 'Functionality',
            Value  => $Map{$Name},
        );
    }
}

=item _BackgroundColorChange()

change the backround color

    my $Result = $CodeObject->_BackgroundColorChange(
        OldColor => 'bbddff',
        NewColor => '003399',
    );

=cut

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

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

    # define the css file
    my $CssFile = $Self->{ConfigObject}->Get('Home') . '/var/httpd/htdocs/css/Standard/agent.css';

    return 1 if -e $CssFile . '.save';

    # read file content
    my $Content = $Self->{MainObject}->FileRead(
        Location        => $CssFile,
        Mode            => 'binmode',
        Result          => 'SCALAR',
        DisableWarnings => 1,
    );

    return if !$Content;
    return if ref $Content ne 'SCALAR';
    return if !${$Content};

    # change background color
    ${$Content} =~ s{
        background-color\:\#$Param{OldColor}\;
    }{background-color\:\#$Param{NewColor}\;}xms;

    # write new content to file
    $Self->{MainObject}->FileWrite(
        Location => $CssFile,
        Content  => $Content,
        Mode     => 'binmode',
    );

    return 1;
}

=item _CIPDefaultMatrixSet()

set the default CIP matrix

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

=cut

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

    # get current allocation list
    my $List = $Self->{CIPAllocateObject}->AllocateList(
        UserID => 1,
    );

    return if !$List;
    return if ref $List ne 'HASH';

    # set no matrix if already defined
    return if %{$List};

    # define the allocations
    my %Allocation;
    $Allocation{'1 very low'}->{'1 very low'}   = '1 very low';
    $Allocation{'1 very low'}->{'2 low'}        = '1 very low';
    $Allocation{'1 very low'}->{'3 normal'}     = '2 low';
    $Allocation{'1 very low'}->{'4 high'}       = '2 low';
    $Allocation{'1 very low'}->{'5 very high'}  = '3 normal';
    $Allocation{'2 low'}->{'1 very low'}        = '1 very low';
    $Allocation{'2 low'}->{'2 low'}             = '2 low';
    $Allocation{'2 low'}->{'3 normal'}          = '2 low';
    $Allocation{'2 low'}->{'4 high'}            = '3 normal';
    $Allocation{'2 low'}->{'5 very high'}       = '4 high';
    $Allocation{'3 normal'}->{'1 very low'}     = '2 low';
    $Allocation{'3 normal'}->{'2 low'}          = '2 low';
    $Allocation{'3 normal'}->{'3 normal'}       = '3 normal';
    $Allocation{'3 normal'}->{'4 high'}         = '4 high';
    $Allocation{'3 normal'}->{'5 very high'}    = '4 high';
    $Allocation{'4 high'}->{'1 very low'}       = '2 low';
    $Allocation{'4 high'}->{'2 low'}            = '3 normal';
    $Allocation{'4 high'}->{'3 normal'}         = '4 high';
    $Allocation{'4 high'}->{'4 high'}           = '4 high';
    $Allocation{'4 high'}->{'5 very high'}      = '5 very high';
    $Allocation{'5 very high'}->{'1 very low'}  = '3 normal';
    $Allocation{'5 very high'}->{'2 low'}       = '4 high';
    $Allocation{'5 very high'}->{'3 normal'}    = '4 high';
    $Allocation{'5 very high'}->{'4 high'}      = '5 very high';
    $Allocation{'5 very high'}->{'5 very high'} = '5 very high';

    # get impact list
    my $ImpactList = $Self->{GeneralCatalogObject}->ItemList(
        Class => 'ITSM::Core::Impact',
    );
    my %ImpactListReverse = reverse %{$ImpactList};

    # get criticality list
    my $CriticalityList = $Self->{GeneralCatalogObject}->ItemList(
        Class => 'ITSM::Core::Criticality',
    );
    my %CriticalityListReverse = reverse %{$CriticalityList};

    # get priority list
    my %PriorityList = $Self->{PriorityObject}->PriorityList(
        UserID => 1,
    );
    my %PriorityListReverse = reverse %PriorityList;

    # create the allocation matrix
    my %AllocationMatrix;
    IMPACT:
    for my $Impact ( keys %Allocation ) {

        next IMPACT if !$ImpactListReverse{$Impact};

        # extract impact id
        my $ImpactID = $ImpactListReverse{$Impact};

        CRITICALITY:
        for my $Criticality ( keys %{ $Allocation{$Impact} } ) {

            next CRITICALITY if !$CriticalityListReverse{$Criticality};

            # extract priority
            my $Priority = $Allocation{$Impact}->{$Criticality};

            next CRITICALITY if !$PriorityListReverse{$Priority};

            # extract criticality id and priority id
            my $CriticalityID = $CriticalityListReverse{$Criticality};
            my $PriorityID    = $PriorityListReverse{$Priority};

            $AllocationMatrix{$ImpactID}->{$CriticalityID} = $PriorityID;
        }
    }

    # save the matrix
    $Self->{CIPAllocateObject}->AllocateUpdate(
        AllocateData => \%AllocationMatrix,
        UserID       => 1,
    );

    return 1;
}

=item _GroupAdd()

add a group

    my $Result = $CodeObject->_GroupAdd(
        Name        => 'the-group-name',
        Description => 'The group description.',
    );

=cut

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

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

    # get valid list
    my %ValidList = $Self->{ValidObject}->ValidList(
        UserID => 1,
    );
    my %ValidListReverse = reverse %ValidList;

    # check if group already exists
    my $GroupID = $Self->{GroupObject}->GroupLookup(
        Group  => $Param{Name},
        UserID => 1,
    );

    # reactivate the group
    if ($GroupID) {

        # get current group data
        my %GroupData = $Self->{GroupObject}->GroupGet(
            ID     => $GroupID,
            UserID => 1,
        );

        # reactivate group
        $Self->{GroupObject}->GroupUpdate(
            %GroupData,
            ValidID => $ValidListReverse{valid},
            UserID  => 1,
        );

        return 1;
    }

    # add the group
    else {
        return if !$Self->{GroupObject}->GroupAdd(
            Name    => $Param{Name},
            Comment => $Param{Description},
            ValidID => $ValidListReverse{valid},
            UserID  => 1,
        );
    }

    # lookup the new group id
    my $NewGroupID = $Self->{GroupObject}->GroupLookup(
        Group  => $Param{Name},
        UserID => 1,
    );

    # add user root to the group
    $Self->{GroupObject}->GroupMemberAdd(
        GID        => $NewGroupID,
        UID        => 1,
        Permission => {
            ro        => 1,
            move_into => 1,
            create    => 1,
            owner     => 1,
            priority  => 1,
            rw        => 1,
        },
        UserID => 1,
    );

    return 1;
}

=item _GroupDeactivate()

deactivate a group

    my $Result = $CodeObject->_GroupDeactivate(
        Name => 'the-group-name',
    );

=cut

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

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

    # lookup group id
    my $GroupID = $Self->{GroupObject}->GroupLookup(
        Group => $Param{Name},
    );

    return if !$GroupID;

    # get valid list
    my %ValidList = $Self->{ValidObject}->ValidList(
        UserID => 1,
    );
    my %ValidListReverse = reverse %ValidList;

    # get current group data
    my %GroupData = $Self->{GroupObject}->GroupGet(
        ID     => $GroupID,
        UserID => 1,
    );

    # deactivate group
    $Self->{GroupObject}->GroupUpdate(
        %GroupData,
        ValidID => $ValidListReverse{invalid},
        UserID  => 1,
    );

    return 1;
}

=item _FillupEmptyServiceTypeID()

fillup empty entries in the type_id column of the service table

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

=cut

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

    # get service type list
    my $ServiceTypeList = $Self->{GeneralCatalogObject}->ItemList(
        Class => 'ITSM::Service::Type',
    );

    # error handling
    if ( !$ServiceTypeList || ref $ServiceTypeList ne 'HASH' || !%{$ServiceTypeList} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't find any item in general catalog class ITSM::Service::Type!",
        );
        return;
    }

    # sort ids
    my @ServiceTypeKeyList = sort keys %{$ServiceTypeList};

    # update type_id
    return $Self->{DBObject}->Do(
        SQL => "UPDATE service "
            . "SET type_id = $ServiceTypeKeyList[0] "
            . "WHERE type_id = 0 OR type_id IS NULL",
    );
}

=item _FillupEmptyServiceCriticalityID()

fillup empty entries in the criticality_id column of the service table

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

=cut

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

    # get criticality list
    my $CriticalityList = $Self->{GeneralCatalogObject}->ItemList(
        Class => 'ITSM::Core::Criticality',
    );

    # error handling
    if ( !$CriticalityList || ref $CriticalityList ne 'HASH' || !%{$CriticalityList} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't find any item in general catalog class ITSM::Core::Criticality!",
        );
        return;
    }

    # sort ids
    my @CriticalityKeyList = sort keys %{$CriticalityList};

    # update criticality_id
    return $Self->{DBObject}->Do(
        SQL => "UPDATE service "
            . "SET criticality_id = $CriticalityKeyList[0] "
            . "WHERE criticality_id = 0 OR criticality_id IS NULL",
    );
}

=item _FillupEmptySLATypeID()

fillup empty entries in the type_id column of the sla table

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

=cut

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

    # get sla type list
    my $SLATypeList = $Self->{GeneralCatalogObject}->ItemList(
        Class => 'ITSM::SLA::Type',
    );

    # error handling
    if ( !$SLATypeList || ref $SLATypeList ne 'HASH' || !%{$SLATypeList} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't find any item in general catalog class ITSM::SLA::Type!",
        );
        return;
    }

    # sort ids
    my @SLATypeKeyList = sort keys %{$SLATypeList};

    # update type_id
    return $Self->{DBObject}->Do(
        SQL => "UPDATE sla "
            . "SET type_id = $SLATypeKeyList[0] "
            . "WHERE type_id = 0 OR type_id IS NULL",
    );
}

1;

=back

=head1 TERMS AND CONDITIONS

This Software is part of the OTRS project (http://otrs.org/).

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

=cut

=head1 VERSION

$Revision: 1.16 $ $Date: 2009/10/07 13:27:33 $

=cut

ITSM::Core::Criticality
1 very low
1
current_timestamp
1
current_timestamp
1
ITSM::Core::Criticality
2 low
1
current_timestamp
1
current_timestamp
1
ITSM::Core::Criticality
3 normal
1
current_timestamp
1
current_timestamp
1
ITSM::Core::Criticality
4 high
1
current_timestamp
1
current_timestamp
1
ITSM::Core::Criticality
5 very high
1
current_timestamp
1
current_timestamp
1
ITSM::Core::Impact
1 very low
1
current_timestamp
1
current_timestamp
1
ITSM::Core::Impact
2 low
1
current_timestamp
1
current_timestamp
1
ITSM::Core::Impact
3 normal
1
current_timestamp
1
current_timestamp
1
ITSM::Core::Impact
4 high
1
current_timestamp
1
current_timestamp
1
ITSM::Core::Impact
5 very high
1
current_timestamp
1
current_timestamp
1
ITSM::Core::IncidentState
Operational
1
current_timestamp
1
current_timestamp
1
ITSM::Core::IncidentState
Warning
1
current_timestamp
1
current_timestamp
1
ITSM::Core::IncidentState
Incident
1
current_timestamp
1
current_timestamp
1
ITSM::Service::Type
End User Service
1
current_timestamp
1
current_timestamp
1
ITSM::Service::Type
Front End
1
current_timestamp
1
current_timestamp
1
ITSM::Service::Type
Back End
1
current_timestamp
1
current_timestamp
1
ITSM::Service::Type
IT Management
1
current_timestamp
1
current_timestamp
1
ITSM::Service::Type
Reporting
1
current_timestamp
1
current_timestamp
1
ITSM::Service::Type
IT Operational
1
current_timestamp
1
current_timestamp
1
ITSM::Service::Type
Demonstration
1
current_timestamp
1
current_timestamp
1
ITSM::Service::Type
Project
1
current_timestamp
1
current_timestamp
1
ITSM::Service::Type
Training
1
current_timestamp
1
current_timestamp
1
ITSM::Service::Type
Underpinning Contract
1
current_timestamp
1
current_timestamp
1
ITSM::Service::Type
Other
1
current_timestamp
1
current_timestamp
1
ITSM::SLA::Type
Availability
1
current_timestamp
1
current_timestamp
1
ITSM::SLA::Type
Response Time
1
current_timestamp
1
current_timestamp
1
ITSM::SLA::Type
Recovery Time
1
current_timestamp
1
current_timestamp
1
ITSM::SLA::Type
Resolution Rate
1
current_timestamp
1
current_timestamp
1
ITSM::SLA::Type
Transactions
1
current_timestamp
1
current_timestamp
1
ITSM::SLA::Type
Errors
1
current_timestamp
1
current_timestamp
1
ITSM::SLA::Type
Other
1
current_timestamp
1
current_timestamp
1