ImportExport
3.3.3
OTRS AG
http://otrs.org/
GNU AFFERO GENERAL PUBLIC LICENSE Version 3, November 2007
Build for OTRS::ITSM 3.3.3.
Build for OTRS::ITSM 3.3.2.
Build for OTRS::ITSM 3.3.1.
Build for OTRS::ITSM 3.3.0 rc1.
Build for OTRS::ITSM 3.3.0 beta5.
Build for OTRS::ITSM 3.3.0 beta4.
Build for OTRS::ITSM 3.3.0 beta3.
Build for OTRS::ITSM 3.3.0 beta2.
Build for OTRS::ITSM 3.3.0 beta1.
The ImportExport package.
Das ImportExport Paket.
El paquete ImportExport.
3.3.x
</br>
<strong>ATTENTION</strong>
</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>
((enjoy))</br>
</br>
</br>
<strong>ACHTUNG</strong>
</br>
</br>
Bei der Deinstallation werden die von diesem Paket angelegten Datenbank-Tabellen gelöscht.
Alle darin enthaltenen Daten gehen unwiderruflich verloren!
</br>
</br>
((enjoy))</br>
</br>
<br/>
<strong>ATENCIÓN</strong>
<br/>
<br/>
Si usted desinstala este paquete, todas las tablas de la base de datos que se crearon durante la instalación
del mismo, se eliminarán. ¡La totalidad de los datos que contenían dichas tablas, se perderá irrevocablemente!
<br/>
<br/>
((enjoy))<br/>
<br/>
# define function name
my $FunctionName = 'CodeUpgradeFromBefore_2_0_3';
# 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."
);
}
}
2013-12-05 16:54:25
opms.otrs.com
IyEvdXNyL2Jpbi9wZXJsCiMgLS0KIyBvdHJzLkltcG9ydEV4cG9ydC5wbCAtIGltcG9ydC9leHBvcnQgc2NyaXB0CiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBwcm9ncmFtIGlzIGZyZWUgc29mdHdhcmU7IHlvdSBjYW4gcmVkaXN0cmlidXRlIGl0IGFuZC9vciBtb2RpZnkKIyBpdCB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEdOVSBBRkZFUk8gR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBhcyBwdWJsaXNoZWQgYnkKIyB0aGUgRnJlZSBTb2Z0d2FyZSBGb3VuZGF0aW9uOyBlaXRoZXIgdmVyc2lvbiAzIG9mIHRoZSBMaWNlbnNlLCBvcgojIGFueSBsYXRlciB2ZXJzaW9uLgojCiMgVGhpcyBwcm9ncmFtIGlzIGRpc3RyaWJ1dGVkIGluIHRoZSBob3BlIHRoYXQgaXQgd2lsbCBiZSB1c2VmdWwsCiMgYnV0IFdJVEhPVVQgQU5ZIFdBUlJBTlRZOyB3aXRob3V0IGV2ZW4gdGhlIGltcGxpZWQgd2FycmFudHkgb2YKIyBNRVJDSEFOVEFCSUxJVFkgb3IgRklUTkVTUyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UuICBTZWUgdGhlCiMgR05VIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgZm9yIG1vcmUgZGV0YWlscy4KIwojIFlvdSBzaG91bGQgaGF2ZSByZWNlaXZlZCBhIGNvcHkgb2YgdGhlIEdOVSBBZmZlcm8gR2VuZXJhbCBQdWJsaWMgTGljZW5zZQojIGFsb25nIHdpdGggdGhpcyBwcm9ncmFtOyBpZiBub3QsIHdyaXRlIHRvIHRoZSBGcmVlIFNvZnR3YXJlCiMgRm91bmRhdGlvbiwgSW5jLiwgNTEgRnJhbmtsaW4gU3QsIEZpZnRoIEZsb29yLCBCb3N0b24sIE1BICAwMjExMC0xMzAxIFVTQQojIG9yIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSBGaWxlOjpCYXNlbmFtZTsKdXNlIEZpbmRCaW4gcXcoJFJlYWxCaW4pOwp1c2UgbGliIGRpcm5hbWUoJFJlYWxCaW4pOwp1c2UgbGliIGRpcm5hbWUoJFJlYWxCaW4pIC4gJy9LZXJuZWwvY3Bhbi1saWInOwoKdXNlIEdldG9wdDo6U3RkOwp1c2UgS2VybmVsOjpDb25maWc7CnVzZSBLZXJuZWw6OlN5c3RlbTo6REI7CnVzZSBLZXJuZWw6OlN5c3RlbTo6RW5jb2RlOwp1c2UgS2VybmVsOjpTeXN0ZW06OkltcG9ydEV4cG9ydDsKdXNlIEtlcm5lbDo6U3lzdGVtOjpMb2c7CnVzZSBLZXJuZWw6OlN5c3RlbTo6TWFpbjsKCnVzZSB2YXJzIHF3KCRSZWFsQmluKTsKCiMgY3JlYXRlIGNvbW1vbiBvYmplY3RzCm15ICVDb21tb25PYmplY3Q7CiRDb21tb25PYmplY3R7Q29uZmlnT2JqZWN0fSA9IEtlcm5lbDo6Q29uZmlnLT5uZXcoKTsKJENvbW1vbk9iamVjdHtMb2dPYmplY3R9ICAgID0gS2VybmVsOjpTeXN0ZW06OkxvZy0+bmV3KAogICAgTG9nUHJlZml4ID0+ICdPVFJTLUltcG9ydEV4cG9ydCcsCiAgICAlQ29tbW9uT2JqZWN0LAopOwokQ29tbW9uT2JqZWN0e0VuY29kZU9iamVjdH0gICAgICAgPSBLZXJuZWw6OlN5c3RlbTo6RW5jb2RlLT5uZXcoJUNvbW1vbk9iamVjdCk7CiRDb21tb25PYmplY3R7TWFpbk9iamVjdH0gICAgICAgICA9IEtlcm5lbDo6U3lzdGVtOjpNYWluLT5uZXcoJUNvbW1vbk9iamVjdCk7CiRDb21tb25PYmplY3R7REJPYmplY3R9ICAgICAgICAgICA9IEtlcm5lbDo6U3lzdGVtOjpEQi0+bmV3KCVDb21tb25PYmplY3QpOwokQ29tbW9uT2JqZWN0e0ltcG9ydEV4cG9ydE9iamVjdH0gPSBLZXJuZWw6OlN5c3RlbTo6SW1wb3J0RXhwb3J0LT5uZXcoJUNvbW1vbk9iamVjdCk7CgojIGdldCBvcHRpb25zCm15ICVPcHRzOwpnZXRvcHRzKCAnaG46YTppOm86JywgXCVPcHRzICk7CgppZiAoICRPcHRze2h9ICkgewoKICAgIHByaW50IFNURE9VVCAib3Rycy5JbXBvcnRFeHBvcnQucGwgLSBhbiBpbXBvcnQvZXhwb3J0IHRvb2xcbiI7CiAgICBwcmludCBTVERPVVQgIkNvcHlyaWdodCAoQykgMjAwMS0yMDEzIE9UUlMgQUcsIGh0dHA6Ly9vdHJzLmNvbS9cbiI7CiAgICBwcmludCBTVERPVVQgIlxuIjsKICAgIHByaW50IFNURE9VVCAidXNhZ2U6IG90cnMuSW1wb3J0RXhwb3J0LnBsIC1uIDxUZW1wbGF0ZU51bWJlcj4gLWEgaW1wb3J0fGV4cG9ydCAiOwogICAgcHJpbnQgU1RET1VUICJbLWkgPFNvdXJjZUZpbGU+XSBbLW8gPERlc3RpbmF0aW9uRmlsZT5dXG4iOwogICAgcHJpbnQgU1RET1VUICJcbiI7CiAgICBwcmludCBTVERPVVQgIiAgIGV4YW1wbGVzOlxuIjsKICAgIHByaW50IFNURE9VVCAiICAgICAgIG90cnMuSW1wb3J0RXhwb3J0LnBsIC1uIDAwMDA0IC1hIGltcG9ydCAtaSAvdG1wL2ltcG9ydC5jc3ZcbiI7CiAgICBwcmludCBTVERPVVQgIiAgICAgICBvdHJzLkltcG9ydEV4cG9ydC5wbCAtbiAwMDAwNCAtYSBleHBvcnQgLW8gL3RtcC9leHBvcnQuY3N2XG4iOwoKICAgIGV4aXQgMTsKfQoKIyBjaGVjayB0ZW1wbGF0ZSBudW1iZXIKaWYgKCAhJE9wdHN7bn0gKSB7CiAgICBwcmludCBTVERFUlIgIkVSUk9SOiBOZWVkIC1uIFRlbXBsYXRlTnVtYmVyXG4iOwogICAgZXhpdCAxOwp9CmlmICggJE9wdHN7bn0gIX4gbXsgXEEgXGQrIFx6IH14bXMgKSB7CiAgICBwcmludCBTVERFUlIgIkVSUk9SOiBJbnZhbGlkIFRlbXBsYXRlTnVtYmVyXG4iOwogICAgZXhpdCAxOwp9Cm15ICRUZW1wbGF0ZUlEID0gaW50ICRPcHRze259OwoKIyBjaGVjayBhY3Rpb24gbW9kZQppZiAoICEkT3B0c3thfSApIHsKICAgIHByaW50IFNUREVSUiAiRVJST1I6IE5lZWQgLWEgaW1wb3J0fGV4cG9ydFxuIjsKICAgIGV4aXQgMTsKfQppZiAoIGxjICRPcHRze2F9IG5lICdpbXBvcnQnICYmIGxjICRPcHRze2F9IG5lICdleHBvcnQnICkgewogICAgcHJpbnQgU1RERVJSICJFUlJPUjogSW52YWxpZCBhY3Rpb25cbiI7CiAgICBleGl0IDE7Cn0KCiMgZ2V0IHRlbXBsYXRlIGRhdGEKbXkgJFRlbXBsYXRlRGF0YSA9ICRDb21tb25PYmplY3R7SW1wb3J0RXhwb3J0T2JqZWN0fS0+VGVtcGxhdGVHZXQoCiAgICBUZW1wbGF0ZUlEID0+ICRUZW1wbGF0ZUlELAogICAgVXNlcklEICAgICA9PiAxLAopOwoKaWYgKCAhJFRlbXBsYXRlRGF0YS0+e1RlbXBsYXRlSUR9ICkgewogICAgcHJpbnQgU1RERVJSICJFUlJPUjogVGVtcGxhdGUgJE9wdHN7bn0gbm90IGZvdW5kIVxuIjsKICAgIHByaW50IFNUREVSUiAiRXhwb3J0IGFib3J0ZWQuXG4iOwogICAgZXhpdCAxOwp9CgojIHRpbWUgdG8gc3RhcnQKaWYgKCBsYyAkT3B0c3thfSBlcSAnaW1wb3J0JyApIHsKCiAgICBteSAkU291cmNlQ29udGVudCA9IFxkbyB7Jyd9OwogICAgaWYgKCAkT3B0c3tpfSApIHsKCiAgICAgICAgcHJpbnQgU1RET1VUICJSZWFkIEZpbGUgJE9wdHN7aX0uXG4iOwoKICAgICAgICAjIHJlYWQgc291cmNlIGZpbGUKICAgICAgICAkU291cmNlQ29udGVudCA9ICRDb21tb25PYmplY3R7TWFpbk9iamVjdH0tPkZpbGVSZWFkKAogICAgICAgICAgICBMb2NhdGlvbiA9PiAkT3B0c3tpfSwKICAgICAgICAgICAgUmVzdWx0ICAgPT4gJ1NDQUxBUicsCiAgICAgICAgICAgIE1vZGUgICAgID0+ICdiaW5tb2RlJywKICAgICAgICApOwoKICAgICAgICBkaWUgIkNhbid0IHJlYWQgZmlsZSAkT3B0c3tpfS5cbkltcG9ydCBhYm9ydGVkLlxuIiBpZiAhJFNvdXJjZUNvbnRlbnQ7CiAgICB9CgogICAgcHJpbnQgU1RET1VUICJJbXBvcnQgaW4gcHJvY2Vzcy4uLlxuIjsKCiAgICAjIGltcG9ydCBkYXRhCiAgICBteSAkUmVzdWx0ID0gJENvbW1vbk9iamVjdHtJbXBvcnRFeHBvcnRPYmplY3R9LT5JbXBvcnQoCiAgICAgICAgVGVtcGxhdGVJRCAgICA9PiAkVGVtcGxhdGVJRCwKICAgICAgICBTb3VyY2VDb250ZW50ID0+ICRTb3VyY2VDb250ZW50LAogICAgICAgIFVzZXJJRCAgICAgICAgPT4gMSwKICAgICk7CgogICAgZGllICJcbkVycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIHRoZSBPVFJTIGxvZyBmb3IgZGV0YWlscy5cbiIgaWYgIWRlZmluZWQgJFJlc3VsdDsKCiAgICAjIFByaW50IHJlc3VsdAogICAgcHJpbnQgU1RET1VUICJcbiI7CiAgICBwcmludCBTVERPVVQKICAgICAgICAiSW1wb3J0IG9mICRSZXN1bHQtPntDb3VudGVyfSAkUmVzdWx0LT57T2JqZWN0fSByZWNvcmRzOiAiCiAgICAgICAgLiAiJFJlc3VsdC0+e0ZhaWxlZH0gZmFpbGVkLCAkUmVzdWx0LT57U3VjY2Vzc30gc3VjY2VlZGVkXG4iOwogICAgZm9yIG15ICRSZXRDb2RlICggc29ydCBrZXlzICV7ICRSZXN1bHQtPntSZXRDb2RlfSB9ICkgewogICAgICAgIG15ICRDb3VudCA9ICRSZXN1bHQtPntSZXRDb2RlfS0+eyRSZXRDb2RlfSB8fCAwOwogICAgICAgIHByaW50IFNURE9VVAogICAgICAgICAgICAiSW1wb3J0IG9mICRSZXN1bHQtPntDb3VudGVyfSAkUmVzdWx0LT57T2JqZWN0fSByZWNvcmRzOiAkQ291bnQgJFJldENvZGVcbiIsCiAgICB9CiAgICBpZiAoICRSZXN1bHQtPntGYWlsZWR9ICkgewogICAgICAgIHByaW50IFNURE9VVAogICAgICAgICAgICAiTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGU6ICRSZXN1bHQtPntDb3VudGVyfVxuIjsKICAgIH0KfQplbHNpZiAoIGxjICRPcHRze2F9IGVxICdleHBvcnQnICkgewoKICAgIHByaW50IFNURE9VVCAiRXhwb3J0IGluIHByb2Nlc3MuLi5cbiI7CgogICAgIyBleHBvcnQgZGF0YQogICAgbXkgJFJlc3VsdCA9ICRDb21tb25PYmplY3R7SW1wb3J0RXhwb3J0T2JqZWN0fS0+RXhwb3J0KAogICAgICAgIFRlbXBsYXRlSUQgPT4gJFRlbXBsYXRlSUQsCiAgICAgICAgVXNlcklEICAgICA9PiAxLAogICAgKTsKCiAgICBkaWUgIlxuRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLlxuIiBpZiAhZGVmaW5lZCAkUmVzdWx0OwoKICAgIHByaW50IFNURE9VVAogICAgICAgICJcbiIsCiAgICAgICAgIlN1Y2Nlc3M6ICRSZXN1bHQtPntTdWNjZXNzfVxuIiwKICAgICAgICAiRmFpbGVkIDogJFJlc3VsdC0+e0ZhaWxlZH1cbiIsCiAgICAgICAgIlxuIjsKCiAgICBpZiAoICRPcHRze299ICkgewoKICAgICAgICBteSAkRmlsZUNvbnRlbnQgPSBqb2luICJcbiIsIEB7ICRSZXN1bHQtPntEZXN0aW5hdGlvbkNvbnRlbnR9IH07CgogICAgICAgICMgc2F2ZSBkZXN0aW5hdGlvbiBjb250ZW50IHRvIGZpbGUKICAgICAgICBteSAkU3VjY2VzcyA9ICRDb21tb25PYmplY3R7TWFpbk9iamVjdH0tPkZpbGVXcml0ZSgKICAgICAgICAgICAgTG9jYXRpb24gPT4gJE9wdHN7b30sCiAgICAgICAgICAgIENvbnRlbnQgID0+IFwkRmlsZUNvbnRlbnQsCiAgICAgICAgKTsKCiAgICAgICAgZGllICJDYW4ndCB3cml0ZSBmaWxlICRPcHRze299LlxuRXhwb3J0IGFib3J0ZWQuXG4iIGlmICEkU3VjY2VzczsKCiAgICAgICAgcHJpbnQgU1RET1VUICJGaWxlICRPcHRze299IHNhdmVkLlxuIjsKICAgIH0KCiAgICBwcmludCBTVERPVVQgIkV4cG9ydCBjb21wbGV0ZS5cbiI7Cn0KCmV4aXQgMDsKCj1iYWNrCgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKClRoaXMgU29mdHdhcmUgaXMgcGFydCBvZiB0aGUgT1RSUyBwcm9qZWN0IChodHRwOi8vb3Rycy5vcmcvKS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIEw8aHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0Pi4KCj1jdXQK
H4sIAAAAAAACA+1d35ObthZ+z1/BOC/tjKNFP0BQN+ncdtq5nWknd5r0JS8e2Wa93GDwAN6N+3D/9iuBvWswGAzIiXePM/HuGlmfhPUdHenTOf7xpy+rwLj34sSPwrcjjMyR4YXzaOGHy7ejvz/+9sYZ/fTu1Y8LX/wg/y9jsTLkO8JE/fV2dJem6x9ubh4eHlCwTUQaxSjwNyjxbv4ngkDcyEI3o3evDOOwgoVIhXpt96pI09ifbVLPCMXKezuaifnnZRxtwsUoL7UrN4+CKDbuRfB29Po2e4xudtXcFOo5UfdaLL1Z7InP9VWb8uG6Xapee3G52tU6SnxZJN2uj4rU1KOeD8rsSiWyULh89/pf+HXepN0LT3VVNbQWJF2JeOmHxzjy3gT5jSDIIYTIm2G5jmtyxvH+npwPN7ssXHBZuPiycH4yXUdxGgs/PYacRVHgiTBHvRVB4nUHSuYikGPsVL9MZBHqYMuklqO6xynpjnfrp2k0UJfylx8JeC6Zl7G/OM3lQomaWh78RXo3/dJwA7vfsLz+rbb67/3EnwVeVQ/8MM0BcP/qt8NUX/6EMqP+9PaeIyKvrXbWWDie5VnnzxrLjb/wkoahVixTU9PdrthN010vl2t7Y/KXSnN4BhGIrRfvqv/5afY2dh+wnG7ijTcyxDz17/d/HdyTaPZfb57uOvshFeFCxAvjjfHR+5KODnyTkeEv3o7em8W7Ve6frE3ax6PbtY4ex5Q1xqTc/WMDUlf3bFauOpatF+Ey8B6rN5H0IawJsZHDTQWHbIy7QabqJjSM9FKZOnOeeQyFciXn4v3Hvz4YFFHjN/kJew9R/LnG2ThBxTrbHoVpFbR63bgVKz/YyhaKMBkZSbpVY0Y6ovlb/+0F917qz8Xh0D6/BXeev7yrbMPhNG2bhQfuh1kch81jsQtG0TDVmCd+q/71A5LOwDJcedWfoxduVrsZpyVKpdk5jxv3WZPK7HhqCm2gXP5ibn5a2KOfvX98L/7DD70Dq2TurBLua5XMsY01WqU3JsLSS7OdcfYb5nhiOdkvElddsruBz7x/plkv2vetggXW2HJPFuDIdFx7bBHEbdM+WdQZm926Mo/i0Iun6nNP6geVeQReee0cYOlle9Myj49dDKH+9UDI3MXjYfLkKhKz1vqdDZZZ8fq72HECXojkLvDC5al+kGFZ/8lffhLLEuv3vgjpy3pOJPe4OyYWotzUaQBKSJNhkGO5AGmkfy3UVy96dl+j2C/OgKVxjdtZh1PvO6dNYpNG0tlOi45d47K5My/+/vMPSYlfApEkFXMg7csGl48tjRRwbeRaYyafJpjIlTG3xoQipyOkF3iraT5naOlSVn+DzZbNtwhzDx49sI485BIYdhA/hHJ5N7DSDmdhCeKvvC9qd2uaeqt1IFKvcgVyDlqSerEXqeFbh9kbQjqxqwqrMFj9YibfK1fPvSled4s263XsJcn08UKiHSpae7FIpfnQBrXfXmrZqWxXohdQyy71B9qNN2137iEW6wt8QBmMuJX8nM7vxJERfTShrLM/n92maeaPVnuNg2GkYrkcYhLu5b/jAf33xuVIDtEN4dYPgmaEosh3/raZ3j6EUbwSwbS0sXW8qTWPNtJnjI3Qezje2/olv9jR+dtNC1+zDeso2K6ieH3nz5ubsYrCKFmLuTdwI+bKH1W/fhOfxjfRmr1damxC6z3Xjtxo9ipP7Ln2GYktgKlbfAxAwxaorAjqDDHmW8DyQWBrRnkjPuk/jHVh1PuHZZ1lswoeCzfrLUcHOcpLncVg6kppdXMMFaaDYclbvTkJNhjS8bJKH1bmWvuBn26HUTfar+RannPocPsUQ6fJPKoeG53haoSby3Bmvz2Q70pdjED3IlZLlO+kh/09MAmY9IyYdKt8NWASMAmY1I1J6jrQB+gD9OlEH3WIZzG94GIoWYkggBUR0OdZ0GevBV18BiIwAwGFngeFYk+k3jT1L+jHLeqONgB7gD3XyJ7ZFnazgTpAnbOocyfCJUw8wB5gT3f2wMQD1HlG1Dnr0PvTEc2bhtNw+WFqXccT9/UnxQDo1jE2NZG+rG88AaZYb6xvBmAik08wY4ibjor1Ze63H+r7+0qJf79mEiAE+WoO8i0PQwjzvdowX6u3SWIuYjYeOxzZdtcjh+3yELjIsTl+ivdV0JZNFbTDKdMY71vbyeOiDI9d81QBG7cO/bUJhP5C6O+FQn/tvpaA2WPGNPKfWSraEbuIWBOXI9OScPJZZ6BvqUsVDOXNBWSL6akibrnI4EG8LQN1h7IBQwbxPhqPMJSjQS0NjszK/ooh188L5YTJEZ1GWfiu8XRVkokc5wgq1Xomn/4TBduaeZX3ZhPXzCaGHMchY0aRTR0yYXtGURs73SMWWnCqiTLM6txzL1xMRRxHDyesNelZ9bTJaBdSlHWDaJzerGEnhpr1qtN3EFOOOMlHFdcZCL/HoQg73JrIn9zOcW3eFfiSq9dNIm+98Z2/+P6FrV3xxdeuNUNS2xK2HJIJS9ihHVe391Trap5qHeW4EoLY5RxXdwDHVbbYanBci0XAcdXjuNqDOK6nc93g3mkoHWtMTY00clhGIzdLd2MyxCkfU7Ua1Jnupken2uS7kQs/2+GWxiQ3LsLFQFmqMcnNiSg2SHEDKW4gxQ2kuIEUN5DiBlLcQIobSHEDKW4gxQ2kuIEUN3A2E85mPuNjzfuzlJcMiZ75yxAIBAR6FgRSX000/extIScHUAgo1J1C2QiEtAJAIgiyud4gmwYRs/e3VmFi6lUxMXYPZEzqIk4d/TJmn269QB3zRA5B0DFBxwQdE3RM0DFBxwQdE3RM0DFBxwQdE3RM0DFBx4TtL9AxQccEAgGBQMcECgGFQMcEEgGJQMc8X8ckvXVMk40Z1aljmlTpmIxkOiYxkW07Y0v+0Ktj9uhWKx3TRjbWqmNyZBV3Q2yNOuZKrNfyFRAyQcgEIROETBAyQcgEIROETBAyQcgEIROETBAyQciE/S8QMkHIBAIBgVoTKKtaLuph6gHmgPTy3KQXOkAeTMvSngfTYod5MG2GmO48mJZ13fFjrLhEZvp1F8iHCfILyC8gv4D8AvILyC/95JfQX802iaHkDyN44QLM7l4o2WHwe9FFgtHZHhBhvlkR5jKqCwGRBUQW2OmCna6r2iPe7wCAxgL8Af5AsBhQCCgEwWJAIiARKJZDK5ZsiKSXWiXLXdLLnWbJTMSYpTRL/UkvdYqWxETUfTY5L/crFsh9CZolaJagWYJmCZolaJYQMgYhY6BWQsgYhIyBmglqJuyCgZoJaibwB/gDaiZQCCgEaiaQCEgEauaQaqbVP/UlHduu1tSXKunl2Hby1Jcki8Dkjm41s0e3XuBX+CWest4gY4KMCTImyJggY4KMCTImyJggY4KMCTImyJggY4KMCdtfkPkSdEwgEBAIdEygEFAIdEwgEZAIdMy2OuaHVIQLES+MN8Ynf/lJLP/wQ+9A0cR7RdMeIKMsZYhrlDR5Fp/pINOauFw+Z3hdI0Lj9G6atf6sPlUUMdsUcRA+VcLlxRJn9ySK/Qp1rsYUlK7hE9c6bhyLTRrF0SYdUC6YR2Eoh4Pi4dFUuL9i3MnBvtvkSyM1quUvT5eVPH3U14p34927afHNR9tJR00alJB8iIDpyzFSfbvmJSh53KuKMthpVYY4DbRtLPAMeP2i7AG+XnvgDPJtu/xi9sBkyh4oQL324KhXlURtUQTm6K/DSXK9nHSHOAbI6eU4SRUnFaBmTpZ7VUm4FkWAk1+Hk9bVcpKYQ3w1iuSMzi9HyY7mMvtgnlSIupeyxV5VFKFtishmn+RkNiHbwEoNrKR9WFmeZy9MSzxI/i9X61RZ4GW+nlWI2tezxW5VrVV5uzK0id70RZD7RRkF9k0bhY/el7TKHJDe5oAil2NrjDEixNEaSbNHMpFNqTs5QKa8K7I6vNt0FqtUpjb4Iy6OQX3CU+mkZc/Tll1acHQ+rvKQuGPWfotaF9DiQDxrMHaTEotHulsd6+4k8AX+MqxTLc+XEQfQ2u6zJtXPDU2xWAMZJzqAccKOnE6ViWAm12ucciQTcWbxyQEyc7sig3F6ZsapejCCcbpG49Q/kTJGDrOsSxinPdLeOD0hg3EC43RyMIJxukbj1D8vAiXINcnYtZBLbKbVc3pEohSzyePfNjJ5V2QwTs/Lc6oZjGCcrtE42f2Nk4mwLccDRZQQvcu6HVLmKZEJZgzZLlPI3OFX4Dm9//jXB+P3jx/+NOS9emGWiiAmPyvGKHOZ6WDmcv2GqmZggqG6RkPFhzBUBGcTF+HaDVWGROUUiW3pRbnIYpZCth1Ov31D9ftKZWL6NcvH9MLsFEbcpo58EGyalnwml7BTleMS7NQ12ilnCDtl42zNZXHCNNspOx95Ludqtcfksi9D5oRcwWqPmJi+weYbE8O67xJmqnJYgpn6qmYq/zsQWy9+9yr/Q/5fxmL17tX/AeeK0mlUOwEA
iVBORw0KGgoAAAANSUhEUgAABh8AAAP2CAIAAAB42hv6AAAABmJLR0QA/wD/AP+gvaeTAAAgAElEQVR4nOzdeXxTZb748SfdoDsNXUgXVi0IilqgVoQZtbZYUARUynixw8iMekexd9Qqw4sfIyMddOgMVxnnXqQV9I6j4NJBFBRUrlSjsvSClkKhlKVNKU1pQ/cmTfL74+gxJE2ablnK5/2Hr5wn5zzLKSbnfPN9nqMwm80CAAAAAAAA6BU/d3cAAACP09bWVl9fP2TIkMjISNe3funSpbq6utDQ0OjoaNe3DgAAAPQU0SUAAKwFBgbGxcW5q3WTyWQymUguBgAAgLfwcXcHAADAZaS4kkKhcHdHAAAAAKcQXQIAwLMQXQIAAIB3IboEAIBnYU4cAAAAvAvRJQAAPAu5SwAAAPAuRJcAAPBERJcAAADgLYguAQDgWchdAgAAgHchugQAgGchugQAAADvQnQJAADPwqreAAAA8C5+7u4AAAAe59KlS83NzSEhIeHh4a5vPSAgICgoKCAgwPVNAwAAAL1AdAkAAGsGg6G1tXXIkCFuaV2pVLqlXQAAAKB3mBkHAIA1Vj4CAAAAnEd0CQAAa6x8BAAAADiP6BIAANbIXQIAAACcR3QJAICuEV0CAAAAnEF0CQAAa8yMAwAAAJxHdAkAAGvMjAMAAACcR3QJAICuEV0CAAAAnEF0CQAAa+QuAQAAAM7zc3cHAADwOO5dd6mystJoNMbGxgYEBLixGwAAAICTiC4BAGDNvblLBoPBaDS6pWkAAACgF4guAQDQNXdFl1Qqldls9vf3d0vrAAAAQE8RXQIAwJp7Z8YFBga6sXUAAACgp1jVGwAAa6zqDQAAADiP6BIAAF0jugQAAAA4g+gSAADWyF0CAAAAnEd0CQAAa+5ddwkAAADwLkSXAACwRu4SAAAA4DyiSwAAdI3oEgAAAOAMoksAAFhjZhwAAADgPAUX0AAAWOno6BBCBAQEuD59yWg0trW1KRSK4OBgFzcNAAAA9I6fuzsAAIDHGTJkiLua1uv158+fDwgIILoEAAAAb8HMOAAAPAgLigMAAMDrEF0CAMCDmEwmQXQJAAAAXoXoEgAAHoTcJQAAAHgdoksAAHgQoksAAADwOkSXAADwIFJ0yceHL2gAAAB4DS5eAQDwIKy7BAAAAK9DdAkAAA/CzDgAAAB4HT93dwAAAM9iMpmampqEEOHh4a5vnegSAAAAvA7RJQAALmMymWpraxUKhRujS6y7BAAAAC9CdAkAAGtBQUHuyh4idwkAAABeh+gSAACX8fPzi4uLc1frrOoNAAAAr0PiPQAAHoTcJQAAAHgdoksAAHgQ1l0CAACA1+HiFQAAD0LuEgAAALwO0SUAADwI6y4BAADA6xBdAgDAg5C7BAAAAK9DdAkAAA/CuksAAADwOly8AgDgQchdAgAAgNfxc3cHAADAT1h3CUB/aW9vr66udncvgIGlUqkCAwPd3QsARJcAALhcS0uLVqsdOnToiBEjXN96aGioyWTy8+MLGkBfHThw4LbbbgsLC3N3R4CB0tTU9PHHH6emprq7IwCILgEAcDmTyWQwGPz9/d3SemRkpFvaBTAoTZ48ubi42N29AAZKSkqKu7sA4AesuwQAwGVY+QgAAADoEaJLAABchugSAAAA0CNElwAAuAzRJQAAAKBHiC4BAHAZKboEAAAAwElElwAAuAy5SwAw0PiMBYBBhugSAACXIboEAPBS3377rUKh4CsMgOsRXQIAoAtcmgPAwPHeOcixsbEuO6oXbrrppt6dW5f1EMBgRXQJAIDLkLsEAJ6ptbW1fyvs7Ow0GAw9OuT8+fO9aMjxUf0+rl7w/B4C8HBElwAAuIwbo0tNTU3l5eXV1dWubxoAXOO7776znbollaxfvz40NFStVn/++efDhg0rLCyUd9BqtdOmTQsODo6JiXn88cflYEd8fLxCoYiPj5c2R48erVAoRo4cKddZWlo6adKkYcOGZWVlNTc3yxU2NTUtWrRo7Nix8fHxGRkZjY2NVp1paGhYsmRJSEhIdna2VH7VVVdJ3VZY6Ha8jo/qclyOz4bjcTn20EMPxcTEjB8//v333+9LDwHAFtElAAAu48boktls9t7ZIgDgjMmTJ9t+0EklRqNx8+bN99xzz6uvvrpp06bMzEx5h5ycnLy8PL1ef+rUqdTU1JycHKm8qqqqvLz8gQceaG5uNplMubm5VVVV586dk+vct2/f559/rtPppkyZ8vTTT8sVrlixYsWKFefOnbtw4UJOTs6KFSusOnPvvffedttt1dXVL730klReXl4uvWW20O14HR/V5bgcnw3H43Lsb3/7W01NzT/+8Y8HH3ywLz0EAFsKrmIBALB04cKFxsbG4cOHK5VKFzdtMpmMRqOPj4+vr6+LmwYw+BQVFWVnZxcXF7u7I11QKKxvQ+QShUJhMpmkDBp5n+HDh9fX18s7R0VF1dbWypsHDhx45JFH4uPj165dO2nSJMs6L1y4EB0dLYTQ6XRjxoxpaGiQ3hoxYkRlZaW/v78QoqOjY9SoUTU1NZYHtrS0BAUFOdPz3o3X8bgcnw0H43LQ3DfffPPoo4+eOHHi2muvPXDggL3z72QPPURKSkpubm5qaqq7OwKA3CUAAC7nxtwlHx8ff39/QksArnC2n8ATJkywzKyxCnAEBQUFBwdXVVW1tbXZq8psNgcEBMjlZrPZZDLJr2370GVoyVZHR4czu9k7yvG4JF1+H9kblwO//OUvn3jiiYsXL+7cubN/ewgAgugSAABWWNUbADzN3Llz//a3v1VWVur1+tOnT//617+W36qoqHjsscfeeeedffv2bdu27dSpU5YHfvLJJxcuXBBCbNmy5f7775fL58+f/8UXX0ivP/jgg3nz5jnZE39/f2kJpPPnz2/YsGH69Ol9OcrBuByzNy4HamtrJ0+e3NTUtGXLFhf0EMCVhugSAACXIboEAAPEctFoq9fC4oPX9sXTTz/d1NT0s5/9LCIi4vHHH3/qqaek8tjY2HHjxpWXl48YMSIkJOSNN9646qqr5EW+hRCpqanz58+Pior6/vvvX3zxRbk8Nzf397//fXBw8NChQ1988cXc3FzLTlp1z9LWrVszMzN9fHxuuumm8vLyDz/80JmB2zuqy3F1ezbsjcve6ZX86U9/uuOOO2699VYpcmQ1tB71EABsse4SAACXqa6ubmlpiYmJCQsLc3dfAKD3PHndJddwvEaS2WxuamoSQoSGhnrXLwq9W/tpUGLdJcBz+Lm7AwAAeBZylwDgSqBQKPgVAQD6CzPjAAC4DNElABgErOaXDRqDdVwAvB3RJQAALkN0CQAGAfkxZwPdUFxcnKIrcXFxA9Gcy8YFAD3CzDgAAC7DJTsAwHkajcbdXQAA9yO6BADAZaSHDbkld6mxsdFoNIaEhPj7+7u+dQAAAKB3iC4BAHAZHx+3TRuvr683GAxDhw4lugSg74xGo16vP336tLs7AgyUjo4Oo9Ho7l4AEILoEgAAnkOalOfG8BaAweTo0aNHjx4dO3asuzsCDKDvvvsuPT3d3b0AQHQJAACPYTKZBAuKA+gnkydPvvHGG4uLi93dEWCgpKSk3Hjjje7uBQAheGYcAACeQ4oukbsEAAAA78L1KwAAHkF+Vh3RJQAAAHgXrl8BAPAIUuKSILoEAAAAb8P1KwAAHoFFlwC4kYMPH4VC0fePJj7cAGBwI7oEAIBHYNElAC6gUqm6LJcn5/boLec5X0loaGjfmwMAuBiXsAAA/MRsNjc0NDQ0NPTL3VSPEF0C4AI1NTVWJTt37rTNTmppaVmyZElISMiNN95YUlJidUhycnJycrKTLXZZv1SyYcOG+Pj4gICAwsJCqVypVDY3Nyt+1IOBAQDcys/dHQAAwIOYTKa6ujohxLBhw1zftCC6BMDlZs+ebTabrUI5zz77rEqlqq+vb2xs3LRpk9UhPYq/d1m/VFJdXV1aWvrZZ59lZmbq9XohRH19vUKhcH18HwDQR3x2AwDwEzm6FB0d7eKmm5qaampqAgMD4+PjXdw0BoGsrKx//vOf7u4FPIvZbB4+fHhtba0QIi4urrq62vLd2NhYjUYjb1rFdJRKZXl5uVKpFEJcvHgxMjKyj3cNtjEjyxJ7rwHHUlJScnNzU1NT3d0RAOQuAQBgwcfHx/VxJQm5S+iLkpISo9E4ZswYd3cEHqS9vd3f3196LQeSehG7kbKK3EKv1wcEBLirdQCA84guAQDgEaT7PaJL6DUfH5+Kigp39wIepKioKDs7u3fHLlq06C9/+ctzzz2n1Wr//Oc/W70rLbq0f//+vnaxK/7+/oWFhXfdddenn366du3affv2DUQrAID+xSUsAAAegdwlAC5gm54pr59tuZD2unXrqqqqIiIiMjIyHnroIeld+RCj0eh8AlSX9cslVv8VQmzdujUzMzMiIiI/P7+goKCX4wQAuBa5SwAAeAQpusQzkgAMqAsXLliVdBknCg4Ofv31119//fUu9zl06JDzLXZZv2Wh1Q7z589341w8AEDv8AMpAAAegdwlAAAAeCkuYQEA8AhElwAAAOCluIQFAMAjEF0CAACAl+ISFgAAj8Az4wAAAOCluIQFAMAjkLsEAAAAL8Uz4wAA+ElLS0tNTU1gYGBsbKyLmw4PDw8JCQkICHBxuwAAAEAfEV0CAOAnJpPJZDJ1+fzsgRYWFub6RgEAAIC+I/0eAICfuCWuBADAoGQ0Gt3dBQAuQnQJAICfSNElhULh7o4AgEtdCZ97CoWi22FeCedBOHcqemrlypXya7PZXFlZWVpaesMNN1jus2/fvsWLF8+fP1+hUAQEBBQWFjout6oWgCdjZhwAAD8hugRgcFOpVOfPn7ct71HmZmtra1BQUP91yimhoaFNTU19qcFsNnf78d7leejRePveTxdw5lT0yMmTJxsaGuTN77//vqKi4p577ikpKbHc7eGHH/76668jIiKEEGq1Oj09vbm52UG5EKKhoaGiomLs2LH92FsAA4HcJQAAfkJ0CcDgVlNTY1Wyc+dOq0wWaXP9+vWhoaFqtbqgoCA0NFRKJ9FqtdOmTQsODo6JiXn88cdbW1ulQ6Kjo5VKpfRaqVTGxMRYVvXZZ59FR0dPnTr17NmzUrnBYFi+fPn48eMTExOfeeYZvV5vub9Op1u6dGloaGhOTo5cZ3Nzs+JH3Q7TXv1CiHPnzqWkpERERDz00EMtLS1S4W233aawIO9sb7zt7e3Z2dkxMTEqlWrLli2JiYnO9DM5OTk5ObnbzlsdbrW5bNmy8ePHBwcHp6Wl7d271/F5s9dPyeeffx4fH2+VK9Sj+mWbNm1KT0+XNydPnjxv3jzbM9DR0SGFkIQQ06dPl8+/vXIhRGpq6saNG7s9aQDcjugSAAA/kaJLPj58PwK4UsyePdsqYUfaNBqNmzdvnjNnTklJyebNmzMzM4UQOTk5eXl5er3+1KlTqampcpRh//79N9100/nz5ysrK5OSkr799lvLqi5evFhZWfnss8/+7ne/k8pXrlwZFxdXVlZ24sSJuLg4efaTtP+CBQtmzpyp0WjuvPNOqby+vl56V9LtoOzVL4TYsmXLu+++e+rUqZCQkGeeeUYq3Lt3b5eV2xvvihUr9Hp9SUlJRUVFcHCw1L1u++lk581mc0pKilqtljbVavXNN98sH7h27dqysjKdTrd8+fK5c+c6Pm/2+inZtWtXWVnZ1q1bpT9uL+qX7d69+5prrul2aJWVlT0qF0JMmDBh9+7d3dYMwO0ULF8KAIDs4sWL9fX1w4YNi4qKcndfgB5ISko6cuQIC+jCUlFRUXZ2dnFxsRAiLi6uurra8t3Y2FiNRiNvKhSX3RfIm1Yvhg8fbhmhiIqKqq2tlV6fPXv20Ucf1ev1BQUFo0ePtqyqoaFh2LBhzc3NsbGxjY2NQogRI0YcP3582LBhQoiGhoaJEyfK8/UUCkVTU1NISIjVcKx66Ji9+hUKRVVVVVxcnBCivr5+woQJcv+7bMjeeFUq1dGjR+V0rV73057du3evW7duz549Qoi0tLRnnnkmLS1NCHHy5Mknn3xSrVZ3dHQkJSUVFRXJbXV53hz302Aw+Pn5Wfa5p/XLwsPDNRpNt3+1bv+Z2e7W2Ng4cuRInU7XZbspKSm5ubmpqaldvgvAlfhtFgCAnzAzDsCgpNFo5MQZ6YVlaMl5U6ZMMVuwDM1I0YqhQ4f6+vraO1yKZUgchGDshTBkljPd7Ok2xGMymbrNVLU33s7Ozm474GQ/u5Sent7W1lZUVFRUVNTe3i6FloQQCxcunDFjxrFjx3Q63bZt26yOsj1vjvtp+efoXf2yXgfU7MXELcvJhwC8AtElAAB+4q7okslk6uzsNJlMLm4XAHokIyMjPz+/qqrK6vPq7Nmzv/rVr1577bX8/PxHHnlEXl9J8uWXX+r1+l27dslRkqysrE2bNkkhm40bNz744IPdNu3v719YWGgwGHbt2nXHHXc43tlB/Zs2baqqqmpoaMjNzV24cGHvxnvffffl5eUJIbRabWFh4eLFi53pp5PrLklWr169atWqVatWrV69Wi7UaDRTp04NCwsrLS19+eWXu63EQT+71NP6ZaNGjbJKjutSYGCgPOPvwIEDYWFh0om1Vy6EqKqqYklvwCsQXQIA4CfS5azro0vNzc2nT5/u8kFOANCPoqOjrUrkFaMtXwiLT0LLj8QnnnhCq9XOnDlTqVTOmjXrhRdekMqnTJly8ODBkSNHqlSqr776atq0aZZNxMbGjhs3bv369X/961+lkjVr1mi12oSEhISEhIsXL65Zs8ayrS5XxZaWB4qIiMjPzy8oKHA8zC7rlypcsmTJggULxo4d29zcLPdfZpUmY2+8eXl5Op0uOjp60qRJ27dvX7dunTP9NBqNzqfhpKammkwms9l8++23y4UbN25cunSpUqlcvnx5VlaWuPyPZXve7PXT9g8t/ben9cvS09OPHz9uWWL770oI8eabb956660KhSIqKmrevHk7duzw9/d3UC6EOH78+KxZs5w8aQDciHWXAAD4SU1NTVNTU2RkpPzwGte4dOmSVqsNDg5WqVSubBeDBusuwZbluktu1C/rELnMF198kZWVZZV7hW6dOHHipZdeeuWVV/q95scee+ypp56yl77EukuA57CeagsAwJXMXTPjwsPDw8PDXdwoAAw0OXvFwwNM0qrngYGBEydO/Mc//uHu7nifxMREaQ31fqdUKpkZB3gFoksAAPyEVb0BoB95eFBJ1rs1zmEpNzd3IKp9/vnnB6JaAP2OdZcAAPiJdCPU7VOEAABuFxMTo7ARExPj7n4BwJWI3CWgB5577jn5vwAGJXKXAMBbXLhwwd1dAAD8wJuiS3q9/uDBgydOnNBqte3t7UOHDo2MjBw/fvzUqVMDAgLk3Zy/83/uueesdvb39x82bNhVV111yy23hISEWO1vMpkOHTpUUlJSW1vb0dExZMiQyMjI0aNHT5o0acSIEY7bunDhQmlp6ZkzZ+Rjo6Ojr7322qSkJF9f32672sfDhcPTQqwEAGRElwAAwJVpEP+U7nhQg3LIruc10aWzZ8++8847zc3NcklLS0tLS8vZs2e/+uqrhQsXjho1qu+tGAwGrVar1Wq///77pUuXWj4wSK/X/8///E9lZaVc0tbWVllZWVlZWVRU1O0/x//6r/+y3Gxrazt79uzZs2cPHz784IMPDh06dEAPBwA4yWQyCaJLAAAA3mbQR8c8fGjeEV2qrKx84403jEZjbGzsLbfcMmrUqMDAwLa2tnPnzn311VcajeaNN9741a9+FR8fL7o6493+JeS3WlpaKisrP/nkk4aGhs8///zee++V9/nf//3fysrK4ODg22677eqrrw4JCTEYDHV1dadPny4pKel2CFFRURMmTBg1alRMTExQUFB7e/vp06d3796t0Wg+++yzOXPmDOjhtiMFAHSJ3CUAAIBBidvhAeUF0SWj0fjee+8ZjcYbbrhh7ty58kqrISEhEydOnDBhwgcffHD48OF333132bJlTk4Tsyc4OHjChAlBQUGvvfZaRUWF5VtHjx4VQixYsGDcuHFSia+vb3x8fHx8/MyZM7ut+bHHHrNq6Nprrw0KCnrjjTeOHTvWbXioj4cDAJxEdAkAgP5iNBr7eIMGwFt4QXTpu+++0+l0UVFRd999t+1DfHx8fO6++26NRiNNZ7vhhhv63qK0iFJHR4dlYVNTkxAiISGh7/XLpGQrq4ZcdrgtKZT7hz/84dChQ8XFxXV1dQaD4Q9/+IMQQqPRHD58+MyZMw0NDQqFIjw8fPz48TNmzAgMDLSt4bnnnjt8+PCBAwe0Wq2Pj09CQkJqaqp0Vo8cOSKVm83mhISEtLQ02yWrGhoa1Gr1qVOnGhsb/fz8VCrVTTfdNGHCBHmH//zP/9TpdI888ohKpZJKCgsLjxw5IoS4/vrr58+fLxWeP39+48aNERER2dnZ8rGVlZVqtbqysrKtrS0wMHDkyJHTp0+XzqQz58FWR0fH+++/X1ZW5uPjk5GRMW3atB6edQCehegSgCuTQqGQPgAHk8LCwszMzJCQkOuvv37v3r3u7o4Hkb7m+vcvvnLlyjVr1kivzWZzVVVVU1NTZmbm999/L++zb9++V199taWl5V//+pe/v//WrVulS3d75VbVwvX6fn8n31sVFxcfOnSorq5OoVDExcXNmDFj7NixVs315WZNzkuyTFDqabLShx9+ePDgwbCwsIcffthyIebm5uZXX321sbFx6tSpd911lzcOzQW8ILpUVlYmhEhJSbEX9vb19b3ppps+/PDDsrKyfokunT9/XggRFhZmWRgaGnrp0qXy8vKJEyf2vQlJTU2NECI6Ototh9vz0UcfHTx40Kpw06ZNlpt1dXV1dXWlpaW/+c1vgoKCrHbetWvXt99+K2+ePHny7Nmzv/nNbw4cOLB//365/NSpU1VVVY888ohSqbQs3Lp1q16vlzY7OztPnz59+vTpn/3sZ7fffrtUOHbs2OLi4tOnT8vRpdOnT1u9EEJI2WdjxoyRSw4cOLBz5075e7S5ubm0tPTYsWN33XXXlClTnDkPVhoaGt56663a2tqgoKD777/fsi0AXsr2asAFzGZzTU2Nr69vVFQUgS0AA0qlUknXulZ6FGhobW21vQIcaKGhodLPvc7LzMy0jFO4Ri/66Xpms7l/v25OnjzZ0NAgb37//fcVFRX33HOP1RIiDz/88Ndffy0tbqtWq9PT06V1de2VCyEaGhoqKirc8u0MWV/u7yQ7d+48cOCAvFlRUXH69OnZs2db/jbf7zdrvZCRkVFbW3vu3Llt27b98pe/lEIQRqNx69atjY2NI0eOzMjI8NKhuYAXRJeqq6vF5TECW9K70p590draevbs2U8++UQIcd1111m+NXny5KKiovfff//UqVPjx4+Pi4sLDg7udUPSolEff/yxQqH42c9+5uLDHTt06ND06dOTkpKUSqWcLDZ69OgpU6aMHDkyJCREr9dXV1d//vnnGo2mqKho1qxZVjUcOHDglltuSUpKCgsLu3Dhwvbt27Va7ZtvvtnY2GhbXlRUdM8990gHNjY2btu2Ta/XX3fdddOnTx8+fHhbW9vRo0f37t27b9++cePGSWu3jxkzRoouTZ8+XQhRV1fX2NgYEhJiNpsbGxvr6uoiIyPFj5Em+V9OTU3Nrl27zGbzjTfeOGPGjGHDhul0uqKiosOHD+/cuTMhIcEqTtflebB05syZbdu2tba2RkdH/+IXv7BcAx4AesRkMkmX0f3+gwEAWJF+nrS0c+dOaZkF+c5Hijv89a9/XbVq1SeffHLs2LH/+I//eOONN+bPn6/VamfPnn3w4MHo6Oj777//z3/+sxRmio6O7uzsrK+vF0IolUp/f/8LFy7IVX366ae/+MUvRo4c+d5770mXcwaD4f/9v/9XWFhoNpvnzZu3Zs0a6RnQ0v4NDQ1PPfXUtm3bHn300XXr1kl1Njc3ywGRbmNho0aNOnfunBBiwYIFlof0qF0H50EIsWzZst27d1dVVU2fPn3FihW33XZbt/1MTk4WQljejXfJ6nCrzS7btXfe2tvbn3322bffftvHx2ft2rV/+tOfTpw4ITf0+eefZ2Vl1dbWWsbgelS/bNOmTenp6fLm5MmTJ0+ebDu0jo4O+Zp5+vTpLS0tjsuFEKmpqRs3bnzxxRcdnzQMqF7f31nWMGXKlFtuuSU8PPzSpUtffvllcXHxxx9/PHr06KioKNEfN2vyE+H7ktTj6+ubmZn56quvnjt3bufOnXfffbcQ4qOPPqqsrAwPD8/MzLRNefGWobmAF0SXWltbhU0mkRXpXWnPXrD9I1133XVWUZtbb721sbHxyJEjhw4dOnTokNTo6NGjr7/+enklpp62FRUVtWjRovHjx7vlcHsl06ZNs/xukCxZskR+HRgYOG7cuMjIyPXr1584ccI2unTLLbekpqZKr+Pj42fPnv3666/rdLqZM2falltmG33zzTcdHR2Ws9sCAgKmT5/u6+u7a9eu/fv3y9ElIcTZs2elidxyFMlsNpeUlJw+fToyMtJoNJ49e1ZYRJe+/fZbk8mUmJgof9gNHz583rx5zc3N5eXl33zzzdy5c7s9D7KDBw/u3LnTZDKNHz9+wYIFQ4YMsbcnAHTLaDQKIboMZAPAQJs9e7ZVJou0aTQaN2/ePGfOnCVLlmzevDkzM1Ov1+fk5OTl5U2fPr2jo2PPnj05OTmvvPKKEGL//v3//u//fv78+c7OzqSkpPz8fMuqLl68WFlZ+cEHH/zud797//33hRArV66Mj4+X5ii89NJLK1eu/POf/yzvv2DBgqysrPXr18sZAfX19T2auyddB9oe0qN2HZwHIcTatWs3bNhgMBj27ds3d+5cKV/JcT+d7L/ZbL755pv/+te/Sptqtfqpp55Sq9XSZpft2jtvK1as0Ov1JSUlISEhH2iBIegAACAASURBVH74oRT+k+3atausrGz37t3yoHpav2z37t2//vWvux2a5QO4nSkXQkyYMOH5558nuuRevb6/k40fP16K1AghlErl3LlzW1paysrKvv32W2miWf/erNnTZXTGqjA4OHjRokWvvfbaoUOHRowYYTabi4uL/f39Fy1a1GV+iYcMzRN4QXTJ9cLDw3/+859bRSV9fX3nz58/ffr0kpKSs2fP1tTUNDY2fvfdd999993EiRPvvffeXqxX19DQcPDgwfDwcNu1h1xwuD1Tp061LWxsbFSr1RUVFTqdTv7uEUJcunTJduekpCTLzdjYWAfllpnD5eXlQoibbrrJqsJJkybt2rVL/tYJCQmJjo6ura3VaDQjR460nAFXUlJSUVExbdq0qqoqg8EQHR0tT5eVLjJmzJhhVfnMmTPLy8vPnDnjzHmQfPTRR9J36owZM1JTU5nGAqCPTCaTILoEYCDFxcXJaf7SpUtsbKxGo3FwyNNPPy2EuP/++9evXy+9EELs2LHj9ddfl/eJioqSokujR4/+7//+74ceekiv1xcUFIwePdqyqvT09CFDhmRkZCxdulQqef31148fPy69zsrKmjhxohTlkXzwwQfSJZx849pfetFul+fh5MmTTz75pFqt7ujoSEpKkqdxOWYblLFn9erVq1at2rNnjxBi1apVq1evlsodt2vb/7feeuvo0aPSNKX7779f6rxs7dq1fn5+8+fPNxgMvatfdvr0afma3wHp1xTny4UQ8fHxXUYr4Eq9vr+T3XLLLbYlZWVl8h+3H2/W+k6lUs2dO/e99977+OOPpaDw3Llz5VVZrHjX0AaUF0SXgoKCGhsbGxsbbWdvyhobG6U9e9eEFK3s7Oysra3ds2fP6dOn33zzzd/+9rf+/v5We8bExMTExAghzGbzhQsXSktLv/nmm9LS0sjISHlVIGfaamtru3Tp0tGjR7/99ttNmzY9+OCDVt/BA3q4Y7YzvLRa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YGdSUlKyd+/etra29957z7cheY7OJLSQXQIAhDm73R4TE8PAJQAeMplMvq2wt7dXSXN77sKFCwFryz0v7obXwQNhJqQ7E6vVumDBAl+F4R06k9BCdgkAEOY0Gs2oUaPy8vIC33RXV1dLS0tPT0/gmwYi3Oeff953RphSsnXr1pSUlMrKyqNHj6anp5eXlzsOaG5unjJlikajyc7OXrlypePNMD8/X5Kk/Px85ePIkSMlSbr66qsddVZXV3/zm99MT0+/6667DAaDo8Lu7u7FixePHj06Pz9/7ty5XV1dLsG0t7cvW7YsOTn5wQcfVMqvueYaJWzJiSeX7KYtIcS5c+emTp2q1WqXL19uNBqdYxiolYHuhhDiwIEDBQUFSUlJ3/jGN1555ZWxY8deYfBA0IrAzmTEiBEuJ3odxqA3SgixfPny7OzscePGHTx40FFIZxKSZAAA4B/nz5+vqanp6upSOxBfmjZt2vXXX692FHBVXFwcFxendhQDSk5OvnDhQuDb7fusK4TYtGnTyy+/nJGRUVJSsm/fvtjYWMfepUuXvv/++xaLpbu7++DBgz/72c8cu06fPl1aWtrd3W2z2V588cX6+nrnOv/jP/6jsbFRluWnn376Jz/5iWPXypUrP/vsM2X73XffXbFihUswN9100+7duzs7OweNfFBu2hJCPP744+fPn29vb//lL3/pfF1uWhzobuzateu2226rrq7u6elpaWn57//+b51Od4XB+8q8efNeeukltVrHoCoqKq699lqvTz98+PCMGTN8GI/nIqozcXOiF2G4v1GyLBuNRrvd/tFHHyUlJfkq+CukfC9osVhUaT10kV0CAMBfzp07V1NTYzAY1A7El8guBSeyS/3q94XQsWG3212O0el0zt/CZmZmOp/70UcfTZo0ad68eVVVVS51Xrx4Udlub29PT0937MrOzna8n/T09GRnZ7ucaDQaPYx8UG7aEkI43mA7OjpcrmugFge6G6NGjXL/t0l2CQMJs+ySYyPMOhM3J3oRhvsb9eGHH06cODExMXHKlClubnKAkV3yDjPjAADwF7vdLoSIiuJ/WyAY9Z1nMX78eOcH5aamJue9SUlJGo2mvr7+0qVLA1Uly3JcXJyjXJZlpR8QA6wKnJSU5EmoZrN50GMGbUsRGxvrYafk5m44GvKEJ8EDIS3MOhM3rjCMvjdq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eCkmJkav1zc1NV199dWO8Ufbt29/5ZVXRo0aJYTYvHnzddddt2XLFmXXM8884/jGr99znQ2UF3jxxRdfe+21vLw8IcSzzz777W9/+8ovxGg0btu2raysrLe3t6enJzExsaCg4Ne//rXjgA8++MDN6aNGjbr66qu9br22tvZKkmLolyzLZrNZCJGQkKB2LL5XVVU1ceJEtaMAAG98/vnnSg/m/B+98pXAU089tX79+jfffLOnp2fhwoW7du1asGCBckBzc3NxcfEnn3ySlZV1xx13/Pa3v01KShJC5OfnNzQ05OXl1dfXCyFGjhx59uzZq6666ty5c0qdJ06cuOOOOxoaGubPn//73/8+OTlZqbC7u/vee++trKw0m80FBQV79+5NTU11DqatrW316tX79++/++67n3nmGSHENddcU1dXJ746ptv91xiDXpcQYvny5X/605/S09M3bty4cOFC5xP7Dd79dfV7ex1n7dmz58knnzx58qQQYu/evUoYJpPpwQcf3L9//6hRo/74xz+OHz9+0OsCEJwYuwQgVL3yyitHjx6dNGnS2LFjHcOIzp49O378eEmSJEkaPnz4+fPnHcc7p2D6PddZTk5OvzmXhoaG0aNHK9vXXHON8y69Xt/vcKdB9fT0tLe322w2x/Miy2EimH3rW98qLi5WOwoA8MaECRP6Zi6UEpvNtmvXru9973v/9V//tX379pKSEscBpaWlmzdvtlgsdXV1M2fOLC0tVcrr6+tPnz79gx/8wGAw2O32srKy+vr6c+fOOeo8duzY0aNHOzo6Jk+e/MgjjzgqXLdu3bp1686dO3fx4sXS0tJ169a5BPP973//pptu0uv1SmpJCHH69Glll+zE/cUOel1CiOeee66xsfHFF1/80Y9+5HJiv8G7v65+b6/jrDfffPPAgQNms3nv3r2OMNasWaPT6RobGysqKo4cOSJILQGhSwaA4JaQkGA0GpVtZUC481673f7GG28MHz5c+Thu3Lgvv/yybyX9dncu5zq74447du/e3ff0r33taydOnFC2q6qqnKvdtWvX4sWLPb4sV62trevWrUtKSoqLi0tPTzeZTF5XBdXZ7faampqamhqbzaZ2LIgUDz30UGZmptpRAIGTk5NTU1Pj3bl9nwocJUIIu93ucozLb5u6/Fv76KOPJk2aNG/evKqqKpc6L168qGy3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AAAAAGBQePbZZwOBwEUXXdQxWhJC/P73v7/wwgsDgcCzzz7bUzOeeOKJQoi2trboRrfbLYQ47rjjemqWZHR30lmzZjkcji+//PKf//yn0vLVV1999dVXDodj5syZKZdhMBiEEAUFBWm+t7eRLgEAAAAAMChs2rRJCHHNNdfE66Bcev/993tqxi+//FIIEbMSasyYMUKI+fPn//DDDz01UZe6O6nZbFZO9V69erXSojy44oorzGZzCgWEw+E9e/Yo7/CNN96YtnvTg51xAAAAAAAMCg6Ho6WlpampyeFwdNrh8OHDBQUFeXl5hw8f7ni1WzvjDh8+vHXr1jvuuOPAgQOLFy9esmSJ2nPr1q3Tp0/3+XxCiGOOOeakk0765S9/ecYZZ5x55pl6vb7LV5HazrjkJ1UH+frrr0866aTc3NyamhohRHFxcWtr69dffz169OjEb0W8woQQpaWld99996233prkjUd4b9qQLgEAAAAAMCgYjcZQKBQMBpU9Vh0Fg0GTyWQwGILBYMeryaRLHV166aUvvPCC0WiMbvzuu++WLVv21ltvOZ1OtbGsrOzxxx+/8MILE7+K1NKl5CeNHmT8+PGff/75K6+8Isvy5ZdfPn78+M8++6zLieIVJoQwGAxnnHHG0qVLkz9z6kjuTRvSJQAAAAAABoU0rF2KUVZWtn79emVXWkfhcHjnzp27du367LPP3n333b1790qSVFFRcf755yd4FSmnS0lOGj3Is88+e8MNN0yZMkWW5Q8++ODZZ5+97rrrkpwoRigUqqmpefvttxcvXtzW1vb3v/996tSpabg3PUiXAAAAAAAYFCZMmLBjx46NGzdOmzat0w4bN248++yzJ0yY8Omnn3a8mvzOuGAw+OOPPy5duvTFF18sKCj45ptvioqKEtcWiUTuvffehx9+OOZr2rpVRndDn04njR6kra2tpKTE6/UKISwWS21trd1uT2GiaH//+99nzJgxbty4HTt2pPPeXsWp3gAAAAAADArKgpeVK1fG66BcOvJ1MUajccSIEWvXrj377LMbGxvvvffeLm/R6XQLFy4UQuzatesIZ09el5NmZ2dfdNFFsizLsnzxxRcr0dIROuOMMxLM2Hv39irSJQAAAAAABoXrr7/eaDS+8cYbL730UserL7744ptvvmkyma6//voemU6SpCeffNJgMLzwwgt79uzpsr/ybW42m61HZk9Sl5POnz8/5sER+uSTT4QQhYWFab63V5EuAQAAAAAwKAwdOvShhx4SQlx11VXXXnvtjh07PB6P1+v9/PPPr7vuurlz5wohHn744fLy8p6accSIEfPmzQuFQtHLl0aNGnXvvfe+9957lZWVfr/f5/P9+OOP//3f/62cfDRjxoyemj1aypOeccYZytqlyZMnd3fSX//61y+++OK+ffv8fr/b7d6zZ88f//jH2bNnCyHmzJmjdpMkqeOpVUne20dw7hIAAAAAAIPI0qVLFy9eHIlEYtp1Ot2DDz74hz/8Id6NyZ+7FK2mpmb48OE+n++zzz4bN26ciP/tckKIE0444YMPPhgyZEiC+lM7dyn5SZM5UynJc5fiTXrWWWe9/fbbmZmZCUZL8t4+gnQJAAAAAIDBZdeuXU8//fTmzZurq6tlWS4rKzvrrLNuueWWUaNGJbgrtXRJCLFo0aLly5dPnTp106ZNQojdu3e/9dZbW7Zs2bVrV0NDgyzLeXl5o0aNuuCCC66++uqMjIzExaeWLiU/aQ+mSzt27HjppZc+/PDDffv2hUKh/Pz8MWPGXHbZZZdeeqlO9/+byTodLcl7+wjSJQAAAAAAAKSuz8VdAAAAAAAA6EdIlwAAAAAAAJA60iUAAAAAAACkjnQJAAAAAAAAqSNdAgAAAAAAQOpIlwAAAAAAAJA60iUAAAAAAACkjnQJAAAAAAAAqTNoXQAAAAAAAEgHSZJkWda6iv8nSVLHxj5VIZJEugQAAAAAAHpdp9mWtllSX4vb+i92xgEAAAAAACB1pEsAAAAAAAwukiS98sor48ePz8zMtNls06ZN27lz54YNG0477TSr1Zqfn3/ZZZc1NTWpnZ977rnjjjvOZDIde+yxa9eujR7queeeGz58uMlkGj58+MqVK2OmOOWUU2w2m16vVzbBST/pssLrr7/+0ksvjW6ZM2fO9ddfr2FJSEQGAAAAAACDgBoCCCFGjhz5/vvvO53O2traq666yuFw/OIXv9i0aZPactlll6mdhw0b9tFHHzmdzg8//HDo0KHvvfeecmndunVlZWWbN29ub2/fvHlzWVnZ+vXr1btGjBixefNmt9sdM3vHejoKBAKnn376M888ozx9+umnJ0+eHAgEerskpIYdhgAAAAAADArqMUOSJH3++ecnn3yy0t7Y2FhYWPjFF1+MHTtWbTnhhBMOHz6sdF6/fv25556rXHrrrbcef/zxjz76SAhx2mmnLViw4IILLlAuVVRUPProo5988oly15YtWyZPntxx9uiWjkWqfRoaGk477bQ///nPQog5c+b84x//KCwsVG/spZKQGt5HAAAAAAAGheh0KRwO63S66EsdW9TOLS0tOTk5SntLS8vw4cOVfXMOh2Pfvn25ubnqpWOPPba5uVm5y+v1ZmRkdBwwQUuML774Ys6cOQaD4eWXX1aTr14tCanh3CUAAAAAAAad6CApXssRis5xUvPpp5/a7Xa/33/o0KE+UhI6RboEAAAAAAAS2bp1q/r4448//uUvf6k8PuGEE7Zs2RJ96Re/+EW8QQwGQzgcTn7Sr7/++v7773/99ddfe+216667bt++fZqXhHhIlwAAAAAAQCK33HLLxx9/7HK5Pvroo9tuu+2ee+5R2u+6665bb731o48+Ui8tXLgw3iBDhw59//33I5FIMjO63e7Zs2c//vjjxx577Pjx4//zP//zoosu8vl8GpaEBNhhCAAAAADAoBB9lFKXRyBFd/6f//mfFStWHDx4sLy8/L777ps3b57a7dlnn1UuDRs2bOHChddcc028AdetW3fXXXcdOnQoHA6rI3csUrl01VVX6fX6559/Xm2/7LLLLBbL6tWre7UkpIZ0CQAAAAAAxNUHj77ugyUNcuyMAwAAAAAAQOpIlwAAAAAAAJA61pIBAAAAAAAgdaxdAgAAAAAAQOpIlwAAAAAAAJA60iUAAAAAAACkjnQJAAAAAAAAqSNdAgAAAABg0Dl48GBlZaXy2OVyHThwQHns9/ubmpqEELIs79u3b9++fTE3qh1iblR1OhoGNoPWBQAAAAAAgLRav379v/71r+bm5qOPPvqiiy66/fbbTzzxRLPZfN555/3xj39sbGx86623/vWvf23atKmhocFgMCxbtky5ce/evWqH2tpa9cY777xT6RDdGD2adq8V6SDJsqx1DQAAAAAAIN3a29tvuOGGSZMmjRo16t/+7d9mzZq1bt06IcQFF1yg5EGRSOTQoUNLlixZs2ZN9I1Kh6effjrmRiFEx0Z1NAxg7IwDAAAAAGDQkWX57rvvXrBgQWNjY0FBQad9KioqFi1adM455+zfv//zzz/3er3RV6NvVDskGA0DGOkSAAAAAACDzl133XXxxRefdNJJpaWlNTU1QghJkmL6XHjhhX/+859feumlmpqa7777zufzRV+NvlHtkGA0DGD6+++/X+saAAAAAABA+vzpT39644033G733r17L7300gceeOCTTz6ZPHnysGHDFixY8Mknn3i93tbW1jVr1rz55psnnXTSpZdeOnr0aIvF0tzcrHb493//d/XG888/X+lw9NFHdzra6aefrvWLRi/i3CUAAAAAAAY1WZZ9Pp/FYolpdzqdJpPJbDZ368Z4o2EA+1965ho/uMBXfQAAAABJRU5ErkJggg==
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSIgPz4KPG90cnNfY29uZmlnIHZlcnNpb249IjEuMCIgaW5pdD0iRnJhbWV3b3JrIj4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZyb250ZW5kOjpNb2R1bGUjIyNBZG1pbkltcG9ydEV4cG9ydCIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gVHJhbnNsYXRhYmxlPSIxIj5Gcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgYWdlbnQgaW50ZXJmYWNlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkltcG9ydEV4cG9ydDwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZG1pbjo6TW9kdWxlUmVnaXN0cmF0aW9uPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEZyb250ZW5kTW9kdWxlUmVnPgogICAgICAgICAgICAgICAgPEdyb3VwPmFkbWluPC9Hcm91cD4KICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbiBUcmFuc2xhdGFibGU9IjEiPkltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi48L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgPFRpdGxlPkltcG9ydC9FeHBvcnQ8L1RpdGxlPgogICAgICAgICAgICAgICAgPE5hdkJhck5hbWU+QWRtaW48L05hdkJhck5hbWU+CiAgICAgICAgICAgICAgICA8TmF2QmFyTW9kdWxlPgogICAgICAgICAgICAgICAgICAgIDxNb2R1bGU+S2VybmVsOjpPdXRwdXQ6OkhUTUw6Ok5hdkJhck1vZHVsZUFkbWluPC9Nb2R1bGU+CiAgICAgICAgICAgICAgICAgICAgPE5hbWUgVHJhbnNsYXRhYmxlPSIxIj5JbXBvcnQvRXhwb3J0PC9OYW1lPgogICAgICAgICAgICAgICAgICAgIDxCbG9jaz5TeXN0ZW08L0Jsb2NrPgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjcxMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyTW9kdWxlPgogICAgICAgICAgICAgICAgPExvYWRlcj4KICAgICAgICAgICAgICAgICAgICA8Q1NTPklUU00uSW1wb3J0RXhwb3J0LmNzczwvQ1NTPgogICAgICAgICAgICAgICAgPC9Mb2FkZXI+CiAgICAgICAgICAgIDwvRnJvbnRlbmRNb2R1bGVSZWc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iSW1wb3J0RXhwb3J0OjpGb3JtYXRCYWNrZW5kUmVnaXN0cmF0aW9uIyMjQ1NWIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBUcmFuc2xhdGFibGU9IjEiPkZvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+PC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+SW1wb3J0RXhwb3J0PC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Rm9ybWF0QmFja2VuZDo6TW9kdWxlUmVnaXN0cmF0aW9uPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik1vZHVsZSI+S2VybmVsOjpTeXN0ZW06OkltcG9ydEV4cG9ydDo6Rm9ybWF0QmFja2VuZDo6Q1NWPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJOYW1lIj5DU1Y8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+Cjwvb3Ryc19jb25maWc+Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9iZ19JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpiZ19JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTZW1pY29sb24gKDspJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0V4cG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTa2lwcGVkJ30gPSAnJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEZWxldGUgVGVtcGxhdGUnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICcnOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnJzsKCiAgICAjCiAgICAjIE9CU09MRVRFIEVOVFJJRVMgRk9SIFJFRkVSRU5DRSwgRE8gTk9UIFRSQU5TTEFURSEKICAgICMKCn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9jc19JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpjc19JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdOb3bDoSDFoWFibG9uYSB6b2JyYXplbsOtJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdabmFrb3bDoSBzYWRhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ0R2b2p0ZcSNa2EgKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ1Nsb3VwZWMnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uIFNlcGFyYXRvcid9ID0gJ09kZMSbbG92YcSNIFNsb3VwY8WvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICdUZcSNa2EgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdTdMWZZWRuw61rICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bMOhdG9yIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXhwb3J0ZWQgcmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICcnOwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdJbXBvcnQvRXhwb3J0IFNwcsOhdmEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdaYWjDoWppdCBJbXBvcnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAnWmFow6FqaXQgRXhwb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RlbGV0ZSBUZW1wbGF0ZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwJ30gPSAnS3Jvayc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJ0VkaXRhY2Ugb2JlY27DvWNoIGluZm9ybWFjw60nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnRWRpdGFjZSBpbmZvcm1hY8OtIG8gb2JqZWt0dSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJ0VkaXRhY2UgZm9ybcOhdHUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICdFZGl0YWNlIG1hcG92w6Fuw60nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnRWRpdGFjZSB2eWhsZWTDoXbDoW7DrSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ09tZXppdCBFeHBvcnQgdnlobGVkw6F2w6Fuw61tJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0luZm9ybWFjZSBvIEltcG9ydHUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdaZHJvam92w70gU291Ym9yJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJyc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICdJbXBvcnQvRXhwb3J0JzsKCiAgICAjCiAgICAjIE9CU09MRVRFIEVOVFJJRVMgRk9SIFJFRkVSRU5DRSwgRE8gTk9UIFRSQU5TTEFURSEKICAgICMKCn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9jdF9JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpjdF9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdBZmVnaXIgcGxhbnRpbGxhIGRlIG1hcGF0Z2UnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ0Nvbmp1bnQgZGUgY2Fyw6BjdGVycyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEb3MgcHVudHMgKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ0NvbHVtbmEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uIFNlcGFyYXRvcid9ID0gJ1NlcGFyYWRvciBkZSBjb2x1bW5hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICdQdW50ICguKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTZW1pY29sb24gKDspJ30gPSAnUHVudCBpIGNvbWEgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYWRvciAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAnR2VzdGnDsiBkZSBJbXBvcnRhci9FeHBvcnRhcic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgdGVtcGxhdGUnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdDb21lbsOnYXIgaW1wb3J0YWNpw7MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAnQ29tZW7Dp2FyIGV4cG9ydGFjacOzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RlbGV0ZSBUZW1wbGF0ZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwJ30gPSAnUGFzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnRWRpdGFyIGluZm9ybWFjacOzIGNvbXVuYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICdFZGl0YXIgaW5mb3JtYWNpw7MgZFwnb2JqZWN0ZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJ0VkaXRhciBpbmZvcm1hY2nDsyBkZSBmb3JtYXQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICdFZGl0YXIgaW5mb3JtYWNpw7MgZGUgbWFwYXRnZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICdFZGl0YXIgaW5mb3JtYWNpw7MgZGUgcmVjZXJjYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ1Jlc3RyaW5naXIgZXhwb3J0YWNpw7MgcGVyIHJlY2VyY2EnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW1wb3J0YXIgaW5mb3JtYWNpw7MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdGaXR4ZXIgZm9udCc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3J0YXIvRXhwb3J0YXInOwoKICAgICMKICAgICMgT0JTT0xFVEUgRU5UUklFUyBGT1IgUkVGRVJFTkNFLCBETyBOT1QgVFJBTlNMQVRFIQogICAgIwoKfQoKMTsK
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9kYV9JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpkYV9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdUaWxmw7hqIE1hcHBpbmctVGVtcGxhdGUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ1RlZ25zw6Z0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ0tvbG9uICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdLb2xvbm5lJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ1B1bmt0dW0gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdTZW1pa29sb24gKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYXRvciAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0V4cG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTa2lwcGVkJ30gPSAnJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAnSW1wb3J0L0VrcG9ydCBzdHlyaW5nJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAnU3RhcnQgaW1wb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ1N0YXJ0IGVrcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEZWxldGUgVGVtcGxhdGUnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCd9ID0gJ1RyaW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICdSZXQgZsOmbGxlcyBpbmZvcm1hdGlvbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICdSZXQgb2JqZWt0IGluZm9ybWF0aW9uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnUmV0IGZvcm1hdCBpbmZvcm1hdGlvbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eycgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ1JldCBtYXBwaW5nIGluZm9ybWF0aW9uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJ1JldCBzw7hnZWluZm9ybWF0aW9uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnQmVncsOmbnMgZWtwb3J0IHByLiBzw7hnbmluZyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICdJbXBvcnQgaW5mb3JtYXRpb24nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdLaWxkZSBmaWwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ0ltcG9ydC9Fa3BvcnQnOwoKICAgICMKICAgICMgT0JTT0xFVEUgRU5UUklFUyBGT1IgUkVGRVJFTkNFLCBETyBOT1QgVFJBTlNMQVRFIQogICAgIwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uIFNlcGVyYXRvcid9ID0gJ0tvbG9ubmUgYWRza2lsbGVsc2UnOwoKfQoKMTsK
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9kZV9JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpkZV9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdNYXBwaW5nLVRlbXBsYXRlIGhpbnp1ZsO8Z2VuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdaZWljaGVuc2F0eic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEb3BwZWxwdW5rdCAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnU3BhbHRlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdTcGFsdGVudHJlbm5lcic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUHVua3QgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdTZW1pY29sb24gKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYXRvciAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnTWl0IFNwYWx0ZW7DvGJlcnNjaHJpZnRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgc3VtbWFyeSBmb3InfSA9ICdJbXBvcnQtQmVyaWNodCBmw7xyJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydGVkIHJlY29yZHMnfSA9ICdJbXBvcnRpZXJ0ZSBEYXRlbnPDpHR6ZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFeHBvcnRlZCByZWNvcmRzJ30gPSAnRXhwb3J0aWVydGUgRGF0ZW5zw6R0emUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ0RhdGVuc8OkdHplJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICfDnGJlcnNwcnVuZ2VuJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAnSW1wb3J0L0V4cG9ydC1WZXJ3YWx0dW5nJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdFcnN0ZWxsZW4gZWluZXIgVm9ybGFnZSB6dW0gSW1wb3J0aWVyZW4gdW5kIEV4cG9ydGllcmVuIHZvbiBPYmpla3QtSW5mb3JtYXRpb25lbi4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAnSW1wb3J0IHN0YXJ0ZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAnRXhwb3J0IHN0YXJ0ZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRGVsZXRlIFRlbXBsYXRlJ30gPSAnVGVtcGxhdGUgbMO2c2NoZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCd9ID0gJ1NjaHJpdHQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICdBbGxnZW1laW5lIEluZm9ybWF0aW9uZW4gYmVhcmJlaXRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnT2JqZWt0IGlzdCBlcmZvcmRlcmxpY2ghJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICdGb3JtYXQgaXN0IGVyZm9yZGVybGljaCEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICdPYmpla3QtSW5mb3JtYXRpb25lbiBiZWFyYmVpdGVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnRm9ybWF0LUluZm9ybWF0aW9uZW4gYmVhcmJlaXRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eycgaXMgcmVxdWlyZWQhJ30gPSAnIHdpcmQgYmVuw7Z0aWd0ISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICdNYXBwaW5nLUluZm9ybWF0aW9uZW4gYmVhcmJlaXRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnS2VpbmUgTWFwcGluZy1FbGVtZW50ZSBnZWZ1bmRlbi4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ01hcHBpbmctRWxlbWVudCBoaW56dWbDvGdlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJ1N1Y2gtSW5mb3JtYXRpb25lbiBiZWFyYmVpdGVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnRXhwb3J0IHBlciBTdWNoZSBlaW5zY2hyw6Rua2VuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0ltcG9ydC1JbmZvcm1hdGlvbmVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnUXVlbGwtRGF0ZWknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJ0VyZm9sZ3JlaWNoJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJ05pY2h0IGVyZm9sZ3JlaWNoJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ0RvcHBlbHRlIE5hbWVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnWnVsZXR6dCB2ZXJhcmJlaXRldGUgWmVpbGUgZGVyIEltcG9ydC1EYXRlaSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJ09rJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnRm9ybWF0LUJhY2tlbmQgTW9kdWwtUmVnaXN0cmF0aW9uIGRlcyBJbXBvcnQvRXhwb3J0IE1vZHVscy4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0ltcG90aWVyZW4gdW5kIEV4cG9ydGllcmVuIHZvbiBPYmpla3QtSW5mb3JtYXRpb25lbi4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ0ltcG9ydC9FeHBvcnQnOwoKICAgICMKICAgICMgT0JTT0xFVEUgRU5UUklFUyBGT1IgUkVGRVJFTkNFLCBETyBOT1QgVFJBTlNMQVRFIQogICAgIwoKfQoKMTsK
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9lc19JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjplc19JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdBw7FhZGlyIHBsYW50aWxsYSBkZSBtYXBlbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAnSnVlZ28gZGUgY2FyYWN0ZXJlcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEb3MgcHVudG9zICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdDb2x1bW5hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdTZXBhcmFkb3IgZGUgQ29sdW1uYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUHVudG8gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdQdW50byB5IENvbWEgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYWRvciAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0V4cG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTa2lwcGVkJ30gPSAnJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAnR2VzdGnDs24gZGUgSW1wb3J0YWNpw7NuL0V4cG9ydGFjacOzbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ0luaWNpYXIgSW1wb3J0YWNpw7NuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ0luaWNpYXIgRXhwb3J0YWNpw7NuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RlbGV0ZSBUZW1wbGF0ZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwJ30gPSAnUGFzbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJ0VkaXRhciBpbmZvcm1hY2nDs24gY29tw7puJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICfCoURlYmUgZXNwZWNpZmljYXIgT2JqZXRvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnwqFEZWJlIGVzcGVjaWZpY2FyIEZvcm1hdG8hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnRWRpdGFyIGluZm9ybWFjacOzbiBkZSBvYmpldG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICdFZGl0YXIgaW5mb3JtYWNpw7NuIGRlbCBmb3JtYXRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57JyBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gPSAnRWRpdGFyIGluZm9ybWFjacOzbiBkZSBtYXBlbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnTm8gc2UgZW5jb250cmFyb24gZWxlbWVudG9zIGRlIG1hcGVvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnRWRpdGFyIGluZm9ybWFjacOzbiBkZSBiw7pzcXVlZGEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdSZXN0cmluZ2lyIGV4cG9ydGFjacOzbiBwb3IgYsO6c3F1ZWRhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0ltcG9ydGFyIGluZm9ybWFjacOzbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ0FyY2hpdm8gb3JpZ2VuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJyc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ1JlZ2lzdHJvIGRlIG3Ds2R1bG8gZGUgZm9ybWF0byBiYWNrZW5kIHBhcmEgZWwgbcOzZHVsbyBpbXBvcnQvZXhwb3J0Lic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnSW1wb3J0YXIgeSBleHBvcnRhciBpbmZvcm1hY2nDs24gZGUgb2JqZXRvcy4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ0ltcG9ydGFyL0V4cG9ydGFyJzsKCiAgICAjCiAgICAjIE9CU09MRVRFIEVOVFJJRVMgRk9SIFJFRkVSRU5DRSwgRE8gTk9UIFRSQU5TTEFURSEKICAgICMKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NsYXNzIGlzIHJlcXVpcmVkISd9ID0gJ8KhRGViZSBlc3BlY2lmaWNhciBDbGFzZSEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uIFNlcGFyYXRvciBpcyByZXF1aXJlZCEnfSA9ICfCoURlYmUgZXNwZWNpZmljYXIgU2VwYXJhZG9yIGRlIENvbHVtbmEhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbW1hICgsKSd9ID0gJ0NvbWEgKCwpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIGluIG9yZGVyIHRvIGNhbiBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPQogICAgICAgICdBZ3JlZ3VlIHVuYSBwbGFudGlsbGEgbnVldmEgcGFyYSBwb2RlciBpbXBvcnRhciB5IGV4cG9ydGFyLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbXB0eSBmaWVsZHMgaW5kaWNhdGUgdGhhdCB0aGUgY3VycmVudCB2YWx1ZXMgYXJlIGtlcHQnfSA9ICdMb3MgY2FtcG9zIHZhY8Otb3MgaW5kaWNhbiBxdWUgbG9zIHZhbG9yZXMgYWN0dWFsZXMgc2UgbWFudGllbmVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0dvIGJhY2snfSA9ICdSZWdyZXNhcic7Cgp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9mYV9JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpmYV9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICfYp9i22KfZgdmHINqp2LHYr9mGINuM2qkg2YLYp9mE2Kgg2Ybar9in2LTYqic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAn2qnYr9io2YbYr9uMINin2LfZhNin2LnYp9iqJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ9iv2YjZhtmC2LfZhyAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAn2LPYqtmI2YYnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uIFNlcGFyYXRvcid9ID0gJ9is2K/Yp9qp2YbZhtiv2Ycg2LPYqtmI2YbigIzZh9inJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICfZhtmC2LfZhyAoLiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ9iz2YXbjCDaqdin2YTZhiAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGFidWxhdG9yIChUQUIpJ30gPSAn2KzYr9mI2YQg2LPYp9iyIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXhwb3J0ZWQgcmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICcnOwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICfZhdiv24zYsduM2Kog2YjYsdmI2K8v2LXYr9mI2LEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ9iz2KfYrtiqINmC2KfZhNio24wg2KjYsdin24wg2YjYsdmI2K8g2Ygg2LXYr9mI2LEg2KfYt9mE2KfYudin2Kog2KLYqNis2qnYqic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICfYtNix2YjYuSDYudmF2YTbjNin2Kog2YjYsdmI2K8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAn2LTYsdmI2Lkg2LnZhdmE24zYp9iqINi12K/ZiNixJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RlbGV0ZSBUZW1wbGF0ZSd9ID0gJ9it2LDZgSDZgtin2YTYqCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwJ30gPSAn2q/Yp9mFJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAn2YjbjNix2KfbjNi0INin2LfZhNin2LnYp9iqINi52YXZiNmF24wnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ9ii2KjYrNqp2Kog2YXZiNix2K8g2YbbjNin2LIg2KfYs9iqISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAn2YLYp9mE2KjigIzYqNmG2K/bjCDZhdmI2LHYryDZhtuM2KfYsiDYp9iz2KohJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAn2YjbjNix2KfbjNi0INin2LfZhNin2LnYp9iqINii2KjYrNqp2KrbjCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJ9mI24zYsdin24zYtCDYp9i32YTYp9i52KfYqiDZgtin2YTYqOKAjNio2YbYr9uMJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57JyBpcyByZXF1aXJlZCEnfSA9ICcg2YXZiNix2K8g2YbbjNin2LIg2KfYs9iqISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICfZiNuM2LHYp9uM2LQg2KfYt9mE2KfYudin2Kog2Ybar9in2LTYqic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAn2YfbjNqGINi52YbYtdixINmG2q/Yp9i02KrbjCDbjNin2YHYqiDZhti02K8uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICfYp9mB2LLZiNiv2YYg2LnZhti12LEg2Ybar9in2LTYqic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJ9mI24zYsdin24zYtCDYp9i32YTYp9i52KfYqiDYrNiz2KrYrNmIJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAn2YXYrdiv2YjYr9iz2KfYstuMINi52YXZhNuM2KfYqiDYtdiv2YjYsSDYqNmHINin2LLYp9uMINis2LPYqtis2YgnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAn2YjYsdmI2K8g2KfYt9mE2KfYudin2KonOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICfZgdin24zZhCDZhdmG2KjYuSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICcnOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICfZgtin2YTYqOKAjNio2YbYr9uMINir2KjYqiDZhdin2pjZiNmEINio2LHYp9uMINmF2KfamNmI2YQg2YjYsdmI2K8v2LXYr9mI2LEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ9mI2LHZiNivINmIINi12K/ZiNixINin2LfZhNin2LnYp9iqINii2KjYrNqp2KonOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ9mI2LHZiNivL9i12K/ZiNixJzsKCiAgICAjCiAgICAjIE9CU09MRVRFIEVOVFJJRVMgRk9SIFJFRkVSRU5DRSwgRE8gTk9UIFRSQU5TTEFURSEKICAgICMKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RlbXBsYXRlIExpc3QnfSA9ICfZgdmH2LHYs9iqINmC2KfZhNio4oCM2YfYpyc7Cgp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9mcl9JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpmcl9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdBam91dGVyIHVuIHRlbXBsYXRlIGRlIG1hcHBhZ2UnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ0pldSBkZSBjYXJhY3TDqHJlcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEZXV4IHBvaW50cyAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnQ29sb25uZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnU8OpcGFyYXRldXIgZGUgY29sb25uZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUG9pbnQgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdQb2ludCB2aXJndWxlICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bGF0aW9uIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXhwb3J0ZWQgcmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICcnOwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdHZXN0aW9uIGRlIGxcJ0ltcG9ydC9FeHBvcnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdEw6ltYXJyZXIgSW1wb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ0TDqW1hcnJlciBFeHBvcnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRGVsZXRlIFRlbXBsYXRlJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAnfSA9ICdFdGFwZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJ0VkaXRlciBsZXMgaW5mb3JtYXRpb25zIGNvbW11bmVzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJ0VkaXRlciBsZXMgaW5mb3JtYXRpb25zIGRlIGxcJ29iamV0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnRWRpdGVyIGxlcyBpbmZvcm1hdGlvbnMgZGUgZm9ybWF0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57JyBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gPSAnRWRpdGVyIGxlcyBpbmZvcm1hdGlvbnMgZGUgbWFwcGFnZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICdFZGl0ZXIgbGVzIGluZm9ybWF0aW9ucyBkZSByZWNoZXJjaGUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdSZXN0cmVpbmRyZSBsXCdleHBvcnQgcGFyIHJlY2hlcmNoZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICdJbmZvcm1hdGlvbnMgZFwnaW1wb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnRmljaGllciBTb3VyY2UnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ0ltcG9ydGVyL0V4cG9ydGVyJzsKCiAgICAjCiAgICAjIE9CU09MRVRFIEVOVFJJRVMgRk9SIFJFRkVSRU5DRSwgRE8gTk9UIFRSQU5TTEFURSEKICAgICMKCn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9pdF9JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjppdF9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdBZ2dpdW5naSBtYXBwYXR1cmEgbW9kZWxsbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAnQ2hhcnNldCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEdWUgcHVudGkgKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ0NvbG9ubmEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uIFNlcGFyYXRvcid9ID0gJ1NlcGFyYXRvcmUgZGkgY29sb25uYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUHVudG8gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdQdW50byBlIHZpcmdvbGEgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYXRvcmUgKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW5jbHVkZSBDb2x1bW4gSGVhZGVycyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgc3VtbWFyeSBmb3InfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0ZWQgcmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFeHBvcnRlZCByZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2tpcHBlZCd9ID0gJyc7CgogICAgIyBUZW1wbGF0ZTogQWRtaW5JbXBvcnRFeHBvcnQKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQgTWFuYWdlbWVudCd9ID0gJ0dlc3Rpb25lIEltcG9ydGF6aW9uZS9Fc3BvcnRhemlvbmUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdJbml6aWFyZSBJbXBvcnRhemlvbmUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAnSW5pemlhcmUgRXNwb3J0YXppb25lJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RlbGV0ZSBUZW1wbGF0ZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwJ30gPSAnUGFzc28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICdNb2RpZmljYSBpbmZvcm1hemlvbmkgY29tdW5pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJ01vZGlmaWNhIGluZm9ybWF6aW9uaSBvZ2dldHRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnTW9kaWZpY2EgZm9ybWF0byBpbmZvcm1hemlvbmUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICdNb2RpZmljYSBtYXBwYXR1cmEgaW5mb3JtYXppb25pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJ01vZGlmaWNhIGluZm9ybWF6aW9uaSBkaSByaWNlcmNhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnUmVzdHJpbmdlcmUgZXNwb3J0YXppb25lIHBlciByaWNlcmNhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0ltcG9ydGFyZSBpbmZvcm1hemlvbmUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdBcmNoaXZpbyBvcmlnaW5lJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJyc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICdJbXBvcnRhcmUvRXNwb3J0YXJlJzsKCiAgICAjCiAgICAjIE9CU09MRVRFIEVOVFJJRVMgRk9SIFJFRkVSRU5DRSwgRE8gTk9UIFRSQU5TTEFURSEKICAgICMKCn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9qYV9JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpqYV9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTZW1pY29sb24gKDspJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0V4cG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTa2lwcGVkJ30gPSAnJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEZWxldGUgVGVtcGxhdGUnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICcnOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnJzsKCiAgICAjCiAgICAjIE9CU09MRVRFIEVOVFJJRVMgRk9SIFJFRkVSRU5DRSwgRE8gTk9UIFRSQU5TTEFURSEKICAgICMKCn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9tc19JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjptc19JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdUYW1iYWggdGVtcGxhdCBwZXRhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdTZXQga2FyYWt0ZXInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sb24gKDopJ30gPSAnVGl0aWsgYmVydGluZGloICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdLb2x1bSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnS29sdW0gcGVtaXNhaCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnVGl0aWsgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdTZW1pa29sb24gKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYXRvciAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnVGVybWFzdWsgS29sdW0gS2VwYWxhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0V4cG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTa2lwcGVkJ30gPSAnJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAnUGVuZ3VydXNhbiBJbXBvcnQvRWtzcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnQ2lwdGEgdGVtcGxhdCB1bnR1ayBpbXBvcnQgZGFuIGVrc3BvcnQgaW5mb3JtYXNpIG9iamVrLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdNdWxhIGltcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdNdWxhIGVrc3BvcnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRGVsZXRlIFRlbXBsYXRlJ30gPSAnUGFkYW0gdGVtcGxhdCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwJ30gPSAnTGFuZ2thaCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJ0F1ZGl0IGluZm9ybWFzaSBiaWFzYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnT2JqZWsgYWRhbGFoIGRpcGVybHVrYW4hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICdGb3JtYXQgYWRhbGFoIGRpcGVybHVrYW4hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnQXVkaXQgaW5mb3JtYXNpIG9iamVrJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnQXVkaXQgaW5mb3JtYXNpIGZvcm1hdCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eycgaXMgcmVxdWlyZWQhJ30gPSAnIGFkYWxhaCBkaXBlcmx1a2FuISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICdBdWRpdCBpbmZvcm1hc2kgcGV0YSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnVGlhZGEgZWxlbWVuIHBldGEgZGl0ZW11aS4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ1RhbWJhaCBlbGVtZW4gcGV0YSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJ0F1ZGl0IGluZm9ybWFzaSBjYXJpYW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdEaWxhcmFuZyBla3Nwb3J0IHBlciBjYXJpYW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW5mb3JtYXNpIGltcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ1N1bWJlciBmYWlsJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJyc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ0Zvcm1hdCBiYWNrZW5kIG1vZHVsIHBlbmRhZnRhcmFuIHVudHVrIG1vZHVsIGltcG9ydC9la3Nwb3J0Lic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnSW5mb3JtYXNpIG9iamVrIGltcG9ydCBkYW4gZWtzcG9ydC4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ0ltcG9ydC9Fa3Nwb3J0JzsKCiAgICAjCiAgICAjIE9CU09MRVRFIEVOVFJJRVMgRk9SIFJFRkVSRU5DRSwgRE8gTk9UIFRSQU5TTEFURSEKICAgICMKCn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9uYl9OT19JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpuYl9OT19JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdMZWdnIHRpbCBtYWwgZm9yIG1hcHBpbmcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ1RlZ25zZXR0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ0tvbG9uICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdLb2xvbm5lJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdLb2xvbm5lc2VwYXJhdG9yJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICdQdW5rdHVtICguKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTZW1pY29sb24gKDspJ30gPSAnU2VtaWtvbG9uICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bGF0b3IgKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW5jbHVkZSBDb2x1bW4gSGVhZGVycyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgc3VtbWFyeSBmb3InfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0ZWQgcmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFeHBvcnRlZCByZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2tpcHBlZCd9ID0gJyc7CgogICAgIyBUZW1wbGF0ZTogQWRtaW5JbXBvcnRFeHBvcnQKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQgTWFuYWdlbWVudCd9ID0gJ0FkbWluaXN0cmFzam9uIGF2IEltcG9ydC9Fa3Nwb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdPcHByZXR0IGVuIG1hbCBmb3Igw6UgZWtzcG9ydGVyZSBvZyBpbXBvcnRlcmUgaW5mb3JtYXNqb24nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAnU3RhcnQgaW1wb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ1N0YXJ0IGVrc3BvcnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRGVsZXRlIFRlbXBsYXRlJ30gPSAnU2xldHQgbWFsJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAnfSA9ICdTdGVnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnRW5kcmUgZmVsbGVzaW5mb3JtYXNqb24nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ09iamVrdCBlciBww6VrcmV2ZCEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJ0Zvcm1hdCBlciBww6VrcmV2ZCEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICdFbmRyZSBvYmpla3RldCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJ0VuZHJlIGZvcm1hdGV0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57JyBpcyByZXF1aXJlZCEnfSA9ICcgZXIgcMOla3JldmQhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ0VuZHJlIG1hcHBpbmcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJ0luZ2VuIGVsZW1lbnRlciBmdW5uZXQuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICdMZWdnIHRpbCBtYXBwaW5nLWVsZW1lbnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICdFbmRyZSBzw7hrJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnQmVncmVucyBla3Nwb3J0IHBlciBzw7hrJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0ltcG9ydC1pbmZvcm1hc2pvbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ0tpbGRlZmlsJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJyc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ0Jha3NpZGVtb2R1bC1yZWdpc3RyZXJpbmcgZm9yIGZvcm1hdGV0IHRpbCBpbXBvcnQvZWtzcG9ydC1tb2R1bGVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdJbmZvcm1hc2pvbiBmb3IgaW1wb3J0LSBvZyBla3Nwb3J0LW9iamVrdCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3J0L0Vrc3BvcnQnOwoKICAgICMKICAgICMgT0JTT0xFVEUgRU5UUklFUyBGT1IgUkVGRVJFTkNFLCBETyBOT1QgVFJBTlNMQVRFIQogICAgIwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uIFNlcGVyYXRvcid9ID0gJ0tvbG9ubmUgYWRza2lsbGVsc2UnOwoKfQoKMTsK
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9ubF9JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpubF9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdNYXBwaW5ndGVtcGxhdGUgdG9ldm9lZ2VuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdLYXJha3RlcnNldCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEdWJiZWxlIHB1bnQgKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ0tvbG9tJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdLb2xvbXNjaGVpZGluZ3N0ZWtlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUHVudCAoLiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ1B1bnRrb21tYSAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGFidWxhdG9yIChUQUIpJ30gPSAnVGFiJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXhwb3J0ZWQgcmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICcnOwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdJbXBvcnQvRXhwb3J0IGJlaGVlcic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ0ltcG9ydCBzdGFydGVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ0V4cG9ydCBzdGFydGVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RlbGV0ZSBUZW1wbGF0ZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwJ30gPSAnU3RhcCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJ0FsZ2VtZW5lIGluZm9ybWF0aWUgYmV3ZXJrZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ09iamVjdCBpcyB2ZXJwbGljaHQuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICdGb3JtYWF0IGlzIHZlcnBsaWNodC4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICdPYmplY3QtaW5mb3JtYXRpZSBiZXdlcmtlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJ0Zvcm1hdC1pbmZvcm1hdGlvbmVuIGJld2Vya2VuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57JyBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gPSAnTWFwcGluZy1pbmZvcm1hdGllIGJld2Vya2VuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICdHZWVuIGVsZW1lbnRlbiBnZXZvbmRlbi4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJ1pvZWstaW5mb3JtYXRpZSBiZXdlcmtlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ0JlcGVyayBleHBvcnQgdG90IHpvZWtvcGRyYWNodCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICdJbXBvcnQtaW5mb3JtYXRpZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ0Jyb25iZXN0YW5kJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJyc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnSW1wb3J0IGVuIGV4cG9ydCBvYmplY3RpbmZvcm1hdGllJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICdJbXBvcnQvRXhwb3J0JzsKCiAgICAjCiAgICAjIE9CU09MRVRFIEVOVFJJRVMgRk9SIFJFRkVSRU5DRSwgRE8gTk9UIFRSQU5TTEFURSEKICAgICMKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NsYXNzIGlzIHJlcXVpcmVkISd9ID0gJ0tsYXNzZSBpcyB2ZXJwbGljaHQuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3IgaXMgcmVxdWlyZWQhJ30gPSAnU2NoZWlkaW5nc3Rla2VuIGlzIHZlcnBsaWNodCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSBpbiBvcmRlciB0byBjYW4gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0KICAgICAgICAnTWFhayBlZW4gdGVtcGxhdGUgYWFuIG9tIG9iamVjdGVuIHRlIGltcG9ydGVyZW4gb2YgZXhwb3J0ZXJlbi4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRW1wdHkgZmllbGRzIGluZGljYXRlIHRoYXQgdGhlIGN1cnJlbnQgdmFsdWVzIGFyZSBrZXB0J30gPSAnQmlqIGxlZ2UgdmVsZGVuIHdvcmR0IGRlIGh1aWRpZ2Ugd2FhcmRlIGJlaG91ZGVuJzsKCn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9wbF9JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpwbF9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdEb2RhaiBzemFibG9uIG1hcG93YW5pYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAnS29kb3dhbmllJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ0R3dWtyb3BlayAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnS29sdW1uYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnU2VwYXJhdG9yIGtvbHVtbnknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ0tyb3BrYSAoLiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ8WacmVkbmlrICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bGF0b3IgKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW5jbHVkZSBDb2x1bW4gSGVhZGVycyd9ID0gJ1VtaWXFm8SHIG5hZ8WCw7N3a2kga29sdW1uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ1BvZHN1bW93YW5pZSBpbXBvcnR1IGRsYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnWmFpbXBvcnRvd2FuZSByZWtvcmR5JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0V4cG9ydGVkIHJlY29yZHMnfSA9ICdXeWVrc3BvcnRvd2FuZSByZWtvcmR5JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlY29yZHMnfSA9ICdSZWtvcmR5JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICdQb21pbmnEmXRlJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAnWmFyesSFZHphbmllIEltcG9ydGVtL0V4cG9ydGVtJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdVdHfDs3J6IHN6YWJsb24gZG8gaW1wb3J0dSBpIGVrc3BvcnR1IGRhbnljaCBvYmlla3TDs3cuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ1JvenBvY3puaWogaW1wb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ1JvenBvY3puaWogZWtzcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEZWxldGUgVGVtcGxhdGUnfSA9ICdVc3XFhCBzemFibG9uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAnfSA9ICdLcm9rJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnRWR5dHVqIGluZm9ybWFjamUgd3Nww7NsbmUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ09iaWVrdCBqZXN0IHd5bWFnYW55ISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnRm9ybWF0IGplc3Qgd3ltYWdhbnkhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnRWR5dHVqIGluZm9ybWFjamUgb2JpZWt0dSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJ0VkeXR1aiBmb3JtYXQgb2JpZWt0dSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eycgaXMgcmVxdWlyZWQhJ30gPSAnIGplc3Qgd3ltYWdhbmUhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ0VkeXR1aiBpbmZvcm1hY2plIG1hcG93YW5pYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnTmllIHpuYWxlemlvbm8gZWxlbWVudMOzdyBtYXB5Lic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnRG9kYWogZWxlbWVudCBtYXBvd2FuaWEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICdFZHl0dWogaW5mb3JtYWNqZSB3eXN6dWtpd2FuaWEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdPZ3JhbmljeiBla3Nwb3J0IHByemV6IHd5c3p1a2FuaWUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW1wb3J0dWogaW5mb3JtYWNqZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ1BsaWsgxbpyw7NkxYJvd3knOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJ1Bvd29kemVuaWUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnTmllcG93b2R6ZW5pZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICdEdXBsaWthdHkgbmF6dyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJ05yIG9zdGFuaWVqIHByemV0d29yem9uZWogbGluaWkgcGxpa3UgaW1wb3J0b3dlZ28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPayc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ01vZHXFgiBmb3JtYXRvd2FuaWEgYmFja2VuZCBkbGEgbW9kdcWCdSBpbXBvcnQvZWtzcG9ydC4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0ltcG9ydHVqIGkgZWtzcG9ydHVqIGluZm9ybWFjamUgb2JpZWt0w7N3Lic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3J0L2Vrc3BvcnQnOwoKICAgICMKICAgICMgT0JTT0xFVEUgRU5UUklFUyBGT1IgUkVGRVJFTkNFLCBETyBOT1QgVFJBTlNMQVRFIQogICAgIwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2xhc3MnfSA9ICdLbGFzYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb25maWcgSXRlbSd9ID0gJ0NJJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VsZW1lbnQgcmVxdWlyZWQsIHBsZWFzZSBpbnNlcnQgZGF0YSd9ID0gJ0VsZW1lbnQgd3ltYWdhbnksIHBvZGFqIHdhcnRvxZvEhyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbXB0eSBmaWVsZHMgaW5kaWNhdGUgdGhhdCB0aGUgY3VycmVudCB2YWx1ZXMgYXJlIGtlcHQnfSA9ICdQdXN0ZSBwb2xhIG96bmFjemFqxIUgemFjaG93YW5pZSBiaWXFvMSFY2VqIHdhcnRvxZtjaSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydNYXhpbXVtIG51bWJlciBvZiBvbmUgZWxlbWVudCd9ID0gJ01ha3N5bWFsbmEgbGljemJhIHd5c3TEhXBpZcWEIHBvamVkeW5jemVnbyBlbGVtZW50dSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAnTGlzdGEgc3phYmxvbsOzdyc7Cgp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9wdF9CUl9JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpwdF9CUl9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdBZGljaW9uYXIgbW9kZWxvIGRlIG1hcGVhbWVudG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ0NvZGlmaWNhw6fDo28gZGUgQ2FyYWN0ZXJlcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEb2lzIFBvbnRvcyAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnQ29sdW5hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdTZXBhcmFkb3IgZGUgQ29sdW5hcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUG9udG8gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdQb250byBlIFbDrXJndWxhICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bGHDp8OjbyAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0V4cG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTa2lwcGVkJ30gPSAnJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAnR2VyZW5jaWFtZW50byBkZSBJbXBvcnRhw6fDo28vRXhwb3J0YcOnw6NvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdDcmlhciB1bSBtb2RlbG8gcGFyYSBpbXBvcnRhciBlIGV4cG9ydGFyIGluZm9ybWHDp8O1ZXMgZGUgb2JqZXRvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdJbmljaWFyIEltcG9ydGHDp8Ojbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdJbmljaWFyIEV4cG9ydGHDp8Ojbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEZWxldGUgVGVtcGxhdGUnfSA9ICdFeGNsdWlyIE1vZGVsbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwJ30gPSAnUGFzc28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICdFZGl0YXIgaW5mb3JtYcOnw7VlcyBjb211bnMnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ09iamV0byDDqSBuZWNlc3PDoXJpbyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJ08gZm9ybWF0byDDqSBuZWNlc3PDoXJpbyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICdFZGl0YXIgaW5mb3JtYcOnw7VlcyBkbyBvYmpldG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICdFZGl0YXIgaW5mb3JtYcOnw7VlcyBkbyBmb3JtYXRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57JyBpcyByZXF1aXJlZCEnfSA9ICcgw6kgbmVjZXNzw6FyaW8hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ0VkaXRhciBpbmZvcm1hw6fDtWVzIGRvIG1hcGVhbWVudG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJ07Do28gaMOhIGVsZW1lbnRvcyBtYXBhIGVuY29udHJhZG8uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICdBZGljaW9uYXIgZWxlbWVudG8gZGUgbWFwZWFtZW50byc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJ0VkaXRhciBpbmZvcm1hw6fDtWVzIGRlIHBlc3F1aXNhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnUmVzdHJpbmdpciBleHBvcnRhw6fDo28gcG9yIHBlc3F1aXNhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0luZm9ybWHDp8O1ZXMgZGUgaW1wb3J0YcOnw6NvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnQXJxdWl2byBkZSBPcmlnZW0nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnRm9ybWF0byBkZSByZWdpc3RybyBiYWNrZW5kIGRvIG3Ds2R1bG8gZGUgaW1wb3J0YcOnw6NvIC8gZXhwb3J0YcOnw6NvIG3Ds2R1bG8uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdJbXBvcnRhciBlIGV4cG9ydGFyIGluZm9ybWHDp8O1ZXMgZGUgb2JqZXRvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3J0YcOnw6NvL0V4cG9ydGHDp8Ojbyc7CgogICAgIwogICAgIyBPQlNPTEVURSBFTlRSSUVTIEZPUiBSRUZFUkVOQ0UsIERPIE5PVCBUUkFOU0xBVEUhCiAgICAjCgp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9wdF9QVF9JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIENvcHlyaWdodCAoQykgMjAxMiBGQ0NOIC0gUnVpIEZyYW5jaXNjbyA8cnVpLmZyYW5jaXNjb0BmY2NuLnB0PgojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpwdF9QVF9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdBZGljaW9uYXIgbW9kZWxvIGRlIG1hcGVhbWVudG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ0NvZGlmaWNhw6fDo28gZGUgQ2FyYWN0ZXJlcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEb2lzIFBvbnRvcyAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnQ29sdW5hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdTZXBhcmFkb3IgZGUgQ29sdW5hcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUG9udG8gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdQb250byBlIFbDrXJndWxhICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bGHDp8OjbyAoVEFCKSc7CgogICAgIyBUZW1wbGF0ZTogQWRtaW5JbXBvcnRFeHBvcnQKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQgTWFuYWdlbWVudCd9ID0gJ0dlc3TDo28gZGUgSW1wb3J0YcOnw6NvL0V4cG9ydGHDp8Ojbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgdGVtcGxhdGUnfSA9ICdBZGljaW9uYXIgbW9kZWxvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdDcmlhciB1bSBtb2RlbG8gcGFyYSBpbXBvcnRhciBlIGV4cG9ydGFyIGluZm9ybWHDp8O1ZXMgZGUgb2JqZXRvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdJbmljaWFyIEltcG9ydGHDp8Ojbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdJbmljaWFyIEV4cG9ydGHDp8Ojbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEZWxldGUgVGVtcGxhdGUnfSA9ICdBcGFnYXIgTW9kZWxvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAnfSA9ICdQYXNzbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJ0VkaXRhciBpbmZvcm1hw6fDtWVzIGNvbXVucyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnT2JqZXRvIMOpIG5lY2Vzc8OhcmlvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnTyBmb3JtYXRvIMOpIG5lY2Vzc8OhcmlvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJ0VkaXRhciBpbmZvcm1hw6fDtWVzIGRvIG9iamV0byc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJ0VkaXRhciBpbmZvcm1hw6fDtWVzIGRvIGZvcm1hdG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnIGlzIHJlcXVpcmVkISd9ID0gJyDDqSBuZWNlc3PDoXJpbyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gPSAnRWRpdGFyIGluZm9ybWHDp8O1ZXMgZG8gbWFwZWFtZW50byc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnTsOjbyBmb3JhbSBlbmNvbnRyYWRvcyBlbGVtZW50b3MgbWFwYS4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ0FkaWNpb25hciBlbGVtZW50byBkZSBtYXBlYW1lbnRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnRWRpdGFyIGluZm9ybWHDp8O1ZXMgZGUgcGVzcXVpc2EnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdSZXN0cmluZ2lyIGV4cG9ydGHDp8OjbyBwb3IgcGVzcXVpc2EnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW5mb3JtYcOnw7VlcyBkZSBpbXBvcnRhw6fDo28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdGaWNoZWlybyBmb250ZSc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9ICdGb3JtYXRvIGRlIHJlZ2lzdHJvIGJhY2tlbmQgZG8gbcOzZHVsbyBkZSBpbXBvcnRhw6fDo28gLyBleHBvcnRhw6fDo28gbcOzZHVsby4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0ltcG9ydGFyIGUgZXhwb3J0YXIgaW5mb3JtYcOnw7VlcyBkZSBvYmpldG8uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICdJbXBvcnRhw6fDo28vRXhwb3J0YcOnw6NvJzsKCiAgICAjCiAgICAjIE9CU09MRVRFIEVOVFJJRVMgRk9SIFJFRkVSRU5DRSwgRE8gTk9UIFRSQU5TTEFURSEKICAgICMKCn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9ydV9JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpydV9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICfQlNC+0LHQsNCy0LvQtdC90LjQtSDRiNCw0LHQu9C+0L3QsCDRgdC+0L7RgtCy0LXRgtGB0YLQstC40Y8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ9Ca0L7QtNC40YDQvtCy0LrQsCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICfQlNCy0L7QtdGC0L7Rh9C40LUgKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ9Ch0YLQvtC70LHQtdGGJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICfQoNCw0LfQtNC10LvQuNGC0LXQu9GMJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICfQotC+0YfQutCwICguKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTZW1pY29sb24gKDspJ30gPSAn0KLQvtGH0LrQsCDRgSDQt9Cw0L/Rj9GC0L7QuSAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGFidWxhdG9yIChUQUIpJ30gPSAn0KLQsNCx0YPQu9GP0YbQuNGPIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXhwb3J0ZWQgcmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICcnOwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICfQo9C/0YDQsNCy0LvQtdC90LjQtSDQmNC80L/QvtGA0YLQvtC8L9Ct0LrRgdC/0L7RgNGC0L7QvCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ9Cd0LDRh9Cw0YLRjCDQuNC80L/QvtGA0YInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAn0J3QsNGH0LDRgtGMINGN0LrRgdC/0L7RgNGCJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RlbGV0ZSBUZW1wbGF0ZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwJ30gPSAn0KjQsNCzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAn0KDQtdC00LDQutGC0LjRgNC+0LLQsNGC0Ywg0L7QsdGJ0YPRjiDQuNC90YTQvtGA0LzQsNGG0LjRjic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICfQoNC10LTQsNC60YLQuNGA0L7QstCw0YLRjCDQuNC90YTQvtGA0LzQsNGG0LjRjiDQvtCxINC+0LHRitC10LrRgtC1JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAn0KDQtdC00LDQutGC0LjRgNC+0LLQsNGC0Ywg0YTQvtGA0LzQsNGCINC00LDQvdC90YvRhSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eycgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ9Cg0LXQtNCw0LrRgtC40YDQvtCy0LDRgtGMINC40L3RhNC+0YDQvNCw0YbQuNGOINGB0L7QvtGC0LLQtdGC0YHRgtCy0LjRjyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICfQoNC10LTQsNC60YLQuNGA0L7QstCw0YLRjCDQv9C+0LjRgdC60L7QstGD0Y4g0LjQvdGE0L7RgNC80LDRhtC40Y4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICfQntCz0YDQsNC90LjRh9C40YLRjCDRjdC60YHQv9C+0YDRgiDQv9C+0LjRgdC60L7QvCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICfQmNC90YTQvtGA0LzQsNGG0LjRjyDQuNC80L/QvtGA0YLQsCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ9CY0YHRhdC+0LTQvdGL0Lkg0YTQsNC50LsnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ9CY0LzQv9C+0YDRgi/QrdC60YHQv9C+0YDRgic7CgogICAgIwogICAgIyBPQlNPTEVURSBFTlRSSUVTIEZPUiBSRUZFUkVOQ0UsIERPIE5PVCBUUkFOU0xBVEUhCiAgICAjCgp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS96aF9DTl9JbXBvcnRFeHBvcnQucG0gLSB0cmFuc2xhdGlvbiBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgQ29weXJpZ2h0IChDKSAyMDEzIE1pY2hhZWwgU2hpIDxtaWNzaGkgYXQgMTYzLmNvbT4KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6emhfQ05fSW1wb3J0RXhwb3J0OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICAjIFRlbXBsYXRlOiBBQUFJbXBvcnRFeHBvcnQKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBtYXBwaW5nIHRlbXBsYXRlJ30gPSAn5aKe5Yqg5pig5bCE5qih54mIJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICflrZfnrKbpm4YnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sb24gKDopJ30gPSAn5YaS5Y+3ICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICfliJcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29tbWEgKCwpJ30gPSAn6YCX5Y+3KCwpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICfliJfliIbpmpTnrKYnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ+WPpeWPtyAoLiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ+WIhuWPtyAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGFidWxhdG9yIChUQUIpJ30gPSAn5Yi26KGo6ZSuIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICfljIXmi6zliJfmoIfpopgnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAn5a+85YWl5oC757uTJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydGVkIHJlY29yZHMnfSA9ICflr7zlhaXorrDlvZUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXhwb3J0ZWQgcmVjb3Jkcyd9ID0gJ+WvvOWHuuiusOW9lSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAn6K6w5b2VJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICfot7Pov4fnmoQnOwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICflr7zlhaUv5a+85Ye6566h55CGJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICfliJvlu7rmqKHmnb/lr7zlhaXlkozlr7zlh7rlr7nosaHkv6Hmga/jgIInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAn5byA5aeL5a+85YWlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ+W8gOWni+WvvOWHuic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEZWxldGUgVGVtcGxhdGUnfSA9ICfliKDpmaTmqKHmnb8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCd9ID0gJ+atpemqpCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJ+e8lui+keWFseeUqOS/oeaBryc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUaGUgbmFtZSBpcyByZXF1aXJlZCEnfSA9ICflkI3np7DmmK/lv4XpnIDnmoTvvIEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ+WvueixoeaYr+W/hemcgOeahO+8gSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAn5qC85byP5piv5b+F6ZyA55qEJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAn57yW6L6R5a+55YOP5L+h5oGvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAn57yW6L6R5qC85byP5L+h5oGvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57JyBpcyByZXF1aXJlZCEnfSA9ICfmmK/lv4XpnIDnmoQhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ+e8lui+keaYoOWwhOS/oeaBryc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAn5rKh5pyJ5om+5Yiw5pig5bCE55qE5a2X5q61JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICfmt7vliqDmmKDlsITlrZfmrrUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICfnvJbovpHmkJzntKLkv6Hmga8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICfpmZDliLblr7zlh7rmr4/kuKrmkJzlr7snOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAn5a+85YWl5L+h5oGvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAn5rqQ5paH5Lu2JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICfmiJDlip8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAn5aSx6LSlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ+mHjeWkjeeahOWQjeensCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJ+WvvOWFpeaWh+S7tuacgOWQjuWkhOeQhueahOihjOaVsCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAn5qih5p2/5YiX6KGoJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VtcHR5IGZpZWxkcyBpbmRpY2F0ZSB0aGF0IHRoZSBjdXJyZW50IHZhbHVlcyBhcmUga2VwdCd9ID0gJ+epuuWtl+auteihqOekuuS/neaMgeW9k+WJjeeahOWAvCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yIGlzIHJlcXVpcmVkISd9ID0gJ+W/hemhu+aMh+WumuWIl+WIhumalOespiEnOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ+WvvOWFpeWSjOWvvOWHuuWvueixoeS/oeaBryc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAn5a+85YWlL+WvvOWHuic7CgoKICAgICMKICAgICMgT0JTT0xFVEUgRU5UUklFUyBGT1IgUkVGRVJFTkNFLCBETyBOT1QgVFJBTlNMQVRFIQogICAgIwoKfQoKMTsK
# --
# Kernel/Modules/AdminImportExport.pm - admin frontend of import export module
# Copyright (C) 2001-2013 OTRS AG, http://otrs.com/
# --
# 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::AdminImportExport;

use strict;
use warnings;

use Kernel::System::ImportExport;
use Kernel::System::Valid;

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 EncodeObject)) {
        if ( !$Self->{$Object} ) {
            $Self->{LayoutObject}->FatalError( Message => "Got no $Object!" );
        }
    }
    $Self->{ImportExportObject} = Kernel::System::ImportExport->new( %{$Self} );
    $Self->{ValidObject}        = Kernel::System::Valid->new( %{$Self} );

    return $Self;
}

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

    # ------------------------------------------------------------ #
    # template edit (common)
    # ------------------------------------------------------------ #
    if ( $Self->{Subaction} eq 'TemplateEdit1' ) {

        # get object list
        my $ObjectList = $Self->{ImportExportObject}->ObjectList();

        if ( !$ObjectList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No object backend found!' );
            return;
        }

        # get format list
        my $FormatList = $Self->{ImportExportObject}->FormatList();

        if ( !$FormatList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No format backend found!' );
            return;
        }

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

        my $SaveContinue;
        $SaveContinue = $Self->{ParamObject}->GetParam( Param => 'SubmitNext' );

        if ( !$SaveContinue ) {

            # if needed new form
            if ( !$TemplateID ) {
                return $Self->_MaskTemplateEdit1( New => 1, %Param );
            }

            # if there is template id
            # get template data
            $TemplateData = $Self->{ImportExportObject}->TemplateGet(
                TemplateID => $TemplateID,
                UserID     => $Self->{UserID},
            );

            if ( !$TemplateData->{TemplateID} ) {
                $Self->{LayoutObject}->FatalError( Message => 'Template not found!' );
                return;
            }

            # if edit
            if ( $TemplateData->{TemplateID} ) {
                return $Self->_MaskTemplateEdit1( %Param, %{$TemplateData} );
            }

        }

        # if save & continue
        my %ServerError;

        # get all data for params and check for errors
        for my $Param (qw(Comment Object Format Name ValidID TemplateID)) {
            $TemplateData->{$Param} = $Self->{ParamObject}->GetParam( Param => $Param ) || '';
        }

        # is a new template?
        my $New;
        if ( !$TemplateData->{TemplateID} ) {
            $New = 1;
        }

        # check needed fields
        # for new templates
        if ($New) {

            if ( !$TemplateData->{Object} ) {
                $ServerError{Object} = 1;
            }

            if ( !$TemplateData->{Format} ) {
                $ServerError{Format} = 1;
            }

        }

        # for all templates
        if ( !$TemplateData->{Name} ) {
            $ServerError{Name} = 1;
        }

        # if some error
        if ( $ServerError{Format} || $ServerError{Object} || $ServerError{Name} ) {
            return $Self->_MaskTemplateEdit1(
                ServerError => \%ServerError,
                New         => $New,
                %{$TemplateData},
            );
        }

        # save to database
        my $Success;

        if ($New) {
            $TemplateData->{TemplateID} = $Self->{ImportExportObject}->TemplateAdd(
                %{$TemplateData},
                UserID => $Self->{UserID},
            );
            $Success = $TemplateData->{TemplateID};
        }
        else {
            $Success = $Self->{ImportExportObject}->TemplateUpdate(
                UserID => $Self->{UserID},
                %{$TemplateData},
            );
        }

        if ( !$Success ) {
            $Self->{LayoutObject}->FatalError( Message => "Can't insert/update template!" );
            return;
        }

        return $Self->{LayoutObject}->Redirect(
            OP =>
                "Action=$Self->{Action};Subaction=TemplateEdit2;TemplateID=$TemplateData->{TemplateID}",
        );
    }

    # ------------------------------------------------------------ #
    # template edit (object)
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'TemplateEdit2' ) {

        # get object list
        my $ObjectList = $Self->{ImportExportObject}->ObjectList();

        if ( !$ObjectList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No object backend found!' );
            return;
        }

        # get format list
        my $FormatList = $Self->{ImportExportObject}->FormatList();

        if ( !$FormatList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No format backend found!' );
            return;
        }

        # get template id
        my $TemplateID = $Self->{ParamObject}->GetParam( Param => 'TemplateID' );

        if ( !$TemplateID ) {
            $Self->{LayoutObject}->FatalError( Message => 'Needed TemplateID!' );
            return;
        }

        my $SubmitNext = $Self->{ParamObject}->GetParam( Param => 'SubmitNext' );

        if ( !$SubmitNext ) {
            return $Self->_MaskTemplateEdit2( TemplateID => $TemplateID );
        }

        # save template starts here

        # get object attributes
        my $ObjectAttributeList = $Self->{ImportExportObject}->ObjectAttributesGet(
            TemplateID => $TemplateID,
            UserID     => $Self->{UserID},
        );

        my %AttributeValues;

        my $Error = 0;
        my %ServerError;
        my %DataTypeError;

        # get attribute values from form
        for my $Item ( @{$ObjectAttributeList} ) {

            # get form data
            $AttributeValues{ $Item->{Key} } = $Self->{LayoutObject}->ImportExportFormDataGet(
                Item => $Item,
            );

            # reload form if value is required and is not there
            if ( $Item->{Form}->{Invalid} ) {
                $ServerError{ $Item->{Name} } = 1;
                $Error = 1;
            }

            if ( $AttributeValues{ $Item->{Key} } ) {

                # look for regexp for data type allowed
                if (
                    $Item->{Input}->{Regex}
                    &&
                    !$AttributeValues{ $Item->{Key} } =~ $Item->{Input}->{Regex}
                    )
                {

                    $DataTypeError{ $Item->{Name} } = 1;
                    $Error = 1;
                }
            }

        }

        # reload with server errors
        if ($Error) {
            return $Self->_MaskTemplateEdit2(
                ServerError      => \%ServerError,
                DataTypeError    => \%DataTypeError,
                TemplateDataForm => \%AttributeValues,
                TemplateID       => $TemplateID,
            );
        }

        # save the object data
        $Self->{ImportExportObject}->ObjectDataSave(
            TemplateID => $TemplateID,
            ObjectData => \%AttributeValues,
            UserID     => $Self->{UserID},
        );

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

    }

    # ------------------------------------------------------------ #
    # template edit (format)
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'TemplateEdit3' ) {

        # get object list
        my $ObjectList = $Self->{ImportExportObject}->ObjectList();

        if ( !$ObjectList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No object backend found!' );
            return;
        }

        # get format list
        my $FormatList = $Self->{ImportExportObject}->FormatList();

        if ( !$FormatList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No format backend found!' );
            return;
        }

        # get template id
        my $TemplateID = $Self->{ParamObject}->GetParam( Param => 'TemplateID' );

        if ( !$TemplateID ) {
            $Self->{LayoutObject}->FatalError( Message => 'Needed TemplateID!' );
            return;
        }

        my $SubmitNext = $Self->{ParamObject}->GetParam( Param => 'SubmitNext' );

        if ( !$SubmitNext ) {
            return $Self->_MaskTemplateEdit3( TemplateID => $TemplateID );
        }

        # save starting here

        # get format attributes
        my $FormatAttributeList = $Self->{ImportExportObject}->FormatAttributesGet(
            TemplateID => $TemplateID,
            UserID     => $Self->{UserID},
        );

        # get format data
        my $FormatData = $Self->{ImportExportObject}->FormatDataGet(
            TemplateID => $TemplateID,
            UserID     => $Self->{UserID},
        );

        my $Error = 0;
        my %ServerError;

        # get attribute values from form
        my %AttributeValues;
        for my $Item ( @{$FormatAttributeList} ) {

            # get form data
            $AttributeValues{ $Item->{Key} } = $Self->{LayoutObject}->ImportExportFormDataGet(
                Item => $Item,
            );

            # reload form if value is required
            if ( $Item->{Form}->{Invalid} ) {
                $ServerError{ $Item->{Name} } = 1;
                $Error = 1;
            }
        }

        # reload with server errors
        if ($Error) {
            return $Self->_MaskTemplateEdit3(
                ServerError => \%ServerError,
                TemplateID  => $TemplateID,
            );
        }

        # save the format data
        $Self->{ImportExportObject}->FormatDataSave(
            TemplateID => $TemplateID,
            FormatData => \%AttributeValues,
            UserID     => $Self->{UserID},
        );

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

    # ------------------------------------------------------------ #
    # template edit (mapping)
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'TemplateEdit4' ) {

        # get object list
        my $ObjectList = $Self->{ImportExportObject}->ObjectList();

        if ( !$ObjectList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No object backend found!' );
            return;
        }

        # get format list
        my $FormatList = $Self->{ImportExportObject}->FormatList();

        if ( !$FormatList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No format backend found!' );
            return;
        }

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

        # get template data
        $TemplateData = $Self->{ImportExportObject}->TemplateGet(
            TemplateID => $TemplateData->{TemplateID},
            UserID     => $Self->{UserID},
        );

        if ( !$TemplateData->{TemplateID} ) {
            $Self->{LayoutObject}->FatalError( Message => 'Template not found!' );
            return;
        }

        # generate ObjectOptionStrg
        my $ObjectOptionStrg = $Self->{LayoutObject}->BuildSelection(
            Data         => $ObjectList,
            Name         => 'Object',
            SelectedID   => $TemplateData->{Object},
            PossibleNone => 1,
            Translation  => 1,
        );

        # generate FormatOptionStrg
        my $FormatOptionStrg = $Self->{LayoutObject}->BuildSelection(
            Data         => $FormatList,
            Name         => 'Format',
            SelectedID   => $TemplateData->{Format},
            PossibleNone => 1,
            Translation  => 1,
        );

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

        $Self->{LayoutObject}->Block( Name => 'ActionList' );
        $Self->{LayoutObject}->Block( Name => 'ActionOverview' );

        # output headline
        $Self->{LayoutObject}->Block(
            Name => 'TemplateEdit4',
            Data => {
                %{$TemplateData},
                ObjectName => $ObjectList->{ $TemplateData->{Object} },
                FormatName => $FormatList->{ $TemplateData->{Format} },
            },
        );

        # get mapping data list
        my $MappingList = $Self->{ImportExportObject}->MappingList(
            TemplateID => $TemplateData->{TemplateID},
            UserID     => $Self->{UserID},
        );

        # get object attributes
        my $MappingObjectAttributes = $Self->{ImportExportObject}->MappingObjectAttributesGet(
            TemplateID => $TemplateData->{TemplateID},
            UserID     => $Self->{UserID},
        );

        # get format attributes
        my $MappingFormatAttributes = $Self->{ImportExportObject}->MappingFormatAttributesGet(
            TemplateID => $TemplateData->{TemplateID},
            UserID     => $Self->{UserID},
        );

        # create headers for object and add common headers
        my @Headers;

        for my $Header ( @{$MappingObjectAttributes} ) {
            push @Headers, $Header->{Name};
        }

        for my $CommonHeader ( 'Column', 'Up', 'Down', 'Delete' ) {
            push @Headers, $CommonHeader;
        }

        for my $Header (@Headers) {

            # output attribute row
            $Self->{LayoutObject}->Block(
                Name => 'TemplateEdit4TableHeader',
                Data => {
                    Header => $Header,
                },
            );
        }

        # to use in colspan for 'no data found' message
        my $HeaderCounter = @Headers;

        my $EmptyMap            = 1;
        my $AtributteRowCounter = 0;
        for my $MappingID ( @{$MappingList} ) {

            $EmptyMap = 0;

            # output attribute row
            $Self->{LayoutObject}->Block(
                Name => 'TemplateEdit4Row',
                Data => {
                    MappingID => $MappingID,
                },
            );

            # get mapping object data
            my $MappingObjectData = $Self->{ImportExportObject}->MappingObjectDataGet(
                MappingID => $MappingID,
                UserID    => $Self->{UserID},
            );

            # get mapping format data
            my $MappingFormatData = $Self->{ImportExportObject}->MappingFormatDataGet(
                MappingID => $MappingID,
                UserID    => $Self->{UserID},
            );

            for my $Item ( @{$MappingObjectAttributes} ) {

                # create form input
                my $InputString = $Self->{LayoutObject}->ImportExportFormInputCreate(
                    Item   => $Item,
                    Prefix => 'Object::' . $AtributteRowCounter . '::',
                    Value  => $MappingObjectData->{ $Item->{Key} },
                    ID     => $Item->{Key} . $AtributteRowCounter,
                );

                # output attribute row
                $Self->{LayoutObject}->Block(
                    Name => 'TemplateEdit4Column',
                    Data => {
                        Name      => $Item->{Name},
                        ID        => 'Object::' . $AtributteRowCounter . '::' . $Item->{Key},
                        InputStrg => $InputString,
                        Counter   => $AtributteRowCounter,
                    },
                );
            }

            for my $Item ( @{$MappingFormatAttributes} ) {

                # create form input
                my $InputString = $Self->{LayoutObject}->ImportExportFormInputCreate(
                    Item   => $Item,
                    Prefix => 'Format::' . $AtributteRowCounter . '::',
                    Value  => $MappingFormatData->{ $Item->{Key} },
                );

                # output attribute row
                $Self->{LayoutObject}->Block(
                    Name => 'TemplateEdit4MapNumberColumn',
                    Data => {
                        Name      => $Item->{Name},
                        InputStrg => $InputString,
                        Counter   => $AtributteRowCounter,
                    },
                );
            }

            # hide the up button for first element and down button for the last element
            my $UpBlock;
            my $DownBlock;
            my $NumberOfElements = @{$MappingList};

            if ( $AtributteRowCounter == 0 ) {
                $UpBlock = 'TemplateEdit4NoUpButton';
            }
            else {
                $UpBlock = 'TemplateEdit4UpButton';
            }

            # check if this is the last element
            if ( $AtributteRowCounter == ( $NumberOfElements - 1 ) ) {
                $DownBlock = 'TemplateEdit4NoDownButton';
            }
            else {
                $DownBlock = 'TemplateEdit4DownButton';
            }

            # up button block
            $Self->{LayoutObject}->Block(
                Name => $UpBlock,
                Data => { MappingID => $MappingID },
            );

            # down button block
            $Self->{LayoutObject}->Block(
                Name => $DownBlock,
                Data => { MappingID => $MappingID },
            );

            $AtributteRowCounter++;
        }

        # output an empty list
        if ($EmptyMap) {

            # output list
            $Self->{LayoutObject}->Block(
                Name => 'TemplateEdit4NoMapFound',
                Data => {
                    Columns => $HeaderCounter,
                },
            );
        }

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

        # start template output
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AdminImportExport',
            Data         => \%Param,
        );

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

    # ------------------------------------------------------------ #
    # template save (mapping)
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'TemplateSave4' ) {

        # get template id
        my $TemplateID = $Self->{ParamObject}->GetParam( Param => 'TemplateID' );

        my %Submit = (
            SubmitNext => 'TemplateEdit5',
            SubmitBack => 'TemplateEdit3',
            Reload     => 'TemplateEdit4',
            MappingAdd => 'TemplateEdit4',
        );

        # get submit action
        my $Subaction    = $Submit{Reload};
        my $SubmitButton = '';

        PARAM:
        for my $SubmitKey ( sort keys %Submit ) {
            next PARAM if !$Self->{ParamObject}->GetParam( Param => $SubmitKey );

            $Subaction    = $Submit{$SubmitKey};
            $SubmitButton = $SubmitKey;
            last PARAM;
        }

        # get mapping data list
        my $MappingList = $Self->{ImportExportObject}->MappingList(
            TemplateID => $TemplateID,
            UserID     => $Self->{UserID},
        );

        # get object attributes
        my $MappingObjectAttributes = $Self->{ImportExportObject}->MappingObjectAttributesGet(
            TemplateID => $TemplateID,
            UserID     => $Self->{UserID},
        );

        # get format attributes
        my $MappingFormatAttributes = $Self->{ImportExportObject}->MappingFormatAttributesGet(
            TemplateID => $TemplateID,
            UserID     => $Self->{UserID},
        );

        my $Counter = 0;
        MAPPINGID:
        for my $MappingID ( @{$MappingList} ) {

            # get object attribute values
            my %ObjectAttributeValues;
            for my $Item ( @{$MappingObjectAttributes} ) {

                # get object form data
                $ObjectAttributeValues{ $Item->{Key} }
                    = $Self->{LayoutObject}->ImportExportFormDataGet(
                    Item   => $Item,
                    Prefix => 'Object::' . $Counter . '::',
                    );
            }

            # save the mapping object data
            $Self->{ImportExportObject}->MappingObjectDataSave(
                MappingID         => $MappingID,
                MappingObjectData => \%ObjectAttributeValues,
                UserID            => $Self->{UserID},
            );

            # get format attribute values
            my %FormatAttributeValues;
            for my $Item ( @{$MappingFormatAttributes} ) {

                # get format form data
                $FormatAttributeValues{ $Item->{Key} }
                    = $Self->{LayoutObject}->ImportExportFormDataGet(
                    Item   => $Item,
                    Prefix => 'Format::' . $Counter . '::',
                    );
            }

            # save the mapping format data
            $Self->{ImportExportObject}->MappingFormatDataSave(
                MappingID         => $MappingID,
                MappingFormatData => \%FormatAttributeValues,
                UserID            => $Self->{UserID},
            );

            $Counter++;
        }

        MAPPINGID:
        for my $MappingID ( @{$MappingList} ) {

            # delete this mapping row
            if ( $Self->{ParamObject}->GetParam( Param => "MappingDelete::$MappingID" ) ) {
                $Self->{ImportExportObject}->MappingDelete(
                    MappingID  => $MappingID,
                    TemplateID => $TemplateID,
                    UserID     => $Self->{UserID},
                );

                next MAPPINGID;
            }

            # move mapping data row up
            if ( $Self->{ParamObject}->GetParam( Param => "MappingUp::$MappingID" ) ) {
                $Self->{ImportExportObject}->MappingUp(
                    MappingID  => $MappingID,
                    TemplateID => $TemplateID,
                    UserID     => $Self->{UserID},
                );

                next MAPPINGID;
            }

            # move mapping data row down
            if ( $Self->{ParamObject}->GetParam( Param => "MappingDown::$MappingID" ) ) {
                $Self->{ImportExportObject}->MappingDown(
                    MappingID  => $MappingID,
                    TemplateID => $TemplateID,
                    UserID     => $Self->{UserID},
                );

                next MAPPINGID;
            }
        }

        # add a new mapping row
        if ( $SubmitButton eq 'MappingAdd' ) {
            $Self->{ImportExportObject}->MappingAdd(
                TemplateID => $TemplateID,
                UserID     => $Self->{UserID},
            );
        }

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

    # ------------------------------------------------------------ #
    # template edit (search)
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'TemplateEdit5' ) {

        # get object list
        my $ObjectList = $Self->{ImportExportObject}->ObjectList();

        if ( !$ObjectList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No object backend found!' );
            return;
        }

        # get format list
        my $FormatList = $Self->{ImportExportObject}->FormatList();

        if ( !$FormatList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No format backend found!' );
            return;
        }

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

        # get template data
        $TemplateData = $Self->{ImportExportObject}->TemplateGet(
            TemplateID => $TemplateData->{TemplateID},
            UserID     => $Self->{UserID},
        );

        if ( !$TemplateData->{TemplateID} ) {
            $Self->{LayoutObject}->FatalError( Message => 'Template not found!' );
            return;
        }

        # generate ObjectOptionStrg
        my $ObjectOptionStrg = $Self->{LayoutObject}->BuildSelection(
            Data         => $ObjectList,
            Name         => 'Object',
            SelectedID   => $TemplateData->{Object},
            PossibleNone => 1,
            Translation  => 1,
        );

        # generate FormatOptionStrg
        my $FormatOptionStrg = $Self->{LayoutObject}->BuildSelection(
            Data         => $FormatList,
            Name         => 'Format',
            SelectedID   => $TemplateData->{Format},
            PossibleNone => 1,
            Translation  => 1,
        );

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

        $Self->{LayoutObject}->Block( Name => 'ActionList' );
        $Self->{LayoutObject}->Block( Name => 'ActionOverview' );

        # get search data
        my $SearchData = $Self->{ImportExportObject}->SearchDataGet(
            TemplateID => $TemplateData->{TemplateID},
            UserID     => $Self->{UserID},
        );

        # create rescrict export string
        my $RestrictExportStrg = $Self->{LayoutObject}->ImportExportFormInputCreate(
            Item => {
                Key   => 'RestrictExport',
                Input => {
                    Type => 'Checkbox',
                },
            },
            Value => scalar keys %{$SearchData},
        );

        # output list
        $Self->{LayoutObject}->Block(
            Name => 'TemplateEdit5',
            Data => {
                %{$TemplateData},
                RestrictExportStrg => $RestrictExportStrg,
            },
        );

        # get search attributes
        my $SearchAttributeList = $Self->{ImportExportObject}->SearchAttributesGet(
            TemplateID => $TemplateData->{TemplateID},
            UserID     => $Self->{UserID},
        );

        # output object attributes
        for my $Item ( @{$SearchAttributeList} ) {

            # create form input
            my $InputString = $Self->{LayoutObject}->ImportExportFormInputCreate(
                Item  => $Item,
                Value => $SearchData->{ $Item->{Key} },
            );

            # output attribute row
            $Self->{LayoutObject}->Block(
                Name => 'TemplateEdit5Element',
                Data => {
                    Name => $Item->{Name} || '',
                    InputStrg => $InputString,
                    ID        => $Item->{Key},
                },
            );
        }

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

        # start template output
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AdminImportExport',
            Data         => \%Param,
        );

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

    # ------------------------------------------------------------ #
    # template save (search)
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'TemplateSave5' ) {

        # get template id
        my $TemplateID = $Self->{ParamObject}->GetParam( Param => 'TemplateID' );

        my %Submit = (
            SubmitNext => 'Overview',
            SubmitBack => 'TemplateEdit4',
            Reload     => 'TemplateEdit5',
        );

        # get submit action
        my $Subaction = $Submit{Reload};

        PARAM:
        for my $SubmitKey ( sort keys %Submit ) {
            next PARAM if !$Self->{ParamObject}->GetParam( Param => $SubmitKey );

            $Subaction = $Submit{$SubmitKey};
            last PARAM;
        }

        # delete all search restrictions
        if ( !$Self->{ParamObject}->GetParam( Param => 'RestrictExport' ) ) {

            # delete all search data
            $Self->{ImportExportObject}->SearchDataDelete(
                TemplateID => $TemplateID,
                UserID     => $Self->{UserID},
            );

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

        # get search attributes
        my $SearchAttributeList = $Self->{ImportExportObject}->SearchAttributesGet(
            TemplateID => $TemplateID,
            UserID     => $Self->{UserID},
        );

        # get attribute values from form
        my %AttributeValues;
        for my $Item ( @{$SearchAttributeList} ) {

            # get form data
            $AttributeValues{ $Item->{Key} } = $Self->{LayoutObject}->ImportExportFormDataGet(
                Item => $Item,
            );

            # reload form if value is required
            if ( $Item->{Form}->{Invalid} ) {
                $Subaction = $Submit{Reload};
            }
        }

        # save the search data
        $Self->{ImportExportObject}->SearchDataSave(
            TemplateID => $TemplateID,
            SearchData => \%AttributeValues,
            UserID     => $Self->{UserID},
        );

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

    # ------------------------------------------------------------ #
    # template delete
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'TemplateDelete' ) {

        # get template id
        my $TemplateID = $Self->{ParamObject}->GetParam( Param => 'TemplateID' );

        # delete template from database
        $Self->{ImportExportObject}->TemplateDelete(
            TemplateID => $TemplateID,
            UserID     => $Self->{UserID},
        );

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

    # ------------------------------------------------------------ #
    # import information
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'ImportInformation' ) {

        # get object list
        my $ObjectList = $Self->{ImportExportObject}->ObjectList();

        if ( !$ObjectList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No object backend found!' );
            return;
        }

        # get format list
        my $FormatList = $Self->{ImportExportObject}->FormatList();

        if ( !$FormatList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No format backend found!' );
            return;
        }

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

        # get template data
        $TemplateData = $Self->{ImportExportObject}->TemplateGet(
            TemplateID => $TemplateData->{TemplateID},
            UserID     => $Self->{UserID},
        );

        if ( !$TemplateData->{TemplateID} ) {
            $Self->{LayoutObject}->FatalError( Message => 'Template not found!' );
            return;
        }

        # generate ObjectOptionStrg
        my $ObjectOptionStrg = $Self->{LayoutObject}->BuildSelection(
            Data         => $ObjectList,
            Name         => 'Object',
            SelectedID   => $TemplateData->{Object},
            PossibleNone => 1,
            Translation  => 1,
        );

        # generate FormatOptionStrg
        my $FormatOptionStrg = $Self->{LayoutObject}->BuildSelection(
            Data         => $FormatList,
            Name         => 'Format',
            SelectedID   => $TemplateData->{Format},
            PossibleNone => 1,
            Translation  => 1,
        );

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

        $Self->{LayoutObject}->Block( Name => 'ActionList' );
        $Self->{LayoutObject}->Block( Name => 'ActionOverview' );

        # output list
        $Self->{LayoutObject}->Block(
            Name => 'ImportInformation',
            Data => {
                %{$TemplateData},
            },
        );

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

        # start template output
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AdminImportExport',
            Data         => \%Param,
        );

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

    # ------------------------------------------------------------ #
    # import
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'Import' ) {

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

        # get template data
        $TemplateData = $Self->{ImportExportObject}->TemplateGet(
            TemplateID => $TemplateData->{TemplateID},
            UserID     => $Self->{UserID},
        );

        if ( !$TemplateData->{TemplateID} ) {
            $Self->{LayoutObject}->FatalError( Message => 'Template not found!' );
            return;
        }

        # get source file
        my %SourceFile = $Self->{ParamObject}->GetUploadAll(
            Param  => 'SourceFile',
            Source => 'String',
        );

        $SourceFile{Content} ||= '';

        # import data
        my $Result = $Self->{ImportExportObject}->Import(
            TemplateID    => $TemplateData->{TemplateID},
            SourceContent => \$SourceFile{Content},
            UserID        => $Self->{UserID},
        );

        if ( !$Result ) {
            $Self->{LayoutObject}->FatalError(
                Message => 'Error occurred. Import impossible! See Syslog for details.',
            );
            return;
        }

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

        # output import results
        $Self->{LayoutObject}->Block(
            Name => 'ImportResult',
            Data => {
                %{$Result},
            },
        );

        # get all return codes and collect the duplicate names
        my @DuplicateNames;
        RETURNCODE:
        for my $RetCode ( sort keys %{ $Result->{RetCode} } ) {

            # just get the duplicate name
            if ( $RetCode =~ m{ \A DuplicateName \s+ (.+) }xms ) {
                push @DuplicateNames, $1;
            }
            else {
                $Self->{LayoutObject}->Block(
                    Name => 'ImportResultReturnCode',
                    Data => {
                        ReturnCodeName  => $RetCode,
                        ReturnCodeCount => $Result->{RetCode}->{$RetCode},
                    },
                );
            }
        }

        # output duplicate names if neccessary
        if (@DuplicateNames) {

            my $DuplicateNamesString = join ', ', @DuplicateNames;

            $Self->{LayoutObject}->Block(
                Name => 'ImportResultDuplicateNames',
                Data => {
                    DuplicateNames => $DuplicateNamesString,
                },
            );
        }

        # output last processed line mumber of import file
        if ( $Result->{Failed} ) {
            $Self->{LayoutObject}->Block(
                Name => 'ImportResultLastLineNumber',
                Data => {
                    LastLineNumber => $Result->{Counter},
                },
            );
        }

        # start output
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AdminImportExport',
            Data         => {
                %Param,
            },
        );

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

    # ------------------------------------------------------------ #
    # export
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'Export' ) {

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

        # get template data
        $TemplateData = $Self->{ImportExportObject}->TemplateGet(
            TemplateID => $TemplateData->{TemplateID},
            UserID     => $Self->{UserID},
        );

        if ( !$TemplateData->{TemplateID} ) {
            $Self->{LayoutObject}->FatalError( Message => 'Template not found!' );
            return;
        }

        # export data
        my $Result = $Self->{ImportExportObject}->Export(
            TemplateID => $TemplateData->{TemplateID},
            UserID     => $Self->{UserID},
        );

        if ( !$Result ) {
            $Self->{LayoutObject}->FatalError(
                Message => 'Error occurred. Export impossible! See Syslog for details.',
            );
            return;
        }

        my $FileContent = join "\n", @{ $Result->{DestinationContent} };

        return $Self->{LayoutObject}->Attachment(
            Type        => 'attachment',
            Filename    => 'Export.csv',
            ContentType => 'text/csv',
            Content     => $FileContent,
        );
    }

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

        # get object list
        my $ObjectList = $Self->{ImportExportObject}->ObjectList();

        if ( !$ObjectList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No object backend found!' );
            return;
        }

        # get format list
        my $FormatList = $Self->{ImportExportObject}->FormatList();

        if ( !$FormatList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No format backend found!' );
            return;
        }

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

        $Self->{LayoutObject}->Block( Name => 'ActionList' );
        $Self->{LayoutObject}->Block( Name => 'ActionAdd' );

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

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

        my $EmptyDatabase = 1;

        CLASS:
        for my $Object ( sort { $ObjectList->{$a} cmp $ObjectList->{$b} } keys %{$ObjectList} ) {

            # get template list
            my $TemplateList = $Self->{ImportExportObject}->TemplateList(
                Object => $Object,
                UserID => $Self->{UserID},
            );

            if ( !$TemplateList || ref $TemplateList ne 'ARRAY' || !@{$TemplateList} ) {
                next CLASS;
            }

            $EmptyDatabase = 0;

            # output list
            $Self->{LayoutObject}->Block(
                Name => 'OverviewList',
                Data => {
                    ObjectName => $ObjectList->{$Object},
                },
            );

            for my $TemplateID ( @{$TemplateList} ) {

                # get template data
                my $TemplateData = $Self->{ImportExportObject}->TemplateGet(
                    TemplateID => $TemplateID,
                    UserID     => $Self->{UserID},
                );

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

        # output an empty list
        if ($EmptyDatabase) {

            # output list
            $Self->{LayoutObject}->Block(
                Name => 'OverviewList',
                Data => {
                    ObjectName => 'Template List',
                },
            );
            $Self->{LayoutObject}->Block( Name => 'NoDataFoundMsg' );
        }

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

        # start template output
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AdminImportExport',
            Data         => \%Param,
        );

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

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

    my %ServerError;
    if ( $Param{ServerError} ) {
        %ServerError = %{ $Param{ServerError} };
    }

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

    $Self->{LayoutObject}->Block( Name => 'ActionList' );
    $Self->{LayoutObject}->Block( Name => 'ActionOverview' );

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

    my $Class = ' Validate_Required ';

    if ( $ServerError{Name} ) {
        $Class .= 'ServerError';
    }

    $Self->{LayoutObject}->Block(
        Name => 'TemplateEdit1',
        Data => {
            %Param,
            ValidOptionStrg => $ValidOptionStrg,
            NameClass       => $Class,
        },
    );

    if ( $Param{TemplateID} ) {
        $Self->{LayoutObject}->Block(
            Name => 'EditObjectFormat',
            Data => {
                %Param,
                ObjectName => $Param{Object},
                FormatName => $Param{Format},
                }
        );
    }

    if ( $Param{New} ) {

        # get object list
        my $ObjectList = $Self->{ImportExportObject}->ObjectList();

        if ( !$ObjectList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No object backend found!' );
            return;
        }

        # get format list
        my $FormatList = $Self->{ImportExportObject}->FormatList();

        if ( !$FormatList ) {
            $Self->{LayoutObject}->FatalError( Message => 'No format backend found!' );
            return;
        }

        $Class = ' Validate_Required ';

        if ( $ServerError{Object} ) {
            $Class .= 'ServerError';
        }

        # generate ObjectOptionStrg
        my $ObjectOptionStrg = $Self->{LayoutObject}->BuildSelection(
            Data         => $ObjectList,
            Name         => 'Object',
            SelectedID   => $Param{Object} || '',
            PossibleNone => 1,
            Translation  => 1,
            Class        => $Class,
        );

        $Class = ' Validate_Required ';
        if ( $ServerError{Format} ) {
            $Class .= 'ServerError';
        }

        # generate FormatOptionStrg
        my $FormatOptionStrg = $Self->{LayoutObject}->BuildSelection(
            Data         => $FormatList,
            Name         => 'Format',
            SelectedID   => $Param{Format} || '',
            PossibleNone => 1,
            Translation  => 1,
            Class        => $Class,
        );

        $Self->{LayoutObject}->Block(
            Name => 'NewObjectFormat',
            Data => {
                ObjectOption => $ObjectOptionStrg,
                FormatOption => $FormatOptionStrg,
            },
        );
    }

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

    # start template output
    $Output .= $Self->{LayoutObject}->Output(
        TemplateFile => 'AdminImportExport',
        Data         => \%Param,
    );

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

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

    my %ServerError;
    if ( $Param{ServerError} ) {
        %ServerError = %{ $Param{ServerError} };
    }

    my %DataTypeError;
    if ( $Param{DataTypeError} ) {
        %DataTypeError = %{ $Param{DataTypeError} };
    }

    my $TemplateID;
    if ( $Param{TemplateID} ) {
        $TemplateID = $Param{TemplateID};
    }
    else {
        $Self->{LayoutObject}->FatalError( Message => 'Needed TemplateID!' );
        return;
    }

    # get template data
    my $TemplateData;
    $TemplateData = $Self->{ImportExportObject}->TemplateGet(
        TemplateID => $TemplateID,
        UserID     => $Self->{UserID},
    );

    if ( !$TemplateData->{TemplateID} ) {
        $Self->{LayoutObject}->FatalError( Message => 'Template not found!' );
        return;
    }

    $Param{BackURL} = "Action=$Self->{Action};Subaction=TemplateEdit1;TemplateID=$TemplateID";

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

    $Self->{LayoutObject}->Block( Name => 'ActionList' );
    $Self->{LayoutObject}->Block( Name => 'ActionOverview' );

    # output list
    $Self->{LayoutObject}->Block(
        Name => 'TemplateEdit2',
        Data => $TemplateData,
    );

    # get object attributes
    my $ObjectAttributeList = $Self->{ImportExportObject}->ObjectAttributesGet(
        TemplateID => $TemplateData->{TemplateID},
        UserID     => $Self->{UserID},
    );

    # get object data
    my $ObjectData = $Self->{ImportExportObject}->ObjectDataGet(
        TemplateID => $TemplateData->{TemplateID},
        UserID     => $Self->{UserID},
    );

    # javascript validation class per datatype
    my %JSClass;
    my %PredefinedErrorMessages;

    $JSClass{Number}                = 'Validate_Number';
    $JSClass{NumberBiggerThanZero}  = 'Validate_NumberBiggerThanZero';
    $JSClass{Integer}               = 'Validate_NumberInteger';
    $JSClass{IntegerBiggerThanZero} = 'Validate_NumberIntegerBiggerThanZero';

    $PredefinedErrorMessages{Number}                = 'number';
    $PredefinedErrorMessages{NumberBiggerThanZero}  = 'number bigger than zero';
    $PredefinedErrorMessages{Integer}               = 'integer';
    $PredefinedErrorMessages{IntegerBiggerThanZero} = 'integer bigger than zero';

    # output object attributes
    for my $Item ( @{$ObjectAttributeList} ) {

        my $Class = ' ';
        my $Value;

        my $DataTypeError;
        my $ErrorMessage;

        if ( $Item->{Input}->{Required} ) {
            $Class        = 'Validate_Required';
            $ErrorMessage = 'Element required, please insert data';
        }

        if ( $Item->{Input}->{DataType} ) {
            $Class .= " $JSClass{ $Item->{Input}->{DataType} }";
            $ErrorMessage = 'Invalid data, please insert a valid ';
            $ErrorMessage .= "$PredefinedErrorMessages{$Item->{Input}->{DataType}}";
        }

        # get data from form or from database
        # ServerError = show the wrong data in form
        # !ServerError = show database data or new fields

        if ( $Param{ServerError} || $Param{DataTypeError} ) {
            $Value = $Param{TemplateDataForm}->{ $Item->{Key} };
        }
        else {
            $Value = $ObjectData->{ $Item->{Key} };
        }

        # error area

        # prepare different data & message per error
        if ( $ServerError{ $Item->{Name} } || $DataTypeError{ $Item->{Name} } ) {
            $Class .= ' ServerError';
        }

        # create form input
        my $InputString = $Self->{LayoutObject}->ImportExportFormInputCreate(
            Item  => $Item,
            Class => $Class,
            Value => $Value,
        );

        # build id
        my $ID;
        if ( $Item->{Prefix} ) {
            $ID = "$Item->{Prefix}$Item->{Key}";
        }
        else {
            $ID = $Item->{Key};
        }

        # output attribute row
        $Self->{LayoutObject}->Block(
            Name => 'TemplateEdit2Element',
            Data => {
                Name => $Item->{Name} || '',
                InputStrg    => $InputString,
                ID           => $ID,
                ErrorMessage => $ErrorMessage,
            },
        );
    }

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

    # start template output
    $Output .= $Self->{LayoutObject}->Output(
        TemplateFile => 'AdminImportExport',
        Data         => \%Param,
    );

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

}

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

    my %ServerError;
    if ( $Param{ServerError} ) {
        %ServerError = %{ $Param{ServerError} };
    }

    my $TemplateID;
    if ( $Param{TemplateID} ) {
        $TemplateID = $Param{TemplateID};
    }

    if ( !$TemplateID ) {
        $Self->{LayoutObject}->FatalError( Message => 'Template not found!' );
        return;
    }

    # get template data
    my $TemplateData;
    $TemplateData = $Self->{ImportExportObject}->TemplateGet(
        TemplateID => $TemplateID,
        UserID     => $Self->{UserID},
    );

    if ( !$TemplateData->{TemplateID} ) {
        $Self->{LayoutObject}->FatalError( Message => 'Template not found!' );
        return;
    }

    $Param{BackURL} = "Action=$Self->{Action};Subaction=TemplateEdit2;TemplateID=$TemplateID";

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

    $Self->{LayoutObject}->Block( Name => 'ActionList' );
    $Self->{LayoutObject}->Block( Name => 'ActionOverview' );

    # output list
    $Self->{LayoutObject}->Block(
        Name => 'TemplateEdit3',
        Data => $TemplateData,
    );

    # get format attributes
    my $FormatAttributeList = $Self->{ImportExportObject}->FormatAttributesGet(
        TemplateID => $TemplateData->{TemplateID},
        UserID     => $Self->{UserID},
    );

    # get format data
    my $FormatData = $Self->{ImportExportObject}->FormatDataGet(
        TemplateID => $TemplateData->{TemplateID},
        UserID     => $Self->{UserID},
    );

    if ( !$FormatData ) {
        $Self->{LayoutObject}->FatalError( Message => 'Format not found!' );
        return;
    }

    # output format attributes
    for my $Item ( @{$FormatAttributeList} ) {

        # build id
        my $ID;
        if ( $Item->{Prefix} ) {
            $ID = "$Item->{Prefix}$Item->{Key}";
        }
        else {
            $ID = "$Item->{Key}";
        }

        my $Class = ' ';
        if ( $Item->{Input}->{Required} ) {
            $Class = 'Validate_Required ';
        }

        if ( $ServerError{ $Item->{Name} } ) {
            $Class .= ' ServerError';
        }

        # create form input
        my $InputString = $Self->{LayoutObject}->ImportExportFormInputCreate(
            Item  => $Item,
            Class => $Class,
            Value => $FormatData->{ $Item->{Key} },
        );

        # output attribute row
        $Self->{LayoutObject}->Block(
            Name => 'TemplateEdit3Element',
            Data => {
                Name => $Item->{Name} || '',
                InputStrg => $InputString,
                ID        => $ID,
            },
        );

        # output required notice
        if ( $Item->{Input}->{Required} ) {
            $Self->{LayoutObject}->Block(
                Name => 'TemplateEdit3ElementRequired',
                Data => {
                    Name => $Item->{Name} || '',
                    ID => $ID,
                },
            );
        }
    }

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

    # start template output
    $Output .= $Self->{LayoutObject}->Output(
        TemplateFile => 'AdminImportExport',
        Data         => \%Param,
    );

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

}

1;

IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnRMYXlvdXRDaGVja2JveC5wbSAtIGxheW91dCBiYWNrZW5kIG1vZHVsZQojIENvcHlyaWdodCAoQykgMjAwMS0yMDEzIE9UUlMgQUcsIGh0dHA6Ly9vdHJzLmNvbS8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydExheW91dENoZWNrYm94OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRDaGVja2JveCAtIGxheW91dCBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIFNZTk9QU0lTCgpBbGwgbGF5b3V0IGZ1bmN0aW9ucyBmb3IgY2hlY2tib3ggZWxlbWVudHMgaW4gaW1wb3J0L2V4cG9ydC4KCj1vdmVyIDQKCj1jdXQKCj1pdGVtIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgJEJhY2tlbmRPYmplY3QgPSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0TGF5b3V0Q2hlY2tib3gtPm5ldygKICAgICAgICAlUGFyYW0sCiAgICApOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBvYmplY3RzCiAgICBmb3IgbXkgJE9iamVjdCAocXcoQ29uZmlnT2JqZWN0IExvZ09iamVjdCBNYWluT2JqZWN0IFBhcmFtT2JqZWN0IExheW91dE9iamVjdCkpIHsKICAgICAgICAkU2VsZi0+eyRPYmplY3R9ID0gJFBhcmFteyRPYmplY3R9IHx8IGRpZSAiR290IG5vICRPYmplY3QhIjsKICAgIH0KCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1pdGVtIEZvcm1JbnB1dENyZWF0ZSgpCgpjcmVhdGUgYSBpbnB1dCBzdHJpbmcKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+Rm9ybUlucHV0Q3JlYXRlKAogICAgICAgIEl0ZW0gICA9PiAkSXRlbVJlZiwKICAgICAgICBQcmVmaXggPT4gJ1ByZWZpeDo6JywgICMgKG9wdGlvbmFsKQogICAgICAgIFZhbHVlICA9PiAnVmFsdWUnLCAgICAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1JbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIEl0ZW0hJywKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAkUGFyYW17UHJlZml4fSB8fD0gJyc7CgogICAgbXkgJENoZWNrZWQgPSAkUGFyYW17VmFsdWV9ID8gJ2NoZWNrZWQ9ImNoZWNrZWQiJyA6ICcnOwoKICAgIHJldHVybgogICAgICAgIHFxezxpbnB1dCBpZD0iJFBhcmFte1ByZWZpeH0kUGFyYW17SXRlbX0tPntLZXl9IiB0eXBlPSJjaGVja2JveCIgbmFtZT0iJFBhcmFte1ByZWZpeH0kUGFyYW17SXRlbX0tPntLZXl9IiAkQ2hlY2tlZCAvPn07Cn0KCj1pdGVtIEZvcm1EYXRhR2V0KCkKCmdldCBmb3JtIGRhdGEKCiAgICBteSAkRm9ybURhdGEgPSAkQmFja2VuZE9iamVjdC0+Rm9ybURhdGFHZXQoCiAgICAgICAgSXRlbSAgID0+ICRJdGVtUmVmLAogICAgICAgIFByZWZpeCA9PiAnUHJlZml4OjonLCAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1EYXRhR2V0IHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgSXRlbSEnLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICRQYXJhbXtQcmVmaXh9IHx8PSAnJzsKCiAgICAjIGdldCBmb3JtIGRhdGEKICAgIG15ICRGb3JtRGF0YSA9ICRTZWxmLT57UGFyYW1PYmplY3R9LT5HZXRQYXJhbSgKICAgICAgICBQYXJhbSA9PiAkUGFyYW17UHJlZml4fSAuICRQYXJhbXtJdGVtfS0+e0tleX0sCiAgICApOwoKICAgIHJldHVybiAkRm9ybURhdGE7Cn0KCjE7Cgo9YmFjawoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIHNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoTDxodHRwOi8vb3Rycy5vcmcvPikuCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dD4uCgo9Y3V0Cg==
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnRMYXlvdXREVEwucG0gLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMyBPVFJTIEFHLCBodHRwOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXREVEw7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydExheW91dERUTCAtIGxheW91dCBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIFNZTk9QU0lTCgpBbGwgbGF5b3V0IGZ1bmN0aW9ucyBmb3IgZGlzcGxheSBEVEwgY29kZQoKPW92ZXIgNAoKPWN1dAoKPWl0ZW0gbmV3KCkKCmNyZWF0ZSBhbiBvYmplY3QKCiAgICAkQmFja2VuZE9iamVjdCA9IEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXREVEwtPm5ldygKICAgICAgICAlUGFyYW0sCiAgICApOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBvYmplY3RzCiAgICBmb3IgbXkgJE9iamVjdCAocXcoQ29uZmlnT2JqZWN0IExvZ09iamVjdCBNYWluT2JqZWN0IFBhcmFtT2JqZWN0IExheW91dE9iamVjdCkpIHsKICAgICAgICAkU2VsZi0+eyRPYmplY3R9ID0gJFBhcmFteyRPYmplY3R9IHx8IGRpZSAiR290IG5vICRPYmplY3QhIjsKICAgIH0KCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1pdGVtIEZvcm1JbnB1dENyZWF0ZSgpCgpjcmVhdGUgYSBpbnB1dCBzdHJpbmcKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+Rm9ybUlucHV0Q3JlYXRlKAogICAgICAgIEl0ZW0gICA9PiAkSXRlbVJlZiwKICAgICk7Cgo9Y3V0CgpzdWIgRm9ybUlucHV0Q3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZyggUHJpb3JpdHkgPT4gJ2Vycm9yJywgTWVzc2FnZSA9PiAnTmVlZCBJdGVtIScgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgcmV0dXJuICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e0RhdGF9Owp9Cgo9aXRlbSBGb3JtRGF0YUdldCgpCgpnZXQgZm9ybSBkYXRhCgogICAgbXkgJEZvcm1EYXRhID0gJEJhY2tlbmRPYmplY3QtPkZvcm1EYXRhR2V0KCk7Cgo9Y3V0CgpzdWIgRm9ybURhdGFHZXQgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm47Cn0KCjE7Cgo9YmFjawoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIHNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoaHR0cDovL290cnMub3JnLykuCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dD4uCgo9Y3V0Cg==
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnRMYXlvdXRTZWxlY3Rpb24ucG0gLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMyBPVFJTIEFHLCBodHRwOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRTZWxlY3Rpb247Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydExheW91dFNlbGVjdGlvbiAtIGxheW91dCBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIFNZTk9QU0lTCgpBbGwgbGF5b3V0IGZ1bmN0aW9ucyBmb3Igc2VsZWN0aW9uIGVsZW1lbnRzCgo9b3ZlciA0Cgo9Y3V0Cgo9aXRlbSBuZXcoKQoKY3JlYXRlIGFuIG9iamVjdAoKICAgICRCYWNrZW5kT2JqZWN0ID0gS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydExheW91dFNlbGVjdGlvbi0+bmV3KAogICAgICAgICVQYXJhbSwKICAgICk7Cgo9Y3V0CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHt9OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgICMgY2hlY2sgbmVlZGVkIG9iamVjdHMKICAgIGZvciBteSAkT2JqZWN0IChxdyhDb25maWdPYmplY3QgTG9nT2JqZWN0IE1haW5PYmplY3QgUGFyYW1PYmplY3QgTGF5b3V0T2JqZWN0KSkgewogICAgICAgICRTZWxmLT57JE9iamVjdH0gPSAkUGFyYW17JE9iamVjdH0gfHwgZGllICJHb3Qgbm8gJE9iamVjdCEiOwogICAgfQoKICAgIHJldHVybiAkU2VsZjsKfQoKPWl0ZW0gRm9ybUlucHV0Q3JlYXRlKCkKCmNyZWF0ZSBhIGlucHV0IHN0cmluZwoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtSW5wdXRDcmVhdGUoCiAgICAgICAgSXRlbSAgID0+ICRJdGVtUmVmLAogICAgICAgIFByZWZpeCA9PiAnUHJlZml4OjonLCAgIyAob3B0aW9uYWwpCiAgICAgICAgVmFsdWUgID0+ICdWYWx1ZScsICAgICAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgRm9ybUlucHV0Q3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZyggUHJpb3JpdHkgPT4gJ2Vycm9yJywgTWVzc2FnZSA9PiAnTmVlZCBJdGVtIScgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBzZXQgZGVmYXVsdCB2YWx1ZQogICAgJFBhcmFte1ByZWZpeH0gfHw9ICcnOwogICAgJFBhcmFte1ZhbHVlfSAgfHw9ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1ZhbHVlRGVmYXVsdH07CgogICAgaWYgKCAkUGFyYW17VmFsdWV9ICYmICRQYXJhbXtWYWx1ZX0gPX4gbXsgIyMjIyMgfXhtcyApIHsKICAgICAgICBteSBAVmFsdWVzID0gc3BsaXQgJyMjIyMjJywgJFBhcmFte1ZhbHVlfTsKICAgICAgICAkUGFyYW17VmFsdWV9ID0gXEBWYWx1ZXM7CiAgICB9CgogICAgIyBnZW5lcmF0ZSBvcHRpb24gc3RyaW5nCiAgICBteSAkU3RyaW5nID0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5CdWlsZFNlbGVjdGlvbigKICAgICAgICBJRCAgICAgICAgICAgPT4gJFBhcmFte1ByZWZpeH0gLiAkUGFyYW17SXRlbX0tPntLZXl9LAogICAgICAgIENsYXNzICAgICAgICA9PiAkUGFyYW17Q2xhc3N9LAogICAgICAgIE5hbWUgICAgICAgICA9PiAkUGFyYW17UHJlZml4fSAuICRQYXJhbXtJdGVtfS0+e0tleX0sCiAgICAgICAgRGF0YSAgICAgICAgID0+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e0RhdGF9IHx8IHt9LAogICAgICAgIFNlbGVjdGVkSUQgICA9PiAkUGFyYW17VmFsdWV9LAogICAgICAgIFRyYW5zbGF0aW9uICA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntUcmFuc2xhdGlvbn0sCiAgICAgICAgUG9zc2libGVOb25lID0+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1Bvc3NpYmxlTm9uZX0sCiAgICAgICAgTXVsdGlwbGUgICAgID0+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e011bHRpcGxlfSwKICAgICAgICBTaXplICAgICAgICAgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57U2l6ZX0sCiAgICApOwoKICAgIHJldHVybiAkU3RyaW5nOwp9Cgo9aXRlbSBGb3JtRGF0YUdldCgpCgpnZXQgZm9ybSBkYXRhCgogICAgbXkgJEZvcm1EYXRhID0gJEJhY2tlbmRPYmplY3QtPkZvcm1EYXRhR2V0KAogICAgICAgIEl0ZW0gICA9PiAkSXRlbVJlZiwKICAgICAgICBQcmVmaXggPT4gJ1ByZWZpeDo6JywgICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBGb3JtRGF0YUdldCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coIFByaW9yaXR5ID0+ICdlcnJvcicsIE1lc3NhZ2UgPT4gJ05lZWQgSXRlbSEnICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICRQYXJhbXtQcmVmaXh9IHx8PSAnJzsKCiAgICAjIGdldCBmb3JtIGRhdGEKICAgIG15IEBGb3JtRGF0YXMgPSAkU2VsZi0+e1BhcmFtT2JqZWN0fS0+R2V0QXJyYXkoCiAgICAgICAgUGFyYW0gPT4gJFBhcmFte1ByZWZpeH0gLiAkUGFyYW17SXRlbX0tPntLZXl9LAogICAgKTsKCiAgICBteSAkRm9ybURhdGEgPSBqb2luICcjIyMjIycsIEBGb3JtRGF0YXM7CgogICAgcmV0dXJuICRGb3JtRGF0YSBpZiAkRm9ybURhdGE7CiAgICByZXR1cm4gJEZvcm1EYXRhIGlmICEkUGFyYW17SXRlbX0tPntJbnB1dH0tPntSZXF1aXJlZH07CgogICAgIyBzZXQgaW52YWxpZCBwYXJhbQogICAgJFBhcmFte0l0ZW19LT57Rm9ybX0tPntJbnZhbGlkfSA9IDE7CgogICAgcmV0dXJuICRGb3JtRGF0YTsKfQoKMTsKCj1iYWNrCgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKClRoaXMgc29mdHdhcmUgaXMgcGFydCBvZiB0aGUgT1RSUyBwcm9qZWN0IChMPGh0dHA6Ly9vdHJzLm9yZy8+KS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIEw8aHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0Pi4KCj1jdXQK
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnRMYXlvdXRUZXh0LnBtIC0gbGF5b3V0IGJhY2tlbmQgbW9kdWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0TGF5b3V0VGV4dDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0TGF5b3V0VGV4dCAtIGxheW91dCBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIFNZTk9QU0lTCgpBbGwgbGF5b3V0IGZ1bmN0aW9ucyBmb3IgdGV4dCBlbGVtZW50cwoKPW92ZXIgNAoKPWN1dAoKPWl0ZW0gbmV3KCkKCmNyZWF0ZSBhbiBvYmplY3QKCiAgICAkQmFja2VuZE9iamVjdCA9IEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRUZXh0LT5uZXcoCiAgICAgICAgJVBhcmFtLAogICAgKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgIyBjaGVjayBuZWVkZWQgb2JqZWN0cwogICAgZm9yIG15ICRPYmplY3QgKHF3KENvbmZpZ09iamVjdCBMb2dPYmplY3QgTWFpbk9iamVjdCBQYXJhbU9iamVjdCBMYXlvdXRPYmplY3QpKSB7CiAgICAgICAgJFNlbGYtPnskT2JqZWN0fSA9ICRQYXJhbXskT2JqZWN0fSB8fCBkaWUgIkdvdCBubyAkT2JqZWN0ISI7CiAgICB9CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aXRlbSBGb3JtSW5wdXRDcmVhdGUoKQoKY3JlYXRlIGEgaW5wdXQgc3RyaW5nCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkZvcm1JbnB1dENyZWF0ZSgKICAgICAgICBJdGVtICAgPT4gJEl0ZW1SZWYsCiAgICAgICAgUHJlZml4ID0+ICdQcmVmaXg6OicsICAjIChvcHRpb25hbCkKICAgICAgICBWYWx1ZSAgPT4gJ1ZhbHVlJywgICAgICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBGb3JtSW5wdXRDcmVhdGUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgaWYgKCAhJFBhcmFte0l0ZW19ICkgewogICAgICAgICRTZWxmLT57TG9nT2JqZWN0fS0+TG9nKCBQcmlvcml0eSA9PiAnZXJyb3InLCBNZXNzYWdlID0+ICdOZWVkIEl0ZW0hJyApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAkUGFyYW17UHJlZml4fSB8fD0gJyc7CgogICAgbXkgJFZhbHVlID0gJFBhcmFte1ZhbHVlfSB8fCAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntWYWx1ZURlZmF1bHR9OwogICAgbXkgJFNpemUgPSAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntTaXplfSB8fCA0MDsKICAgIG15ICRTaXplQ2xhc3M7CiAgICBpZiAoICRTaXplIDwgMTUgKSB7CiAgICAgICAgJFNpemVDbGFzcyA9ICdXMTBwYyc7CiAgICB9CiAgICBlbHNpZiAoICRTaXplIDwgMzUgKSB7CiAgICAgICAgJFNpemVDbGFzcyA9ICdXMzNwYyc7CiAgICB9CiAgICBlbHNpZiAoICRTaXplIDwgNTAgKSB7CiAgICAgICAgJFNpemVDbGFzcyA9ICdXNTBwYyc7CiAgICB9CiAgICBlbHNlIHsKICAgICAgICAkU2l6ZUNsYXNzID0gJ1c3NXBjJzsKICAgIH0KCiAgICAjIHByZXBhcmUgZGF0YQogICAgbXkgJElEID0gKCAkUGFyYW17UHJlZml4fSB8fCAnJyApIC4gKCAkUGFyYW17SXRlbX0tPntLZXl9ICk7CiAgICBteSAkTmFtZSA9ICggJFBhcmFte1ByZWZpeH0gfHwgJycgKSAuICggJFBhcmFte05hbWV9IHx8ICRJRCApOwogICAgbXkgJENsYXNzID0gKCAkU2l6ZUNsYXNzIHx8ICcnICkgLiAoICRQYXJhbXtDbGFzc30gfHwgJycgKTsKCiAgICBteSAkU3RyaW5nCiAgICAgICAgPSAiPGlucHV0IGlkPVwiJElEXCIgdHlwZT1cInRleHRcIiBuYW1lPVwiJE5hbWVcIiBjbGFzcz1cIiRDbGFzc1wiICI7CgogICAgaWYgKCRWYWx1ZSkgewoKICAgICAgICAjIHRyYW5zbGF0ZQogICAgICAgIGlmICggJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VHJhbnNsYXRpb259ICkgewogICAgICAgICAgICAkVmFsdWUgPSAkU2VsZi0+e0xheW91dE9iamVjdH0tPntMYW5ndWFnZU9iamVjdH0tPkdldCgkVmFsdWUpOwogICAgICAgIH0KCiAgICAgICAgIyB0cmFuc2Zvcm0gYXNjaWkgdG8gaHRtbAogICAgICAgICRWYWx1ZSA9ICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+QXNjaWkySHRtbCgKICAgICAgICAgICAgVGV4dCAgICAgICAgICAgPT4gJFZhbHVlLAogICAgICAgICAgICBIVE1MUmVzdWx0TW9kZSA9PiAxLAogICAgICAgICk7CgogICAgICAgICRTdHJpbmcgLj0gInZhbHVlPVwiJFZhbHVlXCIgIjsKICAgIH0KCiAgICAjIGFkZCBtYXhpbXVtIGxlbmd0aAogICAgaWYgKCAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntNYXhMZW5ndGh9ICkgewogICAgICAgICRTdHJpbmcgLj0gIm1heGxlbmd0aD1cIiRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e01heExlbmd0aH1cIiAiOwogICAgfQoKICAgICRTdHJpbmcgLj0gIi8+ICI7CgogICAgcmV0dXJuICRTdHJpbmc7Cn0KCj1pdGVtIEZvcm1EYXRhR2V0KCkKCmdldCBmb3JtIGRhdGEKCiAgICBteSAkRm9ybURhdGEgPSAkQmFja2VuZE9iamVjdC0+Rm9ybURhdGFHZXQoCiAgICAgICAgSXRlbSAgID0+ICRJdGVtUmVmLAogICAgICAgIFByZWZpeCA9PiAnUHJlZml4OjonLCAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1EYXRhR2V0IHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZyggUHJpb3JpdHkgPT4gJ2Vycm9yJywgTWVzc2FnZSA9PiAnTmVlZCBJdGVtIScgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgJFBhcmFte1ByZWZpeH0gfHw9ICcnOwoKICAgICMgZ2V0IGZvcm0gZGF0YQogICAgbXkgJEZvcm1EYXRhID0gJFNlbGYtPntQYXJhbU9iamVjdH0tPkdldFBhcmFtKAogICAgICAgIFBhcmFtID0+ICRQYXJhbXtQcmVmaXh9IC4gJFBhcmFte0l0ZW19LT57S2V5fSwKICAgICk7CgogICAgIyByZWdleCBjaGVjawogICAgaWYgKCAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntSZWdleH0gJiYgJEZvcm1EYXRhICF+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1JlZ2V4fSApIHsKCiAgICAgICAgJFBhcmFte0l0ZW19LT57Rm9ybX0tPntJbnZhbGlkfSA9IDE7CiAgICAgICAgcmV0dXJuICRGb3JtRGF0YTsKICAgIH0KCiAgICByZXR1cm4gJEZvcm1EYXRhIGlmICRGb3JtRGF0YTsKICAgIHJldHVybiAkRm9ybURhdGEgaWYgISRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1JlcXVpcmVkfTsKCiAgICAjIHNldCBpbnZhbGlkIHBhcmFtCiAgICAkUGFyYW17SXRlbX0tPntGb3JtfS0+e0ludmFsaWR9ID0gMTsKCiAgICByZXR1cm4gJEZvcm1EYXRhOwp9CgoxOwoKPWJhY2sKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKEw8aHR0cDovL290cnMub3JnLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQ+LgoKPWN1dAo=
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9MYXlvdXRJbXBvcnRFeHBvcnQucG0gLSBwcm92aWRlcyBnZW5lcmljIEhUTUwgb3V0cHV0IGZvciBJbXBvcnRFeHBvcnQKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMyBPVFJTIEFHLCBodHRwOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXRJbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgo9b3ZlcgoKPWl0ZW0gSW1wb3J0RXhwb3J0Rm9ybUlucHV0Q3JlYXRlKCkKCnJldHVybnMgYSBpbnB1dCBmaWVsZCBodG1sIHN0cmluZwoKICAgIG15ICRTdHJpbmcgPSAkTGF5b3V0T2JqZWN0LT5JbXBvcnRFeHBvcnRGb3JtSW5wdXRDcmVhdGUoCiAgICAgICAgSXRlbSAgPT4gJEl0ZW1SZWYsCiAgICAgICAgVmFsdWUgPT4gJ1ZhbHVlJywgICAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgSW1wb3J0RXhwb3J0Rm9ybUlucHV0Q3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgSXRlbSEnCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBsb2FkIGJhY2tlbmQKICAgIG15ICRCYWNrZW5kT2JqZWN0ID0gJFNlbGYtPl9JbXBvcnRFeHBvcnRMb2FkTGF5b3V0QmFja2VuZCgKICAgICAgICBUeXBlID0+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1R5cGV9LAogICAgKTsKCiAgICByZXR1cm4gJycgaWYgISRCYWNrZW5kT2JqZWN0OwoKICAgICMgbG9va3VwIGl0ZW0gdmFsdWUKICAgIG15ICRTdHJpbmcgPSAkQmFja2VuZE9iamVjdC0+Rm9ybUlucHV0Q3JlYXRlKCVQYXJhbSk7CgogICAgcmV0dXJuICRTdHJpbmc7Cn0KCj1pdGVtIEltcG9ydEV4cG9ydEZvcm1EYXRhR2V0KCkKCnJldHVybnMgdGhlIHZhbHVlcyBmcm9tIHRoZSBodG1sIGZvcm0gYXMgaGFzaCByZWZlcmVuY2UKCiAgICBteSAkRm9ybURhdGEgPSAkTGF5b3V0T2JqZWN0LT5JbXBvcnRFeHBvcnRGb3JtRGF0YUdldCgKICAgICAgICBJdGVtID0+ICRJdGVtUmVmLAogICAgKTsKCj1jdXQKCnN1YiBJbXBvcnRFeHBvcnRGb3JtRGF0YUdldCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coIFByaW9yaXR5ID0+ICdlcnJvcicsIE1lc3NhZ2UgPT4gJ05lZWQgSXRlbSEnICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICMgbG9hZCBiYWNrZW5kCiAgICBteSAkQmFja2VuZE9iamVjdCA9ICRTZWxmLT5fSW1wb3J0RXhwb3J0TG9hZExheW91dEJhY2tlbmQoCiAgICAgICAgVHlwZSA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntUeXBlfSwKICAgICk7CgogICAgcmV0dXJuIGlmICEkQmFja2VuZE9iamVjdDsKCiAgICAjIGdldCBmb3JtIGRhdGEKICAgIG15ICRGb3JtRGF0YSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtRGF0YUdldCglUGFyYW0pOwoKICAgIHJldHVybiAkRm9ybURhdGE7Cn0KCj1pdGVtIF9JbXBvcnRFeHBvcnRMb2FkTGF5b3V0QmFja2VuZCgpCgp0byBsb2FkIGEgaW1wb3J0L2V4cG9ydCBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCiAgICBteSAkQmFja2VuZCA9ICRMYXlvdXRPYmplY3QtPl9JbXBvcnRFeHBvcnRMb2FkTGF5b3V0QmFja2VuZCgKICAgICAgICBUeXBlID0+ICdTZWxlY3Rpb24nLAogICAgKTsKCkFuIGluc3RhbmNlIG9mIHRoZSBsb2FkZWQgYmFja2VuZCBtb2R1bGUgaXMgcmV0dXJuZWQuCgo9Y3V0CgpzdWIgX0ltcG9ydEV4cG9ydExvYWRMYXlvdXRCYWNrZW5kIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgaWYgKCAhJFBhcmFte1R5cGV9ICkgewogICAgICAgICRTZWxmLT57TG9nT2JqZWN0fS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAnTmVlZCBUeXBlIScsCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgbXkgJEdlbmVyaWNNb2R1bGUgPSAiS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydExheW91dCRQYXJhbXtUeXBlfSI7CgogICAgIyBsb2FkIHRoZSBiYWNrZW5kIG1vZHVsZQogICAgaWYgKCAhJFNlbGYtPntNYWluT2JqZWN0fS0+UmVxdWlyZSgkR2VuZXJpY01vZHVsZSkgKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJDYW4ndCBsb2FkIGJhY2tlbmQgbW9kdWxlICRQYXJhbXtUeXBlfSEiCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBjcmVhdGUgbmV3IGluc3RhbmNlCiAgICBteSAkQmFja2VuZE9iamVjdCA9ICRHZW5lcmljTW9kdWxlLT5uZXcoCiAgICAgICAgJXskU2VsZn0sCiAgICAgICAgJVBhcmFtLAogICAgICAgIExheW91dE9iamVjdCA9PiAkU2VsZiwKICAgICk7CgogICAgaWYgKCAhJEJhY2tlbmRPYmplY3QgKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJDYW4ndCBjcmVhdGUgYSBuZXcgaW5zdGFuY2Ugb2YgYmFja2VuZCBtb2R1bGUgJFBhcmFte1R5cGV9ISIsCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgcmV0dXJuICRCYWNrZW5kT2JqZWN0Owp9CgoxOwoKPWJhY2sK
IyAtLQojIEFBQUltcG9ydEV4cG9ydC5kdGwgLSAkVGV4dCB2YXJpYWJsZSBmaWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKIyBDb21tb24KJFRleHR7IkFkZCBtYXBwaW5nIHRlbXBsYXRlIn0KJFRleHR7IkNoYXJzZXQifQokVGV4dHsiQ29sb24gKDopIn0KJFRleHR7IkNvbHVtbiJ9CiRUZXh0eyJDb2x1bW4gU2VwYXJhdG9yIn0KJFRleHR7IkRvdCAoLikifQokVGV4dHsiU2VtaWNvbG9uICg7KSJ9CiRUZXh0eyJUYWJ1bGF0b3IgKFRBQikifQokVGV4dHsiSW5jbHVkZSBDb2x1bW4gSGVhZGVycyJ9CiRUZXh0eyJJbXBvcnQgc3VtbWFyeSBmb3IifQokVGV4dHsiSW1wb3J0ZWQgcmVjb3JkcyJ9CiRUZXh0eyJFeHBvcnRlZCByZWNvcmRzIn0KJFRleHR7IlJlY29yZHMifQokVGV4dHsiU2tpcHBlZCJ9Cg==
# --
# AdminImportExport.dtl - provides HTML form for AdminImportExport
# Copyright (C) 2001-2013 OTRS AG, http://otrs.com/
# --
# 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 -->
<div class="MainBox ARIARoleMain LayoutFixedSidebar SidebarFirst">
    <h1>$Text{"Import/Export Management"}</h1>

    <div class="SidebarColumn">

<!-- dtl:block:ActionList -->
        <div class="WidgetSimple">
            <div class="Header">
                <h2>$Text{"Actions"}</h2>
            </div>
            <div class="Content">
                <ul class="ActionList">
<!-- dtl:block:ActionOverview -->
                    <li>
                        <a href="$Env{"Baselink"}Action=$Env{"Action"}" class="CallForAction"><span>$Text{"Go to overview"}</span></a>
                    </li>
<!-- dtl:block:ActionOverview -->
<!-- dtl:block:ActionAdd -->
                    <li>
                        <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=TemplateEdit1" class="CallForAction Plus"><span>$Text{"Add template"}</span></a>
                    </li>
<!-- dtl:block:ActionAdd -->
                </ul>
            </div>
        </div>
<!-- dtl:block:ActionList -->
        <div class="WidgetSimple">
            <div class="Header">
                <h2>$Text{"Note"}</h2>
            </div>
            <div class="Content">
                <p class="FieldExplanation">
                    $Text{"Create a template to import and export object information."}
                </p>
            </div>
        </div>
    </div>

    <div class="ContentColumn">
        <div class="WidgetSimple">
<!-- dtl:block:OverviewList -->
            <div class="Header">
                <h2>$Text{"$Data{"ObjectName"}"}</h2>
            </div>
            <div class="Content">
                <table class="DataTable">
                    <thead>
                        <tr>
                            <th>$Text{"Number"}</th>
                            <th>$Text{"Name"}</th>
                            <th>$Text{"Format"}</th>
                            <th>$Text{"valid"}/$Text{"invalid"}</th>
                            <th>$Text{"Delete"}</th>
                            <th>$Text{"Start Import"}</th>
                            <th>$Text{"Start Export"}</th>
                        </tr>
                    </thead>
                    <tbody>
<!-- dtl:block:NoDataFoundMsg -->
                        <tr>
                            <td colspan="7">
                                $Text{"No data found."}
                            </td>
                        </tr>
<!-- dtl:block:NoDataFoundMsg -->
<!-- dtl:block:OverviewListRow -->
                        <tr>
                            <td>
                                <a class="AsBlock" href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=TemplateEdit1;TemplateID=$LQData{"TemplateID"}">
                                    $QData{"Number"}
                                </a>
                            </td>
                            <td>$QData{"Name"}</td>
                            <td>$Text{"$Data{"FormatName"}"}</td>
                            <td>$Text{"$Data{"Valid"}"}</td>
                            <td>
                                <a class="TrashCan" title="Delete Template" href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=TemplateDelete;TemplateID=$LQData{"TemplateID"}">
                                    $Text{"Delete Template"}
                                </a>
                            </td>
                            <td>
                                <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=ImportInformation;TemplateID=$LQData{"TemplateID"}">
                                    $Text{"Import"}
                                </a>
                            </td>
                            <td>
                                <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction=Export;TemplateID=$LQData{"TemplateID"}">
                                    $Text{"Export"}
                                </a>
                            </td>
                        </tr>
<!-- dtl:block:OverviewListRow -->
                    </tbody>
                </table>
            </div>
<!-- dtl:block:OverviewList -->

<!-- dtl:block:TemplateEdit1 -->
            <div class="Header">
                <h2>$Text{"Step"} 1 $Text{"of"} 5 - $Text{"Edit common information"}:</h2>
            </div>
            <div class="Content">
                <form action="$Env{"CGIHandle"}" method="post" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="$Env{"Action"}" />
                    <input type="hidden" name="Subaction" value="TemplateEdit1" />
                    <input type="hidden" name="TemplateID" value="$Data{"TemplateID"}" />

                    <fieldset class="TableLike">

                        <label for="Name">$Text{"Name"}:</label>
                        <div class="Field">

<!-- Validate_Required -->
                            <input id="Name" class="$Data{"NameClass"}" type="text" name="Name" value="$QData{"Name"}" size="50" maxlength="200" />
                            <div id="NameError" class="TooltipErrorMessage">
                                <p>$Text{"Name is required!"}</p>
                            </div>
                            <div id="NameServerError" class="TooltipErrorMessage">
                                <p>$Text{"Name is required!"}</p>
                            </div>

                        </div>
                        <div class="Clear"></div>

<!-- dtl:block:NewObjectFormat -->
                        <label for="Object">$Text{"Object"}:</label>
                        <div class="Field">

                            $Data{"ObjectOption"}
                            <div id="ObjectError" class="TooltipErrorMessage">
                                <p>$Text{"Object is required!"}</p>
                            </div>
                            <div id="ObjectServerError" class="TooltipErrorMessage">
                                <p>$Text{"Object is required!"}</p>
                            </div>

                        </div>
                        <div class="Clear"></div>

                        <label for="Format">$Text{"Format"}:</label>
                        <div class="Field">

                            $Data{"FormatOption"}
                            <div id="FormatError" class="TooltipErrorMessage">
                                <p>$Text{"Format is required!"}</p>
                            </div>
                            <div id="FormatServerError" class="TooltipErrorMessage">
                                <p>$Text{"Format is required!"}</p>
                            </div>

                        </div>
<!-- dtl:block:NewObjectFormat -->
<!-- dtl:block:EditObjectFormat -->
                        <label>$Text{"Object"}:</label>
                        <div class="Field">
                            <span>$QData{"ObjectName"}</span>
                            <input type="hidden" name="Object" value="$QData{"Object"}" />
                        </div>
                        <div class="Clear"></div>

                        <label>$Text{"Format"}:</label>
                        <div class="Field">
                            <span>$QData{"FormatName"}</span>
                            <input type="hidden" name="Format" value="$QData{"Format"}" />
                        </div>
<!-- dtl:block:EditObjectFormat -->

                        <div class="Clear"></div>

                        <label for="ValidID">$Text{"Valid"}:</label>
                        <div class="Field">
                            $Data{"ValidOptionStrg"}
                        </div>
                        <div class="Clear"></div>

                        <label for="Comment">$Text{"Comment"}:</label>
                        <div class="Field">
                            <input id="Comment" type="text" name="Comment" value="$QData{"Comment"}" size="50" maxlength="200" />
                        </div>
                        <div class="Clear"></div>

                        <div class="Field SpacingTop">
                            <button class="Primary" type="submit" name="SubmitNext" value="SubmitNext">$Text{"Next"}</button>
                            $Text{"or"}
                            <a href="$Env{"Baselink"}Action=$Env{"Action"};Subaction='Overview'">$Text{"Cancel"} </a>
                        </div>
                    </fieldset>
                </form>
            </div>
<!-- dtl:block:TemplateEdit1 -->

<!-- dtl:block:TemplateEdit2 -->
            <div class="Header">
                <h2>$Text{"Step"} 2 $Text{"of"} 5 - $Text{"Edit object information"}:</h2>
            </div>
            <div class="Content">
                <form action="$Env{"CGIHandle"}" method="post" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="$Env{"Action"}" />
                    <input type="hidden" name="Subaction" value="TemplateEdit2" />
                    <input type="hidden" name="TemplateID" value="$QData{"TemplateID"}" />

                    <fieldset class="TableLike">

                        <label>$Text{"Name"}:</label>
                        <div class="Field">
                            $QData{"Name"}
                        </div>
                        <div class="Clear"></div>

                        <label>$Text{"Object"}:</label>
                        <div class="Field">
                            $QData{"Object"}
                        </div>
                        <div class="Clear"></div>

<!-- dtl:block:TemplateEdit2Element -->
                        <label for="$Data{"ID"}">$Text{"$Data{"Name"}"}: </label>
                        <div class="Field">
                            $Data{"InputStrg"}
                            <div id="$Data{"ID"}Error" class="TooltipErrorMessage">
                                <p>$Text{"$Data{"ErrorMessage"}"}</p>
                            </div>
                            <div id="$Data{"ID"}ServerError" class="TooltipErrorMessage">
                                <p>$Text{"$Data{"ErrorMessage"}"}</p>
                            </div>
                        </div>
<!-- dtl:block:TemplateEdit2Element -->
                        <div class="Field SpacingTop">
                            <button class="Back" type="button" name="Back" >$Text{"Back"}</button>
                            <button class="Primary" type="submit" name="SubmitNext" value="SubmitNext" >$Text{"Next"}</button>
                        </div>
                    </fieldset>
                </form>
            </div>
<!-- dtl:block:TemplateEdit2 -->

<!-- dtl:block:TemplateEdit3 -->
            <div class="Header">
                <h2>$Text{"Step"} 3 $Text{"of"} 5 - $Text{"Edit format information"}:</h2>
            </div>
            <div class="Content">
                <form action="$Env{"CGIHandle"}" method="post" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="$Env{"Action"}" />
                    <input type="hidden" name="Subaction" value="TemplateEdit3" />
                    <input type="hidden" name="TemplateID" value="$QData{"TemplateID"}" />

                    <fieldset class="TableLike">

                        <label>$Text{"Name"}:</label>
                        <div class="Field">
                            $QData{"Name"}
                        </div>
                        <div class="Clear"></div>

                        <label>$Text{"Format"}:</label>
                        <div class="Field">
                            $QData{"Format"}
                        </div>
                        <div class="Clear"></div>

<!-- dtl:block:TemplateEdit3Element -->
                        <label for="$Data{"ID"}">$Text{"$Data{"Name"}"}: </label>
                        <div class="Field">
                            $Data{"InputStrg"}
<!-- dtl:block:TemplateEdit3ElementRequired -->
                            <div id="$Data{"ID"}Error" class="TooltipErrorMessage">
                                <p>$Text{"$Data{"Name"} is required!"}</p>
                            </div>
                            <div id="$Data{"ID"}ServerError" class="TooltipErrorMessage">
                                <p>$Text{"$Data{"Name"} is required!"}</p>
                            </div>
<!-- dtl:block:TemplateEdit3ElementRequired -->
                        </div>
<!-- dtl:block:TemplateEdit3Element -->
                        <div class="Field SpacingTop">
                            <button class="Back" type="button" name="Back" >$Text{"Back"}</button>
                            <button class="Primary" type="submit" name="SubmitNext" value="SubmitNext">$Text{"Next"}</button>
                        </div>
                    </fieldset>
                </form>
            </div>
<!-- dtl:block:TemplateEdit3 -->

<!-- dtl:block:TemplateEdit4 -->
            <div class="Header">
                <h2>$Text{"Step"} 4 $Text{"of"} 5 - $Text{"Edit mapping information"}:</h2>
            </div>
            <div class="Content">
                <div class="MapHeaderRow SpacingTop">
                    <div class="Header">
                        <label>$Text{"Name"}:</label>
                        <div class="Field">
                            $QData{"Name"}
                        </div>
                    </div>

                    <div class="Header">
                        <label>$Text{"Object"}:</label>
                        <div class="Field">
                            $QData{"ObjectName"}
                        </div>
                    </div>

                    <div class="Header">
                        <label>$Text{"Format"}:</label>
                        <div class="Field">
                            $QData{"FormatName"}
                        </div>
                    </div>
                </div>
                <form action="$Env{"CGIHandle"}" method="post" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="$Env{"Action"}" />
                    <input type="hidden" name="Subaction" value="TemplateSave4" />
                    <input type="hidden" name="TemplateID" value="$QData{"TemplateID"}" />
                    <table class="DataTable SpacingTop">
                        <thead>
                            <tr>
<!-- dtl:block:TemplateEdit4TableHeader -->
                                <th class="Center">$Text{"$Data{"Header"}"}</th>
<!-- dtl:block:TemplateEdit4TableHeader -->
                            </tr>
                        </thead>
                        <tbody>
<!-- dtl:block:TemplateEdit4NoMapFound -->
                            <tr>
                                <td colspan="$Data{"Columns"}">
                                    $Text{"No map elements found."}
                                </td>
                            </tr>
<!-- dtl:block:TemplateEdit4NoMapFound -->

<!-- dtl:block:TemplateEdit4Row -->
                            <tr>
<!-- dtl:block:TemplateEdit4Column -->
                                <td class="Center">
                                    $Data{"InputStrg"}
                                </td>
<!-- dtl:block:TemplateEdit4Column -->
<!-- dtl:block:TemplateEdit4MapNumberColumn -->
                                <td class="Center">
                                    $Data{"InputStrg"}
                                </td>
<!-- dtl:block:TemplateEdit4MapNumberColumn-->
                                <td class="Center">
<!-- dtl:block:TemplateEdit4UpButton-->
                                    <button class="ArrowUp"type="submit" name="MappingUp::$QData{"MappingID"}" value="$Text{"Up"}"> $Text{"Up"} </button>
<!-- dtl:block:TemplateEdit4UpButton-->
<!-- dtl:block:TemplateEdit4NoUpButton-->
                                    <button class="ArrowUp" type="submit" disabled="disabled"> $Text{"Up"}</button>
<!-- dtl:block:TemplateEdit4NoUpButton-->
                                </td>
                                <td class="Center">
<!-- dtl:block:TemplateEdit4DownButton-->
                                    <button class="ArrowDown" type="submit" name="MappingDown::$QData{"MappingID"}" value="$Text{"Down"}"> $Text{"Down"}</button>
<!-- dtl:block:TemplateEdit4DownButton-->
<!-- dtl:block:TemplateEdit4NoDownButton-->
                                    <button class="ArrowDown" type="submit" disabled="disabled"> $Text{"Down"}</button>
<!-- dtl:block:TemplateEdit4NoDownButton-->
                                </td>
                                <td class="Center">
                                    <button class="TrashCan" type="submit" name="MappingDelete::$QData{"MappingID"}" value="$Text{"Delete"}"> $Text{"Delete"} </button>
                                </td>
                            </tr>
<!-- dtl:block:TemplateEdit4Row -->
                        </tbody>
                    </table>
                    <div class="W100pc SpacingTopSmall Left">
                        <button class="CallForAction Plus" id="MappingAdd" type="submit" name="MappingAdd" value="$Text{"Add"}"><span> $Text{"Add Mapping Element"}</span> </button>
                    </div>

                    <div class="SpacingTop">
                        <button type="submit" name="SubmitBack" value="SubmitBack">$Text{"Back"}</button>
                        <button class="Primary" type="submit" name="SubmitNext" value="SubmitNext">$Text{"Next"}</button>
                    </div>
                </form>
            </div>

<!-- dtl:js_on_document_complete -->
<script type="text/javascript">//<![CDATA[

    // find the next button and get the first column dropdown
    var $NextButton = $("button.Primary[name='SubmitNext']").first(),
        $FirstColumn = $('#Object\\:\\:0\\:\\:Key');

    // handle changes to the first column selector
    $FirstColumn.bind('change', function (Event) {

    // check if there is at least one column with a value
    if ( $FirstColumn.val() ) {
        // we remove the disabled attribute
        $NextButton.removeAttr("disabled");
    }
    else {
        // we add the disabled attribute
        $NextButton.attr("disabled", "disabled");
    }

}).trigger('change');

//]]></script>
<!-- dtl:js_on_document_complete -->

<!-- dtl:block:TemplateEdit4 -->

<!-- dtl:block:TemplateEdit5 -->
            <div class="Header">
                <h2>$Text{"Step"} 5 $Text{"of"} 5 - $Text{"Edit search information"}:</h2>
            </div>
            <div class="Content">
                <form action="$Env{"CGIHandle"}" method="post" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="$Env{"Action"}" />
                    <input type="hidden" name="Subaction" value="TemplateSave5" />
                    <input type="hidden" name="TemplateID" value="$QData{"TemplateID"}" />
                    <fieldset class="TableLike">
                        <label>$Text{"Template Name"}:</label>
                        <div class="Field">
                            $QData{"Name"}
                        </div>
                        <div class="Clear"></div>

                        <label for="RestrictExport">$Text{"Restrict export per search"}:</label>
                        <div class="Field SpacingBottom">
                            $Data{"RestrictExportStrg"}
                        </div>
                        <div class="Clear"></div>

<!-- dtl:block:TemplateEdit5Element -->
                        <label for="$Data{"ID"}">$Text{"$Data{"Name"}"}: </label>
                        <div class="Field">
                            $Data{"InputStrg"}
                        </div>
<!-- dtl:block:TemplateEdit5Element -->

                        <div class="Field SpacingTop">
                            <button type="submit" name="SubmitBack" value="SubmitBack">$Text{"Back"}</button>
                            <button class="Primary" type="submit" name="SubmitNext" value="SubmitNext">$Text{"Finish"}</button>
                        </div>
                    </fieldset>
                </form>
            </div>
<!-- dtl:block:TemplateEdit5 -->

<!-- dtl:block:ImportInformation -->
            <div class="Header">
                <h2>$Text{"Import information"}:</h2>
            </div>
            <div class="Content">
                <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="$Env{"Action"}" />
                    <input type="hidden" name="Subaction" value="Import" />
                    <input type="hidden" name="TemplateID" value="$QData{"TemplateID"}" />

                    <fieldset class="TableLike">

                        <label for="SourceFile">$Text{"Name"}:</label>
                        <div class="Field">
                            $QData{"Name"}
                        </div>
                        <div class="Clear"></div>

                        <label for="SourceFile">$Text{"Source File"}:</label>
                        <div class="Field">
                            <input type="file" name="SourceFile" size="40" class="fixed" />
                        </div>
                        <div class="Clear"></div>

                        <div class="Field SpacingTop">
                            <button class="Primary" type="submit" value="$Text{"Start Import"}">$Text{"Start Import"}</button>
                        </div>
                    </fieldset>
                </form>
            </div>
<!-- dtl:block:ImportInformation -->
        </div>
    </div>

<!--dtl:js_on_document_complete-->
    <script type="text/javascript">//<![CDATA[
        $('button.Back').bind('click', function () {
            location.href = "$Env{"Baselink"}$QData{"BackURL"}";
        });
    //]]></script>
<!--dtl:js_on_document_complete-->

<!--dtl:js_on_document_complete-->
<script type="text/javascript">//<![CDATA[

    Core.Form.Validate.AddMethod("Validate_NumberBiggerThanZero", function(Value, Element) {
        var Number = parseInt(Value, 10);
        if (isNaN(Number)) {
            return false;
        }

        if (Number > 0) {
            return true;
        }
        return false;

    });

    Core.Form.Validate.AddMethod("Validate_NumberInteger", function(Value, Element) {
        return (Value.match(/^[0-9]+$/)) ? true : false;

    });

    Core.Form.Validate.AddRule("Validate_NumberBiggerThanZero", { Validate_NumberBiggerThanZero: true });
    Core.Form.Validate.AddRule("Validate_NumberInteger", { Validate_NumberInteger: true });
    Core.Form.Validate.AddRule("Validate_NumberIntegerBiggerThanZero", { Validate_NumberInteger: true, Validate_NumberBiggerThanZero: true });

//]]></script>
<!--dtl:js_on_document_complete-->

</div>

<!-- dtl:block:Overview -->

<!-- dtl:block:ImportResult -->
<div class="MainBox AriaRoleMain">
    <div class="W50pc SpacingTopLarge SpacingBottomLarge CenterBox">
        <div class="WidgetSimple">
            <div class="Header">
                <h2>$Text{"Import summary for"} $QData{"Object"}</h2>
            </div>
            <div class="Content">

                <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data">
                    <input type="hidden" name="Action" value="$Env{"Action"}"/>
                    <input type="hidden" name="Subaction" value="Overview"/>

                    <fieldset class="TableLike">
                        <label>$Text{"Records"}:</label>
                        <div class="Value">$QData{"Counter"}</div>
                        <div class="Clear"></div>

                        <label>$Text{"Success"}:</label>
                        <div class="Value">
                            $QData{"Success"}
<!-- dtl:block:ImportResultReturnCode -->
                            ($Text{"$Data{"ReturnCodeName"}"}: $QData{"ReturnCodeCount"})
<!-- dtl:block:ImportResultReturnCode -->
                        </div>
                        <div class="Clear"></div>

                        <label>$Text{"Failed"}:</label>
                        <div class="Value">$QData{"Failed"}</div>
                        <div class="Clear"></div>

<!-- dtl:block:ImportResultDuplicateNames -->
                        <label>$Text{"Duplicate names"}:</label>
                        <div class="Value">$QData{"DuplicateNames"}</div>
                        <div class="Clear"></div>
<!-- dtl:block:ImportResultDuplicateNames -->

<!-- dtl:block:ImportResultLastLineNumber -->
                        <label>$Text{"Last processed line number of import file"}:</label>
                        <div class="Value">$QData{"LastLineNumber"}</div>
                        <div class="Clear"></div>
<!-- dtl:block:ImportResultLastLineNumber -->
                    </fieldset>

                    <p class="Center SpacingTopSmall">
                        <button type="submit" name="Ok" value="$Text{"Ok"}">$Text{"Ok"}</button>
                    </p>
                </form>
            </div>
        </div>
    </div>
</div>
<!-- dtl:block:ImportResult -->

# --
# Kernel/System/ImportExport.pm - all import and export functions
# Copyright (C) 2001-2013 OTRS AG, http://otrs.com/
# --
# 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::ImportExport;

use strict;
use warnings;

use Kernel::System::CheckItem;

=head1 NAME

Kernel::System::ImportExport - import, export lib

=head1 SYNOPSIS

All import and export 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::DB;
    use Kernel::System::ImportExport;
    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 $ImportExportObject = Kernel::System::ImportExport->new(
        ConfigObject => $ConfigObject,
        EncodeObject => $EncodeObject,
        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 my $Object (qw(ConfigObject EncodeObject LogObject DBObject MainObject)) {
        $Self->{$Object} = $Param{$Object} || die "Got no $Object!";
    }
    $Self->{CheckItemObject} = Kernel::System::CheckItem->new( %{$Self} );

    return $Self;
}

=item TemplateList()

return a list of templates as array reference

    my $TemplateList = $ImportExportObject->TemplateList(
        Object => 'Ticket',  # (optional)
        Format => 'CSV'      # (optional)
        UserID => 1,
    );

=cut

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

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

    # create sql string
    my $SQL = 'SELECT id FROM imexport_template WHERE 1=1 ';
    my @BIND;

    if ( $Param{Object} ) {
        $SQL .= 'AND imexport_object = ? ';
        push @BIND, \$Param{Object};
    }
    if ( $Param{Format} ) {
        $SQL .= 'AND imexport_format = ? ';
        push @BIND, \$Param{Format};
    }

    # add order option
    $SQL .= 'ORDER BY id';

    # ask database
    $Self->{DBObject}->Prepare(
        SQL  => $SQL,
        Bind => \@BIND,
    );

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

    return \@TemplateList;
}

=item TemplateGet()

get a import export template

Return
    $TemplateData{TemplateID}
    $TemplateData{Number}
    $TemplateData{Object}
    $TemplateData{Format}
    $TemplateData{Name}
    $TemplateData{ValidID}
    $TemplateData{Comment}
    $TemplateData{CreateTime}
    $TemplateData{CreateBy}
    $TemplateData{ChangeTime}
    $TemplateData{ChangeBy}

    my $TemplateDataRef = $ImportExportObject->TemplateGet(
        TemplateID => 3,
        UserID     => 1,
    );

=cut

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

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

    # check if result is already cached
    return $Self->{Cache}->{TemplateGet}->{ $Param{TemplateID} }
        if $Self->{Cache}->{TemplateGet}->{ $Param{TemplateID} };

    # ask database
    $Self->{DBObject}->Prepare(
        SQL => 'SELECT id, imexport_object, imexport_format, name, valid_id, comments, '
            . 'create_time, create_by, change_time, change_by FROM imexport_template WHERE id = ?',
        Bind  => [ \$Param{TemplateID} ],
        Limit => 1,
    );

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

        $TemplateData{Number} = sprintf "%06d", $TemplateData{TemplateID};
    }

    # cache the result
    $Self->{Cache}->{TemplateGet}->{ $Param{TemplateID} } = \%TemplateData;

    return \%TemplateData;
}

=item TemplateAdd()

add a new import/export template

    my $TemplateID = $ImportExportObject->TemplateAdd(
        Object  => 'Ticket',
        Format  => 'CSV',
        Name    => 'Template Name',
        ValidID => 1,
        Comment => 'Comment',       # (optional)
        UserID  => 1,
    );

=cut

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

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

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

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

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

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

    # abort insert of new template, if template name already exists
    if ($NoAdd) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message =>
                "Can't add new template! Template with same name already exists in this object.",
        );
        return;
    }

    # insert new template
    return if !$Self->{DBObject}->Do(
        SQL => 'INSERT INTO imexport_template '
            . '(imexport_object, imexport_format, name, valid_id, comments, '
            . 'create_time, create_by, change_time, change_by) VALUES '
            . '(?, ?, ?, ?, ?, current_timestamp, ?, current_timestamp, ?)',
        Bind => [
            \$Param{Object}, \$Param{Format}, \$Param{Name}, \$Param{ValidID},
            \$Param{Comment}, \$Param{UserID}, \$Param{UserID},
        ],
    );

    # find id of new template
    $Self->{DBObject}->Prepare(
        SQL   => 'SELECT id FROM imexport_template WHERE imexport_object = ? AND name = ?',
        Bind  => [ \$Param{Object}, \$Param{Name} ],
        Limit => 1,
    );

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

    return $TemplateID;
}

=item TemplateUpdate()

update a existing import/export template

    my $True = $ImportExportObject->TemplateUpdate(
        TemplateID => 123,
        Name       => 'Template Name',
        ValidID    => 1,
        Comment    => 'Comment',        # (optional)
        UserID     => 1,
    );

=cut

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

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

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

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

    # get the object of this template id
    $Self->{DBObject}->Prepare(
        SQL   => 'SELECT imexport_object FROM imexport_template WHERE id = ?',
        Bind  => [ \$Param{TemplateID} ],
        Limit => 1,
    );

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

    if ( !$Object ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't update template because it hasn't been found!",
        );
        return;
    }

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

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

    if ( !$Update ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message =>
                "Can't update template! Template with same name already exists in this object.",
        );
        return;
    }

    # reset cache
    delete $Self->{Cache}->{TemplateGet}->{ $Param{TemplateID} };

    # update template
    return $Self->{DBObject}->Do(
        SQL => 'UPDATE imexport_template SET name = ?,'
            . 'valid_id = ?, comments = ?, '
            . 'change_time = current_timestamp, change_by = ? '
            . 'WHERE id = ?',
        Bind => [
            \$Param{Name}, \$Param{ValidID}, \$Param{Comment},
            \$Param{UserID}, \$Param{TemplateID},
        ],
    );
}

=item TemplateDelete()

delete existing import/export templates

    my $True = $ImportExportObject->TemplateDelete(
        TemplateID => 123,
        UserID     => 1,
    );

    or

    my $True = $ImportExportObject->TemplateDelete(
        TemplateID => [1,44,166,5],
        UserID     => 1,
    );

=cut

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

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

    if ( !ref $Param{TemplateID} ) {
        $Param{TemplateID} = [ $Param{TemplateID} ];
    }
    elsif ( ref $Param{TemplateID} ne 'ARRAY' ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'TemplateID must be an array reference or a string!',
        );
        return;
    }

    # delete existing search data
    $Self->SearchDataDelete(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # delete all mapping data
    for my $TemplateID ( @{ $Param{TemplateID} } ) {
        $Self->MappingDelete(
            TemplateID => $TemplateID,
            UserID     => $Param{UserID},
        );
    }

    # delete existing format data
    $Self->FormatDataDelete(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # delete existing object data
    $Self->ObjectDataDelete(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # create the template id string
    my $TemplateIDString = join q{, }, map {'?'} @{ $Param{TemplateID} };

    # create and add bind parameters
    my @BIND = map { \$_ } @{ $Param{TemplateID} };

    # reset cache
    delete $Self->{Cache}->{TemplateGet};

    # delete templates
    return $Self->{DBObject}->Do(
        SQL  => "DELETE FROM imexport_template WHERE id IN ( $TemplateIDString )",
        Bind => \@BIND,
    );
}

=item ObjectList()

return a list of available objects as hash reference

    my $ObjectList = $ImportExportObject->ObjectList();

=cut

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

    # get config
    my $ModuleList = $Self->{ConfigObject}->Get('ImportExport::ObjectBackendRegistration');

    return if !$ModuleList;
    return if ref $ModuleList ne 'HASH';

    # create the object list
    my $ObjectList = {};
    for my $Module ( sort keys %{$ModuleList} ) {
        $ObjectList->{$Module} = $ModuleList->{$Module}->{Name};
    }

    return $ObjectList;
}

=item ObjectAttributesGet()

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

    my $Attributes = $ImportExportObject->ObjectAttributesGet(
        TemplateID => 123,
        UserID     => 1,
    );

=cut

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

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

    # get template data
    my $TemplateData = $Self->TemplateGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # check template data
    if ( !$TemplateData || !$TemplateData->{Object} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Template with ID $Param{TemplateID} is incomplete!",
        );
        return;
    }

    # load backend
    my $Backend = $Self->_LoadBackend(
        Module => "Kernel::System::ImportExport::ObjectBackend::$TemplateData->{Object}",
    );

    return if !$Backend;

    # get an attribute list of the object
    my $Attributes = $Backend->ObjectAttributesGet(
        UserID => $Param{UserID},
    );

    return $Attributes;
}

=item ObjectDataGet()

get the object data from a template

    my $ObjectDataRef = $ImportExportObject->ObjectDataGet(
        TemplateID => 3,
        UserID     => 1,
    );

=cut

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

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

    # ask database
    $Self->{DBObject}->Prepare(
        SQL  => 'SELECT data_key, data_value FROM imexport_object WHERE template_id = ?',
        Bind => [ \$Param{TemplateID} ],
    );

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

    return \%ObjectData;
}

=item ObjectDataSave()

save the object data of a template

    my $True = $ImportExportObject->ObjectDataSave(
        TemplateID => 123,
        ObjectData => $HashRef,
        UserID     => 1,
    );

=cut

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

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

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

    # delete existing object data
    $Self->ObjectDataDelete(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    DATAKEY:
    for my $DataKey ( sort keys %{ $Param{ObjectData} } ) {

        my $DataValue = $Param{ObjectData}->{$DataKey};

        next DATAKEY if !defined $DataKey;
        next DATAKEY if !defined $DataValue;

        # insert one row
        $Self->{DBObject}->Do(
            SQL => 'INSERT INTO imexport_object '
                . '(template_id, data_key, data_value) VALUES '
                . '(?, ?, ?)',
            Bind => [ \$Param{TemplateID}, \$DataKey, \$DataValue ],
        );
    }

    return 1;
}

=item ObjectDataDelete()

delete the existing object data of a template

    my $True = $ImportExportObject->ObjectDataDelete(
        TemplateID => 123,
        UserID     => 1,
    );

    or

    my $True = $ImportExportObject->ObjectDataDelete(
        TemplateID => [1,44,166,5],
        UserID     => 1,
    );

=cut

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

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

    if ( !ref $Param{TemplateID} ) {
        $Param{TemplateID} = [ $Param{TemplateID} ];
    }
    elsif ( ref $Param{TemplateID} ne 'ARRAY' ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'TemplateID must be an array reference or a string!',
        );
        return;
    }

    # create the template id string
    my $TemplateIDString = join q{, }, map {'?'} @{ $Param{TemplateID} };

    # create and add bind parameters
    my @BIND = map { \$_ } @{ $Param{TemplateID} };

    # delete templates
    return $Self->{DBObject}->Do(
        SQL  => "DELETE FROM imexport_object WHERE template_id IN ( $TemplateIDString )",
        Bind => \@BIND,
    );
}

=item FormatList()

return a list of available formats as hash reference

    my $FormatList = $ImportExportObject->FormatList();

=cut

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

    # get config
    my $ModuleList = $Self->{ConfigObject}->Get('ImportExport::FormatBackendRegistration');

    return if !$ModuleList;
    return if ref $ModuleList ne 'HASH';

    # create the format list
    my $FormatList = {};
    for my $Module ( sort keys %{$ModuleList} ) {
        $FormatList->{$Module} = $ModuleList->{$Module}->{Name};
    }

    return $FormatList;
}

=item FormatAttributesGet()

get the attributes of a format backend as array/hash reference

    my $Attributes = $ImportExportObject->FormatAttributesGet(
        TemplateID => 123,
        UserID     => 1,
    );

=cut

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

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

    # get template data
    my $TemplateData = $Self->TemplateGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # check template data
    if ( !$TemplateData || !$TemplateData->{Format} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Template with ID $Param{TemplateID} is incomplete!",
        );
        return;
    }

    # load backend
    my $Backend = $Self->_LoadBackend(
        Module => "Kernel::System::ImportExport::FormatBackend::$TemplateData->{Format}",
    );

    return if !$Backend;

    # get an attribute list of the format
    my $Attributes = $Backend->FormatAttributesGet(
        UserID => $Param{UserID},
    );

    return $Attributes;
}

=item FormatDataGet()

get the format data from a template

    my $FormatDataRef = $ImportExportObject->FormatDataGet(
        TemplateID => 3,
        UserID     => 1,
    );

=cut

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

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

    # ask database
    $Self->{DBObject}->Prepare(
        SQL  => 'SELECT data_key, data_value FROM imexport_format WHERE template_id = ?',
        Bind => [ \$Param{TemplateID} ],
    );

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

    return \%FormatData;
}

=item FormatDataSave()

save the format data of a template

    my $True = $ImportExportObject->FormatDataSave(
        TemplateID => 123,
        FormatData => $HashRef,
        UserID     => 1,
    );

=cut

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

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

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

    # delete existing format data
    $Self->FormatDataDelete(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    DATAKEY:
    for my $DataKey ( sort keys %{ $Param{FormatData} } ) {

        my $DataValue = $Param{FormatData}->{$DataKey};

        next DATAKEY if !defined $DataKey;
        next DATAKEY if !defined $DataValue;

        # insert one row
        $Self->{DBObject}->Do(
            SQL => 'INSERT INTO imexport_format '
                . '(template_id, data_key, data_value) VALUES (?, ?, ?)',
            Bind => [ \$Param{TemplateID}, \$DataKey, \$DataValue ],
        );
    }

    return 1;
}

=item FormatDataDelete()

delete the existing format data of a template

    my $True = $ImportExportObject->FormatDataDelete(
        TemplateID => 123,
        UserID     => 1,
    );

    or

    my $True = $ImportExportObject->FormatDataDelete(
        TemplateID => [1,44,166,5],
        UserID     => 1,
    );

=cut

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

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

    if ( !ref $Param{TemplateID} ) {
        $Param{TemplateID} = [ $Param{TemplateID} ];
    }
    elsif ( ref $Param{TemplateID} ne 'ARRAY' ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'TemplateID must be an array reference or a string!',
        );
        return;
    }

    # create the template id string
    my $TemplateIDString = join q{, }, map {'?'} @{ $Param{TemplateID} };

    # create and add bind parameters
    my @BIND = map { \$_ } @{ $Param{TemplateID} };

    # delete templates
    return $Self->{DBObject}->Do(
        SQL  => "DELETE FROM imexport_format WHERE template_id IN ( $TemplateIDString )",
        Bind => \@BIND,
    );
}

=item MappingList()

return a list of mapping data ids sorted by position as array reference

    my $MappingList = $ImportExportObject->MappingList(
        TemplateID => 123,
        UserID     => 1,
    );

=cut

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

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

    # ask database
    $Self->{DBObject}->Prepare(
        SQL  => 'SELECT id FROM imexport_mapping WHERE template_id = ? ORDER BY position',
        Bind => [ \$Param{TemplateID} ],
    );

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

    return \@MappingList;
}

=item MappingAdd()

add a new mapping data row

    my $MappingID = $ImportExportObject->MappingAdd(
        TemplateID => 123,
        UserID     => 1,
    );

=cut

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

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

    # find maximum position
    $Self->{DBObject}->Prepare(
        SQL   => 'SELECT max(position) FROM imexport_mapping WHERE template_id = ?',
        Bind  => [ \$Param{TemplateID} ],
        Limit => 1,
    );

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

        if ( defined $Row[0] ) {
            $NewPosition = $Row[0];
            $NewPosition++;
        }
    }

    # insert a new mapping data row
    return if !$Self->{DBObject}->Do(
        SQL => 'INSERT INTO imexport_mapping (template_id, position) VALUES (?, ?)',
        Bind => [ \$Param{TemplateID}, \$NewPosition ],
    );

    # find id of new mapping data row
    $Self->{DBObject}->Prepare(
        SQL   => 'SELECT id FROM imexport_mapping WHERE template_id = ? AND position = ?',
        Bind  => [ \$Param{TemplateID}, \$NewPosition ],
        Limit => 1,
    );

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

    return $MappingID;
}

=item MappingDelete()

delete existing mapping data rows

    my $True = $ImportExportObject->MappingDelete(
        MappingID  => 123,
        TemplateID => 321,
        UserID     => 1,
    );

    or

    my $True = $ImportExportObject->MappingDelete(
        TemplateID => 321,
        UserID     => 1,
    );

=cut

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

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

    if ( defined $Param{MappingID} ) {

        # delete existing object mapping data
        $Self->MappingObjectDataDelete(
            MappingID => $Param{MappingID},
            UserID    => $Param{UserID},
        );

        # delete existing format mapping data
        $Self->MappingFormatDataDelete(
            MappingID => $Param{MappingID},
            UserID    => $Param{UserID},
        );

        # delete one mapping row
        $Self->{DBObject}->Do(
            SQL  => 'DELETE FROM imexport_mapping WHERE id = ?',
            Bind => [ \$Param{MappingID} ],
        );

        # rebuild mapping positions
        $Self->MappingPositionRebuild(
            TemplateID => $Param{TemplateID},
            UserID     => $Param{UserID},
        );

        return 1;
    }
    else {

        # get mapping list
        my $MappingList = $Self->MappingList(
            TemplateID => $Param{TemplateID},
            UserID     => $Param{UserID},
        );

        for my $MappingID ( @{$MappingList} ) {

            # delete existing object mapping data
            $Self->MappingObjectDataDelete(
                MappingID => $MappingID,
                UserID    => $Param{UserID},
            );

            # delete existing format mapping data
            $Self->MappingFormatDataDelete(
                MappingID => $MappingID,
                UserID    => $Param{UserID},
            );
        }

        # delete all mapping rows of this template
        return $Self->{DBObject}->Do(
            SQL  => 'DELETE FROM imexport_mapping WHERE template_id = ?',
            Bind => [ \$Param{TemplateID} ],
        );
    }
}

=item MappingUp()

move an mapping data row up

    my $True = $ImportExportObject->MappingUp(
        MappingID  => 123,
        TemplateID => 321,
        UserID     => 1,
    );

=cut

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

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

    # get mapping data list
    my $MappingList = $Self->MappingList(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    return 1 if $Param{MappingID} == $MappingList->[0];

    # ask database
    $Self->{DBObject}->Prepare(
        SQL  => 'SELECT position FROM imexport_mapping WHERE id = ?',
        Bind => [ \$Param{MappingID} ],
    );

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

    return 1 if !$Position;

    my $PositionUpper = $Position - 1;

    # update positions
    $Self->{DBObject}->Do(
        SQL => 'UPDATE imexport_mapping SET position = ? WHERE template_id = ? AND position = ?',
        Bind => [ \$Position, \$Param{TemplateID}, \$PositionUpper ],
    );
    $Self->{DBObject}->Do(
        SQL => 'UPDATE imexport_mapping SET position = ? WHERE id = ?',
        Bind => [ \$PositionUpper, \$Param{MappingID} ],
    );

    return 1;
}

=item MappingDown()

move an mapping data row down

    my $True = $ImportExportObject->MappingDown(
        MappingID  => 123,
        TemplateID => 321,
        UserID     => 1,
    );

=cut

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

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

    # get mapping data list
    my $MappingList = $Self->MappingList(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    return 1 if $Param{MappingID} == $MappingList->[-1];

    # ask database
    $Self->{DBObject}->Prepare(
        SQL  => 'SELECT position FROM imexport_mapping WHERE id = ?',
        Bind => [ \$Param{MappingID} ],
    );

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

    my $PositionDown = $Position + 1;

    # update positions
    $Self->{DBObject}->Do(
        SQL => 'UPDATE imexport_mapping SET position = ? WHERE template_id = ? AND position = ?',
        Bind => [ \$Position, \$Param{TemplateID}, \$PositionDown ],
    );
    $Self->{DBObject}->Do(
        SQL => 'UPDATE imexport_mapping SET position = ? WHERE id = ?',
        Bind => [ \$PositionDown, \$Param{MappingID} ],
    );

    return 1;
}

=item MappingPositionRebuild()

rebuild the positions of a mapping list

    my $True = $ImportExportObject->MappingPositionRebuild(
        TemplateID => 123,
        UserID     => 1,
    );

=cut

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

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

    # get mapping data list
    my $MappingList = $Self->MappingList(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # update position
    my $Counter = 0;
    for my $MappingID ( @{$MappingList} ) {
        $Self->{DBObject}->Do(
            SQL => 'UPDATE imexport_mapping SET position = ? WHERE id = ?',
            Bind => [ \$Counter, \$MappingID ],
        );
        $Counter++;
    }

    return 1;
}

=item MappingObjectAttributesGet()

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

    my $Attributes = $ImportExportObject->MappingObjectAttributesGet(
        TemplateID => 123,
        UserID     => 1,
    );

=cut

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

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

    # get template data
    my $TemplateData = $Self->TemplateGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # check template data
    if ( !$TemplateData || !$TemplateData->{Object} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Template with ID $Param{TemplateID} is incomplete!",
        );
        return;
    }

    # load backend
    my $Backend = $Self->_LoadBackend(
        Module => "Kernel::System::ImportExport::ObjectBackend::$TemplateData->{Object}",
    );

    return if !$Backend;

    # get an attribute list of the object
    my $Attributes = $Backend->MappingObjectAttributesGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    return $Attributes;
}

=item MappingObjectDataDelete()

delete the existing object data of a mapping

    my $True = $ImportExportObject->MappingObjectDataDelete(
        MappingID => 123,
        UserID    => 1,
    );

    or

    my $True = $ImportExportObject->MappingObjectDataDelete(
        MappingID => [1,44,166,5],
        UserID    => 1,
    );

=cut

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

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

    if ( !ref $Param{MappingID} ) {
        $Param{MappingID} = [ $Param{MappingID} ];
    }
    elsif ( ref $Param{MappingID} ne 'ARRAY' ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'MappingID must be an array reference or a string!',
        );
        return;
    }

    # create the template id string
    my $MappingIDString = join q{, }, map {'?'} @{ $Param{MappingID} };

    # create and add bind parameters
    my @BIND = map { \$_ } @{ $Param{MappingID} };

    # delete mapping object data
    return $Self->{DBObject}->Do(
        SQL  => "DELETE FROM imexport_mapping_object WHERE mapping_id IN ( $MappingIDString )",
        Bind => \@BIND,
    );
}

=item MappingObjectDataSave()

save the object data of a mapping

    my $True = $ImportExportObject->MappingObjectDataSave(
        MappingID         => 123,
        MappingObjectData => $HashRef,
        UserID            => 1,
    );

=cut

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

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

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

    # delete existing object mapping data
    $Self->MappingObjectDataDelete(
        MappingID => $Param{MappingID},
        UserID    => $Param{UserID},
    );

    DATAKEY:
    for my $DataKey ( sort keys %{ $Param{MappingObjectData} } ) {

        my $DataValue = $Param{MappingObjectData}->{$DataKey};

        next DATAKEY if !defined $DataKey;
        next DATAKEY if !defined $DataValue;

        # insert one mapping object row
        $Self->{DBObject}->Do(
            SQL => 'INSERT INTO imexport_mapping_object '
                . '(mapping_id, data_key, data_value) VALUES (?, ?, ?)',
            Bind => [ \$Param{MappingID}, \$DataKey, \$DataValue ],
        );
    }

    return 1;
}

=item MappingObjectDataGet()

get the object data of a mapping

    my $ObjectDataRef = $ImportExportObject->MappingObjectDataGet(
        MappingID => 123,
        UserID    => 1,
    );

=cut

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

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

    # ask database
    $Self->{DBObject}->Prepare(
        SQL  => 'SELECT data_key, data_value FROM imexport_mapping_object WHERE mapping_id = ?',
        Bind => [ \$Param{MappingID} ],
    );

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

    return \%MappingObjectData;
}

=item MappingFormatAttributesGet()

get the attributes of an format backend as array/hash reference

    my $Attributes = $ImportExportObject->MappingFormatAttributesGet(
        TemplateID => 123,
        UserID     => 1,
    );

=cut

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

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

    # get template data
    my $TemplateData = $Self->TemplateGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # check template data
    if ( !$TemplateData || !$TemplateData->{Format} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Template with ID $Param{TemplateID} is incomplete!",
        );
        return;
    }

    # load backend
    my $Backend = $Self->_LoadBackend(
        Module => "Kernel::System::ImportExport::FormatBackend::$TemplateData->{Format}",
    );

    return if !$Backend;

    # get an attribute list of the format
    my $Attributes = $Backend->MappingFormatAttributesGet(
        UserID => $Param{UserID},
    );

    return $Attributes;
}

=item MappingFormatDataDelete()

delete the existing format data of a mapping

    my $True = $ImportExportObject->MappingFormatDataDelete(
        MappingID => 123,
        UserID    => 1,
    );

    or

    my $True = $ImportExportObject->MappingFormatDataDelete(
        MappingID => [1,44,166,5],
        UserID    => 1,
    );

=cut

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

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

    if ( !ref $Param{MappingID} ) {
        $Param{MappingID} = [ $Param{MappingID} ];
    }
    elsif ( ref $Param{MappingID} ne 'ARRAY' ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'MappingID must be an array reference or a string!',
        );
        return;
    }

    # create the template id string
    my $MappingIDString = join q{, }, map {'?'} @{ $Param{MappingID} };

    # create and add bind parameters
    my @BIND = map { \$_ } @{ $Param{MappingID} };

    # delete mapping format data
    return $Self->{DBObject}->Do(
        SQL  => "DELETE FROM imexport_mapping_format WHERE mapping_id IN ( $MappingIDString )",
        Bind => \@BIND,
    );
}

=item MappingFormatDataSave()

save the format data of a mapping

    my $True = $ImportExportObject->MappingFormatDataSave(
        MappingID         => 123,
        MappingFormatData => $HashRef,
        UserID            => 1,
    );

=cut

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

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

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

    # delete existing format mapping data
    $Self->MappingFormatDataDelete(
        MappingID => $Param{MappingID},
        UserID    => $Param{UserID},
    );

    DATAKEY:
    for my $DataKey ( sort keys %{ $Param{MappingFormatData} } ) {

        my $DataValue = $Param{MappingFormatData}->{$DataKey};

        next DATAKEY if !defined $DataKey;
        next DATAKEY if !defined $DataValue;

        # insert one mapping format row
        $Self->{DBObject}->Do(
            SQL => 'INSERT INTO imexport_mapping_format '
                . '(mapping_id, data_key, data_value) VALUES (?, ?, ?)',
            Bind => [ \$Param{MappingID}, \$DataKey, \$DataValue ],
        );
    }

    return 1;
}

=item MappingFormatDataGet()

get the format data of a mapping

    my $ObjectDataRef = $ImportExportObject->MappingFormatDataGet(
        MappingID => 123,
        UserID    => 1,
    );

=cut

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

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

    # ask database
    $Self->{DBObject}->Prepare(
        SQL  => 'SELECT data_key, data_value FROM imexport_mapping_format WHERE mapping_id = ?',
        Bind => [ \$Param{MappingID} ],
    );

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

    return \%MappingFormatData;
}

=item SearchAttributesGet()

get the search attributes of a object backend as array/hash reference

    my $Attributes = $ImportExportObject->SearchAttributesGet(
        TemplateID => 123,
        UserID     => 1,
    );

=cut

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

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

    # get template data
    my $TemplateData = $Self->TemplateGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # check template data
    if ( !$TemplateData || !$TemplateData->{Object} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Template with ID $Param{TemplateID} is incomplete!",
        );
        return;
    }

    # load backend
    my $Backend = $Self->_LoadBackend(
        Module => "Kernel::System::ImportExport::ObjectBackend::$TemplateData->{Object}",
    );

    return if !$Backend;

    # get an search attribute list of an object
    my $Attributes = $Backend->SearchAttributesGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    return $Attributes;
}

=item SearchDataGet()

get the search data from a template

    my $SearchDataRef = $ImportExportObject->SearchDataGet(
        TemplateID => 3,
        UserID     => 1,
    );

=cut

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

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

    # ask database
    $Self->{DBObject}->Prepare(
        SQL  => 'SELECT data_key, data_value FROM imexport_search WHERE template_id = ?',
        Bind => [ \$Param{TemplateID} ],
    );

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

    return \%SearchData;
}

=item SearchDataSave()

save the search data of a template

    my $True = $ImportExportObject->SearchDataSave(
        TemplateID => 123,
        SearchData => $HashRef,
        UserID     => 1,
    );

=cut

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

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

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

    # delete existing search data
    $Self->SearchDataDelete(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    DATAKEY:
    for my $DataKey ( sort keys %{ $Param{SearchData} } ) {

        # quote
        my $DataValue = $Param{SearchData}->{$DataKey};

        next DATAKEY if !$DataKey;
        next DATAKEY if !$DataValue;

        # insert one row
        $Self->{DBObject}->Do(
            SQL => 'INSERT INTO imexport_search '
                . '(template_id, data_key, data_value) VALUES (?, ?, ?)',
            Bind => [ \$Param{TemplateID}, \$DataKey, \$DataValue ],
        );
    }

    return 1;
}

=item SearchDataDelete()

delete the existing search data of a template

    my $True = $ImportExportObject->SearchDataDelete(
        TemplateID => 123,
        UserID     => 1,
    );

    or

    my $True = $ImportExportObject->SearchDataDelete(
        TemplateID => [1,44,166,5],
        UserID     => 1,
    );

=cut

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

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

    if ( !ref $Param{TemplateID} ) {
        $Param{TemplateID} = [ $Param{TemplateID} ];
    }
    elsif ( ref $Param{TemplateID} ne 'ARRAY' ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'TemplateID must be an array reference or a string!',
        );
        return;
    }

    # create the template id string
    my $TemplateIDString = join q{, }, map {'?'} @{ $Param{TemplateID} };

    # create and add bind parameters
    my @BIND = map { \$_ } @{ $Param{TemplateID} };

    # delete templates
    return $Self->{DBObject}->Do(
        SQL  => "DELETE FROM imexport_search WHERE template_id IN ( $TemplateIDString )",
        Bind => \@BIND,
    );
}

=item Export()

export function

    my $ResultRef = $ImportExportObject->Export(
        TemplateID => 123,
        UserID     => 1,
    );

returns something like

    $ResultRef = {
        Success   => 2,
        Failed    => 0,
        DestinationContent => [
            [ 'Attr_1a', 'Attr_1b', 'Attr_1c', ],
            [ 'Attr_2a', 'Attr_2b', 'Attr_3c', ],
        ],
    };

=cut

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

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

    # get template data
    my $TemplateData = $Self->TemplateGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # check template data
    if ( !$TemplateData || !$TemplateData->{Object} || !$TemplateData->{Format} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Template with ID $Param{TemplateID} is incomplete!",
        );
        return;
    }

    # load object backend
    my $ObjectBackend = $Self->_LoadBackend(
        Module => "Kernel::System::ImportExport::ObjectBackend::$TemplateData->{Object}",
    );

    return if !$ObjectBackend;

    # load format backend
    my $FormatBackend = $Self->_LoadBackend(
        Module => "Kernel::System::ImportExport::FormatBackend::$TemplateData->{Format}",
    );

    return if !$FormatBackend;

    # get export data
    my $ExportData = $ObjectBackend->ExportDataGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # get format data
    my $FormatData = $Self->FormatDataGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # if column headers should be included in the export
    if ( $FormatData->{IncludeColumnHeaders} ) {

        # get object attributes (the name of the columns)
        my $MappingObjectAttributes = $Self->MappingObjectAttributesGet(
            TemplateID => $Param{TemplateID},
            UserID     => $Param{UserID},
        );

        # create a lookup hash for the object attribute names
        my %AttributeLookup
            = map { $_->{Key} => $_->{Value} } @{ $MappingObjectAttributes->[0]->{Input}->{Data} };

        # get mapping data list
        my $MappingList = $Self->MappingList(
            TemplateID => $Param{TemplateID},
            UserID     => $Param{UserID},
        );

        # get the column names
        my @ColumnNames;
        for my $MappingID ( @{$MappingList} ) {

            # get mapping object data
            my $MappingObjectData = $Self->MappingObjectDataGet(
                MappingID => $MappingID,
                UserID    => $Param{UserID},
            );

            # get the column name
            my $ColumnName = $AttributeLookup{ $MappingObjectData->{Key} };

            push @ColumnNames, $ColumnName;
        }

        # add column headers as first row
        unshift @{$ExportData}, \@ColumnNames;
    }

    my %Result = (
        Success            => 0,
        Failed             => 0,
        DestinationContent => [],
    );

    EXPORTDATAROW:
    for my $ExportDataRow ( @{$ExportData} ) {

        # export one row
        my $DestinationContentRow = $FormatBackend->ExportDataSave(
            TemplateID    => $Param{TemplateID},
            ExportDataRow => $ExportDataRow,
            UserID        => $Param{UserID},
        );

        if ( !defined $DestinationContentRow ) {
            $Result{Failed}++;
            next EXPORTDATAROW;
        }

        # add row to destination content
        push @{ $Result{DestinationContent} }, $DestinationContentRow;
        $Result{Success}++;
    }

    # log result
    $Self->{LogObject}->Log(
        Priority => 'notice',
        Message  => "Export of $Result{Failed} records ($TemplateData->{Object}): failed!",
    );
    $Self->{LogObject}->Log(
        Priority => 'notice',
        Message  => "Export of $Result{Success} records ($TemplateData->{Object}): successful!",
    );

    return \%Result;
}

=item Import()

import function

    my $ResultRef = $ImportExportObject->Import(
        TemplateID    => 123,
        SourceContent => $StringRef,  # (optional)
        UserID        => 1,
    );

=cut

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

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

    # get template data
    my $TemplateData = $Self->TemplateGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # check template data
    if ( !$TemplateData || !$TemplateData->{Object} || !$TemplateData->{Format} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Template with ID $Param{TemplateID} is incomplete!",
        );
        return;
    }

    # load object backend
    my $ObjectBackend = $Self->_LoadBackend(
        Module => "Kernel::System::ImportExport::ObjectBackend::$TemplateData->{Object}",
    );

    return if !$ObjectBackend;

    # load format backend
    my $FormatBackend = $Self->_LoadBackend(
        Module => "Kernel::System::ImportExport::FormatBackend::$TemplateData->{Format}",
    );

    return if !$FormatBackend;

    # get import data
    my $ImportData = $FormatBackend->ImportDataGet(
        TemplateID    => $Param{TemplateID},
        SourceContent => $Param{SourceContent},
        UserID        => $Param{UserID},
    );

    return if !$ImportData;

    # get format data
    my $FormatData = $Self->FormatDataGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # if column headers are activated, the first row must be removed
    if ( $FormatData->{IncludeColumnHeaders} ) {
        shift @{$ImportData};
    }

    # Number of successfully and not successfully imported rows
    my %Result = (
        Object  => $TemplateData->{Object},
        Success => 0,
        Failed  => 0,
        RetCode => {},
        Counter => 0,
    );
    IMPORTDATAROW:
    for my $ImportDataRow ( @{$ImportData} ) {

        $Result{Counter}++;

        # import a single row
        my ( $ID, $RetCode ) = $ObjectBackend->ImportDataSave(
            TemplateID    => $Param{TemplateID},
            ImportDataRow => $ImportDataRow,
            Counter       => $Result{Counter},
            UserID        => $Param{UserID},
        );

        if ( !$ID ) {

            # count DuplicateName entries as errors
            if ( $RetCode && $RetCode =~ m{ \A DuplicateName }xms ) {
                $Result{RetCode}->{$RetCode}++;
            }
            $Result{Failed}++;
        }
        else {
            $Result{RetCode}->{$RetCode}++;
            $Result{Success}++;
        }
    }

    # log result
    $Self->{LogObject}->Log(
        Priority => 'notice',
        Message =>
            "Import of $Result{Counter} $Result{Object} records: "
            . "$Result{Failed} failed, $Result{Success} succeeded",
    );
    for my $RetCode ( sort keys %{ $Result{RetCode} } ) {
        my $Count = $Result{RetCode}->{$RetCode} || 0;
        $Self->{LogObject}->Log(
            Priority => 'notice',
            Message =>
                "Import of $Result{Counter} $Result{Object} records: $Count $RetCode",
        );
    }
    if ( $Result{Failed} ) {
        $Self->{LogObject}->Log(
            Priority => 'notice',
            Message  => "Last processed line number of import file: $Result{Counter}",
        );
    }

    return \%Result;
}

=item _LoadBackend()

to load a import/export backend module

    my $Backend = $ImportExportObject->_LoadBackend(
        Module => 'Kernel::System::ImportExport::ObjectBackend::Ticket',
    );

An instance of the loaded backend module is returned.

=cut

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

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

    # load object backend module
    if ( !$Self->{MainObject}->Require( $Param{Module} ) ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't load backend module $Param{Module}!"
        );
        return;
    }

    # create new instance
    my $BackendObject = $Param{Module}->new(
        %{$Self},
        %Param,
        ImportExportObject => $Self,
    );

    if ( !$BackendObject ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't create a new instance of backend module $Param{Module}!",
        );
        return;
    }

    return $BackendObject;
}

1;

=back

=head1 TERMS AND CONDITIONS

This Software is part of the OTRS project (L<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 L<http://www.gnu.org/licenses/agpl.txt>.

=cut

# --
# Kernel/System/ImportExport/FormatBackend/CSV.pm - import/export backend for CSV
# Copyright (C) 2001-2013 OTRS AG, http://otrs.com/
# --
# 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::ImportExport::FormatBackend::CSV;

use strict;
use warnings;

=head1 NAME

Kernel::System::ImportExport::FormatBackend::CSV - import/export backend for CSV

=head1 SYNOPSIS

All functions to import and export a csv format

=over 4

=cut

=item new()

create an object

    use Kernel::Config;
    use Kernel::System::Encode;
    use Kernel::System::Log;
    use Kernel::System::DB;
    use Kernel::System::Main;
    use Kernel::System::ImportExport::FormatBackend::CSV;

    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 $BackendObject = Kernel::System::ImportExport::FormatBackend::CSV->new(
        ConfigObject       => $ConfigObject,
        EncodeObject       => $EncodeObject,
        LogObject          => $LogObject,
        DBObject           => $DBObject,
        MainObject         => $MainObject,
        ImportExportObject => $ImportExportObject,
    );

=cut

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

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

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

    if ( !$Self->{MainObject}->Require('Text::CSV') ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "CPAN module Text::CSV is required to use the CSV import/export module!",
        );
        return;
    }

    # define available separators
    $Self->{AvailableSeparators} = {
        Tabulator => "\t",
        Semicolon => ';',
        Colon     => ':',
        Dot       => '.',
        Comma     => ',',
    };

    return $Self;
}

=item FormatAttributesGet()

get the format attributes of a format as array/hash reference

    my $Attributes = $FormatBackend->FormatAttributesGet(
        UserID => 1,
    );

=cut

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

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

    my $Attributes = [
        {
            Key   => 'ColumnSeparator',
            Name  => 'Column Separator',
            Input => {
                Type => 'Selection',
                Data => {
                    Tabulator => 'Tabulator (TAB)',
                    Semicolon => 'Semicolon (;)',
                    Colon     => 'Colon (:)',
                    Dot       => 'Dot (.)',
                    Comma     => 'Comma (,)',
                },
                Required     => 1,
                Translation  => 1,
                PossibleNone => 1,
            },
        },
        {
            Key   => 'Charset',
            Name  => 'Charset',
            Input => {
                Type         => 'Text',
                ValueDefault => 'UTF-8',
                Required     => 1,
                Translation  => 0,
                Size         => 20,
                MaxLength    => 20,
            },
        },
        {
            Key   => 'IncludeColumnHeaders',
            Name  => 'Include Column Headers',
            Input => {
                Type => 'Selection',
                Data => {
                    0 => 'No',
                    1 => 'Yes',
                },
                Translation  => 1,
                PossibleNone => 0,
            },
        },
    ];

    return $Attributes;
}

=item MappingFormatAttributesGet()

get the mapping attributes of an format as array/hash reference

    my $Attributes = $FormatBackend->MappingFormatAttributesGet(
        UserID => 1,
    );

=cut

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

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

    my $Attributes = [
        {
            Key   => 'Column',
            Name  => 'Column',
            Input => {
                Type     => 'DTL',
                Data     => '$QData{"Counter"}',
                Required => 0,
            },
        },
    ];

    return $Attributes;
}

=item ImportDataGet()

get import data as 2D-array reference

    my $ImportData = $FormatBackend->ImportDataGet(
        TemplateID    => 123,
        SourceContent => $StringRef,  # (optional)
        UserID        => 1,
    );

=cut

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

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

    return [] if !defined $Param{SourceContent};

    # check source content
    if ( ref $Param{SourceContent} ne 'SCALAR' ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'SourceContent must be a scalar reference',
        );
        return;
    }

    # get format data
    my $FormatData = $Self->{ImportExportObject}->FormatDataGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # check format data
    if ( !$FormatData || ref $FormatData ne 'HASH' ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "No format data found for the template id $Param{TemplateID}",
        );
        return;
    }

    # get charset
    my $Charset = $FormatData->{Charset} ||= '';
    $Charset =~ s{ \s* (utf-8|utf8) \s* }{UTF-8}xmsi;

    # check the charset
    if ( !$Charset ) {

        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "No valid charset found for the template id $Param{TemplateID}",
        );
        return;
    }

    # get separator
    $FormatData->{ColumnSeparator} ||= '';
    my $Separator = $Self->{AvailableSeparators}->{ $FormatData->{ColumnSeparator} } || '';

    # check the separator
    if ( !$Separator ) {

        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "No valid separator found for the template id $Param{TemplateID}",
        );
        return;
    }

    # create the parser object
    my $ParseObject = Text::CSV->new(
        {
            quote_char          => '"',
            escape_char         => '"',
            sep_char            => $Separator,
            eol                 => '',
            always_quote        => 1,
            binary              => 1,
            keep_meta_info      => 0,
            allow_loose_quotes  => 0,
            allow_loose_escapes => 0,
            allow_whitespace    => 0,
            blank_is_undef      => 0,
            verbatim            => 0,
        }
    );

    # create an in memory temp file and open it
    my $FileContent = '';
    open my $FH, '+<', \$FileContent;    ## no critic

    # write source content
    print $FH ${ $Param{SourceContent} };

    # rewind file handle
    seek $FH, 0, 0;

    # parse the content
    my $LineCount = 1;
    my @ImportData;

    # it is important to use this syntax "while ( !eof($FH) )"
    # as the CPAN module Text::CSV_XS might show errors if the
    # getline call is within the while test
    # have a look at http://bugs.otrs.org/show_bug.cgi?id=9270
    while ( !eof($FH) ) {
        my $Column = $ParseObject->getline($FH);
        push @ImportData, $Column;
        $LineCount++;
    }

    # error handling
    my ( $ParseErrorCode, $ParseErrorString ) = $ParseObject->error_diag();
    if ($ParseErrorCode) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "ImportError at line $LineCount, "
                . "ErrorCode: $ParseErrorCode '$ParseErrorString' ",
        );
    }

    # close the in memory file handle
    close $FH;

    return \@ImportData if $Charset ne 'UTF-8';

    # set utf8 flag
    for my $Row (@ImportData) {
        for my $Cell ( @{$Row} ) {
            Encode::_utf8_on($Cell);
        }
    }

    return \@ImportData;
}

=item ExportDataSave()

export one row of the export data

    my $DestinationContent = $FormatBackend->ExportDataSave(
        TemplateID    => 123,
        ExportDataRow => $ArrayRef,
        UserID        => 1,
    );

=cut

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

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

    # check export data row
    if ( ref $Param{ExportDataRow} ne 'ARRAY' ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'ExportDataRow must be an array reference',
        );
        return;
    }

    # get format data
    my $FormatData = $Self->{ImportExportObject}->FormatDataGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # check format data
    if ( !$FormatData || ref $FormatData ne 'HASH' ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "No format data found for the template id $Param{TemplateID}",
        );
        return;
    }

    # get charset
    my $Charset = $FormatData->{Charset} ||= '';
    $Charset =~ s{ \s* (utf-8|utf8) \s* }{UTF-8}xmsi;

    # check the charset
    if ( !$Charset ) {

        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "No valid charset found for the template id $Param{TemplateID}",
        );
        return;
    }

    # get charset
    $FormatData->{ColumnSeparator} ||= '';
    my $Separator = $Self->{AvailableSeparators}->{ $FormatData->{ColumnSeparator} } || '';

    # check the separator
    if ( !$Separator ) {

        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "No valid separator found for the template id $Param{TemplateID}",
        );
        return;
    }

    # create the parser object
    my $ParseObject = Text::CSV->new(
        {
            quote_char          => '"',
            escape_char         => '"',
            sep_char            => $Separator,
            eol                 => '',
            always_quote        => 1,
            binary              => 1,
            keep_meta_info      => 0,
            allow_loose_quotes  => 0,
            allow_loose_escapes => 0,
            allow_whitespace    => 0,
            blank_is_undef      => 0,
            verbatim            => 0,
        }
    );

    if ( !$ParseObject->combine( @{ $Param{ExportDataRow} } ) ) {

        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't combine the export data to a string!",
        );
        return;
    }

    # create the CSV string
    my $String = $ParseObject->string();

    return $String if $Charset ne 'UTF-8';

    # set utf8 flag
    Encode::_utf8_on($String);

    return $String;
}

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

=cut

# --
# ImportExport.t - all general import export tests
# Copyright (C) 2001-2013 OTRS AG, http://otrs.com/
# --
# 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::Encode;
use Kernel::System::ImportExport;
use Kernel::System::User;

$Self->{EncodeObject}       = Kernel::System::Encode->new( %{$Self} );
$Self->{ImportExportObject} = Kernel::System::ImportExport->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 => 'ImportExport' . $Counter,
            UserLastname  => 'UnitTest',
            UserLogin     => 'UnitTest-ImportExport-' . $Counter . int rand 1_000_000,
            UserEmail     => 'UnitTest-ImportExport-' . $Counter . '@localhost',
            ValidID       => 1,
            ChangeUserID  => 1,
        );

        push @UserIDs, $UserID;
    }

    # restore original email check param
    $Self->{ConfigObject}->Set(
        Key   => 'CheckEmailAddresses',
        Value => $CheckEmailAddressesOrg,
    );
}

# create needed random template names
my @TemplateName;

for my $Counter ( 1 .. 5 ) {

    push @TemplateName, 'UnitTest' . int rand 1_000_000;
}

# create needed random object names
my @ObjectName;
push @ObjectName, 'UnitTest' . int rand 1_000_000;

# create needed format names
my @FormatName = ('CSV');

# get original template list for later checks (all elements)
my $TemplateList1All = $Self->{ImportExportObject}->TemplateList(
    UserID => 1,
);

# get original template list for later checks (all elements)
my $TemplateList1Object = $Self->{ImportExportObject}->TemplateList(
    Object => $ObjectName[0],
    UserID => 1,
);

# ------------------------------------------------------------ #
# define general tests
# ------------------------------------------------------------ #

my $ItemData = [

    # this template is NOT complete and must not be added
    {
        Add => {
            Format  => $FormatName[0],
            Name    => $TemplateName[0],
            ValidID => 1,
            UserID  => 1,
        },
    },

    # this template is NOT complete and must not be added
    {
        Add => {
            Object  => $ObjectName[0],
            Name    => $TemplateName[0],
            ValidID => 1,
            UserID  => 1,
        },
    },

    # this template is NOT complete and must not be added
    {
        Add => {
            Object  => $ObjectName[0],
            Format  => $FormatName[0],
            ValidID => 1,
            UserID  => 1,
        },
    },

    # this template is NOT complete and must not be added
    {
        Add => {
            Object => $ObjectName[0],
            Format => $FormatName[0],
            Name   => $TemplateName[0],
            UserID => 1,
        },
    },

    # this template is NOT complete and must not be added
    {
        Add => {
            Object  => $ObjectName[0],
            Format  => $FormatName[0],
            Name    => $TemplateName[0],
            ValidID => 1,
        },
    },

    # this template must be inserted sucessfully
    {
        Add => {
            Object  => $ObjectName[0],
            Format  => $FormatName[0],
            Name    => $TemplateName[0],
            ValidID => 1,
            UserID  => 1,
        },
        AddGet => {
            Object   => $ObjectName[0],
            Format   => $FormatName[0],
            Name     => $TemplateName[0],
            ValidID  => 1,
            Comment  => '',
            CreateBy => 1,
            ChangeBy => 1,
        },
    },

    # this template have the same name as one test before and must not be added
    {
        Add => {
            Object  => $ObjectName[0],
            Format  => $FormatName[0],
            Name    => $TemplateName[0],
            ValidID => 1,
            UserID  => 1,
        },
    },

    # this template must be inserted sucessfully
    {
        Add => {
            Object  => $ObjectName[0],
            Format  => $FormatName[0],
            Name    => $TemplateName[1],
            ValidID => 1,
            Comment => 'TestComment',
            UserID  => 1,
        },
        AddGet => {
            Object   => $ObjectName[0],
            Format   => $FormatName[0],
            Name     => $TemplateName[1],
            ValidID  => 1,
            Comment  => 'TestComment',
            CreateBy => 1,
            ChangeBy => 1,
        },
    },

    # the template one add-test before must be NOT updated (template update arguments NOT complete)
    {
        Update => {
            ValidID => 2,
            UserID  => $UserIDs[0],
        },
    },

    # the template one add-test before must be NOT updated (template update arguments NOT complete)
    {
        Update => {
            Name   => $TemplateName[1] . 'UPDATE1',
            UserID => $UserIDs[0],
        },
    },

    # the template one add-test before must be NOT updated (template update arguments NOT complete)
    {
        Update => {
            Name    => $TemplateName[1] . 'UPDATE2',
            ValidID => 2,
        },
    },

    # the template one add-test before must be updated (template update arguments are complete)
    {
        Update => {
            Name    => $TemplateName[1] . 'UPDATE3',
            Comment => 'TestComment UPDATE3',
            ValidID => 2,
            UserID  => $UserIDs[0],
        },
        UpdateGet => {
            Name     => $TemplateName[1] . 'UPDATE3',
            ValidID  => 2,
            Comment  => 'TestComment UPDATE3',
            CreateBy => 1,
            ChangeBy => $UserIDs[0],
        },
    },

    # the template one add-test before must be updated (template update arguments are complete)
    {
        Update => {
            Name    => $TemplateName[1] . 'UPDATE4',
            ValidID => 1,
            Comment => '',
            UserID  => 1,
        },
        UpdateGet => {
            Name     => $TemplateName[1] . 'UPDATE4',
            ValidID  => 1,
            Comment  => '',
            CreateBy => 1,
            ChangeBy => 1,
        },
    },

    # this template must be inserted sucessfully (check string cleaner function)
    {
        Add => {
            Object  => " \t \n \r " . $ObjectName[0] . " \t \n \r ",
            Format  => " \t \n \r " . $FormatName[0] . " \t \n \r ",
            Name    => " \t \n \r " . $TemplateName[2] . " \t \n \r ",
            ValidID => 1,
            Comment => " \t \n \r Test Comment \t \n \r ",
            UserID  => 1,
        },
        AddGet => {
            Object   => $ObjectName[0],
            Format   => $FormatName[0],
            Name     => $TemplateName[2],
            ValidID  => 1,
            Comment  => 'Test Comment',
            CreateBy => 1,
            ChangeBy => 1,
        },
    },

    # the template one add-test before must be updated (check string cleaner function)
    {
        Update => {
            Name    => " \t \n \r " . $TemplateName[2] . "UPDATE1 \t \n \r ",
            ValidID => 1,
            Comment => " \t \n \r Test Comment UPDATE1 \t \n \r ",
            UserID  => 1,
        },
        UpdateGet => {
            Name     => $TemplateName[2] . 'UPDATE1',
            ValidID  => 1,
            Comment  => 'Test Comment UPDATE1',
            CreateBy => 1,
            ChangeBy => 1,
        },
    },

    # this template must be inserted sucessfully (unicode checks)
    {
        Add => {
            Object  => ' ƕ Ƙ ' . $ObjectName[0] . ' Ƶ ƻ ',
            Format  => ' Ǔ ǣ ' . $FormatName[0] . ' ǥ Ǯ ',
            Name    => ' Ƿ Ȝ ' . $TemplateName[3] . ' Ȟ Ƞ ',
            ValidID => 2,
            Comment => ' Ѡ Ѥ TestComment5 Ϡ Ω ',
            UserID  => 1,
        },
        AddGet => {
            Object   => 'ƕƘ' . $ObjectName[0] . 'Ƶƻ',
            Format   => 'Ǔǣ' . $FormatName[0] . 'ǥǮ',
            Name     => 'Ƿ Ȝ ' . $TemplateName[3] . ' Ȟ Ƞ',
            ValidID  => 2,
            Comment  => 'Ѡ Ѥ TestComment5 Ϡ Ω',
            CreateBy => 1,
            ChangeBy => 1,
        },
    },
];

# ------------------------------------------------------------ #
# run general tests
# ------------------------------------------------------------ #

my $TestCount = 1;
my @AddedTemplateIDs;

TEMPLATE:
for my $Item ( @{$ItemData} ) {

    if ( $Item->{Add} ) {

        # add new template
        my $TemplateID = $Self->{ImportExportObject}->TemplateAdd( %{ $Item->{Add} } );

        if ($TemplateID) {
            push @AddedTemplateIDs, $TemplateID;
        }

        # check if template was added successfully or not
        if ( $Item->{AddGet} ) {
            $Self->True(
                $TemplateID,
                "Test $TestCount: TemplateAdd() - TemplateKey: $TemplateID"
            );
        }
        else {
            $Self->False( $TemplateID, "Test $TestCount: TemplateAdd()" );
        }
    }

    if ( $Item->{AddGet} ) {

        # get template data to check the values after template was added
        my $TemplateGet = $Self->{ImportExportObject}->TemplateGet(
            TemplateID => $AddedTemplateIDs[-1],
            UserID => $Item->{Add}->{UserID} || 1,
        );

        # check template data after creation of template
        for my $TemplateAttribute ( sort keys %{ $Item->{AddGet} } ) {
            $Self->Is(
                $TemplateGet->{$TemplateAttribute} || '',
                $Item->{AddGet}->{$TemplateAttribute} || '',
                "Test $TestCount: TemplateGet() - $TemplateAttribute",
            );
        }
    }

    if ( $Item->{Update} ) {

        # check last template id varaible
        if ( !$AddedTemplateIDs[-1] ) {
            $Self->False(
                1,
                "Test $TestCount: NO LAST ITEM ID GIVEN. Please add a template first."
            );
            last TEMPLATE;
        }

        # update the template
        my $UpdateSucess = $Self->{ImportExportObject}->TemplateUpdate(
            %{ $Item->{Update} },
            TemplateID => $AddedTemplateIDs[-1],
        );

        # check if template was updated successfully or not
        if ( $Item->{UpdateGet} ) {
            $Self->True(
                $UpdateSucess,
                "Test $TestCount: TemplateUpdate() - TemplateKey: $AddedTemplateIDs[-1]",
            );
        }
        else {
            $Self->False(
                $UpdateSucess,
                "Test $TestCount: TemplateUpdate()",
            );
        }
    }

    if ( $Item->{UpdateGet} ) {

        # get template data to check the values after the update
        my $TemplateGet = $Self->{ImportExportObject}->TemplateGet(
            TemplateID => $AddedTemplateIDs[-1],
            UserID => $Item->{Update}->{UserID} || 1,
        );

        # check template data after update
        for my $TemplateAttribute ( sort keys %{ $Item->{UpdateGet} } ) {
            $Self->Is(
                $TemplateGet->{$TemplateAttribute} || '',
                $Item->{UpdateGet}->{$TemplateAttribute} || '',
                "Test $TestCount: TemplateGet() - $TemplateAttribute",
            );
        }
    }
}
continue {

    # increment the counter
    $TestCount++;
}

# ------------------------------------------------------------ #
# TemplateList test 1 (check array references)
# ------------------------------------------------------------ #

# list must be an empty array reference
$Self->True(
    ref $TemplateList1All eq 'ARRAY' && ref $TemplateList1Object eq 'ARRAY',
    "Test $TestCount: TemplateList() - array references",
);

$TestCount++;

# ------------------------------------------------------------ #
# TemplateList test 2 (list must be empty)
# ------------------------------------------------------------ #

# list must be an empty list
$Self->True(
    scalar @{$TemplateList1Object} eq 0,
    "Test $TestCount: TemplateList() - empty list",
);

$TestCount++;

# ------------------------------------------------------------ #
# TemplateList test 2 (check correct number of new items)
# ------------------------------------------------------------ #

# get template list with all elements
my $TemplateList2 = $Self->{ImportExportObject}->TemplateList(
    UserID => 1,
);

# list must be an array reference
$Self->True(
    ref $TemplateList2 eq 'ARRAY',
    "Test $TestCount: TemplateList() - array reference",
);

my $TemplateListCount = scalar @{$TemplateList2} - scalar @{$TemplateList1All};

# check correct number of new items
$Self->True(
    $TemplateListCount eq scalar @AddedTemplateIDs,
    "Test $TestCount: TemplateList() - correct number of new items",
);

$TestCount++;

# ------------------------------------------------------------ #
# TemplateDelete test 1 (add one template and delete it)
# ------------------------------------------------------------ #

# get template list with all elements
my $TemplateDelete1List1 = $Self->{ImportExportObject}->TemplateList(
    Object => $ObjectName[0],
    UserID => 1,
);

# add a test template
my $TemplateDeleteID = $Self->{ImportExportObject}->TemplateAdd(
    Object  => $ObjectName[0],
    Format  => $FormatName[0],
    Name    => $TemplateName[4],
    ValidID => 1,
    UserID  => 1,
);

# get template list with all elements
my $TemplateDelete1List2 = $Self->{ImportExportObject}->TemplateList(
    Object => $ObjectName[0],
    UserID => 1,
);

# list must have one more element
$Self->True(
    scalar @{$TemplateDelete1List1} eq ( scalar @{$TemplateDelete1List2} ) - 1,
    "Test $TestCount: TemplateDelete() - number of listed elements",
);

# delete the new template
my $TemplateDelete1 = $Self->{ImportExportObject}->TemplateDelete(
    TemplateID => $TemplateDeleteID,
    UserID     => 1,
);

# list must be successfull
$Self->True(
    $TemplateDelete1,
    "Test $TestCount: TemplateDelete()",
);

# get template list with all elements
my $TemplateDelete1List3 = $Self->{ImportExportObject}->TemplateList(
    Object => $ObjectName[0],
    UserID => 1,
);

# list must have the original number of elements
$Self->True(
    scalar @{$TemplateDelete1List1} eq scalar @{$TemplateDelete1List3},
    "Test $TestCount: TemplateDelete() - number of listed elements",
);

$TestCount++;

# ------------------------------------------------------------ #
# TemplateDelete test 2 (delete all unittest templates)
# ------------------------------------------------------------ #

for my $TemplateID (@AddedTemplateIDs) {

    # delete the template
    my $Success = $Self->{ImportExportObject}->TemplateDelete(
        TemplateID => $TemplateID,
        UserID     => 1,
    );

    # check success
    $Self->True(
        $Success,
        "Test $TestCount: TemplateDelete() TemplateID $TemplateID",
    );

    $TestCount++;
}

# ------------------------------------------------------------ #
# ObjectList test 1 (check general functionality)
# ------------------------------------------------------------ #

# define test list
my $ObjectList1TestList = {
    UnitTest1 => {
        Module => 'Kernel::System::ImportExport::ObjectBackend::UnitTest1',
        Name   => 'Unit Test 1',
    },
    UnitTest2 => {
        Module => 'Kernel::System::ImportExport::ObjectBackend::UnitTest2',
        Name   => 'Unit Test 2',
    },
};

# get original object list
my $ObjectListOrg = $Self->{ConfigObject}->Get('ImportExport::ObjectBackendRegistration');

# set test list
$Self->{ConfigObject}->Set(
    Key   => 'ImportExport::ObjectBackendRegistration',
    Value => $ObjectList1TestList,
);

# get object list
my $ObjectList1 = $Self->{ImportExportObject}->ObjectList();

# list must be a hash reference
$Self->True(
    ref $ObjectList1 eq 'HASH',
    "Test $TestCount: ObjectList() - hash reference",
);

# check the list
KEY:
for my $Key ( sort keys %{$ObjectList1} ) {

    if ( !$ObjectList1TestList->{$Key} ) {
        $ObjectList1TestList->{Dummy} = 1;
    }

    next KEY if $ObjectList1->{$Key} ne $ObjectList1TestList->{$Key}->{Name};

    delete $ObjectList1TestList->{$Key};
}

$Self->True(
    !%{$ObjectList1TestList},
    "Test $TestCount: ObjectList() - content is valid",
);

# restore original object list
$Self->{ConfigObject}->Set(
    Key   => 'ImportExport::ObjectBackendRegistration',
    Value => $ObjectListOrg,
);

$TestCount++;

# ------------------------------------------------------------ #
# FormatList test 1 (check general functionality)
# ------------------------------------------------------------ #

# define test list
my $FormatList1TestList = {
    UnitTest1 => {
        Module => 'Kernel::System::ImportExport::FormatBackend::UnitTest1',
        Name   => 'Unit Test 1',
    },
    UnitTest2 => {
        Module => 'Kernel::System::ImportExport::FormatBackend::UnitTest2',
        Name   => 'Unit Test 2',
    },
};

# get original format list
my $FormatListOrg = $Self->{ConfigObject}->Get('ImportExport::FormatBackendRegistration');

# set test list
$Self->{ConfigObject}->Set(
    Key   => 'ImportExport::FormatBackendRegistration',
    Value => $FormatList1TestList,
);

# get format list
my $FormatList1 = $Self->{ImportExportObject}->FormatList();

# list must be a hash reference
$Self->True(
    ref $FormatList1 eq 'HASH',
    "Test $TestCount: FormatList() - hash reference",
);

# check the list
KEY:
for my $Key ( sort keys %{$FormatList1} ) {

    if ( !$FormatList1TestList->{$Key} ) {
        $FormatList1TestList->{Dummy} = 1;
    }

    next KEY if $FormatList1->{$Key} ne $FormatList1TestList->{$Key}->{Name};

    delete $FormatList1TestList->{$Key};
}

$Self->True(
    !%{$FormatList1TestList},
    "Test $TestCount: FormatList() - content is valid",
);

# restore original format list
$Self->{ConfigObject}->Set(
    Key   => 'ImportExport::FormatBackendRegistration',
    Value => $FormatListOrg,
);

$TestCount++;

1;

# --
# ImportExportFormatCSV.t - all import export tests for the CSV format backend
# Copyright (C) 2001-2013 OTRS AG, http://otrs.com/
# --
# 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::Encode;
use Kernel::System::ImportExport;
use Kernel::System::ImportExport::FormatBackend::CSV;

$Self->{EncodeObject}        = Kernel::System::Encode->new( %{$Self} );
$Self->{ImportExportObject}  = Kernel::System::ImportExport->new( %{$Self} );
$Self->{FormatBackendObject} = Kernel::System::ImportExport::FormatBackend::CSV->new( %{$Self} );

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# get home directory
$Self->{Home} = $Self->{ConfigObject}->Get('Home');

# add some test templates for later checks
my @TemplateIDs;
for ( 1 .. 30 ) {

    # add a test template for later checks
    my $TemplateID = $Self->{ImportExportObject}->TemplateAdd(
        Object  => 'UnitTest' . int rand 1_000_000,
        Format  => 'CSV',
        Name    => 'UnitTest' . int rand 1_000_000,
        ValidID => 1,
        UserID  => 1,
    );

    push @TemplateIDs, $TemplateID;
}

my $TestCount = 1;

# ------------------------------------------------------------ #
# FormatList test 1 (check CSV item)
# ------------------------------------------------------------ #

# get format list
my $FormatList1 = $Self->{ImportExportObject}->FormatList();

# check format list
$Self->True(
    $FormatList1 && ref $FormatList1 eq 'HASH' && $FormatList1->{CSV},
    "Test $TestCount: FormatList() - CSV exists",
);

$TestCount++;

# ------------------------------------------------------------ #
# FormatAttributesGet test 1 (check attribute hash)
# ------------------------------------------------------------ #

# get format attributes
my $FormatAttributesGet1 = $Self->{ImportExportObject}->FormatAttributesGet(
    TemplateID => $TemplateIDs[0],
    UserID     => 1,
);

# check format attribute reference
$Self->True(
    $FormatAttributesGet1 && ref $FormatAttributesGet1 eq 'ARRAY',
    "Test $TestCount: FormatAttributesGet() - check array reference",
);

# define the reference hash
my $FormatAttributesGet1Reference = [
    {
        Key   => 'ColumnSeparator',
        Name  => 'Column Separator',
        Input => {
            Type => 'Selection',
            Data => {
                Tabulator => 'Tabulator (TAB)',
                Semicolon => 'Semicolon (;)',
                Colon     => 'Colon (:)',
                Dot       => 'Dot (.)',
                Comma     => 'Comma (,)',
            },
            Required     => 1,
            Translation  => 1,
            PossibleNone => 1,
        },
    },
    {
        Key   => 'Charset',
        Name  => 'Charset',
        Input => {
            Type         => 'Text',
            ValueDefault => 'UTF-8',
            Required     => 1,
            Translation  => 0,
            Size         => 20,
            MaxLength    => 20,
        },
    },
    {
        Key   => 'IncludeColumnHeaders',
        Name  => 'Include Column Headers',
        Input => {
            Type => 'Selection',
            Data => {
                0 => 'No',
                1 => 'Yes',
            },
            Translation  => 1,
            PossibleNone => 0,
        },
    },
];

# turn off all pretty print
$Data::Dumper::Indent = 0;

# dump the list from FormatAttributesGet() and the reference table
## no critic
my $FormatAttributesGetDump1 = Data::Dumper::Dumper($FormatAttributesGet1);
my $FormatAttributesRefDump1 = Data::Dumper::Dumper($FormatAttributesGet1Reference);
## use critic

$Self->True(
    $FormatAttributesGetDump1 eq $FormatAttributesRefDump1,
    "Test $TestCount: FormatAttributesGet() - attributes of the row are identical",
);

$TestCount++;

# ------------------------------------------------------------ #
# FormatAttributesGet test 2 (check with non existing template)
# ------------------------------------------------------------ #

# get format attributes
my $FormatAttributesGet2 = $Self->{ImportExportObject}->FormatAttributesGet(
    TemplateID => $TemplateIDs[-1] + 1,
    UserID     => 1,
);

# check false return
$Self->False(
    $FormatAttributesGet2,
    "Test $TestCount: FormatAttributesGet() - check false return",
);

$TestCount++;

# ------------------------------------------------------------ #
# MappingFormatAttributesGet test 1 (check attribute hash)
# ------------------------------------------------------------ #

# get mapping format attributes
my $MappingFormatAttributesGet1 = $Self->{ImportExportObject}->MappingFormatAttributesGet(
    TemplateID => $TemplateIDs[0],
    UserID     => 1,
);

# check mapping format attribute reference
$Self->True(
    $MappingFormatAttributesGet1 && ref $MappingFormatAttributesGet1 eq 'ARRAY',
    "Test $TestCount: MappingFormatAttributesGet() - check array reference",
);

# define the reference hash
my $MappingFormatAttributesGet1Reference = [
    {
        Key   => 'Column',
        Name  => 'Column',
        Input => {
            Type     => 'DTL',
            Data     => '$QData{"Counter"}',
            Required => 0,
        },
    },
];

# turn off all pretty print
$Data::Dumper::Indent = 0;

# dump the list from MappingFormatAttributesGet() and the reference table
## no critic
my $MappingFormatAttributesGetDump1 = Data::Dumper::Dumper($MappingFormatAttributesGet1);
my $MappingFormatAttributesRefDump1 = Data::Dumper::Dumper($MappingFormatAttributesGet1Reference);
## use critic

$Self->True(
    $MappingFormatAttributesGetDump1 eq $MappingFormatAttributesRefDump1,
    "Test $TestCount: MappingFormatAttributesGet() - attributes of the row are identical",
);

$TestCount++;

# ------------------------------------------------------------ #
# MappingFormatAttributesGet test 2 (check with non existing template)
# ------------------------------------------------------------ #

# get mapping format attributes
my $MappingFormatAttributesGet2 = $Self->{ImportExportObject}->MappingFormatAttributesGet(
    TemplateID => $TemplateIDs[-1] + 1,
    UserID     => 1,
);

# check false return
$Self->False(
    $MappingFormatAttributesGet2,
    "Test $TestCount: MappingFormatAttributesGet() - check false return",
);

$TestCount++;

# ------------------------------------------------------------ #
# define general ImportDataGet tests
# ------------------------------------------------------------ #

my $ImportDataTests = [

    # ImportDataGet doesn't contains all data (check required attributes)
    {
        SourceImportData => {
            ImportDataGet => {
                UserID => 1,
            },
        },
    },

    # ImportDataGet doesn't contains all data (check required attributes)
    {
        SourceImportData => {
            ImportDataGet => {
                TemplateID => $TemplateIDs[1],
            },
        },
    },

    # no source content are given (empty array reference must be returned)
    {
        SourceImportData => {
            ImportDataGet => {
                TemplateID => $TemplateIDs[1],
                UserID     => 1,
            },
        },
        ReferenceImportData => [],
    },

    # source content must be a scalar reference (check return false)
    {
        SourceImportData => {
            ImportDataGet => {
                TemplateID    => $TemplateIDs[1],
                SourceContent => [],
                UserID        => 1,
            },
        },
    },

    # source content must be a scalar reference (check return false)
    {
        SourceImportData => {
            ImportDataGet => {
                TemplateID    => $TemplateIDs[1],
                SourceContent => {},
                UserID        => 1,
            },
        },
    },

    # source content must be a scalar reference (check return false)
    {
        SourceImportData => {
            ImportDataGet => {
                TemplateID    => $TemplateIDs[1],
                SourceContent => '',
                UserID        => 1,
            },
        },
    },

    # no existing template id is given (check return false)
    {
        SourceImportData => {
            ImportDataGet => {
                TemplateID    => $TemplateIDs[-1] + 1,
                SourceContent => \do {'Dummy'},
                UserID        => 1,
            },
        },
    },

    # no column Separator and charset are given (check return false)
    {
        SourceImportData => {
            ImportDataGet => {
                TemplateID    => $TemplateIDs[2],
                SourceContent => \do {'Dummy'},
                UserID        => 1,
            },
        },
    },

    # no column Separator is given (check return false)
    {
        SourceImportData => {
            FormatData => {
                Charset => 'UTF-8',
            },
            ImportDataGet => {
                TemplateID    => $TemplateIDs[2],
                SourceContent => \do {'Dummy'},
                UserID        => 1,
            },
        },
    },

    # no charset is given (check return false)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Dummy',
            },
            ImportDataGet => {
                TemplateID    => $TemplateIDs[2],
                SourceContent => \do {'Dummy'},
                UserID        => 1,
            },
        },
    },

    # invalid column Separator is given (check return false)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Dummy',
                Charset         => 'UTF-8',
            },
            ImportDataGet => {
                TemplateID    => $TemplateIDs[2],
                SourceContent => \do {'Dummy'},
                UserID        => 1,
            },
        },
    },

    # required values are given but source content is empty (empty array reference must be returned)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'UTF-8',
            },
            ImportDataGet => {
                TemplateID    => $TemplateIDs[3],
                SourceContent => \do {''},
                UserID        => 1,
            },
        },
        ReferenceImportData => [],
    },

    # source content is only a string with spaces (one cell array with the spaces must be returned)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'UTF-8',
            },
            ImportDataGet => {
                TemplateID    => $TemplateIDs[4],
                SourceContent => \do {'  '},
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            ['  '],
        ],
    },

    # all required values are given (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV001-MSExcel-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
            [ 'Row2-Col1', 'Row2-Col2', 'Row2-Col3' ],
            [ 'Row3-Col1', 'Row3-Col2', 'Row3-Col3' ],
        ],
    },

    # all required values are given, but Tabulator is used as Separator (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV001-MSExcel-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            ['Row1-Col1;Row1-Col2;Row1-Col3'],
            ['Row2-Col1;Row2-Col2;Row2-Col3'],
            ['Row3-Col1;Row3-Col2;Row3-Col3'],
        ],
    },

    # all required values are given (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV001-MSExcel-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
            [ 'Row2-Col1', 'Row2-Col2', 'Row2-Col3' ],
            [ 'Row3-Col1', 'Row3-Col2', 'Row3-Col3' ],
        ],
    },

    # all required values are given, but Semicolon is used as Separator (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV001-MSExcel-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            ["Row1-Col1\tRow1-Col2\tRow1-Col3"],
            ["Row2-Col1\tRow2-Col2\tRow2-Col3"],
            ["Row3-Col1\tRow3-Col2\tRow3-Col3"],
        ],
    },

    # all required values are given (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV001-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
            [ 'Row2-Col1', 'Row2-Col2', 'Row2-Col3' ],
            [ 'Row3-Col1', 'Row3-Col2', 'Row3-Col3' ],
        ],
    },

    # all required values are given (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV001-OpenOffice-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
            [ 'Row2-Col1', 'Row2-Col2', 'Row2-Col3' ],
            [ 'Row3-Col1', 'Row3-Col2', 'Row3-Col3' ],
        ],
    },

    # all required values are given (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV001-OpenOffice-Colon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
            [ 'Row2-Col1', 'Row2-Col2', 'Row2-Col3' ],
            [ 'Row3-Col1', 'Row3-Col2', 'Row3-Col3' ],
        ],
    },

    # all required values are given (newline checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV002-MSExcel-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[6],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ "\nTest 1 - 1", "Test 1 - 2",   "Test 1\n- 3",  'Test \n\t\r\s' ],
            [ "Test 2 \n- 1", "Te\nst 2 - 2", "Test 2 - 3\n", '' ],
        ],
    },

    # all required values are given (newline checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV002-MSExcel-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[6],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ "\nTest 1 - 1", 'Test 1 - 2',   "Test 1\n- 3",  'Test \n\t\r\s' ],
            [ "Test 2 \n- 1", "Te\nst 2 - 2", "Test 2 - 3\n", '' ],
        ],
    },

    # all required values are given (newline checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV002-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[6],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ "\nTest 1 - 1", "Test 1 - 2",   "Test 1\n- 3",  'Test \n\t\r\s' ],
            [ "Test 2 \n- 1", "Te\nst 2 - 2", "Test 2 - 3\n", '' ],
        ],
    },

    # all required values are given (newline checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV002-OpenOffice-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[6],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ "\nTest 1 - 1", "Test 1 - 2",   "Test 1\n- 3",  'Test \n\t\r\s' ],
            [ "Test 2 \n- 1", "Te\nst 2 - 2", "Test 2 - 3\n", '' ],
        ],
    },

    # all required values are given (newline checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV002-OpenOffice-Colon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[6],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ "\nTest 1 - 1", "Test 1 - 2",   "Test 1\n- 3",  'Test \n\t\r\s' ],
            [ "Test 2 \n- 1", "Te\nst 2 - 2", "Test 2 - 3\n", '' ],
        ],
    },

    # all required values are given (spaces checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV003-MSExcel-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[7],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ '  Test  ', '    ', 'Test  ' ],
            [ '    Test', '',     'Test' ],
            [ '',         '',     ' ' ],
        ],
    },

    # all required values are given (spaces checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV003-MSExcel-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[7],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ '  Test  ', '    ', 'Test  ' ],
            [ '    Test', '',     'Test' ],
            [ '',         '',     ' ' ],
        ],
    },

    # all required values are given (spaces checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV003-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[7],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ '  Test  ', '    ', 'Test  ' ],
            [ '    Test', '',     'Test' ],
            [ '',         '',     ' ' ],
        ],
    },

    # all required values are given (spaces checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV003-OpenOffice-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[7],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ '  Test  ', '    ', 'Test  ' ],
            [ '    Test', '',     'Test' ],
            [ '',         '',     ' ' ],
        ],
    },

    # all required values are given (spaces checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV003-OpenOffice-Colon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[7],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ '  Test  ', '    ', 'Test  ' ],
            [ '    Test', '',     'Test' ],
            [ '',         '',     ' ' ],
        ],
    },

    # all required values are given (special character checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV004-MSExcel-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[8],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Test;:_°^!"§$%&/()=?´`*+Test', '><@~\'}{[]\\' ],
            [ '"";;::..--__##',                  '' ],
        ],
    },

    # all required values are given (special character checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV004-MSExcel-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[8],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Test;:_°^!"§$%&/()=?´`*+Test', '><@~\'}{[]\\' ],
            [ '"";;::..--__##',                  '' ],
        ],
    },

    # all required values are given (special character checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV004-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[8],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Test;:_°^!"§$%&/()=?´`*+Test', '><@~\'}{[]\\' ],
            [ '"";;::..--__##',                  '' ],
        ],
    },

    # all required values are given (special character checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV004-OpenOffice-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[8],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Test;:_°^!"§$%&/()=?´`*+Test', '><@~\'}{[]\\' ],
            [ '"";;::..--__##',                  '' ],
        ],
    },

    # all required values are given (special character checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV004-OpenOffice-Colon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[8],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Test;:_°^!"§$%&/()=?´`*+Test', '><@~\'}{[]\\' ],
            [ '"";;::..--__##',                  '' ],
        ],
    },

    # all required values are given (ISO-8859 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV005-MSExcel-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[9],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'üöäß', 'ÜÖÄ' ],
            [ 'ßäöü', 'ÄÖÜ' ],
        ],
    },

    # all required values are given (ISO-8859 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV005-MSExcel-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[9],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'üöäß', 'ÜÖÄ' ],
            [ 'ßäöü', 'ÄÖÜ' ],
        ],
    },

    # all required values are given (ISO-8859 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV005-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[9],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'üöäß', 'ÜÖÄ' ],
            [ 'ßäöü', 'ÄÖÜ' ],
        ],
    },

    # all required values are given (ISO-8859 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV005-OpenOffice-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[9],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'üöäß', 'ÜÖÄ' ],
            [ 'ßäöü', 'ÄÖÜ' ],
        ],
    },

    # all required values are given (ISO-8859 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV005-OpenOffice-Colon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[9],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'üöäß', 'ÜÖÄ' ],
            [ 'ßäöü', 'ÄÖÜ' ],
        ],
    },

    # all required values are given (UTF-8 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'UTF-8',
            },
            SourceFile    => 'ImportExportFormatCSV006-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[10],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'ʩ ʬ ʮ',     ' ʡ ˤ Ό ' ],
            [ '  Η ϗ Ϡ  ', 'Ά Λ Ξ' ],
        ],
    },

    # all required values are given (UTF-8 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'UTF-8',
            },
            SourceFile    => 'ImportExportFormatCSV006-OpenOffice-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[10],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'ʩ ʬ ʮ',     ' ʡ ˤ Ό ' ],
            [ '  Η ϗ Ϡ  ', 'Ά Λ Ξ' ],
        ],
    },

    # all required values are given (UTF-8 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'UTF-8',
            },
            SourceFile    => 'ImportExportFormatCSV006-OpenOffice-Colon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[10],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'ʩ ʬ ʮ',     ' ʡ ˤ Ό ' ],
            [ '  Η ϗ Ϡ  ', 'Ά Λ Ξ' ],
        ],
    },

    # all required values are given (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV007-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Row1-Col1', 'Row1-Col2',  'Row1-Col3' ],
            [ 'Row2-Col1', 'Row2-Col2',  'Row2-Col3' ],
            [ 'Row3-Col1', '0Row3-Col2', 'Row3-Col3' ],
        ],
    },

    # all required values are given (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV008-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Row1-Col1', 'Row1-Col2',  'Row1-Col3' ],
            [ 'Row2-Col1', '0Row2-Col2', 'Row2-Col3' ],
            [ 'Row3-Col1', 'Row3-Col2',  'Row3-Col3' ],
        ],
    },

];

# ------------------------------------------------------------ #
# run general ImportDataGet tests
# ------------------------------------------------------------ #

TEST:
for my $Test ( @{$ImportDataTests} ) {

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

        $Self->True(
            0,
            "Test $TestCount: No SourceImportData found for this test."
        );

        next TEST;
    }

    # set default ImportDataGet
    if ( !$Test->{SourceImportData}->{ImportDataGet} ) {
        $Test->{SourceImportData}->{ImportDataGet} = {};
    }

    # set source content
    if (
        $Test->{SourceImportData}->{SourceFile}
        && $Test->{SourceImportData}->{ImportDataGet}->{SourceContent}
        && $Test->{SourceImportData}->{ImportDataGet}->{SourceContent} eq 'SourceFile'
        )
    {

        my $SourceFile = $Test->{SourceImportData}->{SourceFile};

        # read source file
        my $SourceContent = $Self->{MainObject}->FileRead(
            Location => $Self->{Home} . '/scripts/test/sample/ImportExport/' . $SourceFile,
            Result   => 'SCALAR',
            Mode     => 'binmode',
        );

        $Test->{SourceImportData}->{ImportDataGet}->{SourceContent} = $SourceContent;
    }

    # set the format data
    if (
        $Test->{SourceImportData}->{FormatData}
        && ref $Test->{SourceImportData}->{FormatData} eq 'HASH'
        && $Test->{SourceImportData}->{ImportDataGet}->{TemplateID}
        )
    {

        # save format data
        $Self->{ImportExportObject}->FormatDataSave(
            TemplateID => $Test->{SourceImportData}->{ImportDataGet}->{TemplateID},
            FormatData => $Test->{SourceImportData}->{FormatData},
            UserID     => 1,
        );
    }

    # get import data
    my $ImportData = $Self->{FormatBackendObject}->ImportDataGet(
        %{ $Test->{SourceImportData}->{ImportDataGet} },
    );

    if ( !$Test->{ReferenceImportData} ) {

        $Self->False(
            $ImportData,
            "Test $TestCount: ImportDataGet() - return false"
        );

        next TEST;
    }

    if ( ref $ImportData ne 'ARRAY' ) {

        # check array reference
        $Self->True(
            0,
            "Test $TestCount: ImportDataGet() - return value is an array reference",
        );

        next TEST;
    }

    # check number of rows
    $Self->Is(
        scalar @{$ImportData},
        scalar @{ $Test->{ReferenceImportData} },
        "Test $TestCount: ImportDataGet() - same number of rows",
    );

    # check content of import data
    my $CounterRow = 0;
    ROW:
    for my $ImportRow ( @{$ImportData} ) {

        # extract reference row
        my $ReferenceRow = $Test->{ReferenceImportData}->[$CounterRow];

        if ( ref $ImportRow ne 'ARRAY' || ref $ReferenceRow ne 'ARRAY' ) {

            # check array reference
            $Self->True(
                0,
                "Test $TestCount: ImportDataGet() - import row and reference row matched",
            );

            next TEST;
        }

        # check number of columns
        $Self->Is(
            scalar @{$ImportRow},
            scalar @{$ReferenceRow},
            "Test $TestCount: ImportDataGet() - same number of columns",
        );

        my $CounterColumn = 0;
        for my $Cell ( @{$ImportRow} ) {

            # set content if values are undef
            if ( !defined $Cell ) {
                $Cell = 'UNDEF-unittest';
            }
            if ( !defined $ReferenceRow->[$CounterColumn] ) {
                $ReferenceRow->[$CounterColumn] = 'UNDEF-unittest';
            }

            # check cell data
            $Self->Is(
                $Cell,
                $ReferenceRow->[$CounterColumn],
                "Test $TestCount: ImportDataGet() ",
            );

            $CounterColumn++;
        }

        $CounterRow++;
    }
}
continue {
    $TestCount++;
}

# ------------------------------------------------------------ #
# define general ExportDataSave tests
# ------------------------------------------------------------ #

my $ExportDataTests = [

    # ExportDataSave doesn't contains all data (check required attributes)
    {
        SourceExportData => {
            ExportDataSave => {
                ExportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # ExportDataSave doesn't contains all data (check required attributes)
    {
        SourceExportData => {
            ExportDataSave => {
                TemplateID => $TemplateIDs[20],
                UserID     => 1,
            },
        },
    },

    # ExportDataSave doesn't contains all data (check required attributes)
    {
        SourceExportData => {
            ExportDataSave => {
                TemplateID    => $TemplateIDs[20],
                ExportDataRow => ['Dummy'],
            },
        },
    },

    # export data row must be an array reference (check return false)
    {
        SourceExportData => {
            ExportDataSave => {
                TemplateID    => $TemplateIDs[20],
                ExportDataRow => '',
                UserID        => 1,
            },
        },
    },

    # export data row must be an array reference (check return false)
    {
        SourceExportData => {
            ExportDataSave => {
                TemplateID    => $TemplateIDs[20],
                ExportDataRow => {},
                UserID        => 1,
            },
        },
    },

    # no existing template id is given (check return false)
    {
        SourceExportData => {
            ExportDataSave => {
                TemplateID    => $TemplateIDs[-1] + 1,
                ExportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # no column Separator and charset are given (check return false)
    {
        SourceExportData => {
            ExportDataSave => {
                TemplateID    => $TemplateIDs[21],
                ExportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # no column Separator is given (check return false)
    {
        SourceExportData => {
            FormatData => {
                Charset => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[21],
                ExportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # no charset is given (check return false)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dummy',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[21],
                ExportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # invalid column Separator is given (check return false)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dummy',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[21],
                ExportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # export data are one cells with empty strings (one empty cell must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [''],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '""',
    },

    # export data are one cells with empty strings (one empty cell must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [''],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '""',
    },

    # export data are one cells with empty strings (one empty cell must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [''],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '""',
    },

    # export data are one cells with empty strings (one empty cell must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dot',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [''],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '""',
    },

    # export data are three cells with empty strings (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ '', '', '' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"";"";""',
    },

    # export data are three cells with empty strings (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ '', '', '' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => "\"\"\t\"\"\t\"\"",
    },

    # export data are three cells with empty strings (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ '', '', '' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"":"":""',
    },

    # export data are three cells with empty strings (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dot',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ '', '', '' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '""."".""',
    },

    # export data are three cells with empty and undef content (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ undef, '', undef ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => ';"";',
    },

    # export data are three cells with empty and undef content (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ undef, '', undef ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => "\t\"\"\t",
    },

    # export data are three cells with empty and undef content (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ undef, '', undef ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => ':"":',
    },

    # export data are three cells with empty and undef content (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dot',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ undef, '', undef ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '."".',
    },

    # all required values are given (check the parsed content)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ExportDataRow => [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"Row1-Col1";"Row1-Col2";"Row1-Col3"',
    },

    # all required values are given (check the parsed content)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Comma',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ExportDataRow => [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"Row1-Col1","Row1-Col2","Row1-Col3"',
    },

    # all required values are given (check the parsed content)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ExportDataRow => [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => "\"Row1-Col1\"\t\"Row1-Col2\"\t\"Row1-Col3\"",
    },

    # all required values are given (check the parsed content)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ExportDataRow => [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"Row1-Col1":"Row1-Col2":"Row1-Col3"',
    },

    # all required values are given (check the parsed content)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dot',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ExportDataRow => [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"Row1-Col1"."Row1-Col2"."Row1-Col3"',
    },

    # all required values are given (newline checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ "\nTest 1", "Test \n 2", 'Test 3 \n\t\r\s', "Test 4\n" ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => qq{"\nTest 1";"Test \n 2";"Test 3 \\n\\t\\r\\s";"Test 4\n"},
    },

    # all required values are given (newline checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ "\nTest 1", "Test \n 2", 'Test 3 \n\t\r\s', "Test 4\n" ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent =>
            qq{"\nTest 1"\t"Test \n 2"\t"Test 3 \\n\\t\\r\\s"\t"Test 4\n"},
    },

    # all required values are given (newline checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ "\nTest 1", "Test \n 2", 'Test 3 \n\t\r\s', "Test 4\n" ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => qq{"\nTest 1":"Test \n 2":"Test 3 \\n\\t\\r\\s":"Test 4\n"},
    },

    # all required values are given (newline checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dot',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ "\nTest 1", "Test \n 2", 'Test 3 \n\t\r\s', "Test 4\n" ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => qq{"\nTest 1"."Test \n 2"."Test 3 \\n\\t\\r\\s"."Test 4\n"},
    },

    # all required values are given (spaces checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ '  Test  ', '    ', 'Test  ', '    Test', '', 'Test', '', ' ' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"  Test  ";"    ";"Test  ";"    Test";"";"Test";"";" "',
    },

    # all required values are given (spaces checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ '  Test  ', '    ', 'Test  ', '    Test', '', 'Test', '', ' ' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent =>
            "\"  Test  \"\t\"    \"\t\"Test  \"\t\"    Test\"\t\"\"\t\"Test\"\t\"\"\t\" \"",
    },

    # all required values are given (spaces checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ '  Test  ', '    ', 'Test  ', '    Test', '', 'Test', '', ' ' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"  Test  ":"    ":"Test  ":"    Test":"":"Test":"":" "',
    },

    # all required values are given (spaces checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dot',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ '  Test  ', '    ', 'Test  ', '    Test', '', 'Test', '', ' ' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"  Test  "."    "."Test  "."    Test".""."Test".""." "',
    },

    # all required values are given (special character checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ExportDataRow => [
                    'Test;:_^!"$%&/()=?`*+Test',
                    '><@~\'}{[]\\',
                    '',
                    '"";;::..--__##'
                ],
                UserID => 1,
            },
        },
        ReferenceDestinationContent =>
            '"Test;:_^!""$%&/()=?`*+Test";"><@~\'}{[]\";"";""""";;::..--__##"'
    },

    # all required values are given (special character checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ExportDataRow => [
                    'Test;:_^!"$%&/()=?`*+Test',
                    '><@~\'}{[]\\',
                    '',
                    '"";;::..--__##'
                ],
                UserID => 1,
            },
        },
        ReferenceDestinationContent =>
            '"Test;:_^!""$%&/()=?`*+Test"'
            . "\t"
            . '"><@~\'}{[]\\"'
            . "\t"
            . '""'
            . "\t"
            . '""""";;::..--__##"',
    },

    # all required values are given (special character checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ExportDataRow => [
                    'Test;:_^!"$%&/()=?`*+Test',
                    '><@~\'}{[]\\',
                    '',
                    '"";;::..--__##'
                ],
                UserID => 1,
            },
        },
        ReferenceDestinationContent =>
            '"Test;:_^!""$%&/()=?`*+Test":"><@~\'}{[]\":"":""""";;::..--__##"',
    },

    # all required values are given (special character checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dot',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ExportDataRow => [
                    'Test;:_^!"$%&/()=?`*+Test',
                    '><@~\'}{[]\\',
                    '',
                    '"";;::..--__##'
                ],
                UserID => 1,
            },
        },
        ReferenceDestinationContent =>
            '"Test;:_^!""$%&/()=?`*+Test"."><@~\'}{[]\\"."".""""";;::..--__##"',
    },

    # all required values are given (UTF-8 checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[26],
                ExportDataRow => [ ' Ѫ Ѭ Ѳ', 'ѯ Ѵ ѿ', '҂ Ҋ Җ ' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '" Ѫ Ѭ Ѳ";"ѯ Ѵ ѿ";"҂ Ҋ Җ "',
    },

    # all required values are given (UTF-8 checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[26],
                ExportDataRow => [ ' Ѫ Ѭ Ѳ', 'ѯ Ѵ ѿ', '҂ Ҋ Җ ' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => "\" Ѫ Ѭ Ѳ\"\t\"ѯ Ѵ ѿ\"\t\"҂ Ҋ Җ \"",
    },

    # all required values are given (UTF-8 checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[26],
                ExportDataRow => [ ' Ѫ Ѭ Ѳ', 'ѯ Ѵ ѿ', '҂ Ҋ Җ ' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '" Ѫ Ѭ Ѳ":"ѯ Ѵ ѿ":"҂ Ҋ Җ "',
    },

    # all required values are given (UTF-8 checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dot',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[26],
                ExportDataRow => [ ' Ѫ Ѭ Ѳ', 'ѯ Ѵ ѿ', '҂ Ҋ Җ ' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '" Ѫ Ѭ Ѳ"."ѯ Ѵ ѿ"."҂ Ҋ Җ "',
    },
];

# ------------------------------------------------------------ #
# run general ExportDataSave tests
# ------------------------------------------------------------ #

TEST:
for my $Test ( @{$ExportDataTests} ) {

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

        $Self->True(
            0,
            "Test $TestCount: No SourceExportData found for this test."
        );

        next TEST;
    }

    # set default ExportDataSave
    if ( !$Test->{SourceExportData}->{ExportDataSave} ) {
        $Test->{SourceExportData}->{ExportDataSave} = {};
    }

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

        # save format data
        $Self->{ImportExportObject}->FormatDataSave(
            TemplateID => $Test->{SourceExportData}->{ExportDataSave}->{TemplateID},
            FormatData => $Test->{SourceExportData}->{FormatData},
            UserID     => 1,
        );
    }

    # get export data row
    my $ExportString = $Self->{FormatBackendObject}->ExportDataSave(
        %{ $Test->{SourceExportData}->{ExportDataSave} },
    );

    if ( !defined $Test->{ReferenceDestinationContent} ) {

        $Self->True(
            !defined $ExportString,
            "Test $TestCount: ExportDataSave() - return false"
        );

        next TEST;
    }

    if ( !defined $ExportString ) {

        $Self->True(
            !defined $Test->{ReferenceDestinationContent},
            "Test $TestCount: ExportDataSave() - return false"
        );

        next TEST;
    }

    if ( !$Test->{SourceExportData}->{ExportDataSave}->{ExportDataRow} ) {

        $Self->True(
            defined $ExportString,
            "Test $TestCount: ExportDataSave() - return false"
        );

        next TEST;
    }

    # check the export string
    $Self->Is(
        $ExportString,
        $Test->{ReferenceDestinationContent},
        "Test $TestCount: ExportDataSave()",
    );
}
continue {
    $TestCount++;
}

# ------------------------------------------------------------ #
# clean the system
# ------------------------------------------------------------ #

# delete the test templates
$Self->{ImportExportObject}->TemplateDelete(
    TemplateID => \@TemplateIDs,
    UserID     => 1,
);

1;

Um93MS1Db2wxO1JvdzEtQ29sMjtSb3cxLUNvbDMNClJvdzItQ29sMTtSb3cyLUNvbDI7Um93Mi1Db2wzDQpSb3czLUNvbDE7Um93My1Db2wyO1JvdzMtQ29sMw0K
Um93MS1Db2wxCVJvdzEtQ29sMglSb3cxLUNvbDMNClJvdzItQ29sMQlSb3cyLUNvbDIJUm93Mi1Db2wzDQpSb3czLUNvbDEJUm93My1Db2wyCVJvdzMtQ29sMw0K
IlJvdzEtQ29sMSI6IlJvdzEtQ29sMiI6IlJvdzEtQ29sMyINCiJSb3cyLUNvbDEiOiJSb3cyLUNvbDIiOiJSb3cyLUNvbDMiDQoiUm93My1Db2wxIjoiUm93My1Db2wyIjoiUm93My1Db2wzIg0K
IlJvdzEtQ29sMSI7IlJvdzEtQ29sMiI7IlJvdzEtQ29sMyINCiJSb3cyLUNvbDEiOyJSb3cyLUNvbDIiOyJSb3cyLUNvbDMiDQoiUm93My1Db2wxIjsiUm93My1Db2wyIjsiUm93My1Db2wzIg0K
IlJvdzEtQ29sMSIJIlJvdzEtQ29sMiIJIlJvdzEtQ29sMyINCiJSb3cyLUNvbDEiCSJSb3cyLUNvbDIiCSJSb3cyLUNvbDMiDQoiUm93My1Db2wxIgkiUm93My1Db2wyIgkiUm93My1Db2wzIg0K
IgpUZXN0IDEgLSAxIjtUZXN0IDEgLSAyOyJUZXN0IDEKLSAzIjtUZXN0IFxuXHRcclxzDQoiVGVzdCAyIAotIDEiOyJUZQpzdCAyIC0gMiI7IlRlc3QgMiAtIDMKIjsNCg==
IgpUZXN0IDEgLSAxIglUZXN0IDEgLSAyCSJUZXN0IDEKLSAzIglUZXN0IFxuXHRcclxzDQoiVGVzdCAyIAotIDEiCSJUZQpzdCAyIC0gMiIJIlRlc3QgMiAtIDMKIgkNCg==
IgpUZXN0IDEgLSAxIjoiVGVzdCAxIC0gMiI6IlRlc3QgMQotIDMiOiJUZXN0IFxuXHRcclxzIg0KIlRlc3QgMiAKLSAxIjoiVGUKc3QgMiAtIDIiOiJUZXN0IDIgLSAzCiI6DQo=
IgpUZXN0IDEgLSAxIjsiVGVzdCAxIC0gMiI7IlRlc3QgMQotIDMiOyJUZXN0IFxuXHRcclxzIg0KIlRlc3QgMiAKLSAxIjsiVGUKc3QgMiAtIDIiOyJUZXN0IDIgLSAzCiI7DQo=
IgpUZXN0IDEgLSAxIgkiVGVzdCAxIC0gMiIJIlRlc3QgMQotIDMiCSJUZXN0IFxuXHRcclxzIg0KIlRlc3QgMiAKLSAxIgkiVGUKc3QgMiAtIDIiCSJUZXN0IDIgLSAzCiIJDQo=
ICBUZXN0ICA7ICAgIDtUZXN0ICANCiAgICBUZXN0OztUZXN0DQo7OyANCg==
ICBUZXN0ICAJICAgIAlUZXN0ICANCiAgICBUZXN0CQlUZXN0DQoJCSANCg==
IiAgVGVzdCAgIjoiICAgICI6IlRlc3QgICINCiIgICAgVGVzdCI6OiJUZXN0Ig0KOjoiICINCg==
IiAgVGVzdCAgIjsiICAgICI7IlRlc3QgICINCiIgICAgVGVzdCI7OyJUZXN0Ig0KOzsiICINCg==
IiAgVGVzdCAgIgkiICAgICIJIlRlc3QgICINCiIgICAgVGVzdCIJCSJUZXN0Ig0KCQkiICINCg==
IlRlc3Q7Ol+wXiEiIqckJSYvKCk9P7RgKitUZXN0Ijs+PEB+J317W11cDQoiIiIiIjs7OjouLi0tX18jIyI7DQo=
IlRlc3Q7Ol+wXiEiIqckJSYvKCk9P7RgKitUZXN0Igk+PEB+J317W11cDQoiIiIiIjs7OjouLi0tX18jIyIJDQo=
IlRlc3Q7Ol+wXiEiIqckJSYvKCk9P7RgKitUZXN0IjoiPjxAfid9e1tdXCINCiIiIiIiOzs6Oi4uLS1fXyMjIjoNCg==
IlRlc3Q7Ol+wXiEiIqckJSYvKCk9P7RgKitUZXN0IjsiPjxAfid9e1tdXCINCiIiIiIiOzs6Oi4uLS1fXyMjIjsNCg==
IlRlc3Q7Ol+wXiEiIqckJSYvKCk9P7RgKitUZXN0IgkiPjxAfid9e1tdXCINCiIiIiIiOzs6Oi4uLS1fXyMjIgkNCg==
/Pbk3zvc1sQNCt/k9vw7xNbcDQo=
/Pbk3wnc1sQNCt/k9vwJxNbcDQo=
Ivz25N8iOiLc1sQiDQoi3+T2/CI6IsTW3CINCg==
Ivz25N8iOyLc1sQiDQoi3+T2/CI7IsTW3CINCg==
Ivz25N8iCSLc1sQiDQoi3+T2/CIJIsTW3CINCg==
IsqpIMqsIMquIjsiIMqhIMukIM6MICINCiIgIM6XIM+XIM+gICAiOyLOhiDOmyDOniINCg==
IsqpIMqsIMquIjoiIMqhIMukIM6MICINCiIgIM6XIM+XIM+gICAiOiLOhiDOmyDOniINCg==
IsqpIMqsIMquIgkiIMqhIMukIM6MICINCiIgIM6XIM+XIM+gICAiCSLOhiDOmyDOniINCg==
IlJvdzEtQ29sMSI7IlJvdzEtQ29sMiI7IlJvdzEtQ29sMyINCiJSb3cyLUNvbDEiOyJSb3cyLUNvbDIiOyJSb3cyLUNvbDMiDQoiUm93My1Db2wxIjsiMFJvdzMtQ29sMiI7IlJvdzMtQ29sMyINCg==
IlJvdzEtQ29sMSI7IlJvdzEtQ29sMiI7IlJvdzEtQ29sMyINCiJSb3cyLUNvbDEiOyIwUm93Mi1Db2wyIjsiUm93Mi1Db2wzIg0KIlJvdzMtQ29sMSI7IlJvdzMtQ29sMiI7IlJvdzMtQ29sMyINCg==
IyAtLQojIEltcG9ydEV4cG9ydC5wbSAtIGNvZGUgdG8gZXhjZWN1dGUgZHVyaW5nIHBhY2thZ2UgaW5zdGFsbGF0aW9uCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTMgT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSB2YXI6OnBhY2thZ2VzZXR1cDo6SW1wb3J0RXhwb3J0OyAgICAjIyBubyBjcml0aWMKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCj1oZWFkMSBOQU1FCgpJbXBvcnRFeHBvcnQucG0gLSBjb2RlIHRvIGV4Y2VjdXRlIGR1cmluZyBwYWNrYWdlIGluc3RhbGxhdGlvbgoKPWhlYWQxIFNZTk9QU0lTCgpBbGwgZnVuY3Rpb25zCgo9aGVhZDEgUFVCTElDIElOVEVSRkFDRQoKPW92ZXIgNAoKPWN1dAoKPWl0ZW0gbmV3KCkKCmNyZWF0ZSBhbiBvYmplY3QKCiAgICB1c2UgS2VybmVsOjpDb25maWc7CiAgICB1c2UgS2VybmVsOjpTeXN0ZW06OkVuY29kZTsKICAgIHVzZSBLZXJuZWw6OlN5c3RlbTo6TG9nOwogICAgdXNlIEtlcm5lbDo6U3lzdGVtOjpNYWluOwogICAgdXNlIEtlcm5lbDo6U3lzdGVtOjpUaW1lOwogICAgdXNlIEtlcm5lbDo6U3lzdGVtOjpEQjsKICAgIHVzZSBLZXJuZWw6OlN5c3RlbTo6WE1MOwogICAgdXNlIHZhcjo6cGFja2FnZXNldHVwOjpJbXBvcnRFeHBvcnQ7CgogICAgbXkgJENvbmZpZ09iamVjdCA9IEtlcm5lbDo6Q29uZmlnLT5uZXcoKTsKICAgIG15ICRFbmNvZGVPYmplY3QgPSBLZXJuZWw6OlN5c3RlbTo6RW5jb2RlLT5uZXcoCiAgICAgICAgQ29uZmlnT2JqZWN0ID0+ICRDb25maWdPYmplY3QsCiAgICApOwogICAgbXkgJExvZ09iamVjdCAgICA9IEtlcm5lbDo6U3lzdGVtOjpMb2ctPm5ldygKICAgICAgICBDb25maWdPYmplY3QgPT4gJENvbmZpZ09iamVjdCwKICAgICAgICBFbmNvZGVPYmplY3QgPT4gJEVuY29kZU9iamVjdCwKICAgICk7CiAgICBteSAkTWFpbk9iamVjdCA9IEtlcm5lbDo6U3lzdGVtOjpNYWluLT5uZXcoCiAgICAgICAgQ29uZmlnT2JqZWN0ID0+ICRDb25maWdPYmplY3QsCiAgICAgICAgRW5jb2RlT2JqZWN0ID0+ICRFbmNvZGVPYmplY3QsCiAgICAgICAgTG9nT2JqZWN0ICAgID0+ICRMb2dPYmplY3QsCiAgICApOwogICAgbXkgJFRpbWVPYmplY3QgPSBLZXJuZWw6OlN5c3RlbTo6VGltZS0+bmV3KAogICAgICAgIENvbmZpZ09iamVjdCA9PiAkQ29uZmlnT2JqZWN0LAogICAgICAgIExvZ09iamVjdCAgICA9PiAkTG9nT2JqZWN0LAogICAgKTsKICAgIG15ICREQk9iamVjdCA9IEtlcm5lbDo6U3lzdGVtOjpEQi0+bmV3KAogICAgICAgIENvbmZpZ09iamVjdCA9PiAkQ29uZmlnT2JqZWN0LAogICAgICAgIEVuY29kZU9iamVjdCA9PiAkRW5jb2RlT2JqZWN0LAogICAgICAgIExvZ09iamVjdCAgICA9PiAkTG9nT2JqZWN0LAogICAgICAgIE1haW5PYmplY3QgICA9PiAkTWFpbk9iamVjdCwKICAgICk7CiAgICBteSAkWE1MT2JqZWN0ID0gS2VybmVsOjpTeXN0ZW06OlhNTC0+bmV3KAogICAgICAgIENvbmZpZ09iamVjdCA9PiAkQ29uZmlnT2JqZWN0LAogICAgICAgIEVuY29kZU9iamVjdCA9PiAkRW5jb2RlT2JqZWN0LAogICAgICAgIExvZ09iamVjdCAgICA9PiAkTG9nT2JqZWN0LAogICAgICAgIERCT2JqZWN0ICAgICA9PiAkREJPYmplY3QsCiAgICAgICAgTWFpbk9iamVjdCAgID0+ICRNYWluT2JqZWN0LAogICAgKTsKICAgIG15ICRDb2RlT2JqZWN0ID0gdmFyOjpwYWNrYWdlc2V0dXA6OkltcG9ydEV4cG9ydC0+bmV3KAogICAgICAgIENvbmZpZ09iamVjdCA9PiAkQ29uZmlnT2JqZWN0LAogICAgICAgIEVuY29kZU9iamVjdCA9PiAkRW5jb2RlT2JqZWN0LAogICAgICAgIExvZ09iamVjdCAgICA9PiAkTG9nT2JqZWN0LAogICAgICAgIE1haW5PYmplY3QgICA9PiAkTWFpbk9iamVjdCwKICAgICAgICBUaW1lT2JqZWN0ICAgPT4gJFRpbWVPYmplY3QsCiAgICAgICAgREJPYmplY3QgICAgID0+ICREQk9iamVjdCwKICAgICAgICBYTUxPYmplY3QgICAgPT4gJFhNTE9iamVjdCwKICAgICk7Cgo9Y3V0CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHt9OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgICMgY2hlY2sgbmVlZGVkIG9iamVjdHMKICAgIGZvciBteSAkT2JqZWN0ICgKICAgICAgICBxdyhDb25maWdPYmplY3QgTG9nT2JqZWN0IEVuY29kZU9iamVjdCBNYWluT2JqZWN0IFRpbWVPYmplY3QgREJPYmplY3QgWE1MT2JqZWN0KQogICAgICAgICkKICAgIHsKICAgICAgICAkU2VsZi0+eyRPYmplY3R9ID0gJFBhcmFteyRPYmplY3R9IHx8IGRpZSAiR290IG5vICRPYmplY3QhIjsKICAgIH0KCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1pdGVtIENvZGVJbnN0YWxsKCkKCnJ1biB0aGUgY29kZSBpbnN0YWxsIHBhcnQKCiAgICBteSAkUmVzdWx0ID0gJENvZGVPYmplY3QtPkNvZGVJbnN0YWxsKCk7Cgo9Y3V0CgpzdWIgQ29kZUluc3RhbGwgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gMTsKfQoKPWl0ZW0gQ29kZVJlaW5zdGFsbCgpCgpydW4gdGhlIGNvZGUgcmVpbnN0YWxsIHBhcnQKCiAgICBteSAkUmVzdWx0ID0gJENvZGVPYmplY3QtPkNvZGVSZWluc3RhbGwoKTsKCj1jdXQKCnN1YiBDb2RlUmVpbnN0YWxsIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIDE7Cn0KCj1pdGVtIENvZGVVcGdyYWRlKCkKCnJ1biB0aGUgY29kZSB1cGdyYWRlIHBhcnQKCiAgICBteSAkUmVzdWx0ID0gJENvZGVPYmplY3QtPkNvZGVVcGdyYWRlKCk7Cgo9Y3V0CgpzdWIgQ29kZVVwZ3JhZGUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gMTsKfQoKPWl0ZW0gQ29kZVVwZ3JhZGVGcm9tQmVmb3JlXzJfMF8zKCkKClRoaXMgZnVuY3Rpb24gaXMgb25seSBleGVjdXRlZCBpZiB0aGUgaW5zdGFsbGVkIG1vZHVsZSB2ZXJzaW9uIGlzIHNtYWxsZXIgdGhhbiAyLjAuMy4KCiAgICBteSAkUmVzdWx0ID0gJENvZGVPYmplY3QtPkNvZGVVcGdyYWRlRnJvbUJlZm9yZV8yXzBfMygpOwoKPWN1dAoKc3ViIENvZGVVcGdyYWRlRnJvbUJlZm9yZV8yXzBfMyB7ICAgICMjIG5vIGNyaXRpYwogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGZpeCBhIHR5cG8gaW4gdGhlIGRhdGFiYXNlCiAgICAkU2VsZi0+X0ZpeERhdGFiYXNlVHlwbygpOwoKICAgIHJldHVybiAxOwp9Cgo9aXRlbSBDb2RlVW5pbnN0YWxsKCkKCnJ1biB0aGUgY29kZSB1bmluc3RhbGwgcGFydAoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+Q29kZVVuaW5zdGFsbCgpOwoKPWN1dAoKc3ViIENvZGVVbmluc3RhbGwgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gMTsKfQoKPWJlZ2luIEludGVybmFsOgoKPWl0ZW0gX0ZpeERhdGFiYXNlVHlwbygpCgogICAgbXkgJFJlc3VsdCA9ICRDb2RlT2JqZWN0LT5fRml4RGF0YWJhc2VUeXBvKCk7Cgo9Y3V0CgpzdWIgX0ZpeERhdGFiYXNlVHlwbyB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgZml4IHRoZSBDb2x1bW5TZXBlcmF0b3IgdHlwbyAoY29ycmVjdCBpcyBDb2x1bW5TZXBhcmF0b3IpCiAgICByZXR1cm4gaWYgISRTZWxmLT57REJPYmplY3R9LT5EbygKICAgICAgICBTUUwgPT4gIlVQREFURSBpbWV4cG9ydF9mb3JtYXQgIgogICAgICAgICAgICAuICJTRVQgZGF0YV9rZXkgPSAnQ29sdW1uU2VwYXJhdG9yJyAiCiAgICAgICAgICAgIC4gIldIRVJFIGRhdGFfa2V5ID0gJ0NvbHVtblNlcGVyYXRvciciLAogICAgKTsKCiAgICByZXR1cm4gMTsKfQoKMTsKCj1lbmQgSW50ZXJuYWw6Cgo9YmFjawoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIFNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoaHR0cDovL290cnMub3JnLykuCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCgo9Y3V0Cg==
LyoqCiAqIEBwcm9qZWN0ICAgICBPVFJTIChodHRwOi8vd3d3Lm90cnMub3JnKSAtIEFnZW50IEZyb250ZW5kCiAqIEBjb3B5cmlnaHQgICBPVFJTIEFHCiAqIEBsaWNlbnNlICAgICBBR1BMIChodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQpCiAqLwoKLyoqCiAqIEBwYWNrYWdlICAgICBTa2luICJEZWZhdWx0IgogKiBAc2VjdGlvbiAgICAgSW1wb3J0IEV4cG9ydCBTY3JlZW4KICovCgpAbWVkaWEgc2NyZWVuLHByb2plY3Rpb24sdHYsaGFuZGhlbGQgewoKLyoqCiAqIEBzdWJzZWN0aW9uCiAqLwoKCi5NYXBIZWFkZXJSb3cgbGFiZWwgewogICAgY29sb3I6ICM5MjkyOTI7Cn0KCi5NYXBIZWFkZXJSb3cgLkhlYWRlciwKLk1hcEhlYWRlclJvdyAuRmllbGQgewogICAgZGlzcGxheTogaW5saW5lOwogICAgbWFyZ2luLXJpZ2h0OiAxNXB4OwogICAgcGFkZGluZy1sZWZ0OiAycHg7Cn0KCmJ1dHRvbi5BcnJvd1VwLApidXR0b24uQXJyb3dEb3duIHsKICAgIGhlaWdodDogMTZweDsKICAgIHdpZHRoOiAxNnB4OwogICAgcGFkZGluZzogMXB4OwogICAgbWFyZ2luLXRvcDogMDsKICAgIG1hcmdpbi1ib3R0b206IDFweDsKICAgIGJvcmRlci1zdHlsZTogbm9uZTsKICAgIHRleHQtaW5kZW50OiAtOTk5OXB4OwogICAgZGlzcGxheTogaW5saW5lLWJsb2NrOwogICAgdmVydGljYWwtYWxpZ246IG1pZGRsZTsKICAgIGN1cnNvcjogcG9pbnRlcjsKfQoKYnV0dG9uLkFycm93VXA6YWN0aXZlLApidXR0b24uQXJyb3dEb3duOmFjdGl2ZSB7CiAgICBtYXJnaW4tdG9wOiAxcHg7CiAgICBtYXJnaW4tYm90dG9tOiAwOwp9CgpidXR0b24uQXJyb3dVcFtkaXNhYmxlZD0iZGlzYWJsZWQiXTphY3RpdmUsCmJ1dHRvbi5BcnJvd0Rvd25bZGlzYWJsZWQ9ImRpc2FibGVkIl06YWN0aXZlIHsKICAgIG1hcmdpbi10b3A6IDBweDsKICAgIG1hcmdpbi1ib3R0b206IDFweDsKfQoKYnV0dG9uLkFycm93VXAgewogICAgYmFja2dyb3VuZDogdXJsKC4uL2ltZy9pY29ucy9pbXBvcnRleHBvcnRfYXJyb3dfdXAucG5nKTsKfQoKYnV0dG9uLkFycm93VXBbZGlzYWJsZWQ9ImRpc2FibGVkIl0gewogICAgYmFja2dyb3VuZDogdXJsKC4uL2ltZy9pY29ucy9pbXBvcnRleHBvcnRfYXJyb3dfdXBfZGlzYWJsZWQucG5nKTsKICAgIGN1cnNvcjogZGVmYXVsdDsKfQoKYnV0dG9uLkFycm93RG93biB7CiAgICBiYWNrZ3JvdW5kOiB1cmwoLi4vaW1nL2ljb25zL2ltcG9ydGV4cG9ydF9hcnJvd19kb3duLnBuZyk7Cn0KCmJ1dHRvbi5BcnJvd0Rvd25bZGlzYWJsZWQ9ImRpc2FibGVkIl17CiAgICBiYWNrZ3JvdW5kOiB1cmwoLi4vaW1nL2ljb25zL2ltcG9ydGV4cG9ydF9hcnJvd19kb3duX2Rpc2FibGVkLnBuZyk7CiAgICBjdXJzb3I6IGRlZmF1bHQ7Cn0KCn0gLyogZW5kIEBtZWRpYSAqLwo=
iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAZlJREFUeNqkUztLA0EQnr3bvEhhI9FC09gJInJiZbAQREHUWhTtbK8ULEWwFVLZaJE/YKWFgn2SQgxBgjYSIoYkEsxxuec6c/FxCXmAGZjbmdmZ7+a1TAgBw5AEQxL3K7OnjxAMhyEQCkEgGARJllfJ7jrOjWWaYBkGmM0mPBzOdAdoIyF2MFD9lkfxmxqYgT/Ysix1fzOukHp59apCq1epgT0QFGya6u56XMnnG0C8vTahkI3u+gJ4wYah7m1NKcWiBrregFqtArlcCTaWxhW66wRpKwEdEjTW5EUmS/riXEzhXMB9upSFdMnzYYwl/KXwjgwOsHEgXBds285oQgYHAQ1dB875PJMkmkz/Jjq2DbqmeXJFDoHDwAMwUI9Eo/0BtHod9EbjVy/zKDgugIkARHSy52uUFroDFM9WgI9NQ2T5qJUBj2AJfwD63QnY73mUznuPkRy022MwcfNsJnlMMtlawYMXadItP8Wst1yhmmz9SVQLBbTRUn0iv1CrfpyZ/zXiiOgYQY72WHDq5QfGuF0B/kNfAgwAlIbWNoRkTzIAAAAASUVORK5CYII=
iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAQAAAC1+jfqAAAAAXNSR0IArs4c6QAAAAJiS0dEAP+Hj8y/AAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH2goUEigB30KkogAAAMpJREFUKM+lkT0OAWEQhp/9ESJKF5FPK1oSCQfYXvvVDuAcCola5w5sKdkL6EQhYv+NYpddSyi8xUwxT2bemTGE7zJ/1LGzNKNBnRrWANJNTEjAvAzkclINtFm+dQAQJ9ZjBWstFIhZlCM9Uh4eQxVpcSqAOKGeqAM+J/b0VfhE8hFRT1i40FEWWxfA6GVjckCmKTeSXcoNH7trYlVNJlwBQQiAZhU4c8k6IfiAT+sVOD52AQJ4xsJk6VrRp1PHWXJXgOGWAePvb94BXKRFykgCmkUAAAAASUVORK5CYII=
iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAZBJREFUeNrEU79Lw0AUfpemhZI4CIpCJ92cHE4HJ0HEQcFRQdShIJ0E/5SCOBSkgxZE7CS4iIt06NIMnVxUcKhWCJb+iPl1l/gu1ppKKkIHH1zu3r33ffnu3Tvi+z4MYxIMaXLYWcy/9taEkBxOtOtqqDTzFbtNT0QTeIyFXbq5kgoIzq9rf1PgOk5fsNVyQZTo5/5AgqauQ1JVe75puuB5AI5lfe91OvidiiYwms0gOakogd9ovYPjuuCY5ifYMMC17cEKhDEEGO02yLIMNYsB9wjYSMCwPh7nvx8BK51DzZSjbo5E9biCBBAQoFW6ORpOmegi2nYJEyhNr9EHPOo9qvXAB3V1iY4mAF7OrjS83tLARkJwwbWsbPmoqCkJCXQmwRuPQVyW4Clf1ERM5IQxJNzK6vwWSmcQm1nfxv2Dsb0dyjHcOD4Vf87yu8uCVb2AMKaPAJPEJK5gRF7Y3yCTs7uBsnr1hJUPi7gUd9j2Q6AognEcKRwxMr08FxA83lS6KaLXnxHjRRL8y2v8EGAAqEvNnH+aSSAAAAAASUVORK5CYII=
iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAQAAAC1+jfqAAAAAXNSR0IArs4c6QAAAAJiS0dEAP+Hj8y/AAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH2goUEig6bklNhgAAAMlJREFUKM+lkT0KAjEQhb9IFEE9TToP4BU8gq2exG23FAKCYCmIra25iYL4w/6asVhddlW0cJo3yXy8CS9K+F6NH3N0IVMAVIgBnIwAJlXgVogZGNh8ckgfxzNS9jXgQAeACE8MwLUOnIjpACcyYuBa+uinVc4ZzQVPRI5/XSGhN54cQYhhB+IYVYBkK2ZojkR4+qbNyqltLSixSTBzLRIyNEuXBGJfkhSbBtb16LJwaTmuPBLEZszHoIKmfcvhieR70Ovqnfr7N++lLlCSshf+TwAAAABJRU5ErkJggg==