ImportExport
2.1.91
OTRS AG
http://otrs.org/
GNU AFFERO GENERAL PUBLIC LICENSE Version 3, November 2007
Build for OTRS::ITSM 3.0.0 beta1.
The ImportExport package.
Das ImportExport Paket.
El paquete ImportExport.
3.0.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."
);
}
}
2010-12-22 22:28:26
opms.otrs.com
IyEvdXNyL2Jpbi9wZXJsIC13CiMgLS0KIyBJbXBvcnRFeHBvcnQucGwgLSBpbXBvcnQvZXhwb3J0IHNjcmlwdAojIENvcHlyaWdodCAoQykgMjAwMS0yMDEwIE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogSW1wb3J0RXhwb3J0LnBsLHYgMS4xNCAyMDEwLzAyLzIzIDEyOjA4OjQ2IGJlcyBFeHAgJAojIC0tCiMgVGhpcyBwcm9ncmFtIGlzIGZyZWUgc29mdHdhcmU7IHlvdSBjYW4gcmVkaXN0cmlidXRlIGl0IGFuZC9vciBtb2RpZnkKIyBpdCB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEdOVSBBRkZFUk8gR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBhcyBwdWJsaXNoZWQgYnkKIyB0aGUgRnJlZSBTb2Z0d2FyZSBGb3VuZGF0aW9uOyBlaXRoZXIgdmVyc2lvbiAzIG9mIHRoZSBMaWNlbnNlLCBvcgojIGFueSBsYXRlciB2ZXJzaW9uLgojCiMgVGhpcyBwcm9ncmFtIGlzIGRpc3RyaWJ1dGVkIGluIHRoZSBob3BlIHRoYXQgaXQgd2lsbCBiZSB1c2VmdWwsCiMgYnV0IFdJVEhPVVQgQU5ZIFdBUlJBTlRZOyB3aXRob3V0IGV2ZW4gdGhlIGltcGxpZWQgd2FycmFudHkgb2YKIyBNRVJDSEFOVEFCSUxJVFkgb3IgRklUTkVTUyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UuICBTZWUgdGhlCiMgR05VIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgZm9yIG1vcmUgZGV0YWlscy4KIwojIFlvdSBzaG91bGQgaGF2ZSByZWNlaXZlZCBhIGNvcHkgb2YgdGhlIEdOVSBBZmZlcm8gR2VuZXJhbCBQdWJsaWMgTGljZW5zZQojIGFsb25nIHdpdGggdGhpcyBwcm9ncmFtOyBpZiBub3QsIHdyaXRlIHRvIHRoZSBGcmVlIFNvZnR3YXJlCiMgRm91bmRhdGlvbiwgSW5jLiwgNTkgVGVtcGxlIFBsYWNlLCBTdWl0ZSAzMzAsIEJvc3RvbiwgTUEgIDAyMTExLTEzMDcgIFVTQQojIG9yIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSBGaWxlOjpCYXNlbmFtZTsKdXNlIEZpbmRCaW4gcXcoJFJlYWxCaW4pOwp1c2UgbGliIGRpcm5hbWUoJFJlYWxCaW4pOwp1c2UgbGliIGRpcm5hbWUoJFJlYWxCaW4pIC4gJy9LZXJuZWwvY3Bhbi1saWInOwoKdXNlIEdldG9wdDo6U3RkOwp1c2UgS2VybmVsOjpDb25maWc7CnVzZSBLZXJuZWw6OlN5c3RlbTo6REI7CnVzZSBLZXJuZWw6OlN5c3RlbTo6RW5jb2RlOwp1c2UgS2VybmVsOjpTeXN0ZW06OkltcG9ydEV4cG9ydDsKdXNlIEtlcm5lbDo6U3lzdGVtOjpMb2c7CnVzZSBLZXJuZWw6OlN5c3RlbTo6TWFpbjsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OICRSZWFsQmluKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuMTQgJCkgWzFdOwoKIyBnZXQgb3B0aW9ucwpteSAlT3B0czsKZ2V0b3B0cyggJ2huOmE6aTpvOicsIFwlT3B0cyApOwoKaWYgKCAkT3B0c3tofSApIHsKCiAgICBwcmludCBTVERPVVQgIkltcG9ydEV4cG9ydC5wbCA8UmV2aXNpb24gJFZFUlNJT04+IC0gYW4gaW1wb3J0L2V4cG9ydCB0b29sXG4iOwogICAgcHJpbnQgU1RET1VUICJDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvXG4iOwogICAgcHJpbnQgU1RET1VUICJcbiI7CiAgICBwcmludCBTVERPVVQgInVzYWdlOkltcG9ydEV4cG9ydC5wbCAtbiA8VGVtcGxhdGVOdW1iZXI+IC1hIGltcG9ydHxleHBvcnQgIjsKICAgIHByaW50IFNURE9VVCAiWy1pIDxTb3VyY2VGaWxlPl0gWy1vIDxEZXN0aW5hdGlvbkZpbGU+XVxuIjsKICAgIHByaW50IFNURE9VVCAiXG4iOwogICAgcHJpbnQgU1RET1VUICIgICBleGFtcGxlczpcbiI7CiAgICBwcmludCBTVERPVVQgIiAgICAgICBJbXBvcnRFeHBvcnQucGwgLW4gMDAwMDQgLWEgaW1wb3J0IC1pIC90bXAvaW1wb3J0LmNzdlxuIjsKICAgIHByaW50IFNURE9VVCAiICAgICAgIEltcG9ydEV4cG9ydC5wbCAtbiAwMDAwNCAtYSBleHBvcnQgLW8gL3RtcC9leHBvcnQuY3N2XG4iOwoKICAgIGV4aXQgMTsKfQoKIyBjaGVjayB0ZW1wYWx0ZSBudW1iZXIKaWYgKCAhJE9wdHN7bn0gKSB7CiAgICBwcmludCBTVERFUlIgIkVSUk9SOiBOZWVkIC1uIFRlbXBsYXRlTnVtYmVyXG4iOwogICAgZXhpdCAxOwp9CmlmICggJE9wdHN7bn0gIX4gbXsgXEEgXGQrIFx6IH14bXMgKSB7CiAgICBwcmludCBTVERFUlIgIkVSUk9SOiBJbnZhbGlkIFRlbXBsYXRlTnVtYmVyXG4iOwogICAgZXhpdCAxOwp9Cm15ICRUZW1wbGF0ZUlEID0gaW50ICRPcHRze259OwoKIyBjaGVjayBhY3Rpb24gbW9kZQppZiAoICEkT3B0c3thfSApIHsKICAgIHByaW50IFNUREVSUiAiRVJST1I6IE5lZWQgLWEgaW1wb3J0fGV4cG9ydFxuIjsKICAgIGV4aXQgMTsKfQppZiAoIGxjICRPcHRze2F9IG5lICdpbXBvcnQnICYmIGxjICRPcHRze2F9IG5lICdleHBvcnQnICkgewogICAgcHJpbnQgU1RERVJSICJFUlJPUjogSW52YWxpZCBhY3Rpb25cbiI7CiAgICBleGl0IDE7Cn0KCiMgY3JlYXRlIGNvbW1vbiBvYmplY3RzCm15ICVDb21tb25PYmplY3Q7CiRDb21tb25PYmplY3R7Q29uZmlnT2JqZWN0fSA9IEtlcm5lbDo6Q29uZmlnLT5uZXcoKTsKJENvbW1vbk9iamVjdHtMb2dPYmplY3R9ICAgID0gS2VybmVsOjpTeXN0ZW06OkxvZy0+bmV3KAogICAgTG9nUHJlZml4ID0+ICdPVFJTLUltcG9ydEV4cG9ydCcsCiAgICAlQ29tbW9uT2JqZWN0LAopOwokQ29tbW9uT2JqZWN0e0VuY29kZU9iamVjdH0gICAgICAgPSBLZXJuZWw6OlN5c3RlbTo6RW5jb2RlLT5uZXcoJUNvbW1vbk9iamVjdCk7CiRDb21tb25PYmplY3R7TWFpbk9iamVjdH0gICAgICAgICA9IEtlcm5lbDo6U3lzdGVtOjpNYWluLT5uZXcoJUNvbW1vbk9iamVjdCk7CiRDb21tb25PYmplY3R7REJPYmplY3R9ICAgICAgICAgICA9IEtlcm5lbDo6U3lzdGVtOjpEQi0+bmV3KCVDb21tb25PYmplY3QpOwokQ29tbW9uT2JqZWN0e0ltcG9ydEV4cG9ydE9iamVjdH0gPSBLZXJuZWw6OlN5c3RlbTo6SW1wb3J0RXhwb3J0LT5uZXcoJUNvbW1vbk9iamVjdCk7CgojIGdldCB0ZW1wbGF0ZSBkYXRhCm15ICRUZW1wbGF0ZURhdGEgPSAkQ29tbW9uT2JqZWN0e0ltcG9ydEV4cG9ydE9iamVjdH0tPlRlbXBsYXRlR2V0KAogICAgVGVtcGxhdGVJRCA9PiAkVGVtcGxhdGVJRCwKICAgIFVzZXJJRCAgICAgPT4gMSwKKTsKCmlmICggISRUZW1wbGF0ZURhdGEtPntUZW1wbGF0ZUlEfSApIHsKICAgIHByaW50IFNUREVSUiAiRVJST1I6IFRlbXBsYXRlICRPcHRze259IG5vdCBmb3VuZCFcbiI7CiAgICBwcmludCBTVERFUlIgIkV4cG9ydCBhYm9ydGVkLlxuIjsKICAgIGV4aXQgMTsKfQoKIyB0aW1lIHRvIHN0YXJ0CmlmICggbGMgJE9wdHN7YX0gZXEgJ2ltcG9ydCcgKSB7CgogICAgbXkgJFNvdXJjZUNvbnRlbnQgPSBcZG8geycnfTsKICAgIGlmICggJE9wdHN7aX0gKSB7CgogICAgICAgIHByaW50IFNURE9VVCAiUmVhZCBGaWxlICRPcHRze2l9LlxuIjsKCiAgICAgICAgIyByZWFkIHNvdXJjZSBmaWxlCiAgICAgICAgJFNvdXJjZUNvbnRlbnQgPSAkQ29tbW9uT2JqZWN0e01haW5PYmplY3R9LT5GaWxlUmVhZCgKICAgICAgICAgICAgTG9jYXRpb24gPT4gJE9wdHN7aX0sCiAgICAgICAgICAgIFJlc3VsdCAgID0+ICdTQ0FMQVInLAogICAgICAgICAgICBNb2RlICAgICA9PiAnYmlubW9kZScsCiAgICAgICAgKTsKCiAgICAgICAgZGllICJDYW4ndCByZWFkIGZpbGUgJE9wdHN7aX0uXG5JbXBvcnQgYWJvcnRlZC5cbiIgaWYgISRTb3VyY2VDb250ZW50OwogICAgfQoKICAgIHByaW50IFNURE9VVCAiSW1wb3J0IGluIHByb2Nlc3MuLi5cbiI7CgogICAgIyBpbXBvcnQgZGF0YQogICAgbXkgJFJlc3VsdCA9ICRDb21tb25PYmplY3R7SW1wb3J0RXhwb3J0T2JqZWN0fS0+SW1wb3J0KAogICAgICAgIFRlbXBsYXRlSUQgICAgPT4gJFRlbXBsYXRlSUQsCiAgICAgICAgU291cmNlQ29udGVudCA9PiAkU291cmNlQ29udGVudCwKICAgICAgICBVc2VySUQgICAgICAgID0+IDEsCiAgICApOwoKICAgIGRpZSAiXG5FcnJvciBvY2N1cnJlZC4gSW1wb3J0IGltcG9zc2libGUhIFNlZSB0aGUgT1RSUyBsb2cgZm9yIGRldGFpbHMuXG4iIGlmICFkZWZpbmVkICRSZXN1bHQ7CgogICAgIyBQcmludCByZXN1bHQKICAgIHByaW50IFNURE9VVCAiXG4iOwogICAgcHJpbnQgU1RET1VUCiAgICAgICAgIkltcG9ydCBvZiAkUmVzdWx0LT57Q291bnRlcn0gJFJlc3VsdC0+e09iamVjdH0gcmVjb3JkczogIgogICAgICAgIC4gIiRSZXN1bHQtPntGYWlsZWR9IGZhaWxlZCwgJFJlc3VsdC0+e1N1Y2Nlc3N9IHN1Y2NlZWRlZFxuIjsKICAgIGZvciBteSAkUmV0Q29kZSAoIHNvcnQga2V5cyAleyAkUmVzdWx0LT57UmV0Q29kZX0gfSApIHsKICAgICAgICBteSAkQ291bnQgPSAkUmVzdWx0LT57UmV0Q29kZX0tPnskUmV0Q29kZX0gfHwgMDsKICAgICAgICBwcmludCBTVERPVVQKICAgICAgICAgICAgIkltcG9ydCBvZiAkUmVzdWx0LT57Q291bnRlcn0gJFJlc3VsdC0+e09iamVjdH0gcmVjb3JkczogJENvdW50ICRSZXRDb2RlXG4iLAogICAgfQogICAgaWYgKCAkUmVzdWx0LT57RmFpbGVkfSApIHsKICAgICAgICBwcmludCBTVERPVVQKICAgICAgICAgICAgIkxhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlOiAkUmVzdWx0LT57Q291bnRlcn1cbiI7CiAgICB9Cn0KZWxzaWYgKCBsYyAkT3B0c3thfSBlcSAnZXhwb3J0JyApIHsKCiAgICBwcmludCBTVERPVVQgIkV4cG9ydCBpbiBwcm9jZXNzLi4uXG4iOwoKICAgICMgZXhwb3J0IGRhdGEKICAgIG15ICRSZXN1bHQgPSAkQ29tbW9uT2JqZWN0e0ltcG9ydEV4cG9ydE9iamVjdH0tPkV4cG9ydCgKICAgICAgICBUZW1wbGF0ZUlEID0+ICRUZW1wbGF0ZUlELAogICAgICAgIFVzZXJJRCAgICAgPT4gMSwKICAgICk7CgogICAgZGllICJcbkVycm9yIG9jY3VycmVkLiBFeHBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy5cbiIgaWYgIWRlZmluZWQgJFJlc3VsdDsKCiAgICBwcmludCBTVERPVVQKICAgICAgICAiXG4iLAogICAgICAgICJTdWNjZXNzOiAkUmVzdWx0LT57U3VjY2Vzc31cbiIsCiAgICAgICAgIkZhaWxlZCA6ICRSZXN1bHQtPntGYWlsZWR9XG4iLAogICAgICAgICJcbiI7CgogICAgaWYgKCAkT3B0c3tvfSApIHsKCiAgICAgICAgbXkgJEZpbGVDb250ZW50ID0gam9pbiAiXG4iLCBAeyAkUmVzdWx0LT57RGVzdGluYXRpb25Db250ZW50fSB9OwoKICAgICAgICAjIHNhdmUgZGVzdGluYXRpb24gY29udGVudCB0byBmaWxlCiAgICAgICAgbXkgJFN1Y2Nlc3MgPSAkQ29tbW9uT2JqZWN0e01haW5PYmplY3R9LT5GaWxlV3JpdGUoCiAgICAgICAgICAgIExvY2F0aW9uID0+ICRPcHRze299LAogICAgICAgICAgICBDb250ZW50ICA9PiBcJEZpbGVDb250ZW50LAogICAgICAgICk7CgogICAgICAgIGRpZSAiQ2FuJ3Qgd3JpdGUgZmlsZSAkT3B0c3tvfS5cbkV4cG9ydCBhYm9ydGVkLlxuIiBpZiAhJFN1Y2Nlc3M7CgogICAgICAgIHByaW50IFNURE9VVCAiRmlsZSAkT3B0c3tvfSBzYXZlZC5cbiI7CiAgICB9CgogICAgcHJpbnQgU1RET1VUICJFeHBvcnQgY29tcGxldGUuXG4iOwp9CgpleGl0IDA7Cgo9YmFjawoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIFNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoaHR0cDovL290cnMub3JnLykuCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMi4wLnR4dC4KCj1jdXQKCj1oZWFkMSBWRVJTSU9OCgokUmV2aXNpb246IDEuMTQgJCAkRGF0ZTogMjAxMC8wMi8yMyAxMjowODo0NiAkCgo9Y3V0Cg==
iVBORw0KGgoAAAANSUhEUgAABh8AAAQACAYAAADx1EgnAAAABmJLR0QA/wD/AP+gvaeTAAAAB3RJTUUH2gkIFBcB3n5g2wAAIABJREFUeJzs3T9sG3ea//HPyP9iJd712knuNomye7ukEDg8N24MDm5rk7YBbaNCLlwEIAEXIRsDd4AKFQL2ADcc4xoSyAEp3LCJAMOkuwMWJuEr3PjkYH8id7FrZTeJ4zix41h2bGl+xcyQQ4mkKInkDKX3CxBgc/58nxkqgef7zPN9DNu2bQEAAAAAAAAAAPTH/f1BRwAAALCXPH36VF9//bUOHz6sd955J+hwOvrhhx/05MkTHT58WMeOHQs6HAAAAADAiCH5AAAAMES2bWttbU1hLz599eqVnj17pn379gUdCgAAAABgBI0FHQAAAMBe4iUdDMMIOJLejI3xz0UAAAAAwNbxNAkAADBEYa948KytrQUdAgAAAABghJF8AAAACEDYKx9GrUIDAAAAABAuJB8AAACGaNQm9UclTgAAAABAuJB8AAAAGCKWXQIAAAAA7AUkHwAAAAIwKhUFNJwGAAAAAGwHT5MAAABDNCrLLnmVD2GPEwAAAAAQTiQfAAAAhojljAAAAAAAe8H+oAMAAADYS37+859rfHxc+/eH+59hBw8e1Pj4uA4ePBh0KAAAAACAEWTYo9L1EAAAAAAAAAAAjIL7LLsEAAAAAAAAAAD6iuQDAAAAAAAAAADoK5IPAAAAAAAAAACgr8Ld6RAAAGCX+eqrr/TDDz/ozTff1C9+8Yugw+nou+++0+rqqn72s5/RdBoAAAAAsGU0nAYAABiytbU1GYYhwzCCDqWjJ0+eaHV1Va+//jrJBwAAAADAVt0n+QAAAAAAAAAAAPrpPj0fAAAAAAAAAABAX5F8AAAAAAAAAAAAfUXyAQAAYIhevHihZ8+eaXV1NehQAAAAAAAYGJIPAAAAQ/TNN9/o73//u549exZ0KF198cUXun//vl6+fBl0KAAAAACAEbQ/6AAAAAAQPi9evNDa2pps2w46FAAAAADACKLyAQAAIABjY+H+Z5iXdAh7nAAAAACAcOJpEgAAYIjW1taCDqEnVDwAAAAAAHaC5AMAAAA6Mgwj6BAAAAAAACOI5AMAAEAAwryckb86g+QDAAAAAGA7wvvUCwAAsAuN2nJGJB8AAAAAANtB8gEAAGCIRqHnA5UPAAAAAICdIvkAAAAQgDBP6nvVGWFeGgoAAAAAEG48UQIAAAQgzMkHr/IhzDECAAAAAMJtf9ABAAAA7CXvvPOO1tbWdODAgaBD6WhsbExvvPGG9u3bF3QoAAAAAIARZdij1vUQAAAAAAAAAACE2X2WXQIAAAAAAAAAAH1F8gEAAAAAAAAAAPQVyQcAAIAhWltbazR0BgAAAABgt6LhNAAAwBD9+c9/liT99re/1dhYON8DWVlZ0ZMnT3To0CEdPXo06HAAAAAAACOIhtMAAABo8dNPP2llZUX79u3TG2+8EXQ4AAAAAIDRc5/kAwAAAAAAAAAA6Kf74az1BwAAAAAAAAAAI4vkAwAAAAAAAAAA6CuSDwAAAENi27aePn2qZ8+eBR0KAAAAAAADRfIBAABgSFZXV/Xll1/qyy+/DDqUrr799lv99a9/1ffffx90KAAAAACAEUXyAQAAYEjW1taCDqEnq6urevny5cjECwAAAAAIH5IPAAAAQ2LbdtAh9MRLOhiGEXAkAAAAAIBRRfIBAABgyMbGwv1PMC9JEvY4AQAAAADhxRMlAADAkIxK5cOoxAkAAAAACC+SDwAAAEMyKpP6VD4AAAAAAHaKJ0oAAIAhC/ukvpd8oOcDAAAAAGC7wv3kCwAAsIuMyqQ+DacBAAAAADtF8gEAAGBIWHYJAAAAALBX7A86AAAAgL0m7BUFXuUDAGC0vHr1Sj/88EPQYQDYg15//XUdPHgw6DAAhAzJBwAAgCEZHx/X+++/H/rkw5EjR7S6uqoDBw4EHQoAYAtu376tf/u3fws6DAB70LVr1zQzMxN0GABChuQDAADAkBiGoUOHDgUdxqbefPPNoEMAAGzTv/7rv+ru3btBhwFgDzl37lzQIQAIKRbyBQAAAAAAAAAAfUXyAQAAAAAAAAAA9BXJBwAAgCFZW1vTy5cvaegMAAAAANj1SD4AAAAMyZMnT3T//n09fPgw6FA6Wl1d1T/+8Q999dVXQYcCAAAAABhhNJwGAAAYkqNHj+ro0aNBh9GVYRg6cuSIbNsOOhQAAAAAwAgj+QAAAICGsbExHTlyJOgwAAAAAAAjjmWXAAAAAAAAAABAX5F8AAAAAAAAAAAAfUXyAQAAYEieP3+up0+f6sWLF0GHAgAAtsg0TFn1oKPoo3pZ5a1cz1b377e6tbXvIOh4AQAkHwAAAIblyZMn+vLLL/X06dOgQ+nohx9+0J///Gd9/fXXQYcCAECoVOyKMpGgo+iTuiUzOq+lQe0ftFGLFwB2KZIPAAAAQzY2Ft5/gq2trTV+AAAAAADYrvA++QIAAOwyozCh78VoGEbAkQAAEC6NJX+85X/KlkzDkOH+mFbd3db8LF32n6Euy2xuM4y0vM11y5TRsqRQWemWc5qyrHTzWNNS64pC687tH7huyTTSsiyzuT2aVVVVZaPuGN3ULZlt9+98Pdu6Rz1d57q4DP/4vvNvNV4AwECQfAAAABiyME/se8mHMFdnAAAQvKqy89Knti3btlXLxVXNRmVcbP2skPQmuOuyzKiysZLsxvZFJd3J9UimolKqquwVZ+9yOqlCqqRKY52nqrJZqWTbsu2acsoq2py1X3fumnKLydYEhArKfj7bGNuu5RRXXLma7Rujg0hGlQ37d7+e7d2jza7Tr6x0NKtYyfaNH1c1e0XlLcYLABgcnioBAACGZBSqCmzblkTyAQCAzaRmM/Km7SNnpxVv+9miluqSyleUraZUyicax0cys0pVs3LzDUpczilemJdVtjRfaN1XklKlvJxPIsp86u7b9twRZWZTUmGh5c3+1FTr+Xakh+uRtniPNrtOv3pUl21b/lvU7ly9xAsAGByeKgEAANAwCgkSAACCF9eJaC+fOepLi5IKSrYsE5RUQdKiN1seyejTnJRNZhVrTMB3OHdkUjFV9XnN23xCLUNHT6ybiO8c23b0dD1bvEdtt6+/zsbnESeBUU6vW0pq6/ECAAaH5AMAAMCQjEJVAcsuAQAwIPGcanZzmSDvx7/sUe1zZ/q8sDAC3Qh6uJ7BcXpiGMlF5Wr+paS2Hi8AYHB4qgQAABiSUXjAHYUECQAAoyYyGZOqRd3o1tu5nFaykFLJLilVSK5rVr3u7f/6khaVUmMlpfXnrn2uqmKaHFAeoKfr2ZZNrtNTXlAhnlPNrqiR66h93rHyYXDxAgC64akSAABgyMI8se8lH1h2CQCAPkpcVi5eVfairyFzOS3DMN1+BmWlkwW330FC+VJqXSNmtTavvphVNTXlLM3knfuKrwH1fEHytrfTaTmjXvff9Hq2r+N1rudPJtQtmUnfIkpbiBcAMDjhffIFAADYZbwljcKMng8AAAxCRJlKTTllFfV6DiQXlas5b+6X00kV4jld9mbZE5eVixeUbJQ/xJXLye1ZEFU2VpLtbzBdqSm3mHR7Gazf3k5CUympkDRkpHtZ4mn9/t2vZ/u6Xac/nLxKqaqyUa/fQ1HTtZJSjYRD7/ECAAbHsEeh/h8AAGAXePXqlWzb1v79+0M7uf/999/r1atXOnLkiA4dOhR0OACALbh165YuXbqku3fvBh0K+qluyYwWNb3jif2Q2yvXuQudO3dOMzMzmpmZCToUAOFyf3/QEQAAAOwV+/eH/59eR48eDToEAAAAAMAuwLJLAAAAAAAAe1XdkuktRdT2Z+d9HAAAe1P4X78DAAAAAADYqyIZVezM6J5/1OIAAPQNyQcAAIAhefHihcbGxkLd8wEAMLru3bun//u//9Nrr70WdCgA9pAXL14oEonQ8wHABiQfAAAAhuTrr7/W6uqq3n33XR08eDDocDawbVtffPGFxsbG9M4775AgAYARE41G9eabb+p///d/gw4FwB7y+9//XidPngw6DAAhRPIBAABgSN5///2gQ9jU8ePHtba2FnQYAIBtOHjwoH75y1/qN7/5TdChANhDJiYmqLgC0BbJBwAAAEiSDMPQ+Ph40GEAAAAAAHaBsaADAAAAAAAAAAAAuwvJBwAAAAAAAAAA0FckHwAAAIbAtm09fvxYjx8/DjqUjmzbDjoEAAAAAMAuQc8HAACAIVhbW9ODBw8kST//+c8Djqa9H3/8UV9++aXGx8f17rvvBh0OAAAAAGCEUfkAAAAwBKNQVbC2tibJaTwNAAAAAMBOUPkAAAAwRGNj4X33g+QD9rJMJqP//u//DjoMYEdWV1f1s5/9LOgwAAAAJFH5AAAAMBTexH6YeTGGOUECDMr//M//6OnTp0GHAQxX3ZJppFXusks5bcjYZJ9+jgcAAHYPKh8AAACGwFt2KcxVBSQfsNeNjY3phx9+CDoMYNtu3bqlS5cubfjcMtOarOSVWL8hklFlk1UBE3lbucV0fwLsYbwN6pbMK5Oq5DdEDwAAQo4nSwAAgCEi+QAACAOnoqFDVUM57W4zZKQ31inULVOGYajNpi2P55zLlGl6201ZdV8c0ayqhWTXeAAAQDjxZAkAADAELLsEAAiTRN6WbdvKxddtqFsyk1LJdraXlFS2OrjxIpmKSqmqqrGSbNuWXYope6XsHSS7llM85W6zbdlUQAAAMDJ4sgQAABiiME/se0tDhTlGAECPfNUL2WpBSffPZqOsoL36jaJipeYSTYl8Sal1+0QyFdm2rf7lAeLKXXZPlphSanGpXycGAAAB4skSAABgCOj5AAAYqkS+US2Qi6calQyVTCToyAAAwB7BkyUAAMAQeMmHMPOSD2FOkAAABitydlqLC76+CuUFFdbts52eDzvSqIQoKz3McQEAwI7sDzoAAACAvWB8fFy//e1vgw6jq6NHj+rly5c6ePBg0KEAAAaqrLSRbCYVjIKkuHK1ijKRjGZlqJmHTikeLyhpnpBdyfR9vLM3TCULVamQ1qQ9pQV3v0J60unvEMloNmbIMLKSpHiupgptHwAAGAmGPQqv4QEAAADAAJ08eVL37t3T6upq0KEA23br1i1dunRJd+/eDToUAHvIuXPnNDMzo5mZmaBDARAu91l2CQAAAAAAAAAA9BXJBwAAAAAAAAAA0Ff0fAAAABiCV69e6dWrV9q3b58OHDgQdDgAAAAAAAwUlQ8AAABDsLKyogcPHuj7778POpS2VldX9be//U33798POhQAAAAAwC5A5QMAAMAQHDlyREeOHAk6jI7Gxsb0z//8zzTbBQAAAAD0BckHAAAAyDAMHTp0KOgwAAAAAAC7BMsuAQAAAAAAAACAviL5AAAAAAAAAAAA+orkAwAAwBA8e/ZM3333nZ4/fx50KG2tra1pdXVVtm0HHQoAYFjqlkwjrXLQcYSEZRoyTGvT/cppQ2lumiTnXhgD/h2qW6ZMq+4fVWmj87hOTO23d9xWTq8bAwDQDyQfAAAAhuDJkyd6+PChVlZWgg6lrSdPnugvf/mLvvrqq6BDAQD0mWV2mByOZFSx80oMO6B+qVsy+5gFyFRKSvWwXyJvK9+vm9bnaxi2RN5WLj7IEcq6ko1pNhNpfGKZSanUfty6ZSqpkmzbll2S5n0JhW7blJhSLHuFRBwA9BnJBwAAgCEyDCPoENpaW1uTJI2N8c9DANgL2r8B7r1R7v6ky041gGHKm6etW2Zzu+9zldO+8zXPky43KwqaY66rHKhbMtuds3EeU1bZ28eNt5yWEc2qWki2xNuTjuNtvMbGKVuO2fjmfMf7suG+mkqn0872Hq7BO29vl+Ybx63gWP89t8TZUuXR5V53uwbXQrr9tW9/PG+3BRVSUy0Jskylc/LnRjGmkrcxMaVY8UZP26SEplKF1oQEAGDHeLoEAAAYAm9yn+QDACAMEnlbtr3+7fGE8rWc4kqpZNvKLSZVnK7JLsVUvFGX6pYuFqedN8dtW3ZtWsWLlndC1XJxKX5CCSU0lZJSJWeSOFMpKVXNNt86t0tS0pukLisdLWq65jtn1JuETihv2yqlqsom3X28N9YTedm1nOKpUjOensoRuo0nqZpVtHGNvjgjGVW8cex1FRLd7ouab+p72xYLi417tr1r6MS5X7l4SqVKxh3C/btb4RKZnFXJHas2XfQtNdTlXne7BklSQYVGRUFM2SvNFMJ2x/OUFwqKn4hu+36c0Ocdqhk2bktMpVT9vLbNsQAA7fB0CQAAMERhndz3kg/79u0LOBIAQF80KhEMZasFJd0/97SufeNN81TLcjf1G0VVq9nmm+zRrKrVYuNN90imotp0UYZhKKnSurfTU823zpVQ3ktolBe0mPtUjWEiGX2aW9RCy4xxXLlaxdknkVfFF9OWbTpec+K+Jc4uut6XuqVizHcvIhlV7Ip6vYRIpiLb3toyT5lZ3yR+Oa3i9OVG5UB9ab7xuxDNVtsc3eZeb3oNqZaKgtTiUuNs2xrPZ2lRik3u4PveKl/sAICdC+fTLwAAwC7jTe6H1erqqqTwJkcAAFuUyDfepnfefHf+vJOJ+8hkTPFcrfmWvm3LXjeRXvu8qni8zWL88ROj21tiE73cl6Hy9S+w5tVMINUtXczGGr8LdqmXDhc7MOzxNijrc3X6veu2DQDQLzxdAgAADIFt25LCO7nPsksAgE1t0pS3nHYqHiqVikpKtvYvqGZbqi7KC4uaPhtps/a+szb/VK+zwo031Z0eApv2Rth0vNZ1/615N87NztnpvkQyml6cb+mDUE6vq0Dpcg1b6/nQCEiXc4taSLdWPUjyVbVI1nyht9P1cg2dbGc8n8mYtLjUex+Gs9NSY/f6kjR9tqdtDbHJLccIAOiMp0sAAIAh8JIPYUXyAQD2kmbz4OaSTKasellptwGyM7FcUNJtElzNRpUuJ5QvqbGMzvpGxsmCpMKCyiproSCpcR5J8Zymi9HGcUnNupUBCV12l2ryfhoT5m6T52Shqmy0zbJRkYxmY95yR0kt5mo9LE/UeTzLTKqwLs7i9Kc9VDB0vi+SlPl0WsWo4bv2UrMCZVvXsLlIZlYqqGXZLEUymlWzuXVRcfd71ab3utM1lNPu75Bpyfm9SqpQzTqJpx2M17izbfowOE3Q/b+7zXsdOXuiGWe0qBO+xFG3bZLTXyLVc9YLANALww77k3C/PLqtT67e1HLLhxM68/FHOn1s3b6165q7dqen0566MKfz0d6OmTjzsT7aMFi3MTvE19Ej3f7kqm76LrLrmD0c37i+zfR0z07pwtx5bbdV1N7ifhc6o48/Oq2efwUAAKH1t7/9TT/99JPee+89HT58OOhwNvDie/fddzU+Ph50OMDQnTx5Uvfu3WssQQaMolu3bunSpUu6e/du0KGsU1baXFK+0Uth1O226wmzstLGgqbchtmjP87udO7cOc3MzGhmZiboUEKqputz13Tn1AXN9TTJNkJ6nUPdjdeOXtzfE6+21a7PaW5D4kGSlnXz6pzmPrmtR0OIY/nmVc1dr234vHZ9rsN/qG58bY7ZqKbrc62Jg25jbrQx8SBJd67NqafDAQBAV15lgWEYAUfSHg2nAQCDYpnu2/BbXj4oRNw39b0KBc2SeBgOZwmp+V6WeNqJ8oIWc5dJPABdPLr9iebmrms3ThPu5msL2v6gAxi02vU5OfP6baoIvGqI5Zu6ev14MwMXPa+5ufP+HXt+C71zpYCb5bzzR902o804ated+CbanNuL7841Xf+gewXCo9t/1B0nAF8mscOYbY//zEk8+ONws5d3rl3XBz1WLPRcKQEAwB7jFZuGNflAw2kAwKBkKraGNlVft2RGs6p22Jwq2dtb1iiSUcUm4RCESKaiyqAHSeRVIfMAdPXtN8uS3m67bdTnA7tdG3Zmdz9dPrqtP3oT+3Ntli86dlofzX2sMxOS7lwb8Bv+UZ2/cErSsr751hfitw8kTejM79skNY6d1kcfn9GEpAffdqvNeKSle8vOdbb8lx6VeWZiw5gdj9cpXfAnQKLn9bFzc/QnUn8AAOzIb37zG0UiER06dCjoUNr6p3/6J7399ttUPgAARlsko4pty+7w049+CgAAoDe7uvKhVrmp5U4T+w3HdPr3Z3Tv6k3d+VNN56NBpOnc5EC7II+d1kdzpzc5/phOfzSnjXvVVLnpJBU+6HpZ3+qbZUkTb+n4+jMff1vSspP8iPar84BbSfL2BX381h911Sm5aFamtFkvbn3vitr1OV17cEYffzSpJd9yUc391i0j1XZtuc16ZHSoeHHj23zfNktZbahw6XQvft/mvrmVLDq1rjIHADAqwlr1IElHjhwJOgQAAABgz9jp3Jaz2ovTX1WNlV+kzj1XtztP5XdH1+bubL+Hgzfn1/ZamhUUQV7b3Nzc1q8LHe3iyodHcooKPtTkZnPmxyb1oVNeMLjeD49u6xNnfSW95ZvhPzb5oZzCiznNzfXa32GzoT5xzjV3TXcmzujjzZZMevStHkjS28c35j+Ov6UJScvdSye258Ef9Vnj/6zO9/To9idt+18s37yqT25v/HaWrrf+j2X55lVdr7X5H86da63HP7qtTzbtkXFMkx9OSMv3tOQ7tPYnJ77le0vN35dHS7q3LE18OOncww7n1/JNXW23hlybe9HKSzxM6MzHJB4AAAAAAAB2g23Pbbn+1DI5LzmT6J+oZdcdzVOd0qmJrV5VB9HzunBKrSvQeEvSn7qwYemmkbo2tLWLKx/ct/lPtZlQ3+CYjr8t6c436lSA0Ks71+bUrcf7xJnfty7/dOy0PvpYTm8H5wSa859gG5lEZ50y1/JNXf1Em/aqkKSJt9bXPUg6dlxO7UNvOl5/u54Wy8vSmY8112yAoetupUZrFtOdeL+3pEenfedYvulUGsydbk74X72pO9euuktttfauWG4c/0i3P3Pud9uKCl+PjWOTH2ri5k1fZYqb1JKk5ebvy6Ole06VjZs1cKpu1CGTe0fXrn/Q+r1uuBct/xfV7U+8xEOb5cMAAAAAAAAwerY9t+W5ozt3WueLnLmnZd2s1HTanXva+TzVeX3gq0ZYP1PZfT60Nb7o+Qs6deea7vzxtszocVWu3WmzlHxw14b+2sWVD+Fz6sJcy0R3w7HT+mjOqXy4cGrdtjvXNLc+o7eJ6Hm3imJuzunZsHxTVwfb0GKbTul3LfcjqvNzc5pr/E/skW5/4lZwtD1+3ZJaXgXLht4VptPXw+NWKejUhQ3fR/S80wPkjtfkwk2+NP7+aEn3lid05sIZTeiBvFYc336z7KtYqOlPXq+Rdf/j9M6vO39al3ldfy+anAw4iQcAGGW2bevZs2d69uxZ0KEAAAAACI1tzm35j183X7Rx7qm/81Q75/bFXb6pq94qH22XzB/Fa8N6u7jy4bjeaiylFN3krX9viaaNPQ+2qrW7e7NEauLMxz11fY+en1MzyeYttdOa0duKY6d/rzP3rurmnT+pdj7adfml5XaNJ7wlmXq0pe727e63m+HtrdLibR1v98X2+j2urzJp4f3eRPXBKemO93v07Tda1tv6XfS43tay7i090unT3zr/o/OqbNx71liCqYWzlNPNZSdx0Wij0Snm5Ztu2Zg3Fv9zBIBRtLa2pkePHmltbU3vv/9+0OFs8OOPP+rrr7/W4cOH9ctf/jLocAAAAIA9YodzW22PXzf3pD7OU3WwpflAyV1+6Y6cooffd3jZNhzXhp3ZxZUP7lJK69brb8t7E75dz4MdxnD6IyfL1qlngR7d1icdqxKiOj/XKUvXewzH395sF+ft/rY9L779xlmeqN2STH1X03Vf4mHizMduBccFrS8IGTh3SSVJin5wqvF7VPvTHenUB4q6ya3lb75tJBtOde/qvU0TOvOxc/3LNz/bUgUMACA89u3bp/feey+UiQdJGh8f18TEhN58882gQwEADFPdkmmkVQ46jiDULZmGIcP9Ma160BGNpHLakDHg36G6Za77fspKG53HdWJqv73jtnKa3wFgqJrLmi/frGxzzhOjYBcnH6SoeUYTWtbNz253aSTdXP9/MJPHx3T6o06Tx+5k+7YTC77zzM1p7pN21+nrUdCRWyXim3BvHO0e/HbbNGyf1f4kp7+Ms2RU2yWq+qiZ3Gj341u/LvqBTmlZ33zr3EsnEeM2o77zJ91euqdlnVLj18frk+FvSN3wSEtOpqt9ZntDkB9q8phbjuZWwAAA0G+GYejAgQM6cOBA0KEAAAbAMjtMDkcyqth5JYYdUL/ULZnp7U17128UFSvZsm3np5KJ9Dm4XgPZ/jWEQSJvKxcf5AhlXcnGNOv7fiwzKZXaj1u3TCVVcr7XkjTvSyh026bElGLZK3szEYcR11wSvGnd3FM/56n65NHtz5wlxi+c0YTu6I9t37YdzWtDq12dfNCx0/qdM+uvq+36Jvi7obfpqN4/vsnjlkSIs6RP207tToDNzvanPuiyZJJ7nuWbWj837fzHLE2cMbsc706k646u+RMYteu66jaAHkhepoNGfwUnCHfpqT5y189bvnlVG4pOated5EPLBic5c+ePn+ne8oQ+dJtKHzv+tqQHundved330/w+1vfaqF3v5ftsw1vb7861jTEDAAAAwBa1fwPce6Pc/UmXZZmGDMOUN09bt8zmdt/nKqd952ueJ12Wcw7T8o3pfN7QUoXgO2fjPKassrePG285LSOaVbWQbIm3F5ZpKJqtqpB0jzOtHcSyxXvmH6uHa/CO7e3SfLG446z/njvG0u1erz+3YSqdTss/d7+QbnfPdjKet9uCCqmplgRZpmIr3yFjdqMYU8nbmJhSrHijp21SQlOpQmtCAhgJy7p59XrLS80b5576OU/VLiGwRY9u67Oby9Kp3+l01Jm7bb/axwheGzbYxT0fHNHzc7ogtxP61TndbLdT247qfQ/EXcvspj67Pdl4qz96/mOdeXBVN5cZGmggAAAgAElEQVR3Fl+jU3y77vITZ/R7XxXBo9uf6OpNtTRtafSGWL6pq3OtUZy6sLGL/UBEP9Ap3dGdTr0Y3MqMnScrj+n078/o3tWb7e+XJnTGjLbsP/nhhG7eXNayTul3XgBevMvSxIety1I1vo+217L+/FuM+Y+3dT56eovHAwAAAEBTIm/LzjtVEb5Pla/ltBj9XLN2XkumoeJ0TfbsFZk36sqcvaGLxWnZdsXZvW7JvGgpU8lIibxquUVFiyeUUEJKSZpyJ4kTJX1uJJWMlWTbCTkTz6asWkWZSFnpaFHTNVuViHvOaFqTdl4JJZS3bU2lDSWTUq5mq1JzlsdJZPKyaydkXplUpdNMdAeZiq2zlqkrk5V1k9jbiSWqSrd7lokoMjmrkp1QQt4SQmedSovE9q+hPSfGE2Zak5WM80neVm4xrcmKW+HSKZau9zrSqDawE959KerEZW/cggpyv9tyWsaVsjLu9XS89k3Ga3wjCwXFmwNt+X6c0ILKUpvqno3bElMpJRdqkgKqggG2ZUITE3d0bcPk0yld8M0l9mOe6rizbIozf3nqguZ8528/v7YunrnzijZWoGnG58V287PbmvQ32Q7o2k7PzW26P3q3uysfXNHzc5r7+Iw2NoWf0JmP5zT3UbuO6oOIo93yS8d0+qM5zV1o39Xg1IVe4/P1h/CZOPNxj8c3+1OsH3/QeZmmqM636e9w6sKcnNvTxwzksdP6qM390sQZfTz30YZGN8cmP3R+f1qype5yVWpWQzS1/z506oLm2py/15i9Sh4AwGh5+fKlHj58qMePHwcdSlsvX77Uy5cvtba2FnQoAIB+aVQiGMpWC0pupbdB403zVMtyN/UbRVWr2eab7NGsqtVi4033SKai2nRRhmEoqdK6if1U861zJZQvxVS8UZfKC1rMfarGMJGMPs0taqHlFfi4crWKs08iP7glknYSS4d7Jkn1pfnG/Y9mq1sKKZKpyLY7v+nfTmbWt6RQOa3i9OXGBPvmsbS5vrqlYsz3fUYyqtiV5n3yf7eJKaUWlxpn29Z4PkuLUmxyiMkAX+zAaHhbv1s/nzdxRh/PrX+ReOfzVMdO/645Z9eub2wPvBVaWl90dleMWb6pz1rKHwK6NvTVrq98aDh2Wh/NbfdtcSdB0PXo6HnNzZ3f5DxRnZ+bU9u9ejp+Mz3EKenY6Y/U/lb0dnxbW4q/2zgd7lF0Tv7TR8+3/r37ebf6ebuQ2/3+bHZ8L+ffWmztrxsAEHY//fSTvvvuOx06dEg///nPgw5ng4cPH+rp06d66623dPTo0aDDAQD0QyIv285LcqobGm++70BkMqZ47tOuk/+1z6uKx+PaMM0cPzG6vSV2om7pYjamkl1xrr+cljno+e3ElGLJKypn8lqal2YrkWBiCeLaW5T1uU4os+VtQL+1n+vqz9zWZtu2ut8W5+y2Op/ZaU6y43kCvDb0xZ6ofAAAAAgDwzCCDqEtr+Jh3759AUcCAAi1TZryltNOxUOlUlFJydb+BdVsS9VFeWFR02cjbdbed9bmn+o1U9F4U93pIbCj3s07jaUTX88Ca76wcXuXa9hazwdPQpdzi1pIt1Y99BRLO5GMphfnW3o5lNNbraLZwng+kzFpcan3Pgxnp6XG7vUlafpsT9saYpNbjhEA0BnJBwAAgAHzJvfDnnwYG+OfhgCwNzSbBzeXZDJl1ctKuw2QnYnlgpJuk+BqNqp0OaF8SY1ldNY3Mk4WJBUWVFZZCwVJjfNIiuc0XYw2jktq1l22J6HL7lJN3k9jwtxt/pwsVJWNtlk2KpLRbMxbBiqpxVyth+WJnGtv33B6O7Fscs9qGc2q2VC6qLh7L3dyDZuLZGalglqXgIp0iWWTe535dFrFqOH7/kqqZCIqp93fIdNy721ShWrWSTztYDxPYiql6uetTWSdht7+391mo+rI2RPNOKNFnTjbvP5u2ySnv0Rqx5kmAICfYdu2HXQQAAAAu9kPP/ygr776Sq+//rreeeedoMPZ4K9//atevnyp9957T4cPHw46HCAQJ0+e1L1797S6uhp0KMC23bp1S5cuXdLdu3eDDmWdstLmkvIVFrnBVpWVNhY0Ze982bBwjLM7nTt3TjMzM5qZmQk6lD2ldn1O1+54jZx3l918bXvMfV5vAwAAGLCwN3Km8gEAMEiW6b4Nv9NlkbAHOUtIzfeyxNNOlBe0mLtM4gEjJXp+TnO7dHJ+N1/bXsMTJgAAwIB5haZhndwn+QAAGKRMxZZtOz/9WFKoK3c5H6PDD8mP0RPJVLo2Oe+LRH7wYwDAHrQ/6AAAAAB2uzCvculNBkk0nAYA7AKRjCo2yzsBABAGvN4GAAAwYGGufPAvCRXG+AAAAAAAo4knTAAAgAHzJvgNwwg4ko285rokHgAAAAAA/cRTJgAAwIAZhqGxsbFQJh/o9wAAAAAAGAR6PgAAAAzY8ePHdfz48aDDaOvgwYN65513QpkYAQAAAACMLpIPAAAAe9jY2Jhef/31oMMAAAAAAOwy1NcDAAAAAIA9r5w2ZBhplQc4Rt0yZVp1/6hKG53HdWJqv73jtnJ63RgAAASD5AMAAAAAALuYZQ52Qn23SORt5eKDHKGsK9mYZjORxieWmZRK7cetW6aSKsm2bdklad6XUOi2TYkpxbJX+M4BAIEj+QAAADBgKysrevr0qV69ehV0KBvYth10CACAIatbpgzDlGl6b86b8s9dO9vdbablfuq9oe/+pMuyzNZjW45bd07/9nRPs+K+8dwY1r/p3z5O/7GmrLIlc0N1gP9aTKXT6ZZYF9JbuS+9jOfttqBCakoJ30eZiq18Yv2OjhvFmErexsSUYsUbPW2TEppKFVoTEgAABIDkAwAAwIA9e/ZMjx8/1k8//RR0KBs8fPhQtVpNjx49CjoUAMCQRDIVlVJVVWPem/MxZa80p8ojk7Mq2bZs21Ztuugu4ZNQvpZTXCmVbFu5xaSK0zXZpZiKN+pS3dLF4rRzPtuWXZtW8aLVOYhNJZS3beXiKZUqGeeTvPt3O69Exzibx5ZSVWWTRU3XWqsDvGoDL87FwqJv3IIK2sp92Xw8T3mhoPiJ6Lbvxwl93qGaYeO2xFRK1c9r2xwLAID+IPkAAAAwYMePH9e7776r8fHxoEPZ4K233tJvf/tbHT16NOhQAAD9VE433tLPVgtKun9uTpjHlbvcfHM+tbjUOLS+NN/YP5qttp638eZ+qmX5oPqNoqrVbLMyIJpVtVpsqRyIZCqy7c5v+reTmfVN4pfTKk5fblQOdI3Tu8ZaRZmIpERelUxEqlsqxkrNGCIZVWx3H/e6/BUFPd+XTuP5LC1KsclIm+MGxBc7AABBIPkAAACwx42NjWlsjH8WAsCuksg3qhCcagHnz+snxDeoW7qYjTX2t0upnoaLTMYUz9WalQ+2LbtlUn+719HsX2DNq5nw2Gac2zbs8TYo63OdUPu8TbdtAAAEh6dMAAAAAADQ5OtLYM0XejumhybHW+v50DixLucWtZBurXrYdpyRjKYX51sqMsppf0VIF9sZz2cyJi0u9d6H4ey01Ni9viRNn+1pW0NscssxAgDQTyQfAAAAAADYQ+qWqWShqmw0rbLKShtJFapZGemyFMloVsnG8klFxVXNRpUul5WOZlUtJN2J+oKSbtNlZ3tC+ZIayxIZnZoub0MkMysV1LLMU+c4JdWdps/ONa5fbkrKfDqtYrQZZ1IlVTIRldPuElWmJfV8XzYfz9OuD4Nlrl8aq3nPImdPNOOMFnXibPP6u22TnP4SqSlqIQAAwTJs27aDDgIAAGA3++abb2QYhn7xi19o3759QYfTYmVlRYZh6ODBgyy9hD3t5MmTunfvnlZXV4MOBdi2W7du6dKlS7p7927QoaCtstLGgqbchtmjPw7gOHfunGZmZjQzMxN0KADC5T5PmAAAAAP2+PFjfffddwrjOx9ffvmllpeX9erVq6BDAQBgl3OWkJrvZYmnnSgvaDF3mcQDACBw+4MOAAAAYDfzmm5KkmEYAUezkfeWN1UPAAAMXiRTUWXQgyTyqpB5AACEAE+ZAAAAA+SvdgjbBP/a2lrjz2GLDQAAAAAw2njKBAAAGCD/BH/YKh9IPgAAAAAABoWnTAAAgAHyKh/COLnvJR/C1gQbAAAAADD6wvcUDAAAsIt4E/xhq3qQ6PcAAAAAABgcnjQBAAAGyEs+hHGCn8oHAAAAAMCghO8pGAAAYBcJ87JLVD4AALaqnDZkGIYMI61yux3qlsxO21rO0X2fgZ9ziLrds37cTwAAwoonTQAAgAHykg9hXHaJygcA2Bsss3+T14m8Ldu2lYt32CGSUcXOK7HJOToeP6xzDlG3e9aP+7lB3ZKZJl0BAAgeyQcAAIAB279/fygn+Kl8AIC9qKy04b5tb1rOJ+vevq9bpvv35j49nbnbW/zldPOcW5gY38k5vevY0jx83ZJpNMc0zbTK/nvmjmWZhgzDlFVvHWur96ybTtfujGXKNL3tzThUTsuIZlUtJLd1vwEA6CeeNAEAAAbo9ddf17/8y7/onXfeCTqUDY4ePaqJiQn94he/CDoUAMDQJJS3beXiKZUqGeeTvPt39w37yOSsSrbzRn5tuiizMbO9yZk7vcVft2Qm1ThnSUllqz1GO4BzdlO+UtR0zTmnXZKqVUlKKF/LKa6USrat3GJSxema7FJMxRvOvdnuPeum07VHMhWVUlVVYyU3zpiyV8reQbJrOcVT7jbblp3fUt0EAAB9Q/IBAABgj9q3b59ee+01HThwIOhQAAD95qsKyFYLSrp/9ibFM7PSvDdBXk6rOH25sbRPfWm+sX+0DzP69RtFxUrNpYMS+ZJSQzhnJFORbdvaytx7YiqmbNStGEgWlPKNodSU++eUZjOR1nj6fM82F1fucsILWqnFpSGMCQDA1pB8AAAAAABgt0nkG2++O1UNzp8r3qR5Ykqx7BWVJVnzak6m1y1dzMYa+9ulnaYJRozvvtm1nBbne1hCaa/fMwAAOiD5AAAAAADAnpPQ5dyiFtKtVQ+SfG/4S9Z8YccjRc5Oa3HB13egvKCdnrWXc26950NZ6Xa9JXrR53u2Y41KCKdfBW0fAABBIPkAAAAwQCsrK3r8+LFevHgRdCgbvHr1Smtra0GHAQAISCQzKxXUuoRQJKNZNZsVFxVXNRt1J6+bjZebSzl5zY67bFt3TiMpxeMFJTdtzDyIc26muUSVEc0qNptx4nCbODvLVjXHqWajStcGcM+6bKtbppKFqrJRrxl2UoVqttlYOpLRbCzrxpPUYq62paWnAADoF8O2bTvoIAAAAHarH374QU+fPtWRI0f0xhtvBB1Oi7/97W/66aef9P777+vQoUNBhwME6uTJk7p3755WV1eDDgXYtlu3bunSpUu6e/du0KEA2EPOnTunmZkZzczMBB0KgHC5vz/oCAAAAHazI0eO6MiRI0GH0davfvWroEMAAAAAAOxSLLsEAAAAAACGr27J9JY4avNDnwIAAEYblQ8AAAAAAGD4IhlV7EzQUQAAgAGh8gEAAAAAAAAAAPQVlQ8AAAAD9PDhQ62ururo0aOhaur86tUrraysaP/+/Tp8+HDQ4QAAAAAAdhkqHwAAAAboxx9/1JMnT7S2thZ0KC2eP3+ur776Sg8fPgw6FAAAAADALkTyAQAAYIC8pINhGAFH0mp1dVWStG/fvoAjAQAAAADsRiQfAAAABshLPoyNheufXWGNCwAAAACwO/C0CQAAMEC2bUsK3yQ/lQ8AAASrnDZkGGmVBzhG3TJlWnX/qEobhgzDkGGYatnUiKl9XB23ldPrxgAAwBGup2AAAIBdxks+hG3ZJSofAGDvsMzBTnAPVN2SmR7Z6LtK5G3l4oMcoawr2ZhmM5HGJ5aZlEq2bNuWXYope9FqbKtbppIqudukeV9Cods2JaYUy14Z3d8xAMDA8LQJAAAwIP4m02FLPlD5AAB7Xesb8Ol0uvkWfN2SueHteP/+hox0WZbpbe+2zTul2dy+4XNTptnmbfxyWkY0q2oh2XLuhpY40zLXJVm8MXvLXfiuwbTc4Vvf9G+5BtNqc6wpq+zF5I+ly72WtJBuc+07Gs/bbUGF1JQSvo8yFVt574PElFK+bTeKMZW8jYkpxYo3etomJTSVKrQmJAAAEMkHAACAgfEnH8JWYUDlAwDsbS1vwNemtVhYdLeUlY4WNV1rbitG0yoroXwtp7hSKtm2cotJFadrsksxFW9Eu2yrS3VLF4vTzvm8c7pv3EcyFZVSVVVjpebb+FfcafREXnYtp3iq1Dw235xKL1/xxVmSqtWd3JGE8ratXDylUiXjDu/+3c4rISkyOauSG0dtuuhbasg5tpSqKpt0Y/JVB3S+15JUUEFtrn0H4zXuz0JB8RPRjldct+al2UzH+3FCn3eoZti4LTGVUvXzWsexAAB7E0+bAAAAAxLWfg9SM/lA5QMA7FLldOOt+Wy1oKT7Z9NykgHFWKn5Bnwko4pdUSYiqbygxdynaqzUE8no09yiFryZ5sab9KmW5Xy6bavfKKpazTbf4o9mVa0WfW/5x5W73HyrPrW41NMlJqZiykbdcyYLSpXyLW/5RzIV2bbvTf8eZGZ9k/jltIrTlxvnrC/NN+5jNNsu0xFXrubex0RelUyk+72WJKVaKgr8176t8XyWFqXYZKTNcU5VRbQ4vaV7s6kevzcAwN4RvidhAACAXcKb4A/bkktSc9mlMCZGAAB9kMg3qgWct/edP6+foB6GyGRM8VytWb1g27JbJuC3yXeNdi2nxXlr82M2PWezf4FTGOAGWbd0MRtr3Ee7lOp6mh0b4HjltKFocVp2pVPVgySV9blOqH1uots2AACaeNoEAAAYoEOHDungwYNBh7EBlQ8AsIdFMppenG/pL1BOu1URG9bzd9b7n9rJTPNOGxI33qh3eh04PRzKSrfrc+CztZ4PjWB1ObeohXRr1YMkX2WHZM0Xejtdt3u9me2M5zMZkxaXWsexTMNpHN0m8XB2WmrsXl+Sps/2tK0hNrnlGAEAuxvJBwAAgAE5dOiQ3n//fb333ntBh7LBr371K/3qV7/SgQMHgg4FABCAzKfTKnpLFhnOhLRTFZHQ5emirzm04U7Cl5V2mz87E+cFJd0myNVstNEYut22dDmhfEmNJYTWN3FOFqrKRtMqq6y0kVShmm02lo5kNBvzlmxKajFX8y0V1FxOyohmFevYv2BrIplZqaDWZaUiGc2q2fi6qLh7bWo0vnauw7e81Sb3upx2l8QyLWn9te9gPM+GPgx1S8WqJH8Db18CJ3L2RDPOaFEnzjavv9s2yekvkdpRhgoAsBsZtrcYMQAAAADsUSdPntS9e/caS5IBo+jWrVu6dOmS7t69G3QoCIWy0saCpuz8gJdIGtY4CKtz585pZmZGMzMzQYcCIFzuU/kAAAAAAACw6zhLSM33ssTTTpQXtJi7TOIBALDB/qADAAAAAAAAQP9FMhVVBj1IIq8KmQcAQBskHwAAAAbk+fPnWllZ0aFDhzQ+Ph50OA2rq6t69eqV9u3bp/37+ecgAAAAAKD/WHYJAABgQGzb1suXL/Xq1augQ2nx4sULffnll3rw4EHQoQAAAAAAdiledQMAABiQw4cP6/Dhw0GHscH4+Lh+/etfBx0GAAAAAGAXo/IBAAAAAAAAAAD0FckHAAAAAAAAAADQVyQfAAAAAAAAAABAX5F8AAAAGJBvv/1W//jHP7SyshJ0KC2ePn2qx48f66effgo6FAAABqduyTTSKgcdh085bcgIWUwAAAwKyQcAAIABWVlZ0Y8//qjV1dWgQ2nx/fff68GDB3rx4kXQoQAAhsAyQzDZXbdkpvscxWbnjGRUsfNK9HfUHUnkbeXiPe48iHsGAMAQkXwAAAAYkLW1NUnS2Fi4/snlxbVv376AIwEABKestGHIMAwZhql0Oi2r7v/clFW2ZBqtb+rXLdM9xt2n3jxjyzbT8g2VlhHNqlpINrf7JtW7nbNz+N3P6VQYrK8y8F+zs79lto65WSze9i3lBMrptjFuGG/Q9wwAgCEL15MwAADALhLW5INXiRG2uAAAw2OZSalky7Zt2bVpLRYW3S0J5W1bpVRV2WRR0zVbdkmat+pS3dLF4rRzjHtc8WJzwjwyOauSu602XZTpzYgn8rJrOcVTpeaxebceYZNzdtTtnHIqDGx7fZVBQvlaTnGlVLJt5RaTKk7XZJdiKt7Y/Pq2pW7JudXOOUtKKlttbh7qPQMAYMh44gQAABgQL/lgGEbAkbQKa1IEANBHvrfts9WCku6fTTeJUIyV1Jirj2RUsSvKRPwniCtXcz9L5FXJRFS/UVS1mm2+cR/NqlotNqsGluYb40T9M+xdbHbOgUhNuUsxpTTru+heYolkKrJtW/ke13Kq3ygqVmou/ZTIl5Tyb+/jPQMAIGx44gQAABiQMC5vZNt2KOMCAPRZIt94Mz4XTzXerq+0Zhi2JDIZUzxXa75xb9uyvaRF3dLFbKwxjl1KbXq+Tc85ZEOPpc/3DACAsCH5AAAAMADeZIAUrgoDL/EghSsuAMAQRTKaXpxveaO/nDaaS/50kphSLHulc/PqRkWBZM0XNm5fXPJGU9rrm7DZOTfT7pzb1UMsW+35EDk7rcUF387lBbXcmSDuGQAAQ8ITJwAAwAD4J/nDtOySv99DmOICAAxX5tNpFaPN5stJlZyqiLrTZDpZqCob9S3VJElKKF9SY5mglobOkYxm1WyOXFRc1Wy0OUkfyWg25i0VlNRiruYuXdTlnJvpeM5mY+nmklOmrHpZabeJs3NNBSXdJs9OrDuIpVuMvvtiJKV43B23z/cMAICwMWzvlTwAAAD0zcuXL/XXv/5VhmEoEglg7YgOnj9/ruXlZR04cEC//vWvgw4HCI2TJ0/q3r17jQQdMIpu3bqlS5cu6e7du0GHAmAPOXfunGZmZjQzMxN0KADC5T6VDwAAAAMQ1qbO/soHAAAAAAAGhadOAACAATAMQ+Pj43rttdeCDqUFzaYBACPBXf7J6PCzo94OAABgKPYHHQAAAMBudPDgQb377rtBh7HBkSNH9MYbb4iVNwEAoRbJqGJngo4CAADsAMkHAACAPcZ7axQAAAAAgEFh2SUAAAAAAAAAANBXJB8AAAAAAAAAAEBfsewSAADAAPz4449aWVnR4cOH9frrrwcdTsPKyookpyfFXms6/fe//11fffVV0GEgpFZWVmTbtu7cuRN0KMC2/b//9/+0uroadBgAAACSSD4AAAAMxNjYWCj7Kjx9+lTPnz/X8ePHNT4+HnQ4Q/Vf//Vf+s///E/95je/CToUhNAXX3wh27Y1PT0ddCjAtj1//lyvvfZa0GEAAABIIvkAAAAwEIcPH9bhw4eDDmODt956K+gQAvXv//7v+sMf/hB0GAihkydP6t69e/rzn/8cdCjAtt26dUuXLl0KOoyhsExDWeVkVzLDHbhuyYx+rlk7r8RwRwYAYOTQ8wEAAAAAgF3MMtMqBx1E3ZKZ7l8UmUpJqb6dbQsiGVW2mnjo87UDADAqSD4AAAAAALDnlJU2DBmGIcMwlU6nZdX9n5uyypZMw5BhNJMXdct0j3H3qTfP2LLNtHxDpWVEs6oWks3tvsn4bufcjP/YxinL6dYY1v/dd9xWcgLltBdjazLHOZcp02xzDZtcOwAAuxnJBwAAAAAA9hjLTEolW7Zty65Na7Gw6G5JKG/bKqWqyiaLmq7ZskvSvFWX6pYuFqedY9zjihebE/qRyVmV3G216aJMbwY+kZddyymeKjWPzbu1A5ucs6tqVtHGsSUp6U76J/IqpeLKfZppjN/y921K5J0Yc/HWzyOZikqpqqox9/pKMWWvlDe/dgAAdjmSDwAAAAPwj3/8Q1988YVevHgRdCgNq6urevjwob777rugQwEADJr3tr9hKFstKOn+2XSTCMVYSY058EhGFbuiTMR/grhyNfezRF6VTET1G0VVq9nmG/zRrKrVYuMt//rSfGOcaLbaU5ibnbO7lEqNng8J5UsxFW84ByYuT6voJQDqluY123J9kUxFtm2rf3mAuHKX3ZMlppRaXOrXiQEAGFkkHwAAAAbgxYsXWllZCTqMFq9evdJ3332n77//PuhQAACDlsg33rTPxVONioRKa4ZhSyKTMcVzteYb/LYt20ta1C1dzMYa49il3joydD3nTkQyml6cl1V3EhyxKaoNAAAYNpIPAAAAA7C6uipJGhsLzz+3whgTACAAvol5TzltNJdJ6iQxpVj2Sufm1ampRiNma76wcXujGsDpK5Eu93DOrgrOclCNMRc1fbaZtcjMxlS8YelKcXpDhcN2ej7sSLtrBwBgl+PJEwAAoM+8NzelcE30r62tSZL27dsXcCQAgKBlPp1WMWo0ljtKquRURdSdJtPJQlXZqG+pJknO0kZqLK3U0nw5ktGsmk2Vi4qrmo02J9kjGc3GvOWVklrM1dyEQJdzdmGZSRXiOU0Xo80xpz9trZhI5DVdzGpx+mwf7lizQXdzGSunx0TdMt37lVZZZaWNpArVbLOxdMdrBwBgdzNs78kYAAAAfbG6uqq//OUvkqRIJCLDMAKOyPH48WM9ePBAb7zxhn75y18GHc7Q/cd//Ick6Q9/+EPAkSCMTp48qXv37jUqhIBRdOvWLV26dEl3794NOpTQsMy0Jit5MdcPDM65c+c0MzOjmZmZoEMBEC73w/MqHgAAwC7hVRiMjY2FJvEgsewSAGCvWFelwBpHAAAEgidPAACAPvMnH8KEZZcAACPDXf7J6PDTPZ+QUN7fwJo1jgAACMT+oAMAAADYbcJaYRDWuAAA2CCSUcXOBB0FAADYAZ48AQAA+iyslQ9e8oHKBwAAAADAoIXriRgAAGAXCGvygWWXAAAAAADDwrJLAAAAfbZ//3698cYbOn/OIq8AACAASURBVHToUNChtDAMQ2NjY6FLigAAAAAAdh+SDwAAAH02Pj6u8fHxoMPY4N133w06BAAAAADAHsFrbwAAAAAAAAAAoK9IPgAAAAAAAElSOW3IMNIqBx1IP9QtmSG7ll11fwEA2ATJBwAAAAAAdjHL7H2yO5G3lYsPNJz+qVsy012uLJJRxc4rMbyINrWl+7vZ9QEAEHIkHwAAAPrs22+/1YMHD/T8+fOgQ2lYXV3V48eP9eOPPwYdCgAgFMpKG4YMw5BhmEqn07Lqza0L6eY2/+d1y3Q/N2SY1rrPTZlm++NUTjeOMy3LGdudWG855/rjOoaflhHNqlpINo/1TdSXG/H7Ey/+a3b2t8zWMTeLxdu+pZyA79qNdQd2up+bXd+27hkAAENG8gEAAKDPXnvtNR08eFD79u0LOpSGtbU1rays6OnTp0GHAgAIActMSiVbtm3Lrk1rsbDo21pQQSVnWymm7JXmpHdkclYl2zmuNl2U6c56RzIVlVJVVWNtjqtbcoZzjpsuZlWI52TnE1Ld0sXitHOMG0vxom8SvpNEXnYtp3iq1Dw2n/Btdj5rrTJIKF/LKa6USrat3GJSxema7FJMxRv17cfSzbprLympbLW5udP97Hp9g4gTAIABIPkAAADQZ6+//rqOHj2qAwcOBB1Kw4EDB/TP//z/2buXGLfO+/7/nzOc+0iji+WxpdiKZXOUQJimadICMqkuAhRBSaXAbDILu6j625CwY4AECvUGNf3/g0FjQAVCAoUBEm0Ko6uOF9XCQ+aHdpNapAu0i0KZaCHSiaVIcuTYsnX13Mjnt+B9ruQMNc8h+X4BAiSe2+ecMY6Hz/e5PKtnnnnGdhQAwF6o620fzSYVrI46KDWyz02lVG2r90aUMRlFvJWDQ0pVNgamFVq4Vj1t/tps9VyT9a3okiSfYufXH5efn9NUqjb9UeTtmCo1gfz8nLLZaK0X/2RU2ezck+3JH5ouZwnpQu2mm8rijWRkjFGiybmc1t57IJFSqH77ls9z83NulBMAALeh+AAAAAAAQLcJJKo942O+ULV3faausb1l+bjORaeq5zKp0PbHbMN7ckq+WK7Wi98YmYZCyN7Z8yw7fJ6b5QQAwG0oPgAAAAAA0Eu8Ec0szDb06E+HndqUP1upjhqQ4rPJ5i53dkYLs7VpgdIXo6r28Q9Mayp6sekFsdepjsooreewq/WZm8jS6poP3rMzWrhUt3P6khqe2nbPc6P72+0zAwBgj/TbDgAAAAAAAPZW5O0Z+ScdRSsfhFIyEa/SYae0JoH/lEzmpMJOUElJyfBJmUREF+TIcUqH+Hw+ZaOTCp80On/Nr2AyKyXDOmmmdWnNcW/P+OU45auFQnVTDwWUSF2SUzlpaQelTG2qok15I7ow5VTP64vllAlIpYb6YK2R30lK8imWu6Crk6XCh/9UTjNKKug/JXNB1fvYcZatMtY9Mykkn6983czmzzMR2Or+Nn5mxiR2mhIAgCfCMcYY2yEAAAC6xcrKim7evKn+/n49//zztuNU3bt3T4uLi9q/f79GR0dtx7Hir/7qryRJP/zhDy0ngRt97Wtf089//nMVCgXbUYAdu3z5sl5//XVduXLFdpSt5ePyXzypTLMLJwBwte985zt65ZVX9Morr9iOAsBdbjDtEgAAQBsVCgWtrq66rgHz8ePHun//vpaXl21HAQD0oroFsJ3JOc2cp/AAAEC3Y9olAACANqoUHfr63NXHo5LL4/FYTgIA6EmBBNMCAQDQY9z1rRgAAKDDubWRv1gsSnJfLgAAAABAd6L4AAAA0EaVRn63jnxwWy4AAAAAQHfi2ycAAEAbuXXkg1tzAQAAAAC6E8UHAACANnLj9EbGGBljJDHyAQAAAACwN1hwGgAAoI3cOL1RJZPkrqLIXrt69aquXLmiDz74wHYUuNCNGzdkjNHMzIztKMCOffLJJ/riiy9sxwAAAJBE8QEAAKCt3Di9kRtHY9jw8ccf69e//rX+4A/+wHYUuFA2m9X9+/f57wMdLZ/P66OPPtrVOdJhR8FkSCmTUKBNuZ6EuN9RVDGZTGRvL5yPyz95VRdc/nwAAHADig8AAABt5MYFp904GsOWr3/96wqFQrZjwIX+4R/+QR999BH/faCjXb58WT/5yU/WfR73h3Uy01xjeSBhFFsItz9cm0UyKV31X9v7C3sjypgWj8nH5b94UpkE5QoAQG/hGygAAEAb7d+/XwcOHNDg4KDtKA2GhoZclwkAYFNaYceR4zhyHL/C4bDi+drWS+HatvrP83F/+XNHjj++5nO//P6Nj1M6XD3OH4+Xrh1Orz/n2uO2UX9s+XS1a1Xyrf133XHVY5qQrj6TsOoP2/Le02E5k1Flk8HaPbZyUQAAOhjFBwAAgDY6ePCgJiYmXNXQPzIyouPHj+vYsWO2owAAXCLuD0opI2OMTG5GC8mFuq1JJZUqbUtNKXqx1ljuPXlBKVM6LjczJ3+5ld0bySgVyio7tcFx+bhKlysdNzMXVdIXk0kEpHxc5+ZmSseUs8ydi6sp2agmq8empGC50T+QUCrkU+zt8pRMa/+9Q4FEKWPM1/j5lvceSMjkYvKFUrV7ZAQEAKBHUHwAAAAAAKDb1I00iGaTClZHHeSlfFxzUylV28C9EWVMRhFv5eCQUpWNgWmFFmrTG+WvzVbPNRnNrrmoT7Hz64/Lz89pKlWb9inydkyV9vv8/Jyy2WhtVMBkVNnsXJOjH0JKVdd8CCiRmtLcfOnAwPkZzdUVP2Z1oe7+SgUDY4zaVwfY+N4BAOhlFB8AAAAAAOg2gUS1p33MF6qOOsjUt8C3Kh/XuehU9Vwmtfs1Urwnp+SL5WqjAoyRaSiE7PTEEc0szCqeLxc/phltAADAXqP4AAAAAABAL6lrmK9Ih53qFEpbCk1XRzDEZ5PNXe7sjBZma1MppS9GVR0zEZjWVPSidrYKQlKzdZnjswuaOVurWkQuTGluPq6LczPrRjjsZM2HXamOhCittcGyDwCAXtBvOwAAAEC3WFpa0m9+8xsNDg5qYmLCdpyqBw8eaHV1VWNjY65aiwIAYE/k7Rn5Jx1FKx+EUjIRr9JhR9GsJP8pmcxJhZ2gkpKS4ZMyiYguyJHjlA7x+XzKRicVPml0/ppfwWRWSoZ10kzr0prj3p7xy3HKVwuFVBszEVAidUlO5aSlHZQytWmaNhL3B5X0xRSbm1TltL5YTpn6EROBhGZmHc3N5HbxpCrS1WchSXKSknyK5TI6O7/VvQckb0QXppzq/ftiOWUYiAEA6AGOMcbYDgEAANANCoWCFhcXJUljY2OW09Tcv39fi4uL2r9/v0ZGRmzHsebll1+WJL3//vuWk8CNvva1r+nnP/+5CoWC7SjAjv37v/+7/uiP/kj/+I//aDvK1u78X/0gdVTf/z9fe+KX+r8/+Gcd/f7/0ZO/EtC7/uIv/kJ/+7d/q1Bo91OxAegqNxj5AAAA0CYej8dVRYeK8fFxjY+P244BAHjCbt68qaWlJf3gBz+wHWU9U1ShWOn76KjP06cf/ODSk7qYioWiqj0t/7//X5f6nK0OALALt2/f1gcffGA7BgAXovgAAAAAAEAXmJyc1G/91m/pypUrtqMA6CHf+c539Nu//du2YwBwIRacBgAAAAAAAAAAbUXxAQAAoE1YSgsAAAAAgBKmXQIAAGiTjz76SI8fP9bExIRr1lgoFou6ffu2PB6Pjh49ajsOAOAJ+9nPfibHYX0DAHvrlVdesR0BgAtRfAAAAGiTQqEgY4z6+twzuLRQKOiLL76gIQoAesCZM2cYhQcAAFzDPd+MAQAAOlyhUJAkeTwey0lqisWiJHdlAgAAAAB0P4oPAAAAbeLG4kMlk5tGYwAAAAAAuh/fQgEAANrAGOPKUQZuzAQAAAAA6H4UHwAAANqgMsJAcldDPyMfAAAAAAA28C0UAACgDdw45ZLEyAcAAAAAgB0UHwAAANrArcUHRj4AAAAAAGzgWygAAEAbuHWEgVtzAQAAAAC6G8UHAACANlhdXZXkvkZ+Rj4AAAAAAGzotx0AAACgG4yMjOjIkSMaGBiwHaXB0NCQCoWC63IBAAAAALobxQcAAIA2GBoa0tDQkO0Y6xw+fFiHDx+2HQMAAAAA0GMYfw8AAAAAAAAAANqK4gMAAAAAAAAAAGgrig8AAAAAAAAAAKCtWPMBAACgDT788EP19fXp2LFj6u93x69YxWJRn3zyifr6+nTkyBHbcQAAAAAAPcQd34wBAAA63LFjx1QoFOTxeGxHaTAwMKBisWg7BgAAAACgx1B8AAAAaIPBwUHbEdbp6+vToUOHbMcAAAAAAPQg1nwAAAAAAAAAAABtRfEBAAAAAAAAAAC0FcUHAACAXSoUClpdXZUxxnYUAAAAAABcgTUfAAAAdunzzz/X3bt3deDAAU1MTNiOU3Xv3j3du3dP+/bt0+HDh23HAQAAAAD0EEY+AAAA7FKhUJAkeTwey0karaysaGlpqZoPAAAAAIC9QvEBAABglyqN+/397hpUWiwWJbmvKAIAAAAA6H4UHwAAAHZpdXVVkvsa+StFkb4+fuUDAAAAAOwtvokCAADsEiMfAAAAAABoRPEBAABglxj5AAAAAABAI76JAgAA7EKxWJQxRhIjHwAAAAAAqKD4AAAAsAuVUQ99fX1yHMdymkaMfAAAAAAA2MI3UQAAgF2oNPC7cXQBIx8AAAAAALZQfAAAANiFysgHt025VCmKSIx8AAAAAADsPXd9SwYAAOgwIyMjOnr0qCsb+MfHx2WMcd10UAAAAACA7kfxAQAAYBf6+/u1b98+2zHW8Xg8euaZZ2zHAAAAAAD0KPd10QMAAAAAAAAAAB2N4gMAAAAAAAAAAGgrig8AAAC7YIyxHQEAAAAAANdhzQcAAIBd+OCDD+Q4jl544QV5PB7bcaoePHighw8fat++fdq/f7/tOAAAAACAHkPxAQAAYBdeeuklFQoFVxUeJGlwcFBjY2OuywUAAAAA6A0UHwAAAHbBcRz197vvV6qhoSENDQ3ZjgEAAAAA6FGs+QAAAAAAAAAAANqK4gMAAAAAAAAAAGgr980RAAAA0CGWlpZULBY1ODjI2gpAFzDG6Be/+IXtGACADjE4OKjnnnvOdgwAcC2KDwAAADv0+eef6/79+3rqqad0+PBh23Ea3LhxQ4VCQceOHWPtB6AJfX19Msbod3/3d21HAQB0gM8++0wDAwNaXl62HQUAXIviAwAAwA6trq5KkisXnF5dXVWhUJDjOLajAB1hYmJCg4ODunv3ru0oAIAOcPnyZb3++uu2YwCAq7HmAwAAwA65ufhQKBQklXpzAwAAAACw1/g2CgAAsENuLT4Ui8Xq3yk+AAAAAABs4NsoAADADhSLxWojP8UHAAAAAAAa8W0UAABgB1ZWViSVGvfd1sBfKT54PB7LSQAAAAAAvcpd35QBAAA6RGXKpYGBActJ1mO9BwAAAACAbXwjBQAA2IHKyAe3Tbkk1UY+UHwAAAAAANjCN1IAAIAdcPPIB4oPAAAAAADb+EYKAACwA5Xig5tHPrDmAwAAAADAFvd9WwYAAOgAExMTOnTokCsb+AcHB3Xo0CENDg7ajgIAAAAA6FEUHwAAAHagr69PQ0NDtmNsaGRkRCMjI7ZjAAAAAAB6GNMuAQAAAAAAAACAtqL4AAAAAAAAAAAA2oriAwAAQIsKhYKWlpZUKBRsRwEAAAAAwJUoPgAAALRocXFRH330ke7cuWM7yobu3LmjW7duaXFx0XYUAAAAAECPYsFpAACAFo2NjWlsbMx2jE0dOHBAq6ur6u/nVz0AAAAAgB18IwUAAOgyw8PDtiMAAAAAAHoc0y4BAAAAAAAAAIC2ovgAAAAAAAAAAADaimmXAAAAWmCM0d27d9Xf36/x8XE5jmM70jrGGFfmArC3/vd//1eFQsF2DAA7sG/fPn3lK1+xHQN77N69e8rn87ZjANih48eP6+mnn7Ydw1UoPgAAALSgUCjo7t27chxH4+PjtuOss7Kyog8//FAej0cvvvii7TgALPqd3/kdjY+P68iRI7ajAGjBL37xCz333HP61a9+ZTsK9tjly5f1ne98h9/hgA70i1/8Qm+99ZZee+0121FcheIDAABAC5aXlyVJ/f39rhxdUCwWbUcA4BIHDhzQlStXdPz4cdtRALRgfn5eb775pu0YsOTMmTN67733bMcA0KJXX33VdgRXYs0HAACAFqysrEiSBgcHLSfZWKX40NfHr3kAAAAAAHv4VgoAANCCSvFhYGDAcpKNUXwAAAAAALgB30oBAABaQPEBAAAAAIDt8a0UAACgBRQfAAAAAADYHt9KAQAAWkDxAQAAAACA7fGtFAAAoEmFQqHauE/xAQAAAACAzfGtFAAAoEmVUQ/9/f1yHMdymo1RfAAAAAAAuAHfSgEAAJpUKBTk8XhcO+pBovgAAAAAAHAHvpUCAAA0aWxsTC+++KK+9KUv2Y6yqf379+vIkSMaGRmxHQUAXMvv+BXP207RRvm00u26n3x86+ez3faWr9fG7AC6Vq+/t/NxvxzHkeO258A7HNug+AAAANAit065JEmjo6M6dOgQxQcA2ELGZBTx2k7RJvm4/JOzurZX1/NG2vf89jo7gI7V2+/ttC5GswqljIybngPvcDSB4gMAAAAAAAAAuFH+mhbk06lJ20GA1lF8AAAAAAD0lOr0HZUphNJx+R2nPKWFI388X95W+yycrj9DXnF/bZvjhFXZXJoao35ajLTCDef0Kx4P1471x9U4Y8Wac9dfOB+X3wkrXp1+w5EzGVVWWUUny9doyhbXqJjfJOO6aZc2fxZbXisfl39H2QH0op59b695V9bO3co1w0rv9Lmt2dZwDO9wNIHiAwAAQBOKxaI+++wzPXz40HYUAEBbZRWdld42RsYY5WI+ZaOTcs41fpYMVhqq8or7JxWdSslUty8oWG6M8kYySoWyil4s7Z0OB5UMpZSpzpORVTQqpYyRMTnFFNVkQ2NS/blzii0E1xQHkopevVC9tsnF5JNPsZypu8ZWmrlGVtG5U8ptmHGrczU+i/XbUwolg6WGLW9EmZazA4DUU+/tNe9Kkwjs4JoJBXb03NIKT0Y1lTJ1z82nbPSi0rzD0SSKDwAAAE0wxmhlZUUPHjywHWVLN27c0I0bN7S6umo7CgB0jNCFiCrNJt6zM/Jt+NmCruUlpS8qmg0plQhUj/dGLiiUjarcbqXA+Zh8yVnF03HNJhv3laRQqtIQ5FXk7fK+G57bq8iFkJS81DCaIDTdeL6WNHuN6v2vybjludY8i/y85rI+xc5XtgeUMEaJXcQHAKnH3ttr7eKaLT23/KTOr3lnN2wHmkDxAQAAoAkej0cTExM6evSo7ShbeuaZZ/T000/L4/HYjgIAHWKjebQ3n1s7f21BUlLBhmkogkpKWqi0xngjejsmRYNRTVUbrDY5t/ekppTV1Vxl8yk1XHry1JqGnjbM+73tNUJqaLNam7Fs22eRu6qspnSSDrEA2qoH39tr7eiarT03eb2lokQ6vGbKKKB5FB8AAAC6yNDQkEZGRuQ4ju0oANC9fLHylESNf+qnnchdLTXPJC9tNF1RF2niWQCAdby3d6C09oUTXChN+VSdMgpoHsUHAAAAAACa5D05JWXnNL/VlBPpsILJkFL1axxUrRlBkL+mhfqRBmvP/SRGD2x7jTVTauSvaWGD3rHbPot1PXEBYO91xXt7rb24ZvqSkr6Yciajao0md5WRD2gJxQcAAAAAAJoVOK+YL6voucqiyipPSeEvr4mQVjiYLM8PHlAiFapbvLOkYRHUc1FlQ9OlKT4q575Yt5DpbFKqbN/IJlMibZt/y2vU318l4wWtG8yw3bPwntXM2mv5ndqiqK1mB4Cd6PT39mb308o1d6q+yJGPyx9M1rbxDkcT+m0HAAAA6AS/+tWvNDAwoImJCfX1ubP/hjFGxWJRfX19TLsEAE+MV5FMTvJPatKJlj/zKZYr9QxNh4OlnqLVNZbPK+abVDA8LXO+vG9MClbe06GUTP2ioeVzV1/jDds3EtB0SAoGHSW33bfZa/gUm7mqyQ0zNv8str9Wq9kBYCc6/b29+f00f80dCCSUCjkKTjoqPTWfYrmUpiaDpYKDl3c4tucYY4ztEAAAAG62srKiDz/8UI7j6KWXXnJtw/7jx49169YtDQ4O6stf/rLtOK7z8ssvS5Lef/99y0ngRt/+9rf105/+VEtLS7ajtM3Bgwd15coVHT9+3HYUVOTj8k/OaSaXWT+KoFN0wz243Pz8vN5880299957tqNgC5cvX9brr7+uK1eutO2c/OxdiHcemvTqq6/qzJkzeu2112xHcZMb7uy2BwAA4CLLy8uSpIGBAdcWHiSpWCxKkmtHZgAAusUTnsscAAB0Bb6ZAgAAbGNlZUWSNDg4aDnJ1ioDWik+AECPysfldxw5m/6pzG++M+mwI2eybq5zAMDuPOH3NmAbaz4AAABsozLywe3FB0Y+AIDLeSPKmEjHnj+QMDKJJ3Z6AHCfDn9vA7ZRfAAAANhGpxUf3Dw1FIC9c+/ePdZ/ATrU0NCQ7Qiw4L//+791+fJlfpcDOlSxWGTNhzUoPgAAAGyjfs0HN2PkA4B6Bw4cYMFpoANVFh1G7/m93/s9nTlzhgWngQ5UWXAajfhmCgAAsIVisahCoSDJ/SMfWPMBAAAAAOAWfDMFAADYQmXUQ39/v+sb9Zl2CQAAAADgFu7+Bg0AAGBZp6z3IDHtEgAAAADAPVjzAQAAYAv9/f3av39/Ryz8yLRLAAAAAAC3oPgAAACwhdHRUY2OjtqO0ZSDBw9qdHRUIyMjtqMAAAAAAHocxQcAAIAuMTIyQuEBAAAAAOAKjMkHAAAAAAAAAABtRfEBAAAAAAAAAAC0FcUHAACATTx48EC3bt3SvXv3bEcBgCcrH5ffCSu9xS7psCNnm33aeT0AwDZ4dwNwOYoPAAAAmxgZGdGBAwc0MDBgO0pTfvnLX+rDDz/U6uqq7SgAXCru36TRyBtRxiQU2OLYQMIo5mtTkCaut04+Ln+YJi8AvYd3N4BORfEBAABgE/39/dq3b59GR0dtR2nKc889p2PHjsnj8diOAqCDlHrFbtIzNh0ub3PkbNB4lI/75TiOWmlX2ux6pXP55fdXtvsVz9flmIwqmwxumQcAegXvbgCdgOIDAABAlxgYGNDg4KAcx7EdBUAHCSSMjNmgZ2w+Ln9QSpnS9pSCimaf3PW8kYxSoayyUykZY2RSU4peTFcOksnF5AuVtxkjk2ip7y0AdBXe3QA6AcUHAAAAAOhmdT1go9mkguW/+6tdUzeWn5/TVKo2vUYgkVJozT7eSEbGGLWvLcmn2PnyyQLTCi1ca9eJAaCz8O4G0AUoPgAAAGygWCyqWCzajgEAuxdIVHucxnyham/YTMRrOxkAYDO8uwF0AYoPAAAAG7h3754++OADffzxx7ajNMUYo+XlZRabBtA23rMzWrhUNzd3+pKSa/bZybzhu1LtTZtWeC+vCwAdgnc3ADeh+AAAALCBpaUlSaV1FDrB4uKirl+/rps3b9qOAqCjlBqCGqf1KC8W6o3oguoWCQ1KPl9SQX/8iVwvH/crmMwqOhlWWmmFnaCS2WhtcVJvRBemouU8QS3Ecm2cMgQAOgnvbgCdwTHGGNshAAAA3Ob69etaXl7WsWPHNDY2ZjvOth4/fqxbt25paGhIx48ftx3HlV5++WVJ0vvvv285Cdzo29/+tn76059WC4/d4ODBg7py5QrvBKDDzM/P680339R7771nOwq2cPnyZb3++uu6cuVK287Jzx7oXK+++qrOnDmj1157zXYUN7nByAcAAIA1jDFaWVmRJA0NDVlO05zK+hSO41hOAgAAAAAA0y4BAACss7y8LGOM+vr61N/fbztOUyqDWSk+AAAAAADcgOIDAADAGpVpVzpl1INUG/nQ18evdwAAAAAA+/h2CgAAsEYnFh8qGPkAAAAAAHADig8AAABrdGLxgTUfAAAAAABuQvEBAABgjUrxYXh42HKS5lXWfGDaJQAAAACAG/DtFAAAoM7y8rKKxaL6+vo0ODhoO07TGPkAAAAAAHCTftsBAAAA3GRgYEDPP/+8VldXbUdpSWXkA8UHYOeWl5f10ksv2Y7RNouLi7YjAEBX+9nPftbW/288fvxYzz77bNvOBwC2UXwAAACo4zhOR023VDE+Pq6RkRENDAzYjgJ0pOPHj+vw4cOam5uzHaVtvvWtb9mOsKF02FEwKUkhpUxCAduB8nH5J6/qwhZZSpl3lzfudxRVTCYT2VWWlj2JcwLQ17/+df3P//xPW895+fJl/eu//mtbz9kuvLt5dwM7QfEBAACgCwwNDXXUAtmA2wwPD6u/v1/f/OY3bUdpm43WgIn7wzqZWdOQkY/Lf/GkMom9ad4IJIxMopSlJU8qpzeijNl6l0DCKLbQYt41IpmUrvqv7TpLy3Zyzj3+bwLoRPv27Wv7/zN+/etfy+PxrPucd/cGeHevx7sbLsSaDwAAAADQq9JhOZNRZZNBOY5T+hNOVzfn4/7a545f8bwkpRWuflbaP+6vbY/7HTn+uNLh2j51p9xSw/X88V3mbOb2K8eEtS5iOrzhtdZes9l72yhr/bHNZvHH46XnH06Xz+WX37/xvW92zi2P2+ZZA3AB3t28u3l3o0NQfAAAACgrFApaWlqqrp8AAF0vkJDJxeQLpWSMKf2p9JjMx3Vubqb2eW5Gc+fikgJK5GLyKaSUMYotBDU3k5NJTWluPq9IJqVQNqqgKudMScHmGpW8Jy8oVb5ebmZO/spBO8rZzO2Xjon51mzIx+UPqpolpaCi2aZOubVsVJPVrI3PpdksM3NRJX0xmURA3khGqVBW2anyc0lNKXqx1ti02Tm3PG6rZw3AHXh3FF8IfgAAIABJREFU8+7m3Y0OQfEBAACgbGlpSbdv39bt27dtRwGA9qnreRnNJhWs9sLcukUpPz+nbDZa60E5GVU2O1driApNl6cACelCxLvm6JBS1UaPgBLlxq3t5K/NVvNNNtlitG3OHcjPz2kqVZviJJBIKbRmH28kI2OMWmvbCSlVnTe8ueeyNkvk7Zga26N8ip0vbw1MK7SwzfQguz4OwJ7g3d0y3t2A+1B8AAAAKBsdHdWJEyd07Ngx21FadvPmTX3wwQd69OiR7SgA3CaQqPaCjPlC1V6YmXWNTo28J6fki+VqPSiNkTEZbXNYie9U6wtk5uM6F52q5jOptU1GTyAnALgV724AXYDiAwAAwBqO49iO0LJjx47phRde0OjoqO0oADpRtedkaU7wcFpSYFpT0Yvr57BuRjba0Ds3fWlBM2ebaFGq9siV4rPJJ59zE96zM1q4VD+p9yWtTbOzecOTmq17LvHZ7Z+L9+yMFmZrU5GkL0bVjllEtrXRswbgLry7G/DuFu9uuI8BAAAAesDp06fN6dOnbceAS33ve98zExMTtmO01YEDB8z169cbPov5Qia1wb6pkIxU+uOL5eo3VD8v/QmZlEmZUN2+MZ+MfLHqvqFUyoR8sdLnleNClavWjq398ZnKJRty+Hzl8+0053aazyKFjK9yn2W52Pp8W6k8p/rnUruHrbNUrlV6liETKueofd74cyk9783PufVx2zxrPHHvvvuuOXPmjO0YsGCznz3v7oa75t3Nu9uVXnnlFfPWW2/ZjuE21x1jWFERAAAA3e/ll1+WJL3//vuWk8CN3njjDb3zzju6c+eO7Shtc/DgQV25ckXHjx/f4yunFfZfU6I6PzbaKh+X/+JJZVhItGvNz8/rzTff1HvvvWc7CvaY3Z897+4nind313v11Vd15swZvfbaa7ajuMkNpl0CAACQ9OjRI3366adaXFy0HQUAOlrcH1SyvIgo0z20Sd3Cs87knGbO03gFoL14dz8BvLsB1nwAAACQpIcPH+ru3bsdu2Dz0tKSlpaWxKBWALZFMrXFQ6128MzH5a80+mzwp6Ma1+oWnmVBVgBPAu/uJ4B3N6B+2wEAAADc4IsvvpAkDQ8PW06yMzdu3JAknThxQv39/IoHAPJGlDFMHwIAHYV3N9BVGPkAAAB6XqFQ0MrKiqTOLT5UOI5jOwIAAAAAABQfAAAAKus8DA4OyuPxWE7TumKxWP07xQcAAAAAgBtQfAAAAD2v06dcql/noa+PX+8AAAAAAPbx7RQAAPS8ysiHkZERy0l2hkWmAQAAAABuQ/EBAAD0NGNMtfjQqSMfKphyCQAAAADgFhQfAABAT1taWpIxRh6PR4ODg7bj7EhlzQeKDwB2Kx125Dhhpe1cXWHHkRO2c3Wb9uK55+N++eP5+quWnvcm1y1l2nj7ptvS4TXXALAXeHfbwbsb2B7FBwAA0NM6fb0HiWmXADQv7t+6kSSQMIr5mjxZPi5/WxubAkrkYmr28t2kpee+I2ldjE7pQsRb/STuD0qpja+bj/sVVErGGJmUNFvXKLXVNgWmNRW9aKkBFOhevLvdiXc3sD2KDwAAoKd5PB4NDw937HoP9VhsGsCOpMPlnpDre67m4/7aNn+88ZjJqLLJ4IbHbnpcS7Eaj284p+NXtd2kkr9ynbX/rju2ufa2Sq/S2jnW9hbd/P4qx/oVT8flX9fDtO7cjl/hcFj17T+Xwhvc366uV9ntkpKhaQXqPopkjBKBtTuWzM9NKVXZGJjW1Nx8U9ukgKZDycZGLQBPBu/utVfm3c27Gy7EN1QAANDTxsfH9fzzz+vQoUO2o+xYZdolAGhZPq5SJ0ojY4xSCiqarW32nrxQ3ZabmatNyxBIyORi8oXKvSiNkalrDdn0uBZMngopljMymYiUj+vc3EztWrkZzZ2LV7OkQj7F3o5s/O+WBZQwRjFfSKlMpHzK8r9NQoEt7690bCqUVTQ4p5lcYw/TSo/Vyj0sJBfqrptUstordUrRi7VmqJ1eryJ9KSnfqckdP49TurpJj9j12wLTIWWv5nZ4LQBN4d29Ad7da58H7264AcUHAACADjc0NKQvfelLevbZZ21HAeBGdb1jo9mkguW/++N55efnNJVKVHtVBhIpheoOzV+bre4/Wd+ytY2dHlcR9zuanDulykwT+fk5ZbPRWg/Syaiy2blqD9PA+RnNVRp88nHN6oLqZqmQN5KRMZv3Ft1I5EJdQ1A6rLmZ89XntP39+RTLZUoZAgllIl4pH9fcVKqWwRtRxmTqcoYaeqWGFq5Vz7aj69W5tiBNnfRucNwTUpcdwA7x7ubdzbsbXYDiAwAAQIfr6+vT6OhoV0wdBeAJCCSqvU5LPUBLf1/byLFOPq5z0anq/iYV2nr/3R5XkQwqmq31XJUk78kp+WK5Wu9ZY2TqG3+8Ec0szCqeLzV2TU230FK1mbo5sOOzqs25vdv7a9VeX2+dtK7qlDZ+olttA7ArvLt3hnd3Ge9uuAPFBwAAAADoUd6zM1q4VDfxQvqSkvU71M01HZ9t2FJS7SVZmr+6Oi/3dsdtJZSSSUnB+vmxm1gMM3JhSnPzcV2cm1nXS7a1ecOrF9X52IIuhRt7zpYy7uD+6hrZKtJhp7lpTXbzPCWdnJIWrjU/fcrZGam6e/6aNHO2qW1VUydbzgigeby7t8K7WxLvbriHAQAA6FE3b940d+7cMSsrK7ajYA+cPn3anD592nYMuNT3vvc9MzExYTtGWx04cMBcv3694bOYL2RSa/ZLhWSkyp+Q8flk5Iut2+bz+YwkE0ptfKwvltv48w2O21jKhCSjUMqYVKh6fPW4us8qWdeeMuZrzFGRizWbYaNMWz+zhvvLxYyvIeOaPGu3lwNVz+eL1Z7DRttbvV4tcPVc9c9q0+fZcF6faTjlVtvKWVt/zqj37rvvmjNnztiOAQs2+9nz7t4uw0aZeHfz7t5br7zyinnrrbdsx3Cb644xxuy0cAEAANDJFhcX9fjxYx08eFB9fZ07INQYI8dxbMdwvZdfflmS9P7771tOAjd644039M477+jOnTu2o7TNwYMHdeXKFR0/ftx2lD0T94d1MpNgKol10go7lzRtnvSz2avrdLf5+Xm9+eabeu+992xHwR7r1Z897+7N8O7uJK+++qrOnDmj1157zXYUN7nRud+yAQAAdml4eFiHDx/u6MKDJD18+FC5XE43b960HQUALChNG1JdlLW1+Tl6RGkaktlmpgnZjfQlLcTO03gFoAm8u7fHuxudr992AAAAAOzO/v37tW/fPjGgFYCr5ePyT0aV3WRzKGXWzffdnIASxiixi2i9wBvJKPOkLxJIKEPrFdBdeHdbxbsbnY7iAwAAQBdwyj3HAMC1vBFlTMR2CgBAK3h3A9iFzp5jAAAAYAcKhYKWl5dtxwAAAAAAoGtRfAAAAD3n0aNHun79um7fvm07CgAAAAAAXYlplwAAQM95/PixJGloaMhykvZYWVnRysqK+vv7NTg4aDsOAAAAAACMfAAAAL3niy++kCSNjIxYTtIeDx480K1bt/T555/bjgIAAAAAgCSKDwAAoMcsLy9rdXVVjuN0TfGhWCzajgAAAAAAQAOKDwAAoKdURj0MDw/LcRzLadqr2+4HwN5Lhx05Tlhp20FaVMrdmdl3Yy9+Xvm4X/54vv6qCjuV5+1XwyZt/bPYdFs6vOYaAFrBu7uz8O5GL6H4AAAAekplvYfR0VHLSdqP4gOA7cT9Wzd2BBJGMV+TJ8vH5Q+7o7kokDAypoXsXaKln9eOpHUxOqULEW/1k7g/KKVKz9ukphQ9F69uy8f9CipV3ibN1jVKbbVNgWlNRS/2VOMj0Are3d2Fdzd6CcUHAADQUyrFh26Zckli2iUAu5QOl3s0OnLWNEjl4/7aNn+88ZjJqLLJ4IbHbnrc5iFqPTLL+6/tadn6Odf0Lq3cp7+xwcXZoidoZXtz7XS7uYfKsX7F03H51/UwbeyxGg6HG7JeCm98Dzu/XmW3S0qGphWo+yiSMUpUPghMK1S3bX5uSqnKxsC0pubmm9omBTQdSjY2agHYGu9u3t28u9EBKD4AAICesbi4qGKxqL6+Pg0PD9uOAwD25eMqdYYs9YZMKahotrbZe/JCdVtuZq42vUIgIZOLyRcq94Y0RiYR2P64TQWUMEYxX0ipTKR8ifK/TUKBHZ1zTe/SQELGpGoNLvm4zs3N1PLnZjR3rrmGsfbfQ+nYVCiraHBOM7nGHqYNPVZzM1pILtRdN6lktVfqlKIXa81QO71eRfpSUr5Tk5vecT4+K12IbPo8TunqJj1i128LTIeUvZrb9FoA6vDu5t3NuxsdguIDAADoGY8ePZJUmnKJKYoA9Iy63rHRbFLB8t/98bzy83OaSiWqvSMDiVRDb8j8tdnq/pP1LVvb2OlxkQt1jSjpsOZmzlez7fScm2acn1M2G631Lp2MKpuda+h96o1kZExdb9Enfg8+xXIZRbySAgllIl4pH9fcVKqWwRtRxpT3kSSFGnqlhhau1e5xJ9erc21Bmjrp3eC4Us/cybmZlp7NtuqyAz2Pd/fGGXl3b3y9Ory74SYUHwAAQM9YXl6WJI2NjVlOAgB7KJCo9hAt9eQs/X1tY8U6+bjORaeq+5tUaOv9d3uc1DB/dKljpnf359yE9+SUfLFcrfesMTINDUM7tIf3sKUneL102NHk3IxMZrOes5KU1lWd0sbtW1ttAyCJd/cmeHfvHO9u2EDxAQAA9IyjR4/qhRde0L59+2xHeSL6+vjVDkBrvGdntHCpbgKF9CUl63eomzM6PtuwpaTa27E0D3V1fu3tjttUQOdjC7oUbux1urtzLuhauTNr3B+s3V8TC2W2Nm949cTtvQdvRDMLsw29etNhp6mpS3b+zEpOTkkL1xqvE/c7pcVHN2i8Ojuj6rNW/po0c7apbVVTJ1vOCPQi3t28u7fCuxuuYgAAANDR7ty5Y65du2bu3r1rO4qrnT592pw+fdp2DLjU9773PTMxMWE7RlsdOHDAXL9+veGzmC9kUmv2S4VkpMqfkPH5ZOSLrdvm8/mMJBNKbXysL5bb+PMNjttayoS0dc7Gc6ZMSPX3ICP5TDVOKlSXMVbatxKmblvl/uuvm4u1mn2H95CLGd+ae6h/nuu2lwNVz+eLNT6HtdtbvV4tcO1ZbZRj7TNr2F73M9huWzlr68+5O7377rvmzJkztmPAgs1+9ry7eXfz7na/V155xbz11lu2Y7jNdccYY1opVgAAAMBdVldXVSgU1N/fL4/HYzuOa7388suSpPfff99yErjRG2+8oXfeeUd37tyxHaVtPB6PhoaGNDAwYDsKOpaRKUjyOHqyKyXt1XU6w6NHjzQyMqIHDx7YjoI99sMf/lB//dd/rfHxcdtR0NF4d9tw//59/emf/qn++Z//2XYUN7nRbzsBAAAAdqe/v1/9/fxaB6CRMUapVErHjx+3HQVAC+bn5/XjH//YdgxY8JWvfEUnTpzQf/zHf9iOAqBFf/Inf6Kvfe1rtmO4Dt9SAQBA11taWtLDhw81Njam4eFh23EAYE+Mj4/rxRdfpPgAdJgXX3yxa9enwtaGhob0pS99SS+++KLtKABa9OUvf5nvmhtgVUIAAND1HMdRsVjUw4cPbUcBAAAAAKAnMPIBAAB0vcHBQT399NO2YzwxxWJRUqnI4jjMuAqg5N69e/rjP/5jPfvss7ajAGjB5cuXNTQ0ZDsGLLl8+bJmZmZsxwDQonfeeUdnzpyxHcN1KD4AAAB0uI8//lgPHjzQ008/rYMHD9qOA8Al/u3f/k0rKyu2YwBo0Xe/+10dOXLEdgxY8I1vfENzc3O2YwDYge9+97v6xje+YTuG61B8AAAA6HDPPvssPZsBrDM9PW07AgCgBUePHtV3v/td2zEAoG1Y8wEAAHS1QqFgOwIAAAAAAD2H4gMAAOhaxWJRv/zlL3Xjxg2KEAAAAAAA7CGKDwAAoGs9fvxYxhgVi0V5PB7bcQAAAAAA6Bms+QAAALrWw4cPJUn79u2znOTJ+uKLL1QoFDQ8PKz+fn69AwAAAADYx8gHAADQlYwxevTokaTuLz589tln+uijj6r3CwAAAACAbRQfAABAV3r8+LGKxaL6+/s1PDxsO86ecBzHdgQAAAAAACRRfAAAAF2qMuXS2NiY5SR7h+IDAAAAAMAtKD4AAICu1CtTLgEAAAAA4EYUHwAAQNd5/PixCoWCPB6PRkZGbMcBAAAAAKDnUHwAAABdpzLqYWxsrKemIioWi7YjAAAAAAAgieIDAADoQouLi5KYcgkAAAAAAFv6bQcAAABot+eff16Li4saGhqyHQUAAAAAgJ5E8QEAAHSl4eFh2xEAAAAAAOhZTLsEAADQ4XppXQsAAAAAQGdg5AMAAECHO3LkiA4fPqyBgQHbUQAAAAAAkETxAQAAdJG7d+/KcRyNj4/L4/HYjrNnKDoAAAAAANyGaZcAAEDXGB4e1tLSklZWVmxHAQAAAACgpzHyAQAAdI3R0VGNjo7ajgEAAAAAQM9j5AMAAECHKxQKWllZUbFYtB0FAAAAAABJFB8AAAA63qeffqobN27o/v37tqMAAAAAACCJaZcAAEAXWFpaUn9/f08tMl1vYmJCExMTtmMAAAAAAFBF8QEAAHS8jz/+WEtLSzp69KjGxsZsxwEAAAAAoOcx7RIAAOhoy8vLWlxclCQNDw9bTgMAAAAAACSKDwAAoMM9ePBAkjQyMtKz0y4BAAAAAOA2TLsEAAA6WmWR5fHxcctJ7FlaWtLKyooGBgY0NDRkOw4AAAAAAIx8AAAAnevRo0daXV1VX1+f9u3bZzuONffu3dNHH31UHQUCAAAAAIBtFB8AAEDHqh/14DiO5TQAAAAAAKCC4gMAAOhIq6urevTokSTpwIEDltPYReEFAAAAAOA2FB8AAEBHun//vowxGhkZ0eDgoO04VlWKD8YYy0kAAAAAACih+AAAADrSvXv3JDHqQaL4AAAAAABwH4oPAACg4xSLRY2Ojqq/v7+nF5qu6OvjVzoAAAAAgLv02w4AAADQqr6+Pj3zzDMyxrDeQZ1isWg7AgAAAAAAkhj5AAAAOhiFhxJGPgAAAAAA3IZvqgAAAF2CkQ8AAAAAALeg+AAAADqGMUarq6u2Y7gOIx8AAAAAAG7Dmg8AAKBjLC8v6+bNm9q3b5+eeeYZ23FcY2xsTC+88AJFCAAAmsC0jUBz3n33XZ09e9Z2DAAdjOIDAADoGENDQzpx4oSWlpZsR3GVvr4+Cg8AADTp2Wef1U9+8hMdP37cdhTAtb71rW/ZjgCgC1B8AAAAHaWvr08jIyO2YwAAgA41ODioQ4cO6dChQ7ajAK61f/9+2xEAdAG6yAEAAAAAAAAAgLai+AAAANDhjDFaWVlhOioAAAAAgGtQfAAAAK73ySef6Pbt2zSub2JpaUk3b97Ub37zG9tRAAAAAACQRPEBANDB8nG//PG8lI/L74SV3ngvxf2OwmmV9vPHld/bmNilYrGoe/fu6dGjR1pdXbUdx5WGh4d14sQJPffcc7ajAAAAAAAgieIDAKBT5eM6NzejtyNeyRtRxiQU2O4Yb0Rvz8zpXNxe+eHhw4f68Y9/rGKxaC1Dp7l3756KxaIGBwc1NjZmOw4AAAAAAGgCxQcAQNMaRhesHW2QDstxnPKftaMQ0gpvtC0fl98fVthf9/mW56k748Wopi5E5N0oS/31/Bd1te44b+SCpqIXd/MYduThw4f64Q9/qP379+tXv/qV+vr4X3AzjDH6/PPPJUkHDx60nAYAAAAAADSLlg8AQBukFQ5KKWNkjFEutqBguK4UEA5qIZbbcJuySemCkTEJBbY5T/31LiVDmt5kqEP99cwFKZmt3xrQdCjZjptuSqXocPToUf3N3/yNJOnP/uzP9uz6na4y1ZLH49H4+LjtOAAAAAAAoEn9tgMAALrFgq7lpYBX8kYyMtXP07qU9Gkm55VUGnkQci4pnQiUp0laW0TY7Dx18te04Dul8xvmKBUmLpjS9RQ4r5gv2TD6YfKUb8d32ayHDx/q+9//vn70ox9pdHRUjx8/1tDQkE6fPq3Z2dmGfX/60582dc4TJ07o+PHjbc2Zy+V0+/bttp6znZaWlmSMkcfj0cDAgO04rlZZD6O/n1/vNvNf//VfOn36tO0YAAAAANATGPkAAGiDgBK5Gc1NVqZL8qtxWYWsotVtQSXLBYbWz1OWu6rsBh9LKhUmGj7w6uTUTu5pd5aXl3Xv3j1JpamDKjwez96H6VCFQqH67GhQ35oxRqurq1pdXW347w2NTp8+rbNnz9qOAQBAy/yb/V7cqfJppTv5fvLx7vuZAMATwDd5AEB7eCPKmEjp7+mwnHNxnc2U12RQSKmNFoTe6Jf1Lc9TNnlKm45d8J7UVMM4h7yuLUiabu12duvw4cP6p3/6J128eFF///d/r1gspuXlZf3nf/6nfvKTn2hkZGRvA3WgW7du6fHjxzpw4IAmJiZsx3G9XC4nSXrppZdYUwQAgC6TMRnbEdonH5d/ck4zucD67wedov47CwBgU3wzBQA0bapuxEJ+fq42+iAdluOPb1hLqKyxMFvpFrTVvluep473pKayV5Vr6noXFV0zTCJ3ddNxE213+PBh/d3f/Z1u3rypP//zP5ck/ehHP9qz63eqL774Qo8fP5bjODp06JDtOB3BcRxJUrFYtJwEAAAAAACKDwCAFpyPqTp90jnNKFTZEEgoNRXVpFOeLikopepGKwQSKU1FJ8vbFhR7e81IhibPU7ejpkNJXdpoLWpJgUROM3Pl681KoYZhEqU1IfZapQjx6aef6uDBgzQQb2N1dVV9fX0aHx9nrYcmUXwAAKB7Vaf4qUz3k47LX/md2XHkj+fL22qfhRt+V84r7q9tc5ywKpvzcf+a6U7TCjec0694PFw7dl1noTXnrr9wPi6/E1Y87q9tn4wqW56W1d/MvEU7vec126rH1J9zs/tqdvsWP5PG5196ptWpZdOl5wIA3Y7iAwCgad5IRsYYGWOUiUSUqJtKKZAw1W1m3RRLASWq2zKKeKsnVGbNvlufp26/8zEtzJa/AKw7j1eRTPkcmYQSGaNEeWM+PquF2MZLVe+Fw4cP6/XXX2danG3s379fJ06c0FNPPWU7Sseo/DdF8QEAgG6XVXRWerv8O3Mu5lM2OinnXONnyWClwJBX3D+p6FSq+nt2LragYLkx3RvJKBXKKnqxtHc6HFQylFKm+kt7VtGolDJGxuQUU1ST1Zb1tefOKbYQbCxAKKno1Qu13/FzMfnkUyxn6q7R7ntOKzwZ1VTK1N2zT9noRaXrz7npfTWzfet8DVmcoBZiuXKWC7oajG6+hh0AdBFaPgAAnckb0dszczrXyipv+bjOzc3o7aa/5MCmvr4+FuhuAcUHAAB6R+hCbXSw9+yMfBt+Vp4yNX1R0WxIqURdV53IBYWyUZXrDQqcj8mXnFU8HddssnFfSQqlKh19vIq8Xd53w3N7FbkQkpKXVN9MH5re/eoOLd1zflLnTa0D0rrt291Xk9u3zld5/peUVEgXqt9BAjof23QFOwDoKhQfAAAdyxvJtNBbSqUREhtO4wR0vkrxwRhjOQkAAHiyfDo12cxnJflrC5KSCjZMQRRUUtJCpSXeG9HbMSkajGoqtXb08Zpze09qSlldrSzA5julhktPnlrTyL95tua1ds/yeku/86fDa6Z72uL4tfe17fbmsuSvLax7RpXiCQB0O4oPAADANZaWlmxH6FiMfAAAAJvyxZQz9dObVqZSrXXLyV0tNc0nN1tYraOU11gILiiWq5/uCQCwlyg+AAAAVygUCrp9+7Zu3bplO0pHYsFpAACwEe/JKSk7p/mtZitNhxVMhpQyKYWSwTWLJa/p7Z+/pgWFVJ1Jae25c1eV1ZRO2hxunL6kpC+mXP16c7mra0Y+bHNf225vTun5X1XDgIl1WQCgO1F8AAAAruDxePTCCy/o0KFDtqN0JEY+AACADQXOK+bLKnqutMC0pPJ0RP7y+gVphYPJ8voGASVSobrFkksaFq8+F1U2NF2amqly7ot1C1DPJqXK9o1sOX1RG9UXRfJx+YPJdbtsel9Nbm9KYFohJTVbXSyi9LwBoBf02w4AAABQ4TiORkdHbcfoSBMTE5qYmKgWIQAAAEq8imRykn9Sk060/JlPsVxpVEA6HCyNEqi0qgfOK+abVDA8LXO+vG9MCpZHWSqUkqlfYLp87srmxu0bCWg6JAWDjpLb7rtDgYRSIUfBSUelO/YplktpajJYKnpMbndfzWxvOowSJqWwM6nS4/cpFgtJ0e2OA4DO5xhWJQQAAADQ49544w298847unPnju0oAJ6wL3/5y3rvvfd0/Phx21HcLx+Xf3JOM7m66Yu6wXb39aTvOx2WM3tKJhN5Aidvj9///d/XX/7lX+rs2bO2owDoXDfoGgcAAKxaXV21HQEAAAB4MhqmuJKqU1NNnbSZCgD2BNMuAQAAa4wxunnzpjwej5599lkNDAzYjgQAAIAKb0QZ497e+Tu23X21874DCTXOOeJVJGPUhU8VANah+AAAAKy5d++eVlZWVCwW5fF4bMfpaIVCQSsrK3IcR0NDQ7bjAADgWrdv39ZXv/pVOj0AW7h//76uXLnCtEsAdoXiAwAAsKJYLOru3buSpKeeeoqFkndpcXFRn376qYaHhzUxMWE7DgAArjU0NKR/+Zd/Yc0HYAt/+Id/qJdeesl2DAAdjuIDAACw4u7duyoUChocHNT4+LjtOB1vbGxMY2NjtmMAAOB6Tz31lL75zW9SfAC28NWvfpXfLQHsGl0MAQDAnltZWdHnn38uSTpy5Igcx7GcCAAAAAAAtBPFBwAAsOc++eQTGWM0OjpKjyoAAAAAALoQxQcAALCnFhcX9fDhQ0lk2MyYAAAgAElEQVSlUQ8AAAAAAKD7UHwAAAB76je/+Y0kaXx8XENDQ5bTdA9jjD777DN9+umntqMAAAAAAMCC0wAAYO88fPhQi4uLcv4fe3caGleW3///c1Wq0r4vtmy33O2WzNBoAkMmpFPlgQTmieThjx+J4A70k1BFkwFVIA0D8YMEHGhiCFWENEiBwBDaAYUEP7CrCEmY+eGRSCANw0R0wKput/e2LVt7aanl/B9IVapNe0m3lvcLREv33HvO996ST9vne+45lqWuri67w6koxhjNzc1Jkjo6OlRTwxwTAAAAAIB9SD4AAIBT09zcrJ6eHklSbS1/DSmmzGRDMpkk+QAAAAAAsBX/6gcAAKeqvb3d7hAqlsPhUCKRUCKRILkDlJmf/vSnWlxctDsMoCosLS3ZHQIAAFWBf5UCAABUiFTyIZlM2h0KgEP6u7/7O7W2tqq1tdXuUICKt7m5aXcIAABUBZIPAADgRBljJEmWZdkcSeVLLbWUSCRsjgTAYdXW1urf/u3f9OGHH9odClDxLl68ePCTI0F5Br/SDTOu4V1OCfssjUx4FdrjnGK2BwBAuWAxYAAAcKJWV1f16NEjRaNRu0OpeA6HQxLJBwAADivo8SlcqGBgTFP7JAKGx40C7iIFcoD28kSC8vgKRg8AgK1IPgAAgBPV3Nys7u5uBsRPAckHAACKJ+yzZFmWLKtAYiLs2y6zZBUY+I8EPbIsS4fJCezW3lZdHnk8qXKPgpGMOAb9mp4Y2TMeAADsQPIBAACcuObmZrW0tNgdRsVLLbvEng8AABzf8LiRMQXeaogE5RmRQmarPKQR+adPrr2BsSmFvNOaHgrJGCMTGpL/Vjh1kcxsQG7vdpkxMuMs2AQAKA0kHwAAACoEbz4AAHAIGW8v+KcnNLL9vSf9WkFhkXuTGgrtLI00PB6SN+ecgbEpGWNUvDyAW4FPtysbvibvzINiVQwAwIkh+QAAAIoumUwy+94GDodDlmWlN/kGAAB7GB5Pvy0QcHvTbzJMjQ3YHRkAABWB5AMAACi6ubk5ffvtt1pdXbU7lKrS2tqqgYEBnT171u5QAACoWANXRzVzJ2NfhfAdTeScc5Q9H44l/SZEWL7TbBcAgD3U2h0AAACoLNFoVIuLi5Iky7Jsjqa68LwBACiWsHzWyE5SwZqQ5FZgdkpjA2O6IUs7/9v1yu2e0IjnA5mpsaK3d/WeRyMT09KET5fNNd3ZPm/Cd3lrf4eBMd0YsmRZfkmSOzCrKbZ9AACUAMvwXj4AACiSRCKhx48fKx6Pq729XT09PXaHBAAH8tOf/lT//M//rJcvX9rSvtPp1P379/Xhhx/a0j5QTS5evKj79++rv7/f7lCAkvWjH/1IP/vZz3T16lW7QwFQvh6z7BIAACiaV69eKR6Py+Vyqbu72+5wAAAAAACATUg+AACAolheXtbKyoosy9KZM2dYAsgmq6urWl5eZtNpAAAAAICtSD4AAIBji8fjevXqlSSpo6ND9fX1NkdUvebn57W8vKxkMml3KAAAAACAKsaG0wAA4NhevnypZDKp+vp6dXZ22h1OVbtw4YLdIQAAAAAAwJsPAADgeBYXFxWNRlluCQAAAAAApJF8AAAAxxKLxSRJ3d3dcrlcNkcDAAAAAABKAcsuAQCAY+nu7lZzczP7PAAoe69evdLo6KgtbScSCf35n/+5urq6bGkfqCbLy8t2hwAAQFUg+QAAAI6NxEPpWF1dVTQaVX19vVpaWuwOBygb77//vr7//e/rxz/+sS3t/8u//It+53d+R5cuXbKlfaCa/L//9//sDqGgsM/SyIQkeRUy4xq2O6BIUJ7Br3Rjj1i2Yj5evEGPJb8CMlNjx4rl0E6iTgBAFpIPAAAAFWR9fV0LCwtqa2sj+QAcwpkzZ9Tf3y+v12tL+3/yJ3+ia9eu6cMPP7SlfaCa/NVf/VXesaDHp8tTOYPQkaA8ty5ravx0hqaHx43M+FYsh3JScQ6MacrsfcrwuFFg5pDx5hibCukrz4Njx3JoR6nzlH8nAKDcsecDAAA4lFgspmfPnikej9sdCgqord2aW8LnAwDAMYR9sgb9mp4YkWVZW1++cLo4EvTsHLc8CkYkKSxf+tjW+UHPTnnQY8nyBBX27ZyTUeWestrzBI8Z50FuP3WNT3khhn0F28pt86D3VijWzGsPGosnGNx6/r7wdl0eeTyF7323Ove8bp9nDQDIR/IBAAAcitPpVH19vV6/fm13KCgglXxIJBI2RwIAQBkbHpeZDcjtDckYs/WVmu0eCerjydGd47Ojmvw4KGlY47MBueVVyBgFZkY0OTorExrS5L2IxqZC8k77NaJUnSFp5GAJgYHLNxTabm92dFKe1EVHivMgt791TcCdUxAJyjOidCwhjcg/faAq9zbt12A61uznctBYRif9mnAHZMaHNTA2pZB3WtND288lNCT/rZ1EwW517nndXs8aAFAQyQcAAHBoXV1d6uvrszsMFOBwOCTx5gMAAPvKmDXvn57QSHoG/d7ZgMi9SU1P+3dmvw/6NT09uZNE8F7bXr7JqxtjAzlXexVKD1gPa3w7MbGfyIOb6fgGDzjav2+cRxC5N6mh0M7yVMPjIeUuVjcwNiVjjA43Lu9VKL3nw8GeS24sYz8PKDuX4Fbg0+3S4WvyzuyztNOxrwMA5CL5AAAAUEFYdgkAgAMaHk/PYA+4vekZ9FN5CYNsA5eH5A7M7sx+N0bGTGmfy7a4Pzj85saRoD72D6XjM6GD7U1zrDgBACgCkg8AAGBfsVhMxhR7lz+chNSbDxJLLwEAcGzpWe9b+zn4wpKGr2nIfyt//4GDmPZnvVkRvjOj0asHyAak36aQgjcnTj7OXQxcHdXMncwNGe4oN5qj7fkwoZsZzyV4c//nMnB1VDM3d5aRCt/yqxgrQO2r0LMGABRE8gEAAOwpkUjo2bNnevLkiWKxmN3hYB+WZbH0EgAAxTAwphtDqWWLRjQTmN1eSmhY4yGll0Ha2bg4LN/2hsRbCYYJjWxvDj3tH9wapHYHNDo5mL5uRDe230TY2ax6Zwmo7X0PBsZ0QzubHE/KvVPfkeLcz8FjsUYkt3vnPo8i6BnRRM5zmRz9+YGey89HJzOepTe9BFQk6NHIxLT8g9ufizWiiWn/9gbRu9e593V7PWsAQCGWYRojAADYhTFGz54909rampxOp/r7+1VTw9yFUvf48WNtbGzo3LlzampqsjscoCzcvn1bt2/f1t27d21p3+l06v79+/rwww9taR+oJhcvXtT9+/fV399/yi2H5fM80Hh6bwMUVSQoz63LmiIbUBQ/+tGP9LOf/UxXr161OxQA5esxowcAAGBXr1+/1tramizLUl9fH4mHMsG+DwAAlJ6gZ3sWPUv1FE/GpuHW4KRGPyXxAAClhBEEAABQ0Pz8vBYXFyVJfX19qqurszkiHJTT6ZTL5ZJlWXaHAgAAto1N7Wz8bOvk/EhQnqylmLK/yioxkrFpOJtpA0DpqbU7AAAAUHpWV1c1NzcnSeru7mbpnjLT09NjdwgAAKBUDYxpyrD0EwDg5PHmAwAAyLKxsaEXL15Iktra2tTR0WFzRAAAAAAAoNyQfAAAAGnxeFzPnz+XMUaNjY3MoAcAAAAAAEdC8gEAAKS9fv1a8XhcLpdLfX197BkAAAAAAACOhD0fAABAWm9vr6StfR5qapijUK6MMVpcXFQikVBXV5fd4QAAAAAAqhDJBwAAkOZwONTX12d3GCiC9fV11dbWyhjDGywAAAAAgFPHlEYAAIAKY1mWzp49q+7ubhIPAACUmkhQHsun8GlddxJKKRYAQMki+QAAQBVbXFzUwsKC3WEAAADYKug5xYH0gTFNmXEN5x6PBOXx7RHFbtfZ4Six7Hd/AICKQ/IBAIAq1tjYqPn5ea2trdkdCgAAQMkIeixZnqDCPkuWtfWVNW4eCcpjpco8CkZ2K/PJk5HY2KkvJ9kR9ska9Gt6YiTdnpXR4K7X7RFLJOiRZXnk8ewSZ/qcnHvbx26x7NnePvcHAKhMJB8AAKhiTqdT/f39amhosDsUAACAkjE2FZJ32q8RhWSMkTEhaSQ1mB6Wb3BSo7Nmq2x2VJODGQmGWxllIWl6eqfe4fGt4wF3ToPD4zKzAbm9qfaMzPjw/tftEcvA2JRC3mlND23XGRqS/9bxB/x3i2XP9va5PwBAZSL5AABAlXM4HHaHgBMQjUb17Nkzzc3N2R0KAAClKexLz8L3T09oZPt7T3q6vleh9AD5sMZDQ5q8F5HCdzQT+LnGBraLBsb088CM7qTG2a8NyT+4Pbt/ZELe0AkulbRPLJJbgU+HU4HJO/Mg6/KBsSkZY1S8PMDe7QEAqgvJBwAAqsja2ppWV1ftDgOnIJlMKhqNsqQWAAC7GR5Pz8IPuL0KbX8/lRrJd39wtKRBRr1mNqCZm8FiRg0AQNkg+QAAQJVYX1/X8+fP9eLFC0WjUbvDwQlzOp2SpFgsZnMkAACUqWl/xlsQUvjOjEavDkjD1zQ0eS/r1HuTQ7o2LElh+Qrty3BQ6TcFwvIdZC+GPWPZ31H2fDiWw94fAKCs1dodAAAAOHkbGxt69uyZksmkGhoa2OOhCqSSD4lEQslkUjU1zDkBAOBQ3AGNTg7K8m//7A3JDEjSsD4dvSkrXSC5A7OaSv80oRFrIv2TN2S2vwvLZ40oXWJNSHIrMDu1tWzSwJhuDFnpet2BWU0N73fd7rFEgh6NTExLEz5dNtd0Z7uOCd/lY+y3sHssV+/t096u9wcAqFSWMcbsfxoAAChX6+vr6cRDfX29zp8/z0B0lfj666+VTCbV39+vuro6u8MBStrt27d1+/Zt3b1715b2nU6n7t+/rw8//NCW9oFqYlmWfu/3fk/t7e17nPVaM/+1qqEP3z2tsICSEg6H9fd///f64z/+Y7tDAVC+HvPmAwAAFWxtbU3Pnz8n8VClnE6nNjY2FI/HST4AAJChtbVVfX19u5abZK96/7+t760ah2qsUwoMKCEbGxt2hwCgzJF8AACgQkWjUT1//lzGGDU0NOjcuXMkHqpMKvnAvg8AAOzo7+/XxMSE+vv77Q4FKFkPHjzQu+++a3cYAMocyQcAACrQ6uqqXrx4IWOMmpqa1NfXJ8tiyl61YdNpAAAAAIBdSD4AAFBhNjc304mH5uZmnT17lsRDlaqt3fqrHskHAAB2fPfdd/qP//gP3nwA9vA///M/docAoAKQfAAAoMK4XC51dHQoFovpzJkzJB6qWOrNh3g8bnMkAACUju9///v6m7/5G9XX19sdClCy6urq1NbWZncYAMocyQcAACpQV1eX3SGgBLhcLjU1NcnlctkdCgAAJYMZ3QAAnA6SDwAAVIBkMslm0sjjdDp17tw5u8MAAAAAAFQhRikAAChzKysrevbsmYwxdocCAAAAAAAgieQDAABlr7m5WU6nU2tra3aHAgAAAAAAIIlllwAAqAhnz561OwQAAAAAAIA03nwAAKDMbGxssMQSDiwajerFixean5+3OxQAAAAAQBUh+QAAQBlZWVnRkydP9OLFC7tDQZlwOBxqampSY2Oj3aEAAAAAAKoIyy4BAFAm5ufnNTc3l/7ZGCPLsmyMCOWgrq5OdXV1docBAAAAAKgyJB8AAChxxhi9fPlSy8vLkqS2tjb19vbaHBUAAAAAAMDuSD4AAFDC4vG4Xrx4ofX1dUlSd3e3Ojo6bI4KAAAAAABgbyQfAAAoUdFoVN99950SiYQcDofOnj3Luv04stQm5SzVBQAAAAA4DSQfAAAoQW/fvtWbN28kba3Z39fXJ6fTaXNUKFdzc3NaWFhQZ2enOjs77Q4HAAAAAFAFSD4AAFBiksmklpaWJEmtra3q7e1ltjqOpaamRsYYbW5u2h0KAAAAAKBKkHwAAKDE1NTUqK+vT+vr62pra7M7HFQAl8slSYrFYjZHAgAAAACoFiQfAAAoQXV1daqrq7M7DFSIVPKBNx8AAAAAAKelxu4AAACodvPz85qfn7c7DFSw1H4hyWRS8Xjc5mgAAAAAANWA5AMAADZramrS27dvGRTGibEsK52A4O0HAAAAAMBpIPkAAIDNXC6X3n33XdXWshoiTg5LLwEAAAAAThPJBwAATlEsFtPS0lLecYfDYUM0qCapPURIPgAAAAAATgNTLAEAOCVLS0t6/fq1ksmknE6nGhoa7A4JVST15sPGxobNkQAAAAAAqgHJBwAATlgikdCrV6+0srIiSWpoaGCJJZw6ll0CAAAAAJwmRj4AADhB0WhUL1++VDwel2VZ6uzsVGdnp91hoQq5XC51d3enkxAASksikdD//u//pjeHBwDY57333uPv7ABQBCQfAAA4AYlEQnNzc+n9HVwul86cOaP6+nqbI0O1sixLHR0ddocBYBe1tbX667/+a7W1tdkdCgBUtS+//FJffPGFrl+/bncoAFD2SD4AAFBky8vLev36tRKJhCSpra1N3d3dqqmpsTkyAECp6uvr03/+53+qv7/f7lAAoKr95Cc/sTsEAKgYJB8AACii1P4OyWRSLpdLvb29bCwNAAAAAACqDskHAACKyOFwqKenR7FYTJ2dnbIsy+6QgDyJREIOh8PuMAAAAAAAFYzkAwAAxxCPx1Vbm/2/09bWVpuiAfYWj8f1+PFjSdKlS5dsjgYAAAAAUMlIPgAAcAxPnz5VT0+Pmpqa7A4F2Fdtba0uXLggl8tldygAAAAAgArHzpcAABxDT0+PotGo3WEAB0biAQAAAABwGnjzAQCAA9rc3MwbuG1qauKtBwAAAAAAgBy8+QAAwD4SiYRevnypR48eaWlpye5wAAAAAAAASh5vPgAAsAtjjBYWFvT27Vslk0lJ0sbGhs1RAcdjjNH6+ro2NjbU3t5udzgAAAAAgApF8gEAgAJWVlY0NzenWCwmSaqrq1NPT48aGhpsjgw4vmfPnskYo6amJjmdTrvDAQAAAABUIJIPAABk2Nzc1OvXr9ObSNfW1qqrq0utra02RwYUh2VZcrlc2tjY0MbGBskHAAAAAMCJIPkAAICkZDKp169fp/d0sCxL7e3t6uzsVE0NWyShstTV1aWTD83NzXaHAwAAAACoQCQfAACQVFNTk97Pobm5Wd3d3cwIR8Wqq6uTJK2vr9scCQAAAACgUpF8AABgW29vrySpvr7e5kiAk5X6HWcDdQAAAADASWEdCQBA1YnFYnr06JGMMVnH6+vrSTygKtTV1cmyLCUSifSm6gAAAAAAFBPJBwBA1XE6nXI4HFpZWbE7FMAWlmWx9BIAAAAA4ESRfAAAVLRkMllwcLWvr08tLS02RASUhlTygaWXAAAAAAAngeQDAKAiJRIJvXnzRg8fPtTz58+VTCazyh0Oh02RAaUhtcQYbz4AAAAAAE4CG04DACpKLBbTwsKCFhcX03s6uFwuxeNxuVwum6MDSkcq+ZC79wkAAAAAAMVA8gEAUBE2NjY0Pz+vlZWV9GBqfX29Ojo61NzcbHN0QOlxuVx6//33VVPDi7BApfrlL3+peDxudxgAjqC9vV0//OEP7Q4Dp2xubk6//vWv7Q4DwBF973vf04ULF+wOo6SQfAAAlLW1tTXNz89rdXU1fayxsVEdHR1qbGy0MTKg9JF4ACrbH/zBH+idd95Rb2+v3aEAOIQvv/xSFy5c0JMnT+wOBafsv//7v/WTn/xEv/3bv213KAAO6csvv9Tnn3+uTz75xO5QSgrJBwBAWXv79q2i0agkqbm5WZ2dnemNdAEAqGZtbW361a9+pf7+frtDAXAI9+7d02effWZ3GLDJlStXdP/+fbvDAHBIH330kd0hlCSSDwCAsmGMkWVZWcc6OzvldDrV0dEhp9NpU2QAAAAAAADIxLv2AICykEwm9fDhQyWTyazjDQ0N6u3tJfEAHNHKyopev36tRCJhdygAAAAAgApC8gEAUBZqampUX1+v5eVlu0MBKsrq6ip7PwAAAAAAio5llwAAJSUej2tpaUnJZFLd3d1ZZWfPnmWQFCiyM2fO2B0CAAAAAKACkXwAAJSEtbU1LS4uamVlJb23Q0dHhxwOR/ocEg8AAAAAAADlgeQDAMA28Xhcy8vLWlpa0ubmZvp4fX292tvbSTYAAAAAAACUKZIPAIBTZYzR6uqqlpaWtLq6mlXW1NSkrq4u1dXV2RQdUN1Sbx0BAAAAAHBcJB8AAKcqmUzqu+++kzFG0tZbDq2trWppaeFNB8AmCwsLmp+fV3t7uzo6OuwOBwAAAABQARjlAQCcmEQioVgslnXM4XCora1NHR0dunjxot555x21tbWReABsZIxRPB7X2tqa3aEAqBIey6NgxO4oiigSVric7ycSLJ3PZL9Yih1ruX92wCkpmT6iWMr9z/4p99uRoEeWZckqtd+Dcv8cqwAjPQCAE7O0tKS5ubm84z09Peru7pbL5bIhKgC5GhsbJYnkA4BTM2WmNDZgdxRFEgnKM3hTD+yO4zgGxsrnMylmrJXw2QGnpGz6iIOohD/7p9pvh3XLPy1vyMiU0u9BJXyOVYDkAwCgKKLRaHoppZTW1lbF43GbIgJwUHV1daqpqVEymdT6+rrd4QAAAAAoFZEHmpFbHwzaHQjKEckHAMCRra2t6dWrV/rmm2/07NmzvFnTDodD77zzjk3RATiMhoYGSbz9AOB0pJeKSC0bEQ7KY1nbSzpY8gQj22U7x3zhzBoiCnp2yizLp1Tx1tIQmctChOXLqtOjYNC3c60nqOwVG3Lqzmw4EpTH8imYXn7CkjXo17Sm5R/cbmM/R73nnLL0NZl17nZfBy3f4zPJfv5bzzS9BEd467lknbL3Q9j9GafcO0CsherKi2OXtiJBeQ772QFVjH67SvvtnL5y59ke5pn7FC72M6QPLxskHwAAh7K+vq7Xr1/r4cOHevr0qRYXF5VIJORwOHjLAShjJB8A2Gda/pvSz42RMUazAbem/YOyPs4+NjGSGiSJKOgZlH8oJJMun9HI9qDMwNiUQt5p+W9tnR32jWjCG9JUep2Iafn9UsgYGTOrgPwazBpMyax7VoGZkZzB8Qn5v7qRbtvMBuSWW4FZk9FGse85LN+gX0Mhk3HPbk37b2UMHO11Xwcp3zu+rFisEc0EZrdjuaGvRvyaPuCdH+wZT8s/+YFm941179+F/PKQvBMjWwNbA2OaOtJnB4B+u4r67Zy+0owPH+GZj2u42M+QPrxskHwAAOxrY2NDc3Nz+vbbb/XkyRMtLCwoHo+rpqZGra2tOn/+vN577z21trbaHSqAIyL5AMBO3htjSg0bDFwdlbvgsRk9iEgK35J/2qvQ+HD6+oGxG/JO+7U9bqXhTwNyT9xUMBzUzYnscyXJG0oNhAxo7Ofb5xase0BjN7zSxJ2s2aHea9n1nfg9Rwb1qTHKvI2s8v3u64Dle8eXev53NCGvbqQHeob1acB98Bs/6DNOt79HrPv9LkTuaXLarcCnqfJhjec8RwBHQ79dRf12rmM885N4hihtJB8AAPt6/fq15ufnFYvFVFNTo5aWFp07d06XLl3SmTNn1NjYKMuy7A4TwDHU19en933Y2NiwOxwAVaXQOtK7ry0deTAjaUIjWcswjGhC0kxqNGJgTD8PSP4Rv4bSAze71D1wWUOa1lezqeIPlNX04Ac5Ax3FWPf6cPesgYGtgZmwL2fZkD2uz72vfcsPFkvkwUzeM0oNIB3Yvs/Yq6wxq11i3fd3YfYrTWtIl5kQCxQZ/fa+9VZav53X3FGe+Uk8Q5Q6kg8AgLTNzU0tLi7mHW9ublZzc7POnj2rS5cu6ezZs2pqaiLhAFSY1NsP0WjU5kgAYB/uwPaSPNlfmcsuzH61NTwxcefgOxGUru21ukdmtpa9SC8bgoP8LgAoAfTb9NuHxjOsBCQfAABZ3rx5k3esvb1dfX19amlpIeEAVDCSDwDKwcDlIWl6Uvf2WnIh7NPIhFehzDX+03JmjUYeaCZzpn1u3aUwez58RxPugGbNlNLjdLNf5cz+3Oe+9i0/mK3n/5WyJt7mxbKPfZ9xzpIakQeaKTA7dt/fhbyZuADsQL+dGVemMuq3c53GMz/QM0SpI/kAAFVmc3NT8/PzevLkidbX17PKXC6X2tralEwmbYoOgJ2am5vV09Oj3t5eu0MBgN0Nf6qAe1r+j1ObCmt7SQbP9jrYYflGJrbXyR7WeMibsXnllqxNUD/2a9p7bWuJj1TdtzI2Mr05IaXKC9lzGYwiyhzoiQTlGZnIO2XX+zpg+YEMX5NXE7qZXnR863kf/PqDPOPMzzcV6w3lvcyw3+/CwFWN5rblsXY2RT2tzw6odvTb5d1v59V3hGd+VHs9Q/rwslBrdwAAgJOVTCa1tramaDSq1dVVxWKxdNnq6qrq6+uzzu/q6jrtEAGUCKfTqfb2drvDAIB9DGhsalbyDGrQ8m8fcyswuzUzMuwb2Zopmd5j+FMF3IMa8V2T+XT73IA0knqb0xuSydw0c7vu9MueWeWFDOuaVxoZsTSx77lHNDyukNfSyKClrTt2KzAb0tDgyNagy+B+93WQ8gMHo3ETks8a1NbjdysQ8Er+/a5LOcgzdisw+pUG941179+F/ds6hc8OgOi3t++hbPvtXEd55kew3zMcoA8vB5YxxtgdBACguDY2NhSNRhWNRrW2tqbMrt6yLDU0NKi5uVlNTU2qrSUPDQDA7du3dfv2bd29e9eW9i9evKj79++rv7+/aHW2t7frN7/5TVHrLHuRoDyDkxqdncqfRV/O9ruvk77vsE/WzQ80OzWmE3+slfoZZrh3754+++wz3b9/3+5QqtJPfvITXb9+XdevXz/1tvnsC6jUP/PV1G9XiY8++khXrlzRJ598YncopeQxyy4BQAV6+fKl5ubmFI1GZYyR0+lUW1ub+vr6dOnSJZ0/f15tbW0kHgAAAHA4WUulSOnlNoYun0Rex8cAACAASURBVOIAls1ruQNAOSmJfhvViuQDAJSxaDSqhYWFvOOptxp6enp08eJFvfvuu+rt7VVzc7Nqauj6AezNGKO1tTX2fwGAw4oE5bEsWbt+ZQ7+lKnhcc0GJP9g6p4G5R/aXu7ihO8/7LNkDR5xzXMAKIR+u/LvH7Zi2SUAKAPGGG1sbOTtz7C2tqZXr17p4sWLNkUGoBI9efJEyWRSZ8+eVV1dnd3hAKeiEpddqqmp0RdffKFz584VrU4AJ+8Xv/iF7ty5o1//+td2h1KV7Fx26V//9V/l9/v1j//4j6feNoDj+bM/+zNdv35df/qnf2p3KKXkMettAEAJSiUbUhtFr6+vK5lM6uLFi3K5XOnz6uvr1dHRYWOkACrR+fPneUsKqADGGF2/fl1NTU12hwLgEFZXV9XQ0GB3GLDBV199pSdPnujq1at2hwLgkFZXV/W9733P7jBKDskHACgBuyUbMtXU1CgWi2UlHyzLUmtr62mHC6DCkXgAKkNbWxsbTgNlKLXpMKrPD37wA125coUNp4Ey9NFHH8ntdtsdRskh+QAANovH43r06FHBZENDQ4MaGxvV0NDA0icAAAAAAAAoGyQfAOAULSwsqL6+Pmvvhtra2vQsY5INAAAAAAAAqAQkHwCgyIwx2tzcVDKZzFurNRaLKZlM5m0c/c4776i2li4ZQOmIx+NaWFhQPB7X2bNn7Q4HAAAAAFBmGOkCgGNKJBJaX1/X2tqa1tfXtb6+LmOM6urq8tZYbmlpyVteSRKJBwAlaX5+XpLU09Mjh8NhczQAAAAAgHLCaBcAHNLGxoai0ag2Nja0vr6uWCyWd05NTU3BhELuGw8AUKpqa2tVV1enjY0Nra6usrk9AAAAAOBQSD4AwC6MMYrFYnK5XFnHo9Go5ubmso65XC7V19eroaFB9fX1edcAQDlqbGwk+QAAAAAAOBKSDwAgKZlMyrIsWZaVPhaLxfT06VNdunQp69z6+no1Nzerrq5O9fX1qqurYzkSABWpqalJ8/PzikajMsZk9ZEAAAAAAOyF5AOAqpNIJLSxsZH1tbm5qQsXLmRtEO1yueRwOJRIJLKSCw0NDXkbSQNAJWpoaEj3g2tra2psbLQ7JAAAAABAmaixOwAAOEnGGC0tLWlubk7Pnj3TN998o2+++UbPnj3T3NyclpeXtbm5KWlrL4dcFy9e5K0GAFUtlXBYXV21ORIAJyoSlMfyKbzHKWGfJWufc4rZHgBgH/TdAEocyQcAFWNtbU2JRCLrmDFGL1++TC8bkip3Op1qbm5WV1eXzp07p/fee0/t7e12hA0AJa25uVmStLKyYnMkAIoh6Nll0GhgTFNmXMN7XDs8bhRwFymQA7SXJxKUx8eQF4DqQ98NoFyRfABQVlJLf6yvr+eVpRIMmWpqatTS0qL29nb19vbqnXfe0fvvv693331XfX196uzsVFNTk2prWYUOAAppamqSZVmKx+MF3xADUP62ZsXuMjM27Evvi2UVGDyKBD2yLEuHGVfarb2tujzyeFLlHgUjGXEM+jU9MbJnPABQLei7AZQDkg8ASlI8Hlc0GtXCwoJevXqlp0+fppdMevr0qebn5/Ou2W0t8rNnz6qnp0dtbW2qr69XTQ1dHwAclGVZampqksTbD0ClGh43MqbAzNhIUJ4RKWS2ykMakX/65NobGJtSyDut6aGQjDEyoSH5b4VTF8nMBuT2bpcZIzN+qLm3AFBR6LsBlANG4ACUlFevXunrr7/Ww4cP9ezZM71+/VqLi4tZSyo5nc6C+zC0t7erpaXltEMGgIrX2tqqzs5O+ligXGXMgPVPT2hk+3tPempqYZF7kxoK7SyvMTwekjfnnIGxKRljVLyxJLcCn25XNnxN3pkHxaoYAMoLfTeACsA6IwBOXCKRyEsWLC8va21tTb29vXnnJ5NJWZYlp9Mpl8uV9eV0OnlzAQBOWVNTU/rtBwBlaHhcxoxL2lo3/PLUIdfrBgCcPvpuABWAETwARZFIJLS+vq6lpSW9efNGL1680OPHj/X111/r22+/zTvf4XAUXDu8o6NDFy9e1Pvvv6+LFy+qr69PXV1damlpUV1dHYkHAACAUzJwdVQzdzLW5g7f0UTOOUdZN/xY0rNpw/KdZrsAUCbouwGUEt58AHBkKysrmp+f1+bmppLJ5J7n5r790NDQoLNnz+ad53Q6ix4nAAAAdhOWzxrZGZiyJiS5FZid0tjAmG7IkmWlCr1yuyc04vlAZmqs6O1dvefRyMS0NOHTZXNNd7bPm/Bd3lojfGBMN4YsWZZfkuQOzGqKacAAqhJ9N4DyQPIBQFoymVQsFpPL5ZK18zcVxWIxPXv2TO+++27W+cYYra+vp3+ura1NL5XkdDqzvs+sT1J6WSUAQHnZ2NhQbW1twb13AJS+sanxnCPDGjdGuUfTpeNGJqsw+8yBsSmZQ41l7dFeTl3DBc7LjwcAKh99N4ByRfIBqCLGGMXjccVisayv1LHUhs7vvPOO6uvr09fV1tYqFovJGJOVRGhoaFBfX186yZCbYAAAVJbvvvtO0WhUZ8+eVWNjo93hAAAAAABKGMkHoAKtrq6qsbExKxmwsbGhx48f73utw+FIJyFSLMvSu+++m5dcqK2tVXNzc3GCBgCUvO7ubtXW8tdHAAAAAMD++NcjUCaMMem3FFJfsVhMLS0tebNPX716pfPnz8vlcqWPpQaLUssdpb5SSyWlvnbb0JklkgAAJB4AAAAAAAfFvyCBEmSM0dzcXHoppMwlkXI5nc685ENjY6OMMVnHHA6H3nvvPQaOAAAAAAAAAJw4RiGBE5TaYyEejyuRSKi2tjZrLwVJev78uVpbW7OWL7IsS0tLS0omk1nnpt5aqK2tTb+x0NDQkNfumTNnCsZD4gEAAAAAAADAaWAkEiiCpaWl9JJIiUQiK+GQqbW1NS/5UFNTU/Cths7Ozrxkg8PhONH7AADgIBYWFrS8vKze3l7V1dXZHQ4AAAAAoASRfAC2pd5SSCQSSiQSsiwrbzmjN2/eyOFwqL29Pev4wsKCNjY2CtZrWVbWmwq5enp6Cu6z0NHRcYy7AQDg5ESjUa2vr2tlZYXkAwAAAACgIJIPqFjGGBlj8gb2l5aWtL6+nk4ypN5UyF3iqKGhIS/5YFmWYrFYXlvNzc1qaGiQw+GQw+FIJxsO8rYCbzMAAMpNS0uLVldXtbS0pK6uLrvDASrC48ePdfHixaJOQFlfXy9aXQBQLb777jt99NFH+ulPf3rqbcdiMb3//vun3i4AnBSSDygbxpishEHml6S8wY/FxUVtbGzk7X+wurqqlZWVgm1YlpVOHrhcrrzytra2gtd1dnYe5ZYAAChLTU1NqqmpUTwe19raWsH9hwAczqNHjxSPx4ta5w9+8IOi1ldUkaA8g1/phhnXcIm0F/ZZGpnwKnSMmIIeS34FZKbGjhXLoZ328wQq2N27dxWNRm1p+xe/+IX+4R/+wZa2D4S+m74bOCSSDygJsVhMm5ubampqyjo+Pz+vxcVFJRKJvDcTMjkcjrzkg8PhKHhNc3OzXC5X1lsKqf8WWv4ot04AAKpdTU2NmpubtbS0pOXlZZIPQBH09/cXvU7LsvKOBT0+XZ4qgUGOgTFNmdJqb3jcKDDjO1YzY1MhfeV5cOxYDu0odUaC8ty6rKlx238bgJJy9uxZ29r+v//7v4LjEvTdu6PvBkrb3iOtwCHEYjGtrKxocXFRb9++1dzcnF6+fKnnz5/r6dOnevTokb755ht9/fXXeddubm5qfn4+73gymVQsFstKIjgcDrlcLjU0NKi5uVnt7e15ezBIW0mGvr6+vOMtLS3q6upSe3u7Wlpa1NDQIJfLtW/iAQAA7GhtbZUkLS8vy5jT/FcogOIJy2dZslJfvrCCHkuW5VEwsnVGJOjZKfcE01cGPVs/h3071/vC+5dJyjjuU8bh7bY88nhS5TtxbF+Yrs8TDG7Fnlnxbne5S3u5dRaqK3X/B2im4HUHvfe97m+/53Kk5xn2yRr0a3piZM/7B1CK6Lvpu+m7UT4YbYUSiYRisZjW19cVjUa1vLyclUDIHVDY2NjQ06dP8+pZWVnRixcv9OrVK71580bz8/NaWlrS6uqq1tbWtLm5mX6DIfeNBKfTqfr6+rw6W1tbdeHCBV28eFGXLl3S4OCgLl26pIsXL+rChQvq6+tTT09PwWWPCs30AgAAxdHQ0KDa2lolk8ldlzMEUOqGNT4bkFtehYxRYGZEk6OzMqEhTd7bGukYuHxDoe291GZHJ+XZHgEZmwrJO+3XiELbe62FpJGtAZK9yqStWarGGAXc2dEMjE0p5J3W9ND2daEh+W9tD6pEgvKMKB3L6KRfE+6AzAFmfu7WXm6dIY3IP32c57lt2q/BydFD3fte97fnc9mjzj2vGx6XmQ3I7Q2l98o7yLMEUArou+m76btRPkg+lLlkMql4PK6NjQ2tra3lJQoSiYTm5ubyrlteXtbXX3+t2dlZffPNN/r222/15MkTPXv2TN99911WAiG1p0LKbpsupxIITU1Nam1tVXt7u7q6utTb26u+vj6dP39e/f39eu+99/LeMnC5XOru7i5YZ+rNBJY8AgCgtGS+/QCghGXMyvRPT2gkPUNze1TFe217KQ+vbowNZF0aeXAzff5g3uiOV6H0oMewxjMGvvYu24tbgU+3rxu+Ju/M1jIYkXuTGgrtLDky9vOAcseADiu3zuHxkLw55wyMTckYo8ON7XgVSq8bfrB73//+Cj+X/R31OgC2o+8uiL4bKC/s+WCDVHYy9T+RzOPLy8vpf8inbGxs6M2bN+k3BnZ7e0DaWiu2rq4u69ji4mLewL5lWVnX19TUqKamRg6HI+/73DcInE6nzp8/n9d2c3OzmpubD/4gAABAWWtpadHbt2+1urqqeDyu2lr+agmUpOFxGTMuqcC64XuNq0SC+tg/pJCZ2jo/7FPWktjuD3Zff3yvMgDA/ui7AVQA3nw4gFSyINfKykre8VgsppcvX+rFixd69uyZnjx5osePH+vbb7/V119/rUgkokgkoq+//lrr6+t5db58+TKvzmQymV66aGNjQ/F4PCtxYFmWHA6HnE5nXn01NTUF90NobGxML2U0MDCg999/X++99576+/t14cIFnTt3TmfOnFFPT0/eGweWZcnlcu390AAAQMVzuVzpZRNZegmoUOmZtVLw5kR22bR/ZwaupPCdGY1eHdi/7AgGro5q5ubOuuXhW34dd5WNgaujmrmTuaj3HeXc4RHXDZ/QzYx7D97c/95P4v4OJD2bdmsNeZYOByoEfTd9N1AqTJVYXV01yWQy61g8HjcvX740L168MM+fPzdPnz41jx8/Nt9++615+PChiUQi5sGDB+bBgwdmeXk5r85IJGJisVjWsfX19fQ1+32trKzk1fn06dOCcS4sLJjl5WWzurpq1tbWzMbGhonFYiaRSBTh6QAAABxNNBo1q6urdocBHNsXX3xhrl69ancYRdXW1mYePXqUdSzg9ppQ+qeQ8UpGknEHZk3ALSN3wJiQ10gy3pAxIe9WuSTjdrvTx40JGa87sHXNdrm8oZ169ypTxnHJSG4TmDVmNuDe/tlrQpnnbV+7Uy4jr9d43YEDPIXd2zMm+/4kr3GnnsG2VJve0C7V50g9w8x7d6ca2yeW3e5v7+dy9OeZe/87ccJud+/eNVeuXLE7DNhgt8+evpu+m7679F2/ft18/vnndodRah5ZxhSY0n+CzPZbBMlkUsaY9PI+mVZWVtTQ0JB13BiTfisgdW3qv7nHent71dbWllXnw4cPdf78+awZ+/F4XA8fPjxQ3GfOnMlbDun58+fq7e3NWmIgmUxqcXExvVxRagmj1FfqLYXcJZcAAAAA2Of27du6ffu27t69a3coRdPe3q7f/OY36u/vP4Haw/J5Hmg8vT72QcuKJBKU59ZlTVXqRpuVfn/Y07179/TZZ5/p/v37doeCU3bynz1994mq9PvDnj766CNduXJFn3zyid2hlJLHWQvzFhrIT/03mUzK6XQW3E+gvr4+7/jjx48L1peru7tbHR0dWcfevn2rnp4eNTQ0ZMV20M0MC+VTUksCZHI4HOrs7MxLDuQmDFJfuc6dO5d3rKamJu9+AAAAAKCSBD0jmpiWJiy/vKHsTT33KjuWsE/WSGpxDbcCs1NFqrhEVPr9AbAdffcJqPT7A46pVtp6KyAej+97ckdHR16SYWVlRbW1tXnHNzc3CyYBMtXU1BQ8p7GxMW+wv6amRt3d3enjmW8PZCYNUm8W5Orr68s7ZlmWurq69owRAAAAAJBtbMpot7mxe5UdS8bmq2mRoDyDu6+xXdQBtJNW6P4AoIjou08AfTewp9rdCnIH9C3LylpeKKWlpaXg8b6+voKJgcz6dtPd3V3wOG8UAAAAlKZEIqGlpSU5nU41NzfbHQ6AajEwpilzgkuEAACKj74bqBq1kvTOO+9I0oESA7ly90FIaWpqKkJ4AAAAKAfLy8taX1/PWjYTAAAAAFC9aiUVfHMBAAAAOKj29na1t7fbHQYAAAAAoETk76IMAAAAAAAAAABwDCQfAAAAAAAAAABAUZF8AAAAQFElEgmtr6/bHQYAAAAAwEYkHwAAAFA00WhUDx8+1MuXL+0OBcARhH2WLMunsD2ty2dZsnz2tG6n03jukaBHnmAks9Wt571Lu1sxFS7ftSzsy2kDwGmg77YHfTewP5IPAAAAKJr6+npZlqXNzU1Fo1G7wwGQI+jZe5BkeNwo4D5gZZGgPEUdbBrW+GxAB22+khzquR9JWLf8Q7oxNpA+EvSMSKHC7UaCHo0oJGOMTEi6mTEotVeZhq9pyH/LpgFQoHLRd5cm+m5gfyQfAAAAUDQ1NTVqbW2VJC0sLNgcDYADCfu2Z0Lmz1yNBD07ZZ5g9jWDfk1PjBS8dtfrDhVW9vVZdVoepcdNUvGn2sn9OePag423pWaV7tSRO1t09/tLXetRMByUJ2+GaUbdlkc+n0+Z4z93fAXu71jtpU67ownvNQ1nHBqbMhofzj1xy73JIYVShcPXNDR570Bl0rCueSeyB7UAnAz67tyW6bvpu1GCSD4AAACgqNra2iRJq6urisViNkcDYE+RoLYmURoZYxTSiPzTO8UDl2+ky2ZHJ3eWZRgel5kNyO3dnkVpjEzGaMiu1x3C4AdeBWaNzNSYFAnq48nRnbZmRzX5cTAdS8jrVuDnY4V/PrRhjRujgNur0NTYdpXbP5txDe95f1vXhrzT8o9ManQ2e4ZpasZq6h5mJmYy2p3QRHpW6pD8t3aGoY7aXkr4zoTcHwwe+Xl8oK92mRGbXzZ8zavpr2aP2BaAA6HvLoC+O/d50HejFJB8AAAAQFG5XC41NjZKkhYXF22OBkDm7Fj/9IRGtr/3BCOK3JvUUGg8PatyeDwkb8alkQc30+cPZo5s7eOo16UEPZYGJz9QaqWJyL1JTU/7d2aQDvo1PT2ZnmE6/OmoJlMDPpGgbuqGMlap0MDYlIzZfbZoIWM3MgaCwj5Njn6afk77359bgdmprRiGxzU1NiBFgpocCu3EMDCmKTOVEac3a1aqd+ZBurYjtZfhwYw0dHmgwHUnJCN2AEdE303fTd+NCkDyAQAAAEXX3t4uaSv5kEwmbY4GqHLD4+lZp1szQLe+zx3kyBMJ6mP/UPp8E/Luff5xr0uZGJF/emfmqiQNXB6SOzC7M3vWGJnMwZ+BMY3O3FQwsjXYNXTtECNVu8lYAzt4Uztrbh/3/g7rtNvLE9ZX+kCFn+heZQCOhb77aOi7t9F3ozSQfAAAAEDRNTU1yel0KplManl52e5wAOxi4OqoZu5kLLwQvqOJzBMy1poO3swq2ZKeJbm1fnV6Xe79rtuLNyQTkkYy18c+wGaYYzeGNHkvqFuTo3mzZA+3bni6UX0amNEdX/bM2a0Yj3B/GYNsKWGfdbBlTY7zPCVdHpJmHhx8+ZSro1L69MgDafTqgcrShi4fOkYAB0ffvRf6bkn03SgdBgAAADgB8/Pz5sGDB+bhw4d2hwLs64svvjBXr161O4yiamtrM48ePco6FnB7TSjnvJBXRkp9eY3bLSN3IK/M7XYbScYbKnytOzBb+HiB6woLGa9k5A0ZE/Kmr09fl3EsFWtulQF3dhwps4GDxlAopr2fWdb9zQaMOyvGnHhyy7cDStfnDuw8h0Llh21vJ+B0XZnPatfnmVWv22RVuVfZdqyHf87IdPfuXXPlyhW7w4ANdvvs6bv3i6FQTPTd9N2n6/r16+bzzz+3O4xS88gyxpijJi4AAACA3SSTST18+FDJZFLnzp1TU1OT3SEBu7p9+7Zu376tu3fv2h1K0bS3t+s3v/mN+vv77Q7l1AQ9Pl2eGmcpiTxh+aw7umZO+tmcVjuV7d69e/rss890//59u0PBKavWz56+ezf03eXko48+0pUrV/TJJ5/YHUopecyySwAAADgRNTU1am9vV2trq5xOp93hAKhYW8uGpDdlPdz6HFViaxmSmwdZJuQ4wnc0E/iUwSsAB0DfvT/6bpQ/kg8AAAA4MV1dXTpz5oxcLpfdoQCwWyQoz/ZAU6Gvo487DWs8czPT3EXDIUkaGJvaf6Pa4xoeP/k2AJwu+m5b0Xej3NXaHQAAAAAAoAoMjGnKjNkdBQDgMOi7ARwDbz4AAAAAAAAAAICiIvkAAACAUxGPx7WysmJ3GAAAAACAU0DyAQAAACduc3NTjx490tramt2hAAAAAABOAXs+AAAA4MS5XC69++67cjgcdocCAAAAADgFvPkAAACAU0HiAQAAAACqB8kHAAAAAIAkKeyzZFk+he0O5JC24i7P2I/jND6vSNAjTzCS2ap8Vup5e5RVpL0/i13Lwr6cNgAcBn13eaHvRjUh+QAAAIBTt7m5qWQyaXcYQNUJevYe7BgeNwq4D1hZJCiPrzSGi4bHjYw5ROwV4lCf15GEdcs/pBtjA+kjQc+IFNp63iY0JP/HwXRZJOjRiELbZdLNjEGpvco0fE1D/ltVNfgIHAZ9d2Wh70Y1IfkAAACAU/XmzRs9evRICwsLdocCQJLCvu0ZjZasnAGpSNCzU+YJZl8z6Nf0xEjBa3e9bvcgdmZkbp+fO9Py8HXmzC5N3acne8DF2mMmaKr8YON0x7mH1LUeBcNBefJmmGbPWPX5fFmx3vEVvoejt5c67Y4mvNc0nHFobMpoPHVg+Jq8GWX3JocUShUOX9PQ5L0DlUnDuuadyB7UArA3+m76bvpulAGSDwAAADhVLpdLkrSwsMDbD4DdIkFtTYbcmg0Z0oj80zvFA5dvpMtmRyd3llcYHpeZDcjt3Z4NaYzM+PD+1+1qWOPGKOD2KjQ1tt3E9s9mXMNHqjNndunwuIwJ7Qy4RIL6eHJ0J/7ZUU1+fLCBseLfw9a1Ie+0/COTGp3NnmGaNWN1dlQzEzMZ7U5oIj0rdUj+WzvDUEdtLyV8Z0LuDwZ3veNI8KZ0Y2zX5/GBvtplRmx+2fA1r6a/mt21LQAZ6Lvpu+m7USZIPgAAAOBUtbS0yOVyKZFIaHFx0e5wgMqXMTvWPz2hke3vPcGIIvcmNRQaT8+OHB4PZc2GjDy4mT5/MHNkax9HvW7sRsYgStinydFP07Edtc5dY7w3qelp/87s0kG/pqcns2afDoxNyZiM2aInfg9uBWanNDYgaXhcU2MDUiSoyaHQTgwDY5oy2+dIkrxZs1K9Mw927vEo7WV4MCMNXR4ocN3WzNzBydFDPZt9ZcQOVD367sIx0ncXbi8DfTdKCckHAAAAnLqOjg5J0tu3b3n7AThpw+PpGaJbMzm3vs8drMgTCepj/1D6fBPy7n3+ca+TstaP3pqYOXD8OncxcHlI7sDszuxZY2SyBoaO6BTvYU8n2F7YZ2lwclRmareZs5IU1lf6QIXHt/YqAyCJvnsX9N1HR98NO5B8AAAAwKlraWmR0+lUMplk7wfARgNXRzVzJ2MBhfAdTWSekLFmdPBmVsmW9GzHrXWo0+tr73fdrob1aWBGd3zZs06PV+eMHmxPZg16Rnbu7wAbZR5u3fB0xcW9h4Exjc7czJrVG/ZZB1q65OjPbMvlIWnmQXY7QY+1tflogcGrq6NKP2tFHkijVw9UljZ0+dAxAtWIvpu+ey/03SgpBgAAALDB0tKSefDggYlEIiaRSNgdDqrcF198Ya5evWp3GEXV1tZmHj16lHUs4PaaUM55Ia+MlPryGrdbRu5AXpnb7TaSjDdU+Fp3YLbw8QLX7S1kvNo7zuw6Q8arzHuQkdwmHU7ImxFjYOvcVDAZZan7z2x3NnDY2I94D7MB4865h8znmVe+HVC6Pncg+znklh+2vZ2Ad55VoThyn1lWecZnsF/ZdqyHf86V6e7du+bKlSt2hwEb7PbZ03fTd9N3l77r16+bzz//3O4wSs0jyxhjDpOsAAAAAIrl0aNH2tzcVGdnp7q6uuwOB1Xs9u3bun37tu7evWt3KEVjWZbdIQA4orq6Oq2vr9sdBk7ZX/zFX+gv//Iv7Q4DwBH94R/+of7pn/7J7jBKyeNauyMAAABA9erq6tKLFy+0sLCg9vZ2ORwOu0MCKkZ9fb3+67/+S/39/XaHAuAQ7t27p7/927+1OwzY4Ac/+IF+67d+S7/85S/tDgXAIf3RH/2R3G633WGUHJIPAAAAsE1zc7Pq6+u1vr6ut2/fqqenx+6QgIpRV1enjo6O9AbvAMpDR0eHXC6X3WHABrW1tWptbaXfBspQe3u7amsZas/FEwEAAICturq6tLq6qs7OTrtDAQAAAAAUCckHAAAA2KqxsVGNjY12hwFUnMXFRV25ckW9vb12hwLgEL788ktduHDB7jBgk1/96lf64Q9/aHcYAA7pyy+/1JUrV+wOo+SQfAAAAACACvSLX/xC8Xjc7jAAHEF7e7vdIcAGv/u7WZbHMwAAIABJREFUv6t///d/tzsMAEf0ve99z+4QSg7JBwAAAACoQL//+79vdwgAgEPo7u7Wj3/8Y7vDAICiqbE7AAAAACBlc3NTz58/VzQatTsUAAAAAMAxkHwAAABAyVhaWlJdXZ3q6+vtDgUAAAAAcAwsuwQAAICS0d3dbXcIAAAAAIAi4M0HAAAAAAAAAABQVCQfAAAAAAAAAABAUZF8AAAAQElaX1/Xs2fPFI/H7Q4FAAAAAHBIJB8AAABQkl6/fq1oNKo3b97YHQoAAAAA4JBIPgAAAKAk9fT0SJKWlpa0vr5uczQAAAAAgMMg+QAAAICSVF9fr9bWVknSq1evbI7m/2fvXmLiStf94D9V3O++u933qHFH6jiDSJkEpt8g4B545MGWop6B9uSDSKeVPfCwB5F68IESRYIog62oPWDUg2NQpDNLjM7J2OpBg87e292+td0GbCgooGp9A6iiiipswNjLwO8nlQzr+qxFt2zWf73vAwAAwEEIHwAAeG9duHAhstls5PP5WFpaSrscAAAA9kn4AADAe6uhoSHOnz8fERG///57FAqFlCsCAABgP4QPAAC813p6eqK5uTkKhYLm0wAAAMeE8AEAgPdaJpOJS5cuRUTE0tJS5PP5lCsCAADgdYQPAAC899ra2qKrqysiNJ8GAAA4DoQPAAAcCxcuXIjGxsbo7u5OuxQAAABeozHtAgAAYD8aGxvj888/j0wmk3YpAAAAvIaRDwAAHBuCBwAAgONB+AAAAAAAABwp4QMAAMdSLpeLX3/9NYrFYtqlAAAAsIvwAQCAYydJknj+/HmcOXMmsln/pAUAAHjfaDgNAMCxk8lk4uOPP067DAAAAPbgNTEAAAAAAOBICR8AADgRkiRJuwQAAAC2CR8AADj2Xr58GX/5y19ibW0t7VIAAAAI4QMAACfAyspKFAqFePz4sREQAAAA7wHhAwAAx96lS5eisbExNjY24tmzZ2mXAwAAcOoJHwAAOPay2WxcunQpIiIWFxdjdXU15YoAAABON+EDAAAnQkdHR3R3d0dExJMnT6JYLKZcEQAAwOklfAAA4MS4ePGi6ZcAAADeA8IHAABOjGw2G5cvX46IiKWlpVhZWUm5IgAAgNNJ+AAAwInS3t4eZ8+ejYit6Zc2NzdTrggAAOD0ET4AAHDinD9/PlpaWqJQKEQul0u7HAAAgFOnMe0CAADgqGUymfjggw9ic3Mz2tvb0y4HAADg1BE+AABwIjU3N0dzc3PaZQAAAJxKpl0CAAAAAACOlPABAIBTYWNjIxYWFtIuAwAA4FQQPgAAcOIVCoX45Zdf0i4DAADg1NDzAQCAE6+hoSE+/fTTaGz0z18AAIB3wcgHAABOBcEDAADAuyN8AADgVEqSJO0SAAAATizhAwAAp876+nrcv38/lpaW0i4FAADgRBI+AABw6qysrMT6+no8ffo08vl82uUAAACcOMIHAABOnbNnz0Z7e3skSRIPHz6MYrGYdkkAAAAnivABAIBT6YMPPojGxsbY3NyMR48epV0OAADAiSJ8AADgVGpoaIgPP/wwMplM5HK5ePbsWdolAQAAnBiNaRcAAABpaWlpicuXL8fjx49jYWEhWltbo7OzM+2yADig//t//2/81//6X6OtrS3tUoADWltbi7/7u7+Lf/2v/3XapQBHTPgAAMCp1tXVFfl8PhYWFuLx48fxySefREtLS9plAXAA8/Pz8T//5/+M//yf/3PapQAH9Kc//Slu3rwpfIATSPgAAMCpd+HChcjn85HL5eLRo0fxySefRENDQ9plAXAAf/jDH+I//af/lHYZwAH97//9v9MuAXhL9HwAAIDYakDd1NQUzc3Nkclk0i4HAADgWDPyAQAAYqsB9ccffxyNjf6JDAAA8KaMfAAAgG2CBwAAgKMhfAAAgFfY2NhIuwQAAIBjR/gAAAB1JEkSjx8/jidPnqRdCgAAwLFjXDkAAOyhqakpLl26lHYZAAAAx47wAQAA6shkMnH+/Pm0ywAAADiWTLsEAAAAAAAcKeEDAADs0++//x7Pnj1LuwwAeL358ejP9Mf4fNqFHKH5mZg5ztdzEn8mAK8gfAAAgH1YW1uL58+fx8LCQiwsLKRdDgC8Wu9I3E3uxkhv2oUckfnx6L/6Xfycdh0A7JvwAQAA9qG1tTUuXLgQERHPnj2Lly9fplwRAADA+0v4AAAA+3T27Nno6emJiIgnT55ELpdLuSIA2EPlFD+lr2fGoz+Ticz2p398fnvdzrLhmaqDxHj/zrpMZjhmyofvj0zVFEIzMVx1zP4YHx/e2bd/PKpnG9p17MoTz49Hf2Y4xsf7d9ZfHY3ZmI3Rq9vn2Pc9qKx//9e31/7lc9fUWNr3FddVcme46pgAJ5XwAQAADuDSpUvR1dUVSZLEo0ePYm1tLe2SAGAfZmP0u4g/J0kkSRJzY30xO3o1Mt9UL5scrHyIfjVGr01HUl5/Lwa3Q4TekbsxPTQbo99vbT0zPBiTQ9NxtzzP02yMjkZMJ0kkyVyMxWhcLT+I333suRi7N7jrQf1kjP50q3zuZG4s+qIvxuaSinO8wvx49F+diptzSdUx7g2WAohXX1/ETAxfHY1r00nF+r6YHf2+IqCorHEiBmqOOR1Dk4O7Ao/ZGJ36KuYqjglwUgkfAADggC5fvhzt7e1RLBbj4cOHsbGxkXZJAPBaQ7dGovTYvvf6zeiru+xe/DwfETPfx+jsUExPDJT37x25FUOzo7GdN8TAt2PRN/ldjM+Mx3eT1dtGRAxNT8TWkt4Y+fP2tnWP3Rsjt4YiJn+sGnkwdKP6eAd3Lb6szCl6R+JuksTEwD6ub/5qfFvatt79qVfj/J2Ymu2LsW9LywZiYtcxImrvOcBJJXwAAIADymQyceXKlWhpaYlCoRAPHjyIzc3NtMsCgFfoi6+u7mfZlvmf70XEZAxWTTs0GJMRca/09L13JP48FjE6OBrXykHDHsfu/TKuxWz8NFda/VVUnfrqV7se7O9d2770jsStoYr6d01/9Nrr6+3dCghmhndN/fSKa5z7KWZ3Bx413vC6AI4R4QMAABxCNpuNjz76KJqammJjYyMePnwYSZKkXRYAHJ2+sfL0QJWfymmP5n7aehw/+WOd3gYpG5hIqqY/qunr8Mrr2+phkRm8F2NzlVM/AbBfwgcAADikhoaGcgBx/vx5TSMBODF6v7wWMTsVd17V23lmOAYnh2J6j94G5VEOERHzP8e9GIryLEW7j72vUQOHtTX9UZJMx1BMxo8z+7i+mR9jsm8s5pK7Uc5a5n7aNfJhl5rRGwCnm/ABAADeQFNTU3z22WfR0dGRdikAcHQGvo2xvtkY/abUgDm2pyDq3+rbEDMxPDi53ddhICamhyqaVW+pal79zWjMDt3YmpqpdOzvKxpQfzcZUVpfz+5pm15nZrh6lEPEdgCyPe3Ra68vqsOJ+fHoH5x89Tl7r8fN3dfVXzvlE8Bp0Zh2AQAAcNwZ8QDAydMbI3fnIvqvxtXM6Payvhib2xoJMDM8uDUyoNxb+dsY67sag8M3Ivl2e9uxiMHS35FD05FUNpjePnb5r9Cq9fUMxI2hiMHBTEy+dtuIGJiIZHp419/RO/W/7vqidyKmhzIxeDUTo+V103Ht6uBWAFK3b8Nhrgvg5MokJqYFAIAjtb6+HouLi3Hp0qW0S2Gfbt++Hbdv346///u/T7sU4BBu374dd+7ciR9++CHtUojYGiVwdSpuzlVMWQR7+Prrr+OPf/xjXL9+Pe1SgKN137RLAABwhJIkiYcPH0Zzc3PapQAAAKRG+AAAAEcok8nExx9/HGfOnEm7FAA4mebHoz+Ticyen4q+DQCkRs8HAAA4Yo2N/pkN8C49ePAgbt++HQsLC2mXwraegZ74X//vYPyvt3b8gb2bU0e81XNztGZmZuLGjRtplwG8BX4rAgCAt6xQKMTi4mKcP38+7VIATqS1tbWIiLhy5UrKlQCHkc/n0y4BeAuEDwAA8BYlSRIPHjyIfD4fGxsb8cEHH6RdEsCJ88UXX8Qf/vCH+B//43+kXQpwQE+ePInPP/887TKAt0DPBwAAeIsymUycO3cuMplMvHz5Mh49ehRJkqRdFgAAwFslfAAAgLess7Mzrly5EplMJpaXl+Px48cCCAAA4EQTPgAAwDvQ0dEhgAAAAE4N4QMAALwjuwOIhw8fCiAAAIATSfgAAADvUEdHR3z44YeRyWQil8vFgwcPolgspl0WAADAkRI+AADAO9be3h4fffRRZLPZWF1djadPn6ZdEgAAwJESPgAAQAra2tri448/jra2trhw4ULa5QAAABypxrQLAACA06qlpSU+/vjjtMsAAAA4ckY+AADAe6ZQKKRdAsDJNj8e/ZnhmHnFJjPDmci8ZpujPB8AnDTCBwAAeI8sLi7Gw4cP0y4D4EQY79/jgX/vSNxNJmLgFfsOTCQx1ndEhezjfDXmx6N/WFwBwPElfAAAgPdEkiSxuroaH3zwQdqlAJxYWyMa9hjVMDO8vS4TmToP/ufH+yOTycRBMoG9zrd1rP7o7y+t74/x+Yo6ro7G7OTgK+sBgPeZ8AEAAN4TmUwmrly5Ek1NTWmXAnBiDUwkkSR1RjXMj0f/YMR0srV+OgZjdPbtna935G5MD83G7LXpSJIkkulrMfr9TGmnSObGom9oe12SRDJxoHETAJA64QMAALznVldX0y4B4PioGL0wOjsZg9tf95eHFdQ3f2cqrk3vTI00MDEdQ7u26R25G0mSxNHlAH0x9u32wQZuxNC9n4/qwACQOuEDAAC8x1ZWVuLXX3+NR48eRZIkaZcD8P4bmCiPFhjrGyqPZLg70pt2ZQBwqggfAADgPVYsFiOTycTy8nI8ePAgCoVC2iUBnEi912/GvR8r+irM/BiTu7Y5TM+HN1IeCTETw+/yvABwBIQPAADwHuvq6oqPPvoostlsrK6uxi+//BIbGxtplwVwTG09xK+ekmm70XPvSNyKigbPgxF9fZMx2D/+Vs43P94fg5OzMXp1OGZiJoYzgzE5O7rTWLp3JG5dG92uZzDujc0d4XRPAPD2ZRJjtwEA4L2Xz+fj4cOHsbm5GQ0NDXHlypVoa2tLu6wT4/bt23H79u34+7//+7RLAQ7h9u3bcefOnfjhhx/SLgU4oK+//jr++Mc/xvXr19MuBTha9418AACAY6ClpSU++eSTaGlpiUKhEA8ePIgXL16kXRYAAEBdwgcAADgmGhsb45NPPonOzs5IkiSePHkSz549S7ssAACAGsIHAAA4RjKZTFy5ciXOnTsXERELCwuRy+VSrgoAAKBaY9oFAAAAB3f+/Plobm6O9fX1aG9vT7scAACAKsIHAAA4prq6utIuAQAAoC7TLgEAwAmyubkZS0tLaZcBAACcckY+AADACZEkSTx69Mg0TIf0j//4j/EP//APaZcBHMK9e/eiUCikXQYAUEH4AAAAJ0Qmk4kzZ86YjukQzp07F5ubm/GnP/0p7VKAQ3j+/Hn8m3/zb9IuAwCoIHwAAIATpF7wkCRJZDKZFKo5Pv79v//3sbi4mHYZwCHdvn077ty5k3YZdc0MZ2JwMiJiKKaTiRhIu6D58ei/+lPcekUtWzW/Wb3j/ZkYjbFI7o68US0H9jaOCcCh6PkAAAAnWJIk8euvv8bCwkLapQC8c+P9wzGze+H8ePQP1yx9awYmkkiSJMb6Drjj26qzdyTuvubB/MDEIerdZeTudAwdQS0HdphjvuP/JgBOC+EDAACcYC9evIi1tbV49uxZPHr0KIrFYtolAaRnZjgyV0djdnIwMpnM1qfiofP8eP/O8kx/jM9HRMzEcHnZ1vbj/Tvrx/szkekfj5nhnW32+xy76nz9429Y534uv7RPnVBmZrjuuXaf86DP6Ctrrdx3v7X0j49v3f/hme1j9Ud/f/1r3+uYr9zvNfcagMMTPgAAwAnW09MTFy9ejEwmE8vLy/HLL7/E+vp62mUBpGNgIpK5segbmo4k2RqRkExsvyM/Px7fTN3cWT53M6a+GY+IgZiYG4u+GIrpJImxe4MxdXMukulrMXVnfusN/9nRGIzSMacjBvcXCPR+eSumt883d3Mq+ks7HarO/Vz+HqMw5sejfzDKtUzHYIzO7uuQrzY7GlfLtVbfl/3WcnNqNCb7xiKZGIjekbsxPTQbs9e278v0tRj9fico2OuYr9zvVfcagDcifAAAgBPuzJkz8dFHH0VjY2Osr6/H/fv3Y2lpKe2yAN6OirfmR2cnY7D8Bv2r04D5O1MxOzu68/b71dGYnZ3aCRGGbmxP5TMUt0Z6d+09FNPlB9YDMbEdTLzO/M/fleu7us+n/a+t8xDm70zFtemdqYoGJmqnTOoduRtJksTBnssPxXS558P+7svuWkb+PBbVWUJfjH27vXbgRgzd+3mftRx2PwAOS/gAAACnQFtbW3z66afR3t4eSZLEb7/9Fo8fPzYNE3DyDEyU32Af6xsqv0F/tyYwqNb75bXoG5vbefs9SSJJ7sZrdtvS99XB+xbMj8c3o9fK9SXTr+2Q8OZ1AsA7JHwAAIBToqGhIT766KM4f/58ZDKZePnypWmYgNOp/Nb7Vj+H4ZmIGLgR10a/r+0/sB+zo1UjK2Z+vBc3r+8jDSiPpogY/27y7de5h97rN+Pej5UNGX6M3dUcrufDZHxXcV/Gv3v9fem9fjPufbczjdTM96NxFDNAvVa9ew3AGxE+AADAKXPu3LmqaZiePXuWdkkA707vSNy6Vpq2aDDujc1tTyU0EBPTUZ4Gaadx8UwMbzck3goYJmNwuzn07OjVrYfUfWNxc+pqeb/BuLU9EmGnWfXOFFDbfQ96R+JW7DQ5noq+neMdqs7X2X8tmcGIvr6d6zyM8f7BmNx1X6Zu/nlf9+XPN6cq7uVQeQqo+fH+GJycjdGr2z+XzGBMzo5uN4je+5iv3u9V9xqAN5FJkiRJuwgAAODdKxQK8fTp07h48WI0NDSkXQ7Aod2+fTvu3LkTP/zwQwpnn4nh/p9jotzbgCM1Px79338Zd6UBJ9bXX38df/zjH+P69etplwIcrftGPgAAwCnV0NAQH3zwgeAB4A2M92+/RW+qnqNT0TQ8c3Uqbn4reAA4joQPAABAld9++y2Wl5fTLgPgWBi5u9P4OdWX8+fHo79qKqbqz7EKRiqahmumDXB8NaZdAAAA8P5YXV2NlZWVuHDhQtqlAHAQvSNxNzH1EwDvDyMfAACAsra2tvj0008jm/WrAgAAcHh+owAAAKrU6wGxsLAQxWIxhWoAAIDjSPgAAAC80tLSUjx79iz+9re/RS6XS7scAADgGBA+AAAAr9TS0hJNTU2xubkZDx48iN9++80oCAAA4JWEDwAAwCu1trbGp59+Gj09PRGxNRLi/v37sbq6mnJlAG9mZjgTmcxwzKRz9hjOZCIznM7Z0/Qu7vv8eH/0j89XnnXrfu9x3q2a6q/fc93M8K5zAFBJ+AAAALxWNpuNS5cuxUcffRSNjY2xsbERv/76azx9+jSSJEm7PIC6xvtf/YB7YCKJsb59Hmx+PPqPNCgYiIm5sdjv6U+SA933Q5mJ70evxa2R3vKS8f7BiOn6550f74/BmI4kSSKZjviuIlB41boYuBHXRr9PKbwCeP8JHwAAgH1rb2+Pzz77LLq7uyMiYnFxMf72t78ZBQEcHzPD22+x1446mB/v31nXP169z9XRmJ0crLvvnvsdqKzq/auOmemP8jPvUv2l8+z+vmLf/WUlpREBO8fY/ab/3tdX2rc/xmfGo79mdEDFsTP9MTw8HJXP7n8crnN9b3S+0mY/xuTQjRioWDRyN4mJgd0bbrkzdS2mSysHbsS1qTv7WhcxEDeGJqsDCQDKhA8AAMCBZLPZuHz5cnz44YflURC//fZb2mUBvN78eGy9AJ9EkiQxHYMxOruzuvfLW+V1czendqbUGZiIZG4s+oa234BPkkgqnmTvud8BXP1qKMbmkkjujkTMj8c3Uzd3zjV3M6a+GS/XMj3UF2N/Hqn//YENxESSxFjfUEzfHdk+5Pb3yUQMvPL6tvadHpqN0cGpuDlXPTqgNNqgdA33Ju9VnHcyJssjCq7F6Pc7EcJhz1cy8+Nk9H119dD346v4aY/RDLXrBm4MxexPc4c8F8DJJnwAAAAOpaOjozwK4vLly2mXA7ClYmTD6OxkDG5/3T8+H/N3puLa9ET5jfiBiekYqth1/ufvyttfrUwlXuOw+5WM92fi6tRXUZolaP7OVMzOju68/X91NGZnp8qjAwa+vRlTpYf18+PxXdyKihmGonfkbiTJ3m/61zNyq+Ih/sxwTN38tnyfXn99fTE2d3erhoGJuDvSGzE/HlPXpndq6B2Ju8ndijqHqkYUDN37uXy0Q52vws/3Iq592Vtnv7ekonYAdggfAACAQyuNgmhtba1aniRJFIvFlKoCTrWBifKIga2397e+3v2Ausb8eHwzeq28fTI99Ort33S/ksnBGJ3dGXUQEdH75bXoG5vbGfmQJJFUPrjvHYmb976L8fmtoOLajQOkDHup6F8w/l3s9Et40+s7qHd9vhoz8VN8FfXv6KvWAbCb8AEAADhyi4uL8eTJk7TLAKjSe/1m3PuxYtKcmR9jsnKDij4B499VrdlSfsN9q/dAuafC6/Z7laHpSKYjBit7G+yjkfHIrWsxdWc8vp+6WTPC4WA9H8onjW/H7sWPw9WjHrZqPMT1VQQkJTPDmf1NSfUm9zMivrwWce/n/U99df1mRHnz+Z8jbl7f17qya18euEaA00D4AAAAHLnl5eU4f/582mUAVOsdiVtR0TR6MKKvb3Lrwf+udVPRF7OjV3ce4PeOxK1rpamQBuPe2NzWQ//X7benmRi+OhrlSYW2p1na2m8gJqajPPVQpl5j5YGJuDk1GvfqPQw/9O25FTFZMeqhdN17Xd/8VtPnwcnZGL26M71Vycifb8bU1Z1rGIzpuDvSGzPD21Ni9Y9v3YfMYEzOjm418X6D85VvTZ0+DOP9u6fi2rmfvde/2qnz6lR8dX3n+l+1LmKrv8TQUYw8ATiBMkmSJGkXAQAAnHxJkkQmk0m7DOAEun37dty5cyd++OGHtEt5p8b7h+PLuxOmAaoxE8OZH+NG8rbvzbs6z8n29ddfxx//+Me4fv3ogjTgvXDfyAcAAOCd+P333+OXX36JfD6fdikAx9jWlE/lt/gPNrfSKbE1hdR3+5ni6U3M/Bj3xr4VPADsQfgAAAC8dcViMZaWlmJtbS1++eWXePr0qYbUwMm2PUVQZo/P4TODgZiobES9u+EDERHRO3L39U3G39TAxNs/B8Ax1ph2AQAAwMmXzWbjs88+i6dPn8by8nIsLi7G8vJyXLhwIbq6utIuD+Do9Y7E3WQk7SoAIDVGPgAAAO9EY2NjXLlyJT788MNoamqKzc3NePz4cfz666+xvr6ednkAAMAREj4AAADvVEdHR3z22Wdx/vz5yGQysbq6Gvfv349nz56ZigkAAE4I4QMAAPDOZTKZOHfuXHz22WfR0dERSZLEwsJCPHv2LO3SAACAI6DnAwAAkJqmpqb48MMPY2VlJX7//fc4d+5c2iUBAABHQPgAAACkrqOjIzo6OmqWFwqFiIhoaGh41yUBAABvwLRLAADAe+vp06exsLCQdhnACTUznIlMZjhm0i7kgLbqPp61v4l38fOaH++P/vH5yrPGcKZ0v/ujalW8+mex57qZ4V3nADiZhA8AAMB7KUmSKBQKpmICDm28/9UPqgcmkhjr2+fB5sejf/j9eNQ/MJFEkhyg9hPiQD+vQ5mJ70evxa2R3vKS8f7BiOmt+51MX4vRb8bL6+bH+2MwprfXRXxXESi8al0M3Ihro9+fquAIOJ2EDwAAwHspk8nERx99FNls9a8tGxsbkc/nU6oKOPZmhrffRs9EZleYMD/ev7Ouf7x6n6ujMTs5WHffPffbu4idt+m3t9/9lvzBj7lrZEDpOvurH5ZnXvEWf2n9/jKWN7mG0r79MT4zHv01owOqRxsMDw9X1frjcP1rOPz5Spv9GJNDN2KgYtHI3SQmSgsGbsRQxbo7U9diurRy4EZcm7qzr3URA3FjaLI6kAA4gYQPAADAsfLbb7/F/fv348mTJ7G5uZl2OcBxMj8eWy+yb73JPh2DMTq7s7r3y1vldXM3p3amxhmYiGRuLPqGtt9kT5JIJgZev9+eBmIiSWKsbyim745sn2L7+2QiBg51zF0jAwYmIkmmdx6Wz4/HN1M3d+qfuxlT3+wv1Dj6a9jad3poNkYHp+LmXPXogKrRBnM3497kvYrzTsZkeUTBtRj9fidCOOz5SmZ+nIy+r67uecXz499F3BrZ8358FT/tMZqhdt3AjaGY/Wluz3MBnATCBwAA4NhIkqQ8EuLFixfx17/+NX7//fcoFospVwa8NypGNozOTsbg9tf94/Mxf2cqrk1PlN9sH5iYrnqTff7n78rbX61MJV7jsPuN3Kp4AD4zHFM3vy3Xdthj7lnjnamYnR3dGRlwdTRmZ6eqRg70jtyNJKl40/+tX0NfjM3djZHeiBiYiLsjvRHz4zF1bXqnht6RuJtsbxMREUNVIwqG7v28c42HOV+Fn+9FXPuyt85+W6Mqrk7dPNC9ea2K2gFOIuEDAABwbGQymbhy5Up8/PHH0draGkmSxPPnz+Ovf/1rLC4uRpIkaZcIpG1govx2/9Zb+Ftf737QXGN+PL4ZvVbePpkeevX2b7pfRNXc/1sv1fe++TH30Pvltegbm9sZ+ZAkkVQ91D+kd3gNr/QWzzcznImrUzcjubvXqIeIiJn4Kb6K+tnEq9YBnFzCBwAA4Nhpa2uLTz75JK5cuRLNzc1RKBTi6dOn8be//S1evnyZdnnAe6r3+s2492PF5DczP8Zk5QYV8/2Pf1e1Zkv5TfWtHgLl3ggKl6f2AAAgAElEQVSv229PA/Ht2L34cbh6xMCbHfNe/Lw9EGG8f3Dn+vbR5PhgPR/KBz7aa+gdiZv3vqsakTEznNnXtFOHv2dbvrwWce/n6vOM92e2GkfXCR6u34zyvY75nyNuXt/XurJrXx64RoDjRPgAAAAcW52dnfHpp5/GpUuXorGxMTY2NuLx48fx66+/pl0a8D7qHYlbUdE0ejCir28yBvvHa9ZNRV/Mjl7deRDfOxK3rpWmLRqMe2NzW1PwvG6/15Z0K2KyYsRAnTqrj7nTjHlnWqmdxssjt67F6NXt/W6OxdDs6HZz7IGYmI7ytESVjaHf/LYe8Brmt5o+D07OlmutDBdG/nwzpq7u1DkY03F3pDdmhrevuX98+z4MxmTp+t7gfCU1fRjmx2NqNiIqG41X3LPe61/t1Hl1Kr66vnP9r1oXsdVfYuiGsRDAyZZJjEsGAABOgGKxGIuLi7GwsBBnz56Nc+fOpV0S8I78l//yX+JPf/pTDA8Pp10Kx9pf4x/+v/no/Y//T3x+Is5zPPy3//bf4r//9/8e/+E//Ie0SwGO1v3GtCsAAAA4CtlsNs6dOxc9PT2RyWRq1m9sbERTU1MKlQFv2+PHjyOXy8U//dM/pV0Kx1zXv2uPh//4T/Gk9q+Ro5MUo/3fdcWTf/qnePIWT3Nc5PP5ePr0adplAG+B8AEAADhRGhoaapatrq7G48eP4/PPP68bTADH27/6V/8q/vCHP8QPP/yQdinAAX399dfxL//lv0y7DOAt0PMBAAA48VZXV+Ps2bOCBwAAeEeMfAAAAE68vfo/PH/+PLq6ukzHBCfA/fv345//+Z/TLgM4oIcPH6ZdAvCWCB8AAIBTaWVlJX7//fd4/vx5dHd3x7lz56Kx0a9IcBw1NzfH//k//yf+7b/9t2mXAhzQwsKClwDghMokSZKkXQQAAMC7ls/n49mzZ5HL5SIiIpPJRFdXV5w7d85DEAAAeDP3hQ8AAMCptrq6Gr///nusrq5GxFYI0dnZGefOnYvm5uaUqwMAgGNJ+AAAABCxFUIsLCzEyspKeVkphGhpaUmxMgAAOHbum9AUAAAgItra2qKtrS3y+Xw8f/48lpeXY3l5Odrb24UPAABwQEY+AAAA1JHP52NpaSkuXrwYmUwm7XIAAOA4Me0SAADAQSwvL8fKykpcvnw57VIAAOB9dT+bdgUAAADHyeLiYrS1taVdBgAAvNeMfAAAADiAQqEQ2Wy2aiqmJElicXExuru7o6GhIcXqAADgvWDaJQAAgDf14sWLePLkSWQymeju7o6zZ89GU1NT2mUBAEBa7jemXQEAAMBx19jYGC0tLeUm1UtLS9HZ2Rlnz56N1tbWtMsDAIB3zsgHAACAI5LL5WJhYSFyuVx5WWtra5w5cyY6OzurpmoCAIATzLRLAAAARy2fz8fi4mK8fPkySr9yNTY2xoULF6Krqyvl6gAA4K0TPgAAALwthUIhlpaWYnFxMQqFQly5ciU6OzvTLgsAAN424QMAAMDbliRJLC8v10y9VCwWI5fLCSQAADhp7mfTrgAAAOCky2Qy0dXVVdPz4eXLl7G0tJRSVQAA8PYIHwAAAFKSJEmcPXu2ZvnGxkYK1QAAwNEx7RIAAMB7pFgsxl/+8pdoamqKnp6e6OrqimzWe2MAABwrej4AAAC8T1ZXV+PBgwdR+lUtm81GV1dX9PT0REtLS8rVAQDAvggfAAAA3jfFYjFevHgRS0tLsb6+Xl7e2toaZ86cqWlcDQAA7xnhAwAAwPssl8vF0tJSrKysVI2G6O7ujosXL6ZcHQAA1HW/Me0KAAAA2Ft7e3u0t7fH5uZmeTTE5uZmbG5upl0aAADsycgHAACAY2ZlZSUaGhqitbW1avnS0lJ0dHREY6P3zAAASNX9bNoVAAAAcDAdHR01wUOhUIhnz56lVBEAAFQTPgAAAJwASZLE+fPna0Y95PP5WF5eDoPeAQB4l0y7BAAAcII9efIkXrx4EdlsNrq6uqK7u7tm1AQAABwxDacBAABOsqampmhsbIzNzc1YWlqKpaWlaGpqiu7u7ujq6oqmpqa0SwQA4AQy8gEAAOAUyOVy8eLFi1hZWYlisVhe3tbWFl1dXdHV1RXZrJl5AQA4EveFDwAAAKdIsViMlZWVePHiReRyufLyixcvxpkzZ1KsDACAE8S0SwAAAKdJqfdDV1dXbG5uxsuXL+Ply5fR1dVVtV2SJLG6uhrt7e0pVQoAwHFm5AMAAAA1VlZW4vfff49PP/007VIAADh+7pvQEwAAgBqFQiG6u7trludyucjn8ylUBADAcWLaJQAAAGrUCx4iIp48eRKbm5vR3NwcXV1d0dnZGc3Nze+4OgAA3nemXQIAAGBfisViPHnyJFZWVqLyV8mWlpbo7OwURAAAUHJf+AAAAMCBFIvFWF5ejpcvX8bq6mpVENHc3BydnZ3R09MTjY0G2wMAnFLCBwAAAA6vUCjEyspKLC8vRy6XKwcRn332mVEQAACnl/ABAACAo1EsFmNlZSVWV1fj0qVLVevy+Xzkcrk4e/ZsStUBAPAO3TcGFgAAgCORzWajq6srurq6ata9fPkyvPsGAHB6CB8AAAB469rb2+v2gPj111+joaEhOjo6oqOjIxoaGlKoDgCAoyZ8AAAA4K1rb2+vWba5uRmrq6sREbG8vBwREW1tbdHZ2RkdHR3R1NT0TmsEAODo6PkAAABAatbW1mJlZSVWVlYin89XrWtubi4HEa2trSlVCADAIWg4DQAAwPthY2MjVlZWYnl5OdbW1qp6RHzyyScCCACA40P4AAAAwPunUChELpeL5eXlyOfz8fnnn1et39jYiFwuFz09PekUCADAqwgfAAAAOH4WFhYin8/HBx98kHYpAADUuq/hNAAAAMdOU1NT3WmYHj58GEmSREdHh6bVAAApMvIBAACAE6FYLMY///M/V/WKaGpqivb29vInm82mWCEAwKlh2iUAAABOjnw+H7lcLlZWVmqaVkdEtLa2Rnt7e3R3dxsVAQDw9ggfAAAAOJmKxWKsrq5GLpeLXC4X6+vr5XUffvhhdHR0pFgdAMCJpucDAAAAJ1M2my33foiI2NjYiFwuF6urq9HW1la17ebmZjx48CA+++yzNEoFADhxhA8AAACcCk1NTdHT0xM9PT0163K5XDQ21v6KvLq6GsViMdra2vSLAAA4AOEDAAAAp15nZ2fNaIiIiMXFxVheXo6IrX4RbW1t0d7eHq2trcIIAIBX0PMBAAAA9vDs2bNYXl6OjY2NquWZTKYmjMhkMilVCQDw3tFwGgAAAF5nc3Oz3C8il8vF5uZm1fru7u64fPlyStUBALx3NJwGAACA12lsbIzu7u7o7u6OiOrm1blcru6UTb/99lucOXMmmpub33W5AACpEz4AAADAAe1uXr17UoEkSeLFixdx/vz5quXFYrEcVjQ0NLyzegEA3jXhAwAAALyhev0ePvjgg5qAIZfLxaNHjyIiorm5udw3orW11QgJAOBE0fMBAAAA3pHl5eV4/vx55PP5mnUNDQ1VYUS9qZwAAI4JDacBAADgXSsUCrG2tharq6uxuroa+Xy+auqmbDYbX3zxRYoVAgC8EQ2nAQAA4F1raGiIjo6O6OjoiIitHhH5fD5WV1djbW2t7jROi4uLsbGxERcvXnzX5QIAHJjwAQAAAFKWyWSitbU1Wltb99xmbW2t7vqnT59GRJT3b2pqemt1AgDsl/ABAAAAjoFLly7VXf7ixYsoFovl70u9I1paWsqBxO7G1wAAb5ueDwAAAHBMJUkSL1++jHw+H2trazW9I0qampri/Pnz0dXVlUKVAMAppOcDAAAAHFeZTCa6u7vL35d6R6ytrZXDiPX19djY2KjbR+LZs2dx9uxZIyMAgCMnfAAAAIATol7viGKxWLdfRKFQiMXFxTh//nzV8o2NjVhZWYnW1tZobm6ObDb7TmoHAE4W4QMAAACcYNlsNtrb2+uuu3DhQs2IiFwuV25iHRHR3Nxc7h/R0tISLS0tAgkA4LX0fAAAAADKVlZWYmlpKdbW1qJQKNTdphRIXL58ue50TgDAqafnAwAAALCjo6MjOjo6IiJic3Ozqpl1Pp+Pzc3NWF9fj2KxWBM8rK6uxvr6evT09KRROgDwHhE+AAAAAHU1NjZGY2NjOYyI2OoVsba2FvUmUsjlclEsFmuWLy0tRZIk0dLSEs3NzRpcA8ApIHwAAAAA9q2hoaEqjKjU1tZWtx/E4uJirK+vl79vbGwsBxGlP5ubm03hBAAniJ4PAAAAwFv1/PnzWFtbi/X19djY2Nhzu7Nnz8aFCxfeYWUAwFui5wMAAADwdp07d678dbFYjHw+H+vr61V/FovFutMxPXjwIM6ePRvt7e3vsmQA4A0JHwAAAIB3JpvNRltbW7S1tVUt39zcrDvtUj6fj6ampqplhUIh/vrXv0ZTU1N5yqbSp6mpyfRNAPAeMO0SAAAA8N7a2NioCR9yuVw8ePCg7vaZTKYqlOju7q7ZHwB46+4LHwAAAIBjJUmS2NjYiPX19fKn9H2xWKza9tNPP42WlpaqfZ89exYXL15812UDwGmi5wMAAABwvGQymfLIht02NzerQondox7W19djZWWlJnxYWVmJFy9elKduKv1Zrw8FAPB6wgcAAADgxGhsbIzGxsY9G1Rns9k4e/ZszfK1tbVYXl6uWd7Q0FAVRjQ3N0dnZ+eR1w0AJ43wAQAAADg1mpqaoqenp2Z5Z2dnZLPZ8vRNGxsbsbm5GYVCIQqFQqytrUXEVnixO3zI5/Oxvr4eXV1d7+QaAOA4ED4AAAAAp15LS0tVb4iIiGKxGBsbG1X9JTKZTM2+uVwuNjY2asKH58+fl6d+qvw0NnocA8DJ5287AAAAgDqy2WzdUGK3vfpP5HK5WF1drVmeyWSqwoi2tjZTOQFw4ggfAAAAAN5AR0dH3eXnzp2LfD5fHj1RmsopSZLySIqIrSbZu8OHhYWFaGpqqlmeJEnd0RcA8L4RPgAAAAC8Be3t7TWNr5Mkic3NzapAot7IitXV1brTM/31r3+NJEnK0zdVTuVU+jObzb61awKA/RI+AAAAALwjlVMuvcrZs2drtikFFxERhUJhz32z2WycP38+zpw5U7V8fX1dOAHAOyN8AAAAAHjPtLW11SzLZDLxxRdf1EzjVPl1sViMYrFYd2qmR48exeXLl6O1tbW8rFgsxvPnz6tGTjQ2NkZDQ8NbvT4ATj7hAwAAAMAx8bom2MViMTY3N/cMD3aPptjc3IyFhYWa7TKZTDmIKIURPT09dRtrA0A9mSRJkrSLAAAAAODdK4UPm5ub5VEUe03p9NFHH9X0sPjLX/4Sn332WdVUToVCIdbW1oyiADjd7hv5AAAAAHBKNTY2xsWLF6uWlXpLlD6FQiE2NjZqRj0UCoUoFAo1PSTW1tbi4cOH5e8zmUw0NDSUg4jKP7u6uvSgADihhA8AAAAAlO23KXY2m41PP/207rqWlpZycFEZZuzW3t5eFT5sbGzEkydP4uOPP67artTXohRcGE0B8P4TPgAAAABwYJlMpm4PiI6Ojujo6IiIrVEUhUKhHERUjqbY3NyMxsbqR1Olptm7LS8vx7Nnz6qWlUKIytEUDQ0Ncfbs2boNtwF4t4QPAAAAALwVlY2r96OlpSUuX75cszybzUZzc3N5qqeIqPq68nznzp2rWrayshKLi4vx0UcfVS3P5XKRz+erQoxsNhuNjY2mggI4AsIHAAAAAN4L2Ww2Wlpaapb39PRET09PROyMptj92dzcjCRJavat15ciYms0xdLS0p61lAKJ5ubmuHLlStW6jY2NKBaLdWsFYIvwAQAAAIBj46CjKbq6usrTQFVqbW2NYrFYFWAUi8XytE/1RlaUrKysRD6frxml8dtvv8Xq6mpks9nySIrKPytHVwgugJNO+AAAAADAiZXJZOo2qO7u7o7u7u6a5btHVtSz1wiNjY2NWF9ff21Nra2t8cknn1QtW1xcjIiIM2fOVC1/+fJlFAqFcnBR+lR+D/A+Ej4AAAAAwLb9jKyoF1pERFy8eLFqFMXur0t/1mvUvbGxUTckWVxcjLW1tVfWnM1mo7Ozs2YkxvLycjQ0NERbW1vV8nw+X96vFGIAHDXhAwAAAAAcgXqhwn6dOXMmMplMzfL29vZobGysCjBKn1KPi8qvKy0vL0dbW1tN+PDo0aPY2NioWpbJZKpGVpQCjd0jMXK5XDQ2NtZc68bGRmSz2fJxAIQPAAAAAJCypqamusvPnz+/5z6VQUS9B/7t7e11A5GGhoaaAKNyuqmSelNLLS4uRldXV81x79+/X+6XERFVQUTl193d3dHV1VW178uXL6Otra1mtEk+n49MJlPeV7ABx4vwAQAAAACOodf1fNhreqjKfhNJklSFGJWfeoFIS0vLnkFJpcrG3ZVaW1trtl1cXIyGhoaa8OH+/ft1j707kDh37lzNtf7+++/R3d1dVWuSJPHixYuIiPIUV6X7VzpW6WM6KnhzwgcAAAAAOKVKD9rr9ZuoZ6+RGF988UVERNWIinpf1xtNUW/UQ7FYLE83lSRJ1bRSpe9LAUfliIuS5eXl6OzsrDnmb7/9tq/rvHTpUvT09FQtu3//fnzwwQdVoz4KhUI8evSoPGVWZWBReU+z2Wx0dXXVXP+LFy+is7Ozar9isVju81F5jFIwUnnMUmgCe9n9/8fuadpK/z/t/m+zUCjE5uZmzfJ8Pl/Th6beOTKZTLfwAQAAAAA4Eq8bjVHPhQsX6h7nX/yLf1G1rDLIqAwg6o3EOHPmTN2m4Z2dnVWhRek4uwONetewublZs7xQKMTq6uq+rrO1tbXmQe7Tp0+jo6OjatnGxkY8ePBgX8f88MMPa/afm5uLL774oqrWtbW1ePz4cdV2u6+lNJrk4sWLVQFLsViMJ0+exJUrV6q2z+fzsbCwsOfPu3L52bNnq4KUYrEYCwsLNWHW+vp6rKysvOqSy3p6eqrOUSgUYnl5uSY0yufzkcvlqrbby/nz56vCnI2NjVhZWanb+2R5eXlfdV66dKnq+/X19Xj58mXNtS8vL8eLFy9qgoHdD/aTJInPPvuspp6lpaWan9GzZ89iYWHhtTVms9lygFh5zJcvX8aHH35YtXxpaSmWlpZee8yGhoZO4QMAAAAA8N4rPWjezyiN3Q+gS/vtfjh7EJ988knNuRsbG8vHrHyoXXpgXPnwuF7/jfb29rojF1paWuoeo/R96QH1XqMedi9PkqSmyfhe6j2cr/egfXNzM16+fLmvY3Z1ddWED4uLizUP4PP5fDx79mzfx6wMHzY3N+P58+c1P/u1tbV9H/PcuXNV965YLMaLFy9qwod8Pr+vB/ARW+Ha7pBkZWWl5tpLQcdhlHq2HERlTfUCpIaGhrrBXnNzc7S3t9fdb9dInVwmqYxSAAAAAAA4tjY3N+tOY1UKH3Y/pN79Zn1bW1vVQ+RSr4zdD/U3NjZqQondU/qU7B75UCgU4vnz53Hx4sWq7dbW1sp9OfZSqv/SpUtVxyyFD7tHGqyuru4rJMlmszUjHwqFQrx8+bImfFhbWytf++vCsDNnzlQdc3NzM3K5XE2fksrpjHZPp7V7yq3Sw/+S0rRmu3/ulUFVCj1M7gsfAAAAAACAo3Rfy3YAAAAAAOBICR8AAAAAAIAjJXwAAAAAAACOlPABAAAAAAA4UsIHAAAAAADgSAkfAAAAAACAIyV8AAAAAAAAjpTwAQAAAAAAOFLCBwAAAAAA4EgJHwAAAAAAgCMlfAAAAAAAAI6U8AEAAAAAADhSwgcAAAAAAOBICR8AAAAAAIAjJXwAAAAAAACOlPABAAAAAAA4UsIHAAAAAADgSAkfAAAAAACAIyV8AAAAAAAAjpTwAQAAAAAAOFLCBwAAAAAA4EgJHwAAAAAAgCMlfAAAAAAAAI6U8AEAAAAAADhSwgcAAAAAAOBICR8AAAAAAIAjJXwAAAAAAACOlPABAAAAAAA4UsIHAAAAAADgSAkfAAAAAACAIyV8AAAAAAAAjpTwAQAAAAAAOFLCBwAAAAAA4EgJHwAAAAAAgCMlfAAAAAAAAI6U8AEAAAAAADhSwgcAAAAAAOBICR8AAAAAAIAjJXwAAAAAAACOlPABAAAAADiV5sf7I5PJVH36x+frbDkTw7u2q/8ZjplXHHuvbWvONly7ff269nWVMd6fiUz/eMwf8loOWtfOdv2xd9nbdWUykRne605UHbT63OXr2Y/d17z3vefoCB8AAAAAgFNm62H01dHZmjWzo1ff0cPpyRisOc9WXYOTtVvPjl494AP3o3TYumZj6s4eFc/fiana219/0/H+yOw++exoXN3Pz2l+PPozg1G9d717z1ETPgAAAAAAp8h8jPdvP4zuG4u5JImk/JmLsb6IiMkYrHqgPhAT9bar2X8iBqrONRTTVet3PtNDW+f5rmJowPz4dzEZEX1jc/W3nx2Nbw49AuLw13K4uoZiaChidupO3WBi/s5UzA4NxdBr652J70dna+or3b8fX5MgzHw/GrO7aq937zl6wgcAAAAA4NSYH/8mtp5lz0VydyR6q9b2xsjdJObG+o7oQf/eBiamax68z/00GxFDcWukd8/tZ3+a21m4PRVR1axF9Za9oQPXte3GjaGI2amoHfwwH3emZmPoxo19nH07LNn1sxr4diz6IuLez6/6Gc3Ej1upSfy5ovaBia3AZXbqzj7Oz2EJHwAAAACAU+PO1N4P0kt6R25tPVDf4639ozAzvHsqoIirX/XF3m/kbz+Enxios+7tOnRdAzdiqN7US/N3Ymp2KG68waVsjWjoi5vX9/45xvzPcS8i+m5erwmZrt/si5j96fAF8FrCBwAAAADg1Nh6if9GvPq590Dc2HqdP2rf5z+IyRjco6HzVguD6hCkHHqMXt1fs+mBiUiSJKqe+9db9oYOXNdOMXFjqHZUxNaUS6/7GdSz06R6cLIvxubuxisypLJrX+610b0DV8D+CR8AAAAAAN61oek6PSK2RhFM75qPqfzQP7WG04eva+DGUMTkjxXNnUtTLh0mHZnbCo+2zh6jV/vjlfnH3E+xz57WvAXCBwD+//buXSmxbQsD8L+q9qOAgcUTwBOAiZHpziBkJWaGZiYQQtZpRyYHnkCewDJoiPZDnHSdYCHirbeX1XpKv6+Kqsa5LqM0m3+POQAAAIA/4sHA6dvd++4k69+0JvRne/es69kGSZJVmXaTwxxe6U119Y8z3B8M/a4jl/aHZS8yzCrl378JZNqHdzXy4YQPAAAAAMC3UY8v2P+f+E+5HVR8mHaTL+/P6gBiVab90i6G1jhXVZWqqock/3vtH+TFddVHL8236cPbj1x6/NzTemr0EwOt73t+KHXn3VXwPOEDAAAAAPBtHJ38bnhybTM9zzxPDSpuQH9WH1+0KvP3fg2baXpFkeLZzoZWDj5jr7yBuu6OXnrDkUvb9//7fImnSjtIJ08NDq/rSPfw9c/kxYQPAAAAAMC30Rr/yKS7nVfwqPugHmjcLldJd5IfL5lm/Ab92WI7wPnirlugdZQ6Fxk8vdG/HNVDqpvuxvg3TdTVPkw381xO33Dk0vb9935X9YtzUa6S7kmOnv0z3Q4Ovx/0LEft1LcevaIQXkv4AAAAAAB8I62Mr+rN/6zKtIuiHppcFCmKelM6GWZxNW6+62Fne2RQ5hnsNvRbGf/YzlGYD/Zq2n7qHf5MfuzVtRylKIrcywSe+tm7vKGuR4+oA4R5Wb7hyKVWxmfDJPMM7r17kPkLfh/907r23XDsoqjDkgxz9ofCJWrCBwAAAADgm6kHF68nj8cRdyfrVNWsgZkEv3fbgZH54G6zfDtH4XYu9YPCsq6u8in75e+uq7U97iqvO3LpVn92f8B1/aQsXvL7aI1zVW3Dpnv3/vm/8XdXVFVVfXYRAAAAAADAl/GPzgcAAAAAAKBRwgcAAAAAAKBRwgcAAAAAAKBRwgcAAAAAAKBRwgcAAAAAAKBRwgcAAAAAAKBRwgcAAAAAAKBRwgcAAAAAAKBRwgcAAAAAAKBRwgcAAAAAAKBRwgcAAAAAAKBRwgcAAAAAAN6kV4yy/KyXb6Z3799M0yuKFI8+vUw3H1gHO399dgEAAAAAAPB+wyyqWfqfXQZJdD4AAAAAAPBum0x7vUyno13XQW+6SZYPvidJlhkVo0ynvWe6E5YZ7ToX9joKNtP0eqOMetu1dplV5hm8oOtgObp7/2baSzFa/pE6uCN8AAAAAACgAauUN8epqirVepKU7RSX+98v9kKCecqfJ1lXVapFJ2X7bnN/ORrkerKu71skg940u0xgNU/Oqt0zuy/sdujPFumUF1lmmYuyk8Xs9o5m6+CO8AEAAAAAgAZ0Mzndbuq3jnLS3f9+kE6u82uvs2B4Nk4rSfqnmXRv15a5nA9zNm7VF/WPM1z9zH929w1z/GzSMM/g4cyHXWDQz+nkOoNikOvJ6b2wovk6SIQPAAAAAAB8uG4O27f/buWgs8rNOsnmV67vXdfOYfelzxxmUW27EW4/V9tgIUnr6CTddHNy1Nq750/UQSJ8AAAAAADgU23y63obArQO0rm3ts7Nqpm3LC/KdIadlBfPTYj4mDq+C+EDAAAAAAAfbHUXAiwvUuYkdUNCP8fDec5vJz8vLzPv3q69w2aa8+tJTmenmVyf7w2W/uA6vpG/PrsAAAAAAAC+m26GOU9RDJJ0M1nPdscj9WeLXBbtFGVSH6V0d3TSPa2jnHTLDIpksT5MPfNh/uiy4WKdw/MynbMqrSTjs06Kv6c5ujpovI7K0Omdoqqq6rOLAAAAAADgu1hmVJzncH2V8ad2Evy/1PEl/ePYJQAAAAAAoFE6HwAAAAAAgCbpfAAAAAAAAJolfAAAAAAAABolfAAAAAAAABolfAAAAAAAABolfL1oqpwAAAGCSURBVAAAAAAAABolfAAAAAAAABolfAAAAAAAABolfAAAAAAAABolfAAAAAAAABolfAAAAAAAABolfAAAAAAAABolfAAAAAAAABolfAAAAAAAABolfAAAAAAAABolfAAAAAAA4OtajlIURYretP6+maa3930z7aUoeplu8uT6Iw/Xm77+ixA+AAAAAADwhR1nUVWZ5CbLJJv/3ORkXaU6O8gySWt8lfWks7v64fpyVNThxTageLie9U06iyrrk5tMN6+7/isTPgAAAAAA8HX1+/nVK/Lz5DT9vR9vfl3m1+bZu3br/VmVqqpSVVcZt564v3+cDIq0y+Sg9brrvzLhAwAAAAAAX9ZyVKRcJavyou50OEjKdpH2z8OMW/WxS+1ynrJddyo8XH/o0fryMvPfvP+1138VRVVV1WcXAQAAAAAAfBn/6HwAAAAAAAAaJXwAAAAAAAAaJXwAAAAAAAAaJXwAAAAAAAAaJXwAAAAAAAAa9VeSk88uAgAAAAAA+DL++z8W4DZrQvElpAAAAABJRU5ErkJggg==
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSIgPz4KPG90cnNfY29uZmlnIHZlcnNpb249IjEuMCIgaW5pdD0iRnJhbWV3b3JrIj4KICAgIDxDVlM+JElkOiBJbXBvcnRFeHBvcnQueG1sLHYgMS44IDIwMTAvMDkvMTQgMjA6NTg6MjEgZHogRXhwICQ8L0NWUz4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZyb250ZW5kOjpNb2R1bGUjIyNBZG1pbkltcG9ydEV4cG9ydCIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gVHJhbnNsYXRhYmxlPSIxIj5Gcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgYWdlbnQgaW50ZXJmYWNlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkltcG9ydEV4cG9ydDwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZyb250ZW5kOjpBZG1pbjo6TW9kdWxlUmVnaXN0cmF0aW9uPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEZyb250ZW5kTW9kdWxlUmVnPgogICAgICAgICAgICAgICAgPEdyb3VwPmFkbWluPC9Hcm91cD4KICAgICAgICAgICAgICAgIDxEZXNjcmlwdGlvbiBUcmFuc2xhdGFibGU9IjEiPkltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi48L0Rlc2NyaXB0aW9uPgogICAgICAgICAgICAgICAgPFRpdGxlPkltcG9ydC9FeHBvcnQ8L1RpdGxlPgogICAgICAgICAgICAgICAgPE5hdkJhck5hbWU+QWRtaW48L05hdkJhck5hbWU+CiAgICAgICAgICAgICAgICA8TmF2QmFyTW9kdWxlPgogICAgICAgICAgICAgICAgICAgIDxNb2R1bGU+S2VybmVsOjpPdXRwdXQ6OkhUTUw6Ok5hdkJhck1vZHVsZUFkbWluPC9Nb2R1bGU+CiAgICAgICAgICAgICAgICAgICAgPE5hbWUgVHJhbnNsYXRhYmxlPSIxIj5JbXBvcnQvRXhwb3J0PC9OYW1lPgogICAgICAgICAgICAgICAgICAgIDxCbG9jaz5TeXN0ZW08L0Jsb2NrPgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjcxMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyTW9kdWxlPgogICAgICAgICAgICAgICAgPExvYWRlcj4KICAgICAgICAgICAgICAgICAgICA8Q1NTPklUU00uSW1wb3J0RXhwb3J0LmNzczwvQ1NTPgogICAgICAgICAgICAgICAgPC9Mb2FkZXI+CiAgICAgICAgICAgIDwvRnJvbnRlbmRNb2R1bGVSZWc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iSW1wb3J0RXhwb3J0OjpGb3JtYXRCYWNrZW5kUmVnaXN0cmF0aW9uIyMjQ1NWIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBUcmFuc2xhdGFibGU9IjEiPkZvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+PC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+SW1wb3J0RXhwb3J0PC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Rm9ybWF0QmFja2VuZDo6TW9kdWxlUmVnaXN0cmF0aW9uPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik1vZHVsZSI+S2VybmVsOjpTeXN0ZW06OkltcG9ydEV4cG9ydDo6Rm9ybWF0QmFja2VuZDo6Q1NWPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJOYW1lIj5DU1Y8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+Cjwvb3Ryc19jb25maWc+
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9iZ19JbXBvcnRFeHBvcnQucG0gLSB0aGUgYnVsZ2FyaWFuIHRyYW5zbGF0aW9uIG9mIEltcG9ydEV4cG9ydAojIENvcHlyaWdodCAoQykgMjAwMS0yMDEwIE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyBDb3B5cmlnaHQgKEMpIDIwMDctMjAwOCBNaWxlbiBLb3V0ZXYKIyAtLQojICRJZDogYmdfSW1wb3J0RXhwb3J0LnBtLHYgMS4xNCAyMDEwLzA5LzE0IDIxOjQ5OjE0IGR6IEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmJnX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuMTQgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQnfSAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQgTWFuYWdlbWVudCd9ICAgPSAnJzsKICAgICRMYW5nLT57J0FkZCBtYXBwaW5nIHRlbXBsYXRlJ30gICAgICAgPSAnJzsKICAgICRMYW5nLT57J1N0YXJ0IEltcG9ydCd9ICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1N0YXJ0IEV4cG9ydCd9ICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1N0ZXAnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gICAgPSAnJzsKICAgICRMYW5nLT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gICAgPSAnJzsKICAgICRMYW5nLT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gICAgPSAnJzsKICAgICRMYW5nLT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ICAgPSAnJzsKICAgICRMYW5nLT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gICAgPSAnJzsKICAgICRMYW5nLT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0NvbHVtbid9ICAgICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnJzsKICAgICRMYW5nLT57J1NvdXJjZSBGaWxlJ30gICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0NvbHVtbiBTZXBhcmF0b3InfSAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1RhYnVsYXRvciAoVEFCKSd9ICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1NlbWljb2xvbiAoOyknfSAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0NvbG9uICg6KSd9ICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0RvdCAoLiknfSAgICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0NoYXJzZXQnfSAgICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0Zyb250ZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBhZ2VudCBpbnRlcmZhY2UuJ30gPSAnJzsKICAgICRMYW5nLT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9ICcnOwogICAgJExhbmctPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CiAgICAkTGFuZy0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJExhbmctPnsnQ2xhc3MgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J0NvbHVtbiBTZXBhcmF0b3IgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICcnOwogICAgJExhbmctPnsnRW1wdHkgZmllbGRzIGluZGljYXRlIHRoYXQgdGhlIGN1cnJlbnQgdmFsdWVzIGFyZSBrZXB0J30gPSAnJzsKICAgICRMYW5nLT57J0NyZWF0ZSBhIHRlbXBsYXRlIGluIG9yZGVyIHRvIGNhbiBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKCiAgICByZXR1cm4gMTsKfQoKMTsK
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9jdF9JbXBvcnRFeHBvcnQucG0gLSB0aGUgY2F0YWxhbiB0cmFuc2xhdGlvbiBvZiBJbXBvcnRFeHBvcnQKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgQ29weXJpZ2h0IChDKSAyMDA4IFNpc3RlbWVzIE9USUMgKGlic2FsdXQpIC0gQW50b25pbyBMaW5kZQojIC0tCiMgJElkOiBjdF9JbXBvcnRFeHBvcnQucG0sdiAxLjcgMjAxMC8wOS8xNCAyMTo0OToxNCBkeiBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpjdF9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjcgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQnfSAgICAgICAgICAgICAgPSAnSW1wb3J0YXIvRXhwb3J0YXInOwogICAgJExhbmctPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gICA9ICdHZXN0afIgZGUgSW1wb3J0YXIvRXhwb3J0YXInOwogICAgJExhbmctPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSAgICAgICA9ICdBZmVnaXIgcGxhbnRpbGxhIGRlIG1hcGF0Z2UnOwogICAgJExhbmctPnsnU3RhcnQgSW1wb3J0J30gICAgICAgICAgICAgICA9ICdDb21lbudhciBpbXBvcnRhY2nzJzsKICAgICRMYW5nLT57J1N0YXJ0IEV4cG9ydCd9ICAgICAgICAgICAgICAgPSAnQ29tZW7nYXIgZXhwb3J0YWNp8yc7CiAgICAkTGFuZy0+eydTdGVwJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ1Bhcyc7CiAgICAkTGFuZy0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ICAgID0gJ0VkaXRhciBpbmZvcm1hY2nzIGNvbXVuYSc7CiAgICAkTGFuZy0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ICAgID0gJ0VkaXRhciBpbmZvcm1hY2nzIGRcJ29iamVjdGUnOwogICAgJExhbmctPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSAgICA9ICdFZGl0YXIgaW5mb3JtYWNp8yBkZSBmb3JtYXQnOwogICAgJExhbmctPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gICA9ICdFZGl0YXIgaW5mb3JtYWNp8yBkZSBtYXBhdGdlJzsKICAgICRMYW5nLT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gICAgPSAnRWRpdGFyIGluZm9ybWFjafMgZGUgcmVjZXJjYSc7CiAgICAkTGFuZy0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSAgICAgICAgID0gJ0ltcG9ydGFyIGluZm9ybWFjafMnOwogICAgJExhbmctPnsnQ29sdW1uJ30gICAgICAgICAgICAgICAgICAgICA9ICdDb2x1bW5hJzsKICAgICRMYW5nLT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnUmVzdHJpbmdpciBleHBvcnRhY2nzIHBlciByZWNlcmNhJzsKICAgICRMYW5nLT57J1NvdXJjZSBGaWxlJ30gICAgICAgICAgICAgICAgPSAnRml0eGVyIGZvbnQnOwogICAgJExhbmctPnsnQ29sdW1uIFNlcGFyYXRvcid9ICAgICAgICAgICA9ICdTZXBhcmFkb3IgZGUgY29sdW1uYSc7CiAgICAkTGFuZy0+eydUYWJ1bGF0b3IgKFRBQiknfSAgICAgICAgICAgID0gJ1RhYnVsYWRvciAoVEFCKSc7CiAgICAkTGFuZy0+eydTZW1pY29sb24gKDspJ30gICAgICAgICAgICAgID0gJ1B1bnQgaSBjb21hICg7KSc7CiAgICAkTGFuZy0+eydDb2xvbiAoOiknfSAgICAgICAgICAgICAgICAgID0gJ0RvcyBwdW50cyAoOiknOwogICAgJExhbmctPnsnRG90ICguKSd9ICAgICAgICAgICAgICAgICAgICA9ICdQdW50ICguKSc7CiAgICAkTGFuZy0+eydDaGFyc2V0J30gICAgICAgICAgICAgICAgICAgID0gJ0Nvbmp1bnQgZGUgY2Fy4GN0ZXJzJzsKICAgICRMYW5nLT57J0Zyb250ZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBhZ2VudCBpbnRlcmZhY2UuJ30gPSAnJzsKICAgICRMYW5nLT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9ICcnOwogICAgJExhbmctPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CiAgICAkTGFuZy0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJExhbmctPnsnQ2xhc3MgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J0NvbHVtbiBTZXBhcmF0b3IgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICcnOwogICAgJExhbmctPnsnRW1wdHkgZmllbGRzIGluZGljYXRlIHRoYXQgdGhlIGN1cnJlbnQgdmFsdWVzIGFyZSBrZXB0J30gPSAnJzsKICAgICRMYW5nLT57J0NyZWF0ZSBhIHRlbXBsYXRlIGluIG9yZGVyIHRvIGNhbiBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKCiAgICByZXR1cm4gMTsKfQoKMTsK
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9jel9JbXBvcnRFeHBvcnQucG0gLSB0aGUgY3plY2ggdHJhbnNsYXRpb24gb2YgSW1wb3J0RXhwb3J0CiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTAgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIENvcHlyaWdodCAoQykgMjAxMCBPMkJTLmNvbSwgcyByLm8uIEpha3ViIEhhbnVzCiMgLS0KIyAkSWQ6IGN6X0ltcG9ydEV4cG9ydC5wbSx2IDEuMTIgMjAxMC8wOS8xNCAyMTo0OToxNCBkeiBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpjel9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjEyICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAkTGFuZy0+eydJbXBvcnQvRXhwb3J0J30gICAgICAgICAgICAgID0gJ0ltcG9ydC9FeHBvcnQnOwogICAgJExhbmctPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gICA9ICdJbXBvcnQvRXhwb3J0IFNwcuF2YSc7CiAgICAkTGFuZy0+eydBZGQgbWFwcGluZyB0ZW1wbGF0ZSd9ICAgICAgID0gJ05vduEguWFibG9uYSB6b2JyYXplbu0nOwogICAgJExhbmctPnsnU3RhcnQgSW1wb3J0J30gICAgICAgICAgICAgICA9ICdaYWjhaml0IEltcG9ydCc7CiAgICAkTGFuZy0+eydTdGFydCBFeHBvcnQnfSAgICAgICAgICAgICAgID0gJ1phaOFqaXQgRXhwb3J0JzsKICAgICRMYW5nLT57J1N0ZXAnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnS3Jvayc7CiAgICAkTGFuZy0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ICAgID0gJ0VkaXRhY2Ugb2JlY279Y2ggaW5mb3JtYWPtJzsKICAgICRMYW5nLT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gICAgPSAnRWRpdGFjZSBpbmZvcm1hY+0gbyBvYmpla3R1JzsKICAgICRMYW5nLT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gICAgPSAnRWRpdGFjZSBmb3Jt4XR1JzsKICAgICRMYW5nLT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ICAgPSAnRWRpdGFjZSBtYXBvduFu7Sc7CiAgICAkTGFuZy0+eydFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ICAgID0gJ0VkaXRhY2UgdnlobGVk4Xbhbu0nOwogICAgJExhbmctPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gICAgICAgICA9ICdJbmZvcm1hY2UgbyBJbXBvcnR1JzsKICAgICRMYW5nLT57J0NvbHVtbid9ICAgICAgICAgICAgICAgICAgICAgPSAnU2xvdXBlYyc7CiAgICAkTGFuZy0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ09tZXppdCBFeHBvcnQgdnlobGVk4Xbhbu1tJzsKICAgICRMYW5nLT57J1NvdXJjZSBGaWxlJ30gICAgICAgICAgICAgICAgPSAnWmRyb2pvdv0gU291Ym9yJzsKICAgICRMYW5nLT57J0NvbHVtbiBTZXBhcmF0b3InfSAgICAgICAgICAgPSAnT2Rk7GxvdmHoIFNsb3VwY/knOwogICAgJExhbmctPnsnVGFidWxhdG9yIChUQUIpJ30gICAgICAgICAgICA9ICdUYWJ1bOF0b3IgKFRBQiknOwogICAgJExhbmctPnsnU2VtaWNvbG9uICg7KSd9ICAgICAgICAgICAgICA9ICdTdPhlZG7tayAoOyknOwogICAgJExhbmctPnsnQ29sb24gKDopJ30gICAgICAgICAgICAgICAgICA9ICdEdm9qdGXoa2EgKDopJzsKICAgICRMYW5nLT57J0RvdCAoLiknfSAgICAgICAgICAgICAgICAgICAgPSAnVGXoa2EgKC4pJzsKICAgICRMYW5nLT57J0NoYXJzZXQnfSAgICAgICAgICAgICAgICAgICAgPSAnWm5ha2924SBzYWRhJzsKICAgICRMYW5nLT57J0Zyb250ZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBhZ2VudCBpbnRlcmZhY2UuJ30gPSAnJzsKICAgICRMYW5nLT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9ICcnOwogICAgJExhbmctPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CiAgICAkTGFuZy0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJExhbmctPnsnQ2xhc3MgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J0NvbHVtbiBTZXBhcmF0b3IgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICcnOwogICAgJExhbmctPnsnRW1wdHkgZmllbGRzIGluZGljYXRlIHRoYXQgdGhlIGN1cnJlbnQgdmFsdWVzIGFyZSBrZXB0J30gPSAnJzsKICAgICRMYW5nLT57J0NyZWF0ZSBhIHRlbXBsYXRlIGluIG9yZGVyIHRvIGNhbiBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKCiAgICByZXR1cm4gMTsKfQoKMTsK
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9kYV9JbXBvcnRFeHBvcnQucG0gLSBwcm92aWRlcyBkYSAoRGFuaXNoKSBsYW5ndWFnZSB0cmFuc2xhdGlvbgojIENvcHlyaWdodCAoQykgMjAwMS0yMDEwIE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogZGFfSW1wb3J0RXhwb3J0LnBtLHYgMS40IDIwMTAvMDkvMTQgMjE6NDk6MTQgZHogRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6ZGFfSW1wb3J0RXhwb3J0OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS40ICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAkTGFuZy0+eydJbXBvcnQvRXhwb3J0J30gICAgICAgICAgICAgID0gJ0ltcG9ydC9Fa3BvcnQnOwogICAgJExhbmctPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gICA9ICdJbXBvcnQvRWtwb3J0IHN0eXJpbmcnOwogICAgJExhbmctPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSAgICAgICA9ICdUaWxm+GogTWFwcGluZy1UZW1wbGF0ZSc7CiAgICAkTGFuZy0+eydTdGFydCBJbXBvcnQnfSAgICAgICAgICAgICAgID0gJ1N0YXJ0IGltcG9ydCc7CiAgICAkTGFuZy0+eydTdGFydCBFeHBvcnQnfSAgICAgICAgICAgICAgID0gJ1N0YXJ0IGVrcG9ydCc7CiAgICAkTGFuZy0+eydTdGVwJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ1RyaW4nOwogICAgJExhbmctPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSAgICA9ICdSZXQgZuZsbGVzIGluZm9ybWF0aW9uJzsKICAgICRMYW5nLT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gICAgPSAnUmV0IG9iamVrdCBpbmZvcm1hdGlvbic7CiAgICAkTGFuZy0+eydFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ICAgID0gJ1JldCBmb3JtYXQgaW5mb3JtYXRpb24nOwogICAgJExhbmctPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gICA9ICdSZXQgbWFwcGluZyBpbmZvcm1hdGlvbic7CiAgICAkTGFuZy0+eydFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ICAgID0gJ1JldCBz+GdlaW5mb3JtYXRpb24nOwogICAgJExhbmctPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gICAgICAgICA9ICdJbXBvcnQgaW5mb3JtYXRpb24nOwogICAgJExhbmctPnsnQ29sdW1uJ30gICAgICAgICAgICAgICAgICAgICA9ICdLb2xvbm5lJzsKICAgICRMYW5nLT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnQmVncuZucyBla3BvcnQgcHIuIHP4Z25pbmcnOwogICAgJExhbmctPnsnU291cmNlIEZpbGUnfSAgICAgICAgICAgICAgICA9ICdLaWxkZSBmaWwnOwogICAgJExhbmctPnsnQ29sdW1uIFNlcGVyYXRvcid9ICAgICAgICAgICA9ICdLb2xvbm5lIGFkc2tpbGxlbHNlJzsKICAgICRMYW5nLT57J1RhYnVsYXRvciAoVEFCKSd9ICAgICAgICAgICAgPSAnVGFidWxhdG9yIChUQUIpJzsKICAgICRMYW5nLT57J1NlbWljb2xvbiAoOyknfSAgICAgICAgICAgICAgPSAnU2VtaWtvbG9uICg7KSc7CiAgICAkTGFuZy0+eydDb2xvbiAoOiknfSAgICAgICAgICAgICAgICAgID0gJ0tvbG9uICg6KSc7CiAgICAkTGFuZy0+eydEb3QgKC4pJ30gICAgICAgICAgICAgICAgICAgID0gJ1B1bmt0dW0gKC4pJzsKICAgICRMYW5nLT57J0NoYXJzZXQnfSAgICAgICAgICAgICAgICAgICAgPSAnVGVnbnPmdCc7CiAgICAkTGFuZy0+eydGcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgYWdlbnQgaW50ZXJmYWNlLid9ID0gJyc7CiAgICAkTGFuZy0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPSAnJzsKICAgICRMYW5nLT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwogICAgJExhbmctPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J0NsYXNzIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydDb2x1bW4gU2VwYXJhdG9yIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnJzsKICAgICRMYW5nLT57J0VtcHR5IGZpZWxkcyBpbmRpY2F0ZSB0aGF0IHRoZSBjdXJyZW50IHZhbHVlcyBhcmUga2VwdCd9ID0gJyc7CiAgICAkTGFuZy0+eydDcmVhdGUgYSB0ZW1wbGF0ZSBpbiBvcmRlciB0byBjYW4gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9kZV9JbXBvcnRFeHBvcnQucG0gLSB0aGUgZ2VybWFuIHRyYW5zbGF0aW9uIG9mIEltcG9ydEV4cG9ydAojIENvcHlyaWdodCAoQykgMjAwMS0yMDEwIE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogZGVfSW1wb3J0RXhwb3J0LnBtLHYgMS4yNSAyMDEwLzA5LzE0IDIxOjQ5OjE0IGR6IEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmRlX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuMjUgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQnfSAgICAgICAgICAgICAgPSAnSW1wb3J0L0V4cG9ydCc7CiAgICAkTGFuZy0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSAgID0gJ0ltcG9ydC9FeHBvcnQgVmVyd2FsdHVuZyc7CiAgICAkTGFuZy0+eydBZGQgbWFwcGluZyB0ZW1wbGF0ZSd9ICAgICAgID0gJ01hcHBpbmctVGVtcGxhdGUgaGluenVm77+9Z2VuJzsKICAgICRMYW5nLT57J1N0YXJ0IEltcG9ydCd9ICAgICAgICAgICAgICAgPSAnSW1wb3J0IHN0YXJ0ZW4nOwogICAgJExhbmctPnsnU3RhcnQgRXhwb3J0J30gICAgICAgICAgICAgICA9ICdFeHBvcnQgc3RhcnRlbic7CiAgICAkTGFuZy0+eydTdGVwJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ1NjaHJpdHQnOwogICAgJExhbmctPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSAgICA9ICdBbGxnZW1laW5lIEluZm9ybWF0aW9uZW4gYmVhcmJlaXRlbic7CiAgICAkTGFuZy0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ICAgID0gJ09iamVrdC1JbmZvcm1hdGlvbmVuIGJlYXJiZWl0ZW4nOwogICAgJExhbmctPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSAgICA9ICdGb3JtYXQtSW5mb3JtYXRpb25lbiBiZWFyYmVpdGVuJzsKICAgICRMYW5nLT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ICAgPSAnTWFwcGluZy1JbmZvcm1hdGlvbmVuIGJlYXJiZWl0ZW4nOwogICAgJExhbmctPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSAgICA9ICdTdWNoLUluZm9ybWF0aW9uZW4gYmVhcmJlaXRlbic7CiAgICAkTGFuZy0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSAgICAgICAgID0gJ0ltcG9ydCBJbmZvcm1hdGlvbmVuJzsKICAgICRMYW5nLT57J0NvbHVtbid9ICAgICAgICAgICAgICAgICAgICAgPSAnU3BhbHRlJzsKICAgICRMYW5nLT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnRXhwb3J0IHBlciBTdWNoZSBlaW5zY2hy77+9bmtlbic7CiAgICAkTGFuZy0+eydTb3VyY2UgRmlsZSd9ICAgICAgICAgICAgICAgID0gJ1F1ZWxsLURhdGVpJzsKICAgICRMYW5nLT57J0NvbHVtbiBTZXBhcmF0b3InfSAgICAgICAgICAgPSAnU3BhbHRlbnRyZW5uZXInOwogICAgJExhbmctPnsnVGFidWxhdG9yIChUQUIpJ30gICAgICAgICAgICA9ICdUYWJ1bGF0b3IgKFRBQiknOwogICAgJExhbmctPnsnU2VtaWNvbG9uICg7KSd9ICAgICAgICAgICAgICA9ICdTZW1pY29sb24gKDspJzsKICAgICRMYW5nLT57J0NvbG9uICg6KSd9ICAgICAgICAgICAgICAgICAgPSAnRG9wcGVscHVua3QgKDopJzsKICAgICRMYW5nLT57J0RvdCAoLiknfSAgICAgICAgICAgICAgICAgICAgPSAnUHVua3QgKC4pJzsKICAgICRMYW5nLT57J0NoYXJzZXQnfSAgICAgICAgICAgICAgICAgICAgPSAnWmVpY2hlbnNhdHonOwogICAgJExhbmctPnsnRnJvbnRlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGFnZW50IGludGVyZmFjZS4nfSA9ICcnOwogICAgJExhbmctPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0gJ0Zvcm1hdC1CYWNrZW5kIE1vZHVsIFJlZ2lzdHJhdGlvbiBkZXMgSW1wb3J0L0V4cG9ydCBNb2R1bHMuJzsKICAgICRMYW5nLT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwogICAgJExhbmctPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J0NsYXNzIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydDb2x1bW4gU2VwYXJhdG9yIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnJzsKICAgICRMYW5nLT57J0VtcHR5IGZpZWxkcyBpbmRpY2F0ZSB0aGF0IHRoZSBjdXJyZW50IHZhbHVlcyBhcmUga2VwdCd9ID0gJyc7CiAgICAkTGFuZy0+eydDcmVhdGUgYSB0ZW1wbGF0ZSBpbiBvcmRlciB0byBjYW4gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9lc19JbXBvcnRFeHBvcnQucG0gLSB0aGUgc3BhbmlzaCB0cmFuc2xhdGlvbiBvZiBJbXBvcnRFeHBvcnQKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgQ29weXJpZ2h0IChDKSAyMDA4IEFxdWlsZXMgQ29oZW4KIyAtLQojICRJZDogZXNfSW1wb3J0RXhwb3J0LnBtLHYgMS4xMiAyMDEwLzEwLzAxIDIxOjU5OjU1IGR6IEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmVzX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuMTIgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQnfSAgICAgICAgICAgICAgPSAnSW1wb3J0YXIvRXhwb3J0YXInOwogICAgJExhbmctPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gICA9ICdHZXN0afNuIGRlIEltcG9ydGFjafNuL0V4cG9ydGFjafNuJzsKICAgICRMYW5nLT57J0FkZCBtYXBwaW5nIHRlbXBsYXRlJ30gICAgICAgPSAnQfFhZGlyIHBsYW50aWxsYSBkZSBtYXBlbyc7CiAgICAkTGFuZy0+eydTdGFydCBJbXBvcnQnfSAgICAgICAgICAgICAgID0gJ0luaWNpYXIgSW1wb3J0YWNp824nOwogICAgJExhbmctPnsnU3RhcnQgRXhwb3J0J30gICAgICAgICAgICAgICA9ICdJbmljaWFyIEV4cG9ydGFjafNuJzsKICAgICRMYW5nLT57J1N0ZXAnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnUGFzbyc7CiAgICAkTGFuZy0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ICAgID0gJ0VkaXRhciBpbmZvcm1hY2nzbiBjb236bic7CiAgICAkTGFuZy0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ICAgID0gJ0VkaXRhciBpbmZvcm1hY2nzbiBkZSBvYmpldG8nOwogICAgJExhbmctPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSAgICA9ICdFZGl0YXIgaW5mb3JtYWNp824gZGVsIGZvcm1hdG8nOwogICAgJExhbmctPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gICA9ICdFZGl0YXIgaW5mb3JtYWNp824gZGUgbWFwZW8nOwogICAgJExhbmctPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSAgICA9ICdFZGl0YXIgaW5mb3JtYWNp824gZGUgYvpzcXVlZGEnOwogICAgJExhbmctPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gICAgICAgICA9ICdJbXBvcnRhciBpbmZvcm1hY2nzbic7CiAgICAkTGFuZy0+eydDb2x1bW4nfSAgICAgICAgICAgICAgICAgICAgID0gJ0NvbHVtbmEnOwogICAgJExhbmctPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdSZXN0cmluZ2lyIGV4cG9ydGFjafNuIHBvciBi+nNxdWVkYSc7CiAgICAkTGFuZy0+eydTb3VyY2UgRmlsZSd9ICAgICAgICAgICAgICAgID0gJ0FyY2hpdm8gb3JpZ2VuJzsKICAgICRMYW5nLT57J0NvbHVtbiBTZXBhcmF0b3InfSAgICAgICAgICAgPSAnU2VwYXJhZG9yIGRlIENvbHVtbmEnOwogICAgJExhbmctPnsnVGFidWxhdG9yIChUQUIpJ30gICAgICAgICAgICA9ICdUYWJ1bGFkb3IgKFRBQiknOwogICAgJExhbmctPnsnU2VtaWNvbG9uICg7KSd9ICAgICAgICAgICAgICA9ICdQdW50byB5IENvbWEgKDspJzsKICAgICRMYW5nLT57J0NvbG9uICg6KSd9ICAgICAgICAgICAgICAgICAgPSAnRG9zIHB1bnRvcyAoOiknOwogICAgJExhbmctPnsnRG90ICguKSd9ICAgICAgICAgICAgICAgICAgICA9ICdQdW50byAoLiknOwogICAgJExhbmctPnsnQ29tbWEgKCwpJ30gICAgICAgICAgICAgICAgICA9ICdDb21hICgsKSc7CiAgICAkTGFuZy0+eydDaGFyc2V0J30gICAgICAgICAgICAgICAgICAgID0gJ0p1ZWdvIGRlIGNhcmFjdGVyZXMnOwogICAgJExhbmctPnsnRnJvbnRlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGFnZW50IGludGVyZmFjZS4nfSA9ICdSZWdpc3RybyBkZSBt82R1bG8gZnJvbnRlbmQgcGFyYSBsYSBpbnRlcmZheiBkZWwgYWdlbnRlLic7CiAgICAkTGFuZy0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPSAnUmVnaXN0cm8gZGUgbfNkdWxvIGRlIGZvcm1hdG8gYmFja2VuZCBwYXJhIGVsIG3zZHVsbyBpbXBvcnQvZXhwb3J0Lic7CiAgICAkTGFuZy0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnSW1wb3J0YXIgeSBleHBvcnRhciBpbmZvcm1hY2nzbiBkZSBvYmpldG9zLic7CiAgICAkTGFuZy0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnoURlYmUgZXNwZWNpZmljYXIgT2JqZXRvISc7CiAgICAkTGFuZy0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnoURlYmUgZXNwZWNpZmljYXIgRm9ybWF0byEnOwogICAgJExhbmctPnsnQ2xhc3MgaXMgcmVxdWlyZWQhJ30gPSAnoURlYmUgZXNwZWNpZmljYXIgQ2xhc2UhJzsKICAgICRMYW5nLT57J0NvbHVtbiBTZXBhcmF0b3IgaXMgcmVxdWlyZWQhJ30gPSAnoURlYmUgZXNwZWNpZmljYXIgU2VwYXJhZG9yIGRlIENvbHVtbmEhJzsKICAgICRMYW5nLT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICdObyBzZSBlbmNvbnRyYXJvbiBlbGVtZW50b3MgZGUgbWFwZW8uJzsKICAgICRMYW5nLT57J0VtcHR5IGZpZWxkcyBpbmRpY2F0ZSB0aGF0IHRoZSBjdXJyZW50IHZhbHVlcyBhcmUga2VwdCd9ID0gJ0xvcyBjYW1wb3MgdmFj7W9zIGluZGljYW4gcXVlIGxvcyB2YWxvcmVzIGFjdHVhbGVzIHNlIG1hbnRpZW5lbic7CiAgICAkTGFuZy0+eydDcmVhdGUgYSB0ZW1wbGF0ZSBpbiBvcmRlciB0byBjYW4gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0FncmVndWUgdW5hIHBsYW50aWxsYSBudWV2YSBwYXJhIHBvZGVyIGltcG9ydGFyIHkgZXhwb3J0YXIuJzsKICAgICRMYW5nLT57J0dvIGJhY2snfSAgICAgICAgICAgICAgICAgICAgPSAnUmVncmVzYXInOwoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9mYV9JbXBvcnRFeHBvcnQucG0gLSB0aGUgcGVyc2lhbiAoZmFyc2kpIHRyYW5zbGF0aW9uIG9mIGZhX0ltcG9ydEV4cG9ydAojIENvcHlyaWdodCAoQykgMjAwMS0yMDEwIE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyBDb3B5cmlnaHQgKEMpIDIwMDMtMjAwOSBBZnNoYXIgTW9oZWJiaSA8YWZzaGFyLm1vaGViYmkgYXQgZ21haWwuY29tPgojIC0tCiMgJElkOiBmYV9JbXBvcnRFeHBvcnQucG0sdiAxLjUgMjAxMC8wOS8xNCAyMTo0OToxNCBkeiBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpmYV9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjUgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQnfSAgICAgICAgICAgICAgPSAn2YjYsdmI2K8v2LXYr9mI2LEnOwogICAgJExhbmctPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gICA9ICfZhdiv24zYsduM2Kog2YjYsdmI2K8v2LXYr9mI2LEnOwogICAgJExhbmctPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSAgICAgICA9ICfYp9i22KfZgdmHINqp2LHYr9mGINuM2qkg2YLYp9mE2Kgg2Ybar9in2LTYqic7CiAgICAkTGFuZy0+eydTdGFydCBJbXBvcnQnfSAgICAgICAgICAgICAgID0gJ9i02LHZiNi5INi52YXZhNuM2KfYqiDZiNix2YjYryc7CiAgICAkTGFuZy0+eydTdGFydCBFeHBvcnQnfSAgICAgICAgICAgICAgID0gJ9i02LHZiNi5INi52YXZhNuM2KfYqiDYtdiv2YjYsSc7CiAgICAkTGFuZy0+eydTdGVwJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ9qv2KfZhSc7CiAgICAkTGFuZy0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ICAgID0gJ9mI24zYsdin24zYtCDYp9i32YTYp9i52KfYqiDYudmF2YjZhduMJzsKICAgICRMYW5nLT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gICAgPSAn2YjbjNix2KfbjNi0INin2LfZhNin2LnYp9iqINii2KjYrNqp2KrbjCc7CiAgICAkTGFuZy0+eydFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ICAgID0gJ9mI24zYsdin24zYtCDYp9i32YTYp9i52KfYqiDZgtin2YTYqOKAjNio2YbYr9uMJzsKICAgICRMYW5nLT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ICAgPSAn2YjbjNix2KfbjNi0INin2LfZhNin2LnYp9iqINmG2q/Yp9i02KonOwogICAgJExhbmctPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSAgICA9ICfZiNuM2LHYp9uM2LQg2KfYt9mE2KfYudin2Kog2KzYs9iq2KzZiCc7CiAgICAkTGFuZy0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSAgICAgICAgID0gJ9mI2LHZiNivINin2LfZhNin2LnYp9iqJzsKICAgICRMYW5nLT57J0NvbHVtbid9ICAgICAgICAgICAgICAgICAgICAgPSAn2LPYqtmI2YYnOwogICAgJExhbmctPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICfZhdit2K/ZiNiv2LPYp9iy24wg2LnZhdmE24zYp9iqINi12K/ZiNixINio2Ycg2KfYstin24wg2KzYs9iq2KzZiCc7CiAgICAkTGFuZy0+eydTb3VyY2UgRmlsZSd9ICAgICAgICAgICAgICAgID0gJ9mB2KfbjNmEINmF2YbYqNi5JzsKICAgICRMYW5nLT57J0NvbHVtbiBTZXBhcmF0b3InfSAgICAgICAgICAgPSAn2KzYr9in2qnZhtmG2K/ZhyDYs9iq2YjZhuKAjNmH2KcnOwogICAgJExhbmctPnsnVGFidWxhdG9yIChUQUIpJ30gICAgICAgICAgICA9ICfYrNiv2YjZhCDYs9in2LIgKFRBQiknOwogICAgJExhbmctPnsnU2VtaWNvbG9uICg7KSd9ICAgICAgICAgICAgICA9ICfYs9mF24wg2qnYp9mE2YYgKDspJzsKICAgICRMYW5nLT57J0NvbG9uICg6KSd9ICAgICAgICAgICAgICAgICAgPSAn2K/ZiNmG2YLYt9mHICg6KSc7CiAgICAkTGFuZy0+eydEb3QgKC4pJ30gICAgICAgICAgICAgICAgICAgID0gJ9mG2YLYt9mHICguKSc7CiAgICAkTGFuZy0+eydDaGFyc2V0J30gICAgICAgICAgICAgICAgICAgID0gJ9qp2K/YqNmG2K/bjCDYp9i32YTYp9i52KfYqic7CiAgICAkTGFuZy0+eydGcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgYWdlbnQgaW50ZXJmYWNlLid9ID0gJyc7CiAgICAkTGFuZy0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPSAnJzsKICAgICRMYW5nLT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwogICAgJExhbmctPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J0NsYXNzIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydDb2x1bW4gU2VwYXJhdG9yIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnJzsKICAgICRMYW5nLT57J0VtcHR5IGZpZWxkcyBpbmRpY2F0ZSB0aGF0IHRoZSBjdXJyZW50IHZhbHVlcyBhcmUga2VwdCd9ID0gJyc7CiAgICAkTGFuZy0+eydDcmVhdGUgYSB0ZW1wbGF0ZSBpbiBvcmRlciB0byBjYW4gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9mcl9JbXBvcnRFeHBvcnQucG0gLSB0aGUgZnJlbmNoIHRyYW5zbGF0aW9uIG9mIEltcG9ydEV4cG9ydAojIENvcHlyaWdodCAoQykgMjAwMS0yMDA5IE9saXZpZXIgU2FsbG91IDxvbGl2aWVyLnNhbGxvdSBhdCBpcmlzYS5mcj4KIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IGZyX0ltcG9ydEV4cG9ydC5wbSx2IDEuNiAyMDEwLzA5LzE0IDIxOjQ5OjE0IGR6IEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmZyX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuNiAkKSBbMV07CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgIG15ICRMYW5nID0gJFNlbGYtPntUcmFuc2xhdGlvbn07CgogICAgcmV0dXJuIGlmIHJlZiAkTGFuZyBuZSAnSEFTSCc7CgogICAgJExhbmctPnsnSW1wb3J0L0V4cG9ydCd9ICAgICAgICAgICAgICA9ICdJbXBvcnRlci9FeHBvcnRlcic7CiAgICAkTGFuZy0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSAgID0gJ0dlc3Rpb24gZGUgbFwnSW1wb3J0L0V4cG9ydCc7CiAgICAkTGFuZy0+eydBZGQgbWFwcGluZyB0ZW1wbGF0ZSd9ICAgICAgID0gJ0Fqb3V0ZXIgdW4gdGVtcGxhdGUgZGUgbWFwcGFnZSc7CiAgICAkTGFuZy0+eydTdGFydCBJbXBvcnQnfSAgICAgICAgICAgICAgID0gJ0TpbWFycmVyIEltcG9ydCc7CiAgICAkTGFuZy0+eydTdGFydCBFeHBvcnQnfSAgICAgICAgICAgICAgID0gJ0TpbWFycmVyIEV4cG9ydCc7CiAgICAkTGFuZy0+eydTdGVwJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ0V0YXBlJzsKICAgICRMYW5nLT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gICAgPSAnRWRpdGVyIGxlcyBpbmZvcm1hdGlvbnMgY29tbXVuZXMnOwogICAgJExhbmctPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSAgICA9ICdFZGl0ZXIgbGVzIGluZm9ybWF0aW9ucyBkZSBsXCdvYmpldCc7CiAgICAkTGFuZy0+eydFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ICAgID0gJ0VkaXRlciBsZXMgaW5mb3JtYXRpb25zIGRlIGZvcm1hdCc7CiAgICAkTGFuZy0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSAgID0gJ0VkaXRlciBsZXMgaW5mb3JtYXRpb25zIGRlIG1hcHBhZ2UnOwogICAgJExhbmctPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSAgICA9ICdFZGl0ZXIgbGVzIGluZm9ybWF0aW9ucyBkZSByZWNoZXJjaGUnOwogICAgJExhbmctPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gICAgICAgICA9ICdJbmZvcm1hdGlvbnMgZFwnaW1wb3J0JzsKICAgICRMYW5nLT57J0NvbHVtbid9ICAgICAgICAgICAgICAgICAgICAgPSAnQ29sb25uZSc7CiAgICAkTGFuZy0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ1Jlc3RyZWluZHJlIGxcJ2V4cG9ydCBwYXIgcmVjaGVyY2hlJzsKICAgICRMYW5nLT57J1NvdXJjZSBGaWxlJ30gICAgICAgICAgICAgICAgPSAnRmljaGllciBTb3VyY2UnOwogICAgJExhbmctPnsnQ29sdW1uIFNlcGFyYXRvcid9ICAgICAgICAgICA9ICdT6XBhcmF0ZXVyIGRlIGNvbG9ubmUnOwogICAgJExhbmctPnsnVGFidWxhdG9yIChUQUIpJ30gICAgICAgICAgICA9ICdUYWJ1bGF0aW9uIChUQUIpJzsKICAgICRMYW5nLT57J1NlbWljb2xvbiAoOyknfSAgICAgICAgICAgICAgPSAnUG9pbnQgdmlyZ3VsZSAoOyknOwogICAgJExhbmctPnsnQ29sb24gKDopJ30gICAgICAgICAgICAgICAgICA9ICdEZXV4IHBvaW50cyAoOiknOwogICAgJExhbmctPnsnRG90ICguKSd9ICAgICAgICAgICAgICAgICAgICA9ICdQb2ludCAoLiknOwogICAgJExhbmctPnsnQ2hhcnNldCd9ICAgICAgICAgICAgICAgICAgICA9ICdKZXUgZGUgY2FyYWN06HJlcyc7CiAgICAkTGFuZy0+eydGcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgYWdlbnQgaW50ZXJmYWNlLid9ID0gJyc7CiAgICAkTGFuZy0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPSAnJzsKICAgICRMYW5nLT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwogICAgJExhbmctPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J0NsYXNzIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydDb2x1bW4gU2VwYXJhdG9yIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnJzsKICAgICRMYW5nLT57J0VtcHR5IGZpZWxkcyBpbmRpY2F0ZSB0aGF0IHRoZSBjdXJyZW50IHZhbHVlcyBhcmUga2VwdCd9ID0gJyc7CiAgICAkTGFuZy0+eydDcmVhdGUgYSB0ZW1wbGF0ZSBpbiBvcmRlciB0byBjYW4gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9pdF9JbXBvcnRFeHBvcnQucG0gLSB0aGUgaXRhbGlhbiB0cmFuc2xhdGlvbiBvZiBJbXBvcnRFeHBvcnQKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IGl0X0ltcG9ydEV4cG9ydC5wbSx2IDEuNSAyMDEwLzA5LzE0IDIxOjQ5OjE0IGR6IEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6Oml0X0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuNSAkKSBbMV07CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgIG15ICRMYW5nID0gJFNlbGYtPntUcmFuc2xhdGlvbn07CgogICAgcmV0dXJuIGlmIHJlZiAkTGFuZyBuZSAnSEFTSCc7CgogICAgJExhbmctPnsnSW1wb3J0L0V4cG9ydCd9ICAgICAgICAgICAgICA9ICdJbXBvcnRhcmUvRXNwb3J0YXJlJzsKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQgTWFuYWdlbWVudCd9ICAgPSAnR2VzdGlvbmUgSW1wb3J0YXppb25lL0VzcG9ydGF6aW9uZSc7CiAgICAkTGFuZy0+eydBZGQgbWFwcGluZyB0ZW1wbGF0ZSd9ICAgICAgID0gJ0FnZ2l1bmdpIG1hcHBhdHVyYSBtb2RlbGxvJzsKICAgICRMYW5nLT57J1N0YXJ0IEltcG9ydCd9ICAgICAgICAgICAgICAgPSAnSW5pemlhcmUgSW1wb3J0YXppb25lJzsKICAgICRMYW5nLT57J1N0YXJ0IEV4cG9ydCd9ICAgICAgICAgICAgICAgPSAnSW5pemlhcmUgRXNwb3J0YXppb25lJzsKICAgICRMYW5nLT57J1N0ZXAnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnUGFzc28nOwogICAgJExhbmctPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSAgICA9ICdNb2RpZmljYSBpbmZvcm1hemlvbmkgY29tdW5pJzsKICAgICRMYW5nLT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gICAgPSAnTW9kaWZpY2EgaW5mb3JtYXppb25pIG9nZ2V0dG8nOwogICAgJExhbmctPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSAgICA9ICdNb2RpZmljYSBmb3JtYXRvIGluZm9ybWF6aW9uZSc7CiAgICAkTGFuZy0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSAgID0gJ01vZGlmaWNhIG1hcHBhdHVyYSBpbmZvcm1hemlvbmknOwogICAgJExhbmctPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSAgICA9ICdNb2RpZmljYSBpbmZvcm1hemlvbmkgZGkgcmljZXJjYSc7CiAgICAkTGFuZy0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSAgICAgICAgID0gJ0ltcG9ydGFyZSBpbmZvcm1hemlvbmUnOwogICAgJExhbmctPnsnQ29sdW1uJ30gICAgICAgICAgICAgICAgICAgICA9ICdDb2xvbm5hJzsKICAgICRMYW5nLT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnUmVzdHJpbmdlcmUgZXNwb3J0YXppb25lIHBlciByaWNlcmNhJzsKICAgICRMYW5nLT57J1NvdXJjZSBGaWxlJ30gICAgICAgICAgICAgICAgPSAnQXJjaGl2aW8gb3JpZ2luZSc7CiAgICAkTGFuZy0+eydDb2x1bW4gU2VwYXJhdG9yJ30gICAgICAgICAgID0gJ1NlcGFyYXRvcmUgZGkgY29sb25uYSc7CiAgICAkTGFuZy0+eydUYWJ1bGF0b3IgKFRBQiknfSAgICAgICAgICAgID0gJ1RhYnVsYXRvcmUgKFRBQiknOwogICAgJExhbmctPnsnU2VtaWNvbG9uICg7KSd9ICAgICAgICAgICAgICA9ICdQdW50byBlIHZpcmdvbGEgKDspJzsKICAgICRMYW5nLT57J0NvbG9uICg6KSd9ICAgICAgICAgICAgICAgICAgPSAnRHVlIHB1bnRpICg6KSc7CiAgICAkTGFuZy0+eydEb3QgKC4pJ30gICAgICAgICAgICAgICAgICAgID0gJ1B1bnRvICguKSc7CiAgICAkTGFuZy0+eydDaGFyc2V0J30gICAgICAgICAgICAgICAgICAgID0gJ0NoYXJzZXQnOwogICAgJExhbmctPnsnRnJvbnRlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGFnZW50IGludGVyZmFjZS4nfSA9ICcnOwogICAgJExhbmctPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0gJyc7CiAgICAkTGFuZy0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKICAgICRMYW5nLT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJExhbmctPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydDbGFzcyBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJExhbmctPnsnQ29sdW1uIFNlcGFyYXRvciBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJExhbmctPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJyc7CiAgICAkTGFuZy0+eydFbXB0eSBmaWVsZHMgaW5kaWNhdGUgdGhhdCB0aGUgY3VycmVudCB2YWx1ZXMgYXJlIGtlcHQnfSA9ICcnOwogICAgJExhbmctPnsnQ3JlYXRlIGEgdGVtcGxhdGUgaW4gb3JkZXIgdG8gY2FuIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9ubF9JbXBvcnRFeHBvcnQucG0gLSB0aGUgRHV0Y2ggdHJhbnNsYXRpb24gb2YgSW1wb3J0RXhwb3J0CiMgQ29weXJpZ2h0IChDKSAyMDA5IE1pY2hpZWwgQmVpamVuIDxtaWNoaWVsICdhdCcgYmVlZnJlZWl0Lm5sPgojIC0tCiMgJElkOiBubF9JbXBvcnRFeHBvcnQucG0sdiAxLjcgMjAxMC8wOS8yNCAwODo1ODoxNSBtYiBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpubF9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjcgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQnfSAgICAgICAgICAgICAgPSAnSW1wb3J0L0V4cG9ydCc7CiAgICAkTGFuZy0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSAgID0gJ0ltcG9ydC9FeHBvcnQgYmVoZWVyJzsKICAgICRMYW5nLT57J0FkZCBtYXBwaW5nIHRlbXBsYXRlJ30gICAgICAgPSAnTWFwcGluZ3RlbXBsYXRlIHRvZXZvZWdlbic7CiAgICAkTGFuZy0+eydTdGFydCBJbXBvcnQnfSAgICAgICAgICAgICAgID0gJ0ltcG9ydCBzdGFydGVuJzsKICAgICRMYW5nLT57J1N0YXJ0IEV4cG9ydCd9ICAgICAgICAgICAgICAgPSAnRXhwb3J0IHN0YXJ0ZW4nOwogICAgJExhbmctPnsnU3RlcCd9ICAgICAgICAgICAgICAgICAgICAgICA9ICdTdGFwJzsKICAgICRMYW5nLT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gICAgPSAnQWxnZW1lbmUgaW5mb3JtYXRpZSBiZXdlcmtlbic7CiAgICAkTGFuZy0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ICAgID0gJ09iamVjdC1pbmZvcm1hdGllIGJld2Vya2VuJzsKICAgICRMYW5nLT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gICAgPSAnRm9ybWF0LWluZm9ybWF0aW9uZW4gYmV3ZXJrZW4nOwogICAgJExhbmctPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gICA9ICdNYXBwaW5nLWluZm9ybWF0aWUgYmV3ZXJrZW4nOwogICAgJExhbmctPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSAgICA9ICdab2VrLWluZm9ybWF0aWUgYmV3ZXJrZW4nOwogICAgJExhbmctPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gICAgICAgICA9ICdJbXBvcnQtaW5mb3JtYXRpZSc7CiAgICAkTGFuZy0+eydDb2x1bW4nfSAgICAgICAgICAgICAgICAgICAgID0gJ0tvbG9tJzsKICAgICRMYW5nLT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnQmVwZXJrIGV4cG9ydCB0b3Qgem9la29wZHJhY2h0JzsKICAgICRMYW5nLT57J1NvdXJjZSBGaWxlJ30gICAgICAgICAgICAgICAgPSAnQnJvbmJlc3RhbmQnOwogICAgJExhbmctPnsnQ29sdW1uIFNlcGFyYXRvcid9ICAgICAgICAgICA9ICdLb2xvbXNjaGVpZGluZ3N0ZWtlbic7CiAgICAkTGFuZy0+eydUYWJ1bGF0b3IgKFRBQiknfSAgICAgICAgICAgID0gJ1RhYic7CiAgICAkTGFuZy0+eydTZW1pY29sb24gKDspJ30gICAgICAgICAgICAgID0gJ1B1bnRrb21tYSAoOyknOwogICAgJExhbmctPnsnQ29sb24gKDopJ30gICAgICAgICAgICAgICAgICA9ICdEdWJiZWxlIHB1bnQgKDopJzsKICAgICRMYW5nLT57J0RvdCAoLiknfSAgICAgICAgICAgICAgICAgICAgPSAnUHVudCAoLiknOwogICAgJExhbmctPnsnQ2hhcnNldCd9ICAgICAgICAgICAgICAgICAgICA9ICdLYXJha3RlcnNldCc7CiAgICAkTGFuZy0+eydGcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgYWdlbnQgaW50ZXJmYWNlLid9ID0gJyc7CiAgICAkTGFuZy0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPSAnJzsKICAgICRMYW5nLT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdJbXBvcnQgZW4gZXhwb3J0IG9iamVjdGluZm9ybWF0aWUnOwogICAgJExhbmctPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ09iamVjdCBpcyB2ZXJwbGljaHQuJzsKICAgICRMYW5nLT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICdGb3JtYWF0IGlzIHZlcnBsaWNodC4nOwogICAgJExhbmctPnsnQ2xhc3MgaXMgcmVxdWlyZWQhJ30gPSAnS2xhc3NlIGlzIHZlcnBsaWNodC4nOwogICAgJExhbmctPnsnQ29sdW1uIFNlcGFyYXRvciBpcyByZXF1aXJlZCEnfSA9ICdTY2hlaWRpbmdzdGVrZW4gaXMgdmVycGxpY2h0JzsKICAgICRMYW5nLT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICdHZWVuIGVsZW1lbnRlbiBnZXZvbmRlbi4nOwogICAgJExhbmctPnsnRW1wdHkgZmllbGRzIGluZGljYXRlIHRoYXQgdGhlIGN1cnJlbnQgdmFsdWVzIGFyZSBrZXB0J30gPSAnQmlqIGxlZ2UgdmVsZGVuIHdvcmR0IGRlIGh1aWRpZ2Ugd2FhcmRlIGJlaG91ZGVuJzsKICAgICRMYW5nLT57J0NyZWF0ZSBhIHRlbXBsYXRlIGluIG9yZGVyIHRvIGNhbiBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnTWFhayBlZW4gdGVtcGxhdGUgYWFuIG9tIG9iamVjdGVuIHRlIGltcG9ydGVyZW4gb2YgZXhwb3J0ZXJlbi4nOwoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9wbF9JbXBvcnRFeHBvcnQucG0gLSB0aGUgcG9saXNoIHRyYW5zbGF0aW9uIG9mIEltcG9ydEV4cG9ydAojIENvcHlyaWdodCAoQykgMjAwMS0yMDEwIE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyBDb3B5cmlnaHQgKEMpIDIwMDggTWFjaWVqIExvc3phamMKIyAtLQojICRJZDogcGxfSW1wb3J0RXhwb3J0LnBtLHYgMS44IDIwMTAvMDkvMTQgMjE6NDk6MTQgZHogRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6cGxfSW1wb3J0RXhwb3J0OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS44ICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAkTGFuZy0+eydJbXBvcnQvRXhwb3J0J30gICAgICAgICAgICAgID0gJ0ltcG9ydC9FeHBvcnQnOwogICAgJExhbmctPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gICA9ICdaYXJ6MWR6YW5pZSBJbXBvcnRlbS9FeHBvcnRlbSc7CiAgICAkTGFuZy0+eydBZGQgbWFwcGluZyB0ZW1wbGF0ZSd9ICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydTdGFydCBJbXBvcnQnfSAgICAgICAgICAgICAgID0gJ1JvenBvY3puaWogSW1wb3J0JzsKICAgICRMYW5nLT57J1N0YXJ0IEV4cG9ydCd9ICAgICAgICAgICAgICAgPSAnUm96cG9jem5paiBFeHBvcnQnOwogICAgJExhbmctPnsnU3RlcCd9ICAgICAgICAgICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSAgICA9ICcnOwogICAgJExhbmctPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSAgICA9ICcnOwogICAgJExhbmctPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSAgICA9ICcnOwogICAgJExhbmctPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gICA9ICcnOwogICAgJExhbmctPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSAgICA9ICcnOwogICAgJExhbmctPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gICAgICAgICA9ICcnOwogICAgJExhbmctPnsnQ29sdW1uJ30gICAgICAgICAgICAgICAgICAgICA9ICdLb2x1bW5hJzsKICAgICRMYW5nLT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnJzsKICAgICRMYW5nLT57J1NvdXJjZSBGaWxlJ30gICAgICAgICAgICAgICAgPSAnUGxpayBZcvNkM293eSc7CiAgICAkTGFuZy0+eydDb2x1bW4gU2VwYXJhdG9yJ30gICAgICAgICAgID0gJ1NlcGFyYXRvciBrb2x1bW55JzsKICAgICRMYW5nLT57J1RhYnVsYXRvciAoVEFCKSd9ICAgICAgICAgICAgPSAnVGFidWxhdG9yIChUQUIpJzsKICAgICRMYW5nLT57J1NlbWljb2xvbiAoOyknfSAgICAgICAgICAgICAgPSAnT3JlZG5payAoOyknOwogICAgJExhbmctPnsnQ29sb24gKDopJ30gICAgICAgICAgICAgICAgICA9ICdEd3Vrcm9wZWsgKDopJzsKICAgICRMYW5nLT57J0RvdCAoLiknfSAgICAgICAgICAgICAgICAgICAgPSAnS3JvcGthICguKSc7CiAgICAkTGFuZy0+eydDaGFyc2V0J30gICAgICAgICAgICAgICAgICAgID0gJ0tvZG93YW5pZSc7CiAgICAkTGFuZy0+eydGcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgYWdlbnQgaW50ZXJmYWNlLid9ID0gJyc7CiAgICAkTGFuZy0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPSAnJzsKICAgICRMYW5nLT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwogICAgJExhbmctPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J0NsYXNzIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydDb2x1bW4gU2VwYXJhdG9yIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnJzsKICAgICRMYW5nLT57J0VtcHR5IGZpZWxkcyBpbmRpY2F0ZSB0aGF0IHRoZSBjdXJyZW50IHZhbHVlcyBhcmUga2VwdCd9ID0gJyc7CiAgICAkTGFuZy0+eydDcmVhdGUgYSB0ZW1wbGF0ZSBpbiBvcmRlciB0byBjYW4gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9wdF9CUl9JbXBvcnRFeHBvcnQucG0gLSB0aGUgQnJhemlsaWFuIHRyYW5zbGF0aW9uIG9mIEltcG9ydEV4cG9ydAojIENvcHlyaWdodCAoQykgMjAwMS0yMDEwIE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyBDb3B5cmlnaHQgKEMpIDIwMTAgQ3Jpc3RpYW5vIEtvcm5k9nJmZXIsIGh0dHA6Ly93d3cuZG9yZmVyLmNvbS5ici8KIyAtLQojICRJZDogcHRfQlJfSW1wb3J0RXhwb3J0LnBtLHYgMS41IDIwMTAvMDkvMTQgMjE6NDk6MTQgZHogRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6cHRfQlJfSW1wb3J0RXhwb3J0OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS41ICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAkTGFuZy0+eydJbXBvcnQvRXhwb3J0J30gICAgICAgICAgICAgID0gJ0ltcG9ydGHn428vRXhwb3J0Yefjbyc7CiAgICAkTGFuZy0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSAgID0gJ0dlcmVuY2lhbWVudG8gZGUgSW1wb3J0Yefjby9FeHBvcnRh5+NvJzsKICAgICRMYW5nLT57J0FkZCBtYXBwaW5nIHRlbXBsYXRlJ30gICAgICAgPSAnQWRpY2lvbmFyIE1vZGVsbyBkZSBNYXBlYW1lbnRvJzsKICAgICRMYW5nLT57J1N0YXJ0IEltcG9ydCd9ICAgICAgICAgICAgICAgPSAnSW5pY2lhciBJbXBvcnRh5+NvJzsKICAgICRMYW5nLT57J1N0YXJ0IEV4cG9ydCd9ICAgICAgICAgICAgICAgPSAnSW5pY2lhciBFeHBvcnRh5+NvJzsKICAgICRMYW5nLT57J1N0ZXAnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnUGFzc28nOwogICAgJExhbmctPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSAgICA9ICdFZGl0YXIgaW5mb3JtYef1ZXMgY29tdW5zJzsKICAgICRMYW5nLT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gICAgPSAnRWRpdGFyIGluZm9ybWHn9WVzIGRvIG9iamV0byc7CiAgICAkTGFuZy0+eydFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ICAgID0gJ0VkaXRhciBpbmZvcm1h5/VlcyBkbyBmb3JtYXRvJzsKICAgICRMYW5nLT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ICAgPSAnRWRpdGFyIGluZm9ybWHn9WVzIGRvIG1hcGVhbWVudG8nOwogICAgJExhbmctPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSAgICA9ICdFZGl0YXIgaW5mb3JtYef1ZXMgZGUgcGVzcXVpc2EnOwogICAgJExhbmctPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gICAgICAgICA9ICdJbmZvcm1h5/VlcyBkZSBpbXBvcnRh5+NvJzsKICAgICRMYW5nLT57J0NvbHVtbid9ICAgICAgICAgICAgICAgICAgICAgPSAnQ29sdW5hJzsKICAgICRMYW5nLT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnUmVzdHJpbmdpciBleHBvcnRh5+NvIHBvciBwZXNxdWlzYSc7CiAgICAkTGFuZy0+eydTb3VyY2UgRmlsZSd9ICAgICAgICAgICAgICAgID0gJ0FycXVpdm8gZGUgT3JpZ2VtJzsKICAgICRMYW5nLT57J0NvbHVtbiBTZXBhcmF0b3InfSAgICAgICAgICAgPSAnU2VwYXJhZG9yIGRlIENvbHVuYXMnOwogICAgJExhbmctPnsnVGFidWxhdG9yIChUQUIpJ30gICAgICAgICAgICA9ICdUYWJ1bGHn428gKFRBQiknOwogICAgJExhbmctPnsnU2VtaWNvbG9uICg7KSd9ICAgICAgICAgICAgICA9ICdQb250byBlIFbtcmd1bGEgKDspJzsKICAgICRMYW5nLT57J0NvbG9uICg6KSd9ICAgICAgICAgICAgICAgICAgPSAnRG9pcyBQb250b3MgKDopJzsKICAgICRMYW5nLT57J0RvdCAoLiknfSAgICAgICAgICAgICAgICAgICAgPSAnUG9udG8gKC4pJzsKICAgICRMYW5nLT57J0NoYXJzZXQnfSAgICAgICAgICAgICAgICAgICAgPSAnQ29kaWZpY2Hn428gZGUgQ2FyYWN0ZXJlcyc7CiAgICAkTGFuZy0+eydGcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgYWdlbnQgaW50ZXJmYWNlLid9ID0gJyc7CiAgICAkTGFuZy0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPSAnJzsKICAgICRMYW5nLT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwogICAgJExhbmctPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J0NsYXNzIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydDb2x1bW4gU2VwYXJhdG9yIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnJzsKICAgICRMYW5nLT57J0VtcHR5IGZpZWxkcyBpbmRpY2F0ZSB0aGF0IHRoZSBjdXJyZW50IHZhbHVlcyBhcmUga2VwdCd9ID0gJyc7CiAgICAkTGFuZy0+eydDcmVhdGUgYSB0ZW1wbGF0ZSBpbiBvcmRlciB0byBjYW4gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9ydV9JbXBvcnRFeHBvcnQucG0gLSB0aGUgcnVzc2lhbiB0cmFuc2xhdGlvbiBvZiBJbXBvcnRFeHBvcnQKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgQ29weXJpZ2h0IChDKSAyMDA4IEVnb3IgVHNpbGVua28gPGJnOHMgYXQgc3ltbGluay5ydT4KIyAtLQojICRJZDogcnVfSW1wb3J0RXhwb3J0LnBtLHYgMS42IDIwMTAvMDkvMTQgMjE6NDk6MTQgZHogRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6cnVfSW1wb3J0RXhwb3J0OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS42ICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAkTGFuZy0+eydJbXBvcnQvRXhwb3J0J30gICAgICAgICAgICAgID0gJ8js7+7w8i/d6vHv7vDyJzsKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQgTWFuYWdlbWVudCd9ICAgPSAn0+/w4OLr5e3o5SDI7O/u8PLu7C/d6vHv7vDy7uwnOwogICAgJExhbmctPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSAgICAgICA9ICfE7uHg4uvl7ejlIPjg4evu7eAg8e7u8uLl8vHy4uj/JzsKICAgICRMYW5nLT57J1N0YXJ0IEltcG9ydCd9ICAgICAgICAgICAgICAgPSAnzeD34PL8IOjs7+7w8ic7CiAgICAkTGFuZy0+eydTdGFydCBFeHBvcnQnfSAgICAgICAgICAgICAgID0gJ83g9+Dy/CD96vHv7vDyJzsKICAgICRMYW5nLT57J1N0ZXAnfSAgICAgICAgICAgICAgICAgICAgICAgPSAn2ODjJzsKICAgICRMYW5nLT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gICAgPSAn0OXk4Ory6PDu4uDy/CDu4fnz/iDo7fTu8Ozg9uj+JzsKICAgICRMYW5nLT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gICAgPSAn0OXk4Ory6PDu4uDy/CDo7fTu8Ozg9uj+IO7hIO7h+uXq8uUnOwogICAgJExhbmctPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSAgICA9ICfQ5eTg6vLo8O7i4PL8IPTu8Ozg8iDk4O3t+/UnOwogICAgJExhbmctPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gICA9ICfQ5eTg6vLo8O7i4PL8IOjt9O7w7OD26P4g8e7u8uLl8vHy4uj/JzsKICAgICRMYW5nLT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gICAgPSAn0OXk4Ory6PDu4uDy/CDv7ujx6u7i8/4g6O307vDs4Pbo/ic7CiAgICAkTGFuZy0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSAgICAgICAgID0gJ8jt9O7w7OD26P8g6Ozv7vDy4Cc7CiAgICAkTGFuZy0+eydDb2x1bW4nfSAgICAgICAgICAgICAgICAgICAgID0gJ9Hy7uvh5fYnOwogICAgJExhbmctPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICfO4/Dg7ej36PL8IP3q8e/u8PIg7+7o8eru7Cc7CiAgICAkTGFuZy0+eydTb3VyY2UgRmlsZSd9ICAgICAgICAgICAgICAgID0gJ8jx9e7k7fvpIPTg6esnOwogICAgJExhbmctPnsnQ29sdW1uIFNlcGFyYXRvcid9ICAgICAgICAgICA9ICfQ4Ofk5evo8uXr/Cc7CiAgICAkTGFuZy0+eydUYWJ1bGF0b3IgKFRBQiknfSAgICAgICAgICAgID0gJ9Lg4fPr//bo/yAoVEFCKSc7CiAgICAkTGFuZy0+eydTZW1pY29sb24gKDspJ30gICAgICAgICAgICAgID0gJ9Lu9+rgIPEg5+Dv//Lu6SAoOyknOwogICAgJExhbmctPnsnQ29sb24gKDopJ30gICAgICAgICAgICAgICAgICA9ICfE4u7l8u736OUgKDopJzsKICAgICRMYW5nLT57J0RvdCAoLiknfSAgICAgICAgICAgICAgICAgICAgPSAn0u736uAgKC4pJzsKICAgICRMYW5nLT57J0NoYXJzZXQnfSAgICAgICAgICAgICAgICAgICAgPSAnyu7k6PDu4urgJzsKICAgICRMYW5nLT57J0Zyb250ZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBhZ2VudCBpbnRlcmZhY2UuJ30gPSAnJzsKICAgICRMYW5nLT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9ICcnOwogICAgJExhbmctPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CiAgICAkTGFuZy0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJExhbmctPnsnQ2xhc3MgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J0NvbHVtbiBTZXBhcmF0b3IgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRMYW5nLT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICcnOwogICAgJExhbmctPnsnRW1wdHkgZmllbGRzIGluZGljYXRlIHRoYXQgdGhlIGN1cnJlbnQgdmFsdWVzIGFyZSBrZXB0J30gPSAnJzsKICAgICRMYW5nLT57J0NyZWF0ZSBhIHRlbXBsYXRlIGluIG9yZGVyIHRvIGNhbiBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKCiAgICByZXR1cm4gMTsKfQoKMTsK
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS96aF9DTl9JbXBvcnRFeHBvcnQucG0gLSB0aGUgQ2hpbmVzZSBzaW1wbGUgdHJhbnNsYXRpb24gb2YgSW1wb3J0RXhwb3J0CiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTAgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiB6aF9DTl9JbXBvcnRFeHBvcnQucG0sdiAxLjYgMjAxMC8wOS8xNCAyMTo0OToxNCBkeiBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjp6aF9DTl9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjYgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQnfSAgICAgICAgICAgICAgPSAntbzI6y+1vLP2JzsKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQgTWFuYWdlbWVudCd9ICAgPSAntbzI6y+1vLP2udzA7Sc7CiAgICAkTGFuZy0+eydBZGQgbWFwcGluZyB0ZW1wbGF0ZSd9ICAgICAgID0gJ9T2vNPTs8nkxKOw5ic7CiAgICAkTGFuZy0+eydTdGFydCBJbXBvcnQnfSAgICAgICAgICAgICAgID0gJ7+qyry1vMjrJzsKICAgICRMYW5nLT57J1N0YXJ0IEV4cG9ydCd9ICAgICAgICAgICAgICAgPSAnv6rKvLW8s/YnOwogICAgJExhbmctPnsnU3RlcCd9ICAgICAgICAgICAgICAgICAgICAgICA9ICeyvdboJzsKICAgICRMYW5nLT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gICAgPSAnseC8rbmy08PQxc+iJzsKICAgICRMYW5nLT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gICAgPSAnseC8rbbUz/HQxc+iJzsKICAgICRMYW5nLT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gICAgPSAnseC8rbjxyr3Qxc+iJzsKICAgICRMYW5nLT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ICAgPSAnseC8rdOzyeTQxc+iJzsKICAgICRMYW5nLT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gICAgPSAnseC8rcvRy/fQxc+iJzsKICAgICRMYW5nLT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ICAgICAgICAgPSAntbzI69DFz6InOwogICAgJExhbmctPnsnQ29sdW1uJ30gICAgICAgICAgICAgICAgICAgICA9ICfB0Cc7CiAgICAkTGFuZy0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ8/e1sa1vLP2w7+49svR0bAnOwogICAgJExhbmctPnsnU291cmNlIEZpbGUnfSAgICAgICAgICAgICAgICA9ICfUtM7EvP4nOwogICAgJExhbmctPnsnQ29sdW1uIFNlcGFyYXRvcid9ICAgICAgICAgICA9ICfB0LfWuPS3+yc7CiAgICAkTGFuZy0+eydUYWJ1bGF0b3IgKFRBQiknfSAgICAgICAgICAgID0gJ9bGse28/CAoVEFCKSc7CiAgICAkTGFuZy0+eydTZW1pY29sb24gKDspJ30gICAgICAgICAgICAgID0gJ7fWusUgKDspJzsKICAgICRMYW5nLT57J0NvbG9uICg6KSd9ICAgICAgICAgICAgICAgICAgPSAnw7C6xSAoOiknOwogICAgJExhbmctPnsnRG90ICguKSd9ICAgICAgICAgICAgICAgICAgICA9ICe+5LrFICguKSc7CiAgICAkTGFuZy0+eydDaGFyc2V0J30gICAgICAgICAgICAgICAgICAgID0gJ9fWt/u8ryc7CiMgYWRkIGJ5IE5ldmVyCiAgICAkTGFuZy0+eydUZW1wbGF0ZSd9ICAgICAgICAgICAgICAgICAgID0gJ8SjsOYnOwogICAgJExhbmctPnsnRnJvbnRlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGFnZW50IGludGVyZmFjZS4nfSA9ICcnOwogICAgJExhbmctPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0gJyc7CiAgICAkTGFuZy0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKICAgICRMYW5nLT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJExhbmctPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkTGFuZy0+eydDbGFzcyBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJExhbmctPnsnQ29sdW1uIFNlcGFyYXRvciBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJExhbmctPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJyc7CiAgICAkTGFuZy0+eydFbXB0eSBmaWVsZHMgaW5kaWNhdGUgdGhhdCB0aGUgY3VycmVudCB2YWx1ZXMgYXJlIGtlcHQnfSA9ICcnOwogICAgJExhbmctPnsnQ3JlYXRlIGEgdGVtcGxhdGUgaW4gb3JkZXIgdG8gY2FuIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwoKICAgIHJldHVybiAxOwp9CgoxOwo=
# --
# Kernel/Modules/AdminImportExport.pm - admin frontend of import export module
# Copyright (C) 2001-2010 OTRS AG, http://otrs.org/
# --
# $Id: AdminImportExport.pm,v 1.26 2010/10/20 19:26:04 dz Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Modules::AdminImportExport;

use strict;
use warnings;

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

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

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

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

    # check needed objects
    for my $Object (qw(ConfigObject ParamObject LogObject LayoutObject 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 ( 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 ( 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;
        }

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

    # ------------------------------------------------------------ #
    # 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;

IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnRMYXlvdXRDaGVja2JveC5wbSAtIGxheW91dCBiYWNrZW5kIG1vZHVsZQojIENvcHlyaWdodCAoQykgMjAwMS0yMDEwIE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogSW1wb3J0RXhwb3J0TGF5b3V0Q2hlY2tib3gucG0sdiAxLjYgMjAxMC8wOS8xNCAyMDo1ODoyMCBkeiBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0TGF5b3V0Q2hlY2tib3g7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjYgJCkgWzFdOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRDaGVja2JveCAtIGxheW91dCBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIFNZTk9QU0lTCgpBbGwgbGF5b3V0IGZ1bmN0aW9ucyBmb3IgY2hlY2tib3ggZWxlbWVudHMgaW4gaW1wb3J0L2V4cG9ydC4KCj1vdmVyIDQKCj1jdXQKCj1pdGVtIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgJEJhY2tlbmRPYmplY3QgPSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0TGF5b3V0Q2hlY2tib3gtPm5ldygKICAgICAgICAlUGFyYW0sCiAgICApOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBvYmplY3RzCiAgICBmb3IgbXkgJE9iamVjdCAocXcoQ29uZmlnT2JqZWN0IExvZ09iamVjdCBNYWluT2JqZWN0IFBhcmFtT2JqZWN0IExheW91dE9iamVjdCkpIHsKICAgICAgICAkU2VsZi0+eyRPYmplY3R9ID0gJFBhcmFteyRPYmplY3R9IHx8IGRpZSAiR290IG5vICRPYmplY3QhIjsKICAgIH0KCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1pdGVtIEZvcm1JbnB1dENyZWF0ZSgpCgpjcmVhdGUgYSBpbnB1dCBzdHJpbmcKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+Rm9ybUlucHV0Q3JlYXRlKAogICAgICAgIEl0ZW0gICA9PiAkSXRlbVJlZiwKICAgICAgICBQcmVmaXggPT4gJ1ByZWZpeDo6JywgICMgKG9wdGlvbmFsKQogICAgICAgIFZhbHVlICA9PiAnVmFsdWUnLCAgICAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1JbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIEl0ZW0hJywKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAkUGFyYW17UHJlZml4fSB8fD0gJyc7CgogICAgbXkgJENoZWNrZWQgPSAkUGFyYW17VmFsdWV9ID8gJ2NoZWNrZWQ9ImNoZWNrZWQiJyA6ICcnOwoKICAgIHJldHVybgogICAgICAgIHFxezxpbnB1dCBpZD0iJFBhcmFte1ByZWZpeH0kUGFyYW17SXRlbX0tPntLZXl9IiB0eXBlPSJjaGVja2JveCIgbmFtZT0iJFBhcmFte1ByZWZpeH0kUGFyYW17SXRlbX0tPntLZXl9IiAkQ2hlY2tlZCAvPn07Cn0KCj1pdGVtIEZvcm1EYXRhR2V0KCkKCmdldCBmb3JtIGRhdGEKCiAgICBteSAkRm9ybURhdGEgPSAkQmFja2VuZE9iamVjdC0+Rm9ybURhdGFHZXQoCiAgICAgICAgSXRlbSAgID0+ICRJdGVtUmVmLAogICAgICAgIFByZWZpeCA9PiAnUHJlZml4OjonLCAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1EYXRhR2V0IHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgSXRlbSEnLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICRQYXJhbXtQcmVmaXh9IHx8PSAnJzsKCiAgICAjIGdldCBmb3JtIGRhdGEKICAgIG15ICRGb3JtRGF0YSA9ICRTZWxmLT57UGFyYW1PYmplY3R9LT5HZXRQYXJhbSgKICAgICAgICBQYXJhbSA9PiAkUGFyYW17UHJlZml4fSAuICRQYXJhbXtJdGVtfS0+e0tleX0sCiAgICApOwoKICAgIHJldHVybiAkRm9ybURhdGE7Cn0KCjE7Cgo9YmFjawoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIHNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoTDxodHRwOi8vb3Rycy5vcmcvPikuCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dD4uCgo9Y3V0Cgo9aGVhZDEgVkVSU0lPTgoKJFJldmlzaW9uOiAxLjYgJCAkRGF0ZTogMjAxMC8wOS8xNCAyMDo1ODoyMCAkCgo9Y3V0Cg==
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnRMYXlvdXREVEwucG0gLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IEltcG9ydEV4cG9ydExheW91dERUTC5wbSx2IDEuMyAyMDA5LzA1LzE4IDA5OjQyOjUyIG1oIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXREVEw7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjMgJCkgWzFdOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXREVEwgLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKQWxsIGxheW91dCBmdW5jdGlvbnMgZm9yIGRpc3BsYXkgRFRMIGNvZGUKCj1vdmVyIDQKCj1jdXQKCj1pdGVtIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgJEJhY2tlbmRPYmplY3QgPSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0TGF5b3V0RFRMLT5uZXcoCiAgICAgICAgJVBhcmFtLAogICAgKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgIyBjaGVjayBuZWVkZWQgb2JqZWN0cwogICAgZm9yIG15ICRPYmplY3QgKHF3KENvbmZpZ09iamVjdCBMb2dPYmplY3QgTWFpbk9iamVjdCBQYXJhbU9iamVjdCBMYXlvdXRPYmplY3QpKSB7CiAgICAgICAgJFNlbGYtPnskT2JqZWN0fSA9ICRQYXJhbXskT2JqZWN0fSB8fCBkaWUgIkdvdCBubyAkT2JqZWN0ISI7CiAgICB9CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aXRlbSBGb3JtSW5wdXRDcmVhdGUoKQoKY3JlYXRlIGEgaW5wdXQgc3RyaW5nCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkZvcm1JbnB1dENyZWF0ZSgKICAgICAgICBJdGVtICAgPT4gJEl0ZW1SZWYsCiAgICApOwoKPWN1dAoKc3ViIEZvcm1JbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coIFByaW9yaXR5ID0+ICdlcnJvcicsIE1lc3NhZ2UgPT4gJ05lZWQgSXRlbSEnICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIHJldHVybiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntEYXRhfTsKfQoKPWl0ZW0gRm9ybURhdGFHZXQoKQoKZ2V0IGZvcm0gZGF0YQoKICAgIG15ICRGb3JtRGF0YSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtRGF0YUdldCgpOwoKPWN1dAoKc3ViIEZvcm1EYXRhR2V0IHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuOwp9CgoxOwoKPWJhY2sKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKGh0dHA6Ly9vdHJzLm9yZy8pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgoKPWN1dAoKPWhlYWQxIFZFUlNJT04KCiRSZXZpc2lvbjogMS4zICQgJERhdGU6IDIwMDkvMDUvMTggMDk6NDI6NTIgJAoKPWN1dAo=
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnRMYXlvdXRTZWxlY3Rpb24ucG0gLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IEltcG9ydEV4cG9ydExheW91dFNlbGVjdGlvbi5wbSx2IDEuOSAyMDEwLzA5LzE0IDIwOjU4OjIwIGR6IEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRTZWxlY3Rpb247Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjkgJCkgWzFdOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRTZWxlY3Rpb24gLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKQWxsIGxheW91dCBmdW5jdGlvbnMgZm9yIHNlbGVjdGlvbiBlbGVtZW50cwoKPW92ZXIgNAoKPWN1dAoKPWl0ZW0gbmV3KCkKCmNyZWF0ZSBhbiBvYmplY3QKCiAgICAkQmFja2VuZE9iamVjdCA9IEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRTZWxlY3Rpb24tPm5ldygKICAgICAgICAlUGFyYW0sCiAgICApOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBvYmplY3RzCiAgICBmb3IgbXkgJE9iamVjdCAocXcoQ29uZmlnT2JqZWN0IExvZ09iamVjdCBNYWluT2JqZWN0IFBhcmFtT2JqZWN0IExheW91dE9iamVjdCkpIHsKICAgICAgICAkU2VsZi0+eyRPYmplY3R9ID0gJFBhcmFteyRPYmplY3R9IHx8IGRpZSAiR290IG5vICRPYmplY3QhIjsKICAgIH0KCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1pdGVtIEZvcm1JbnB1dENyZWF0ZSgpCgpjcmVhdGUgYSBpbnB1dCBzdHJpbmcKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+Rm9ybUlucHV0Q3JlYXRlKAogICAgICAgIEl0ZW0gICA9PiAkSXRlbVJlZiwKICAgICAgICBQcmVmaXggPT4gJ1ByZWZpeDo6JywgICMgKG9wdGlvbmFsKQogICAgICAgIFZhbHVlICA9PiAnVmFsdWUnLCAgICAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1JbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coIFByaW9yaXR5ID0+ICdlcnJvcicsIE1lc3NhZ2UgPT4gJ05lZWQgSXRlbSEnICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICMgc2V0IGRlZmF1bHQgdmFsdWUKICAgICRQYXJhbXtQcmVmaXh9IHx8PSAnJzsKICAgICRQYXJhbXtWYWx1ZX0gIHx8PSAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntWYWx1ZURlZmF1bHR9OwoKICAgIGlmICggJFBhcmFte1ZhbHVlfSAmJiAkUGFyYW17VmFsdWV9ID1+IG17ICMjIyMjIH14bXMgKSB7CiAgICAgICAgbXkgQFZhbHVlcyA9IHNwbGl0ICcjIyMjIycsICRQYXJhbXtWYWx1ZX07CiAgICAgICAgJFBhcmFte1ZhbHVlfSA9IFxAVmFsdWVzOwogICAgfQoKICAgICMgZ2VuZXJhdGUgb3B0aW9uIHN0cmluZwogICAgbXkgJFN0cmluZyA9ICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+QnVpbGRTZWxlY3Rpb24oCiAgICAgICAgSUQgICAgICAgICAgID0+ICRQYXJhbXtQcmVmaXh9IC4gJFBhcmFte0l0ZW19LT57S2V5fSwKICAgICAgICBDbGFzcyAgICAgICAgPT4gJFBhcmFte0NsYXNzfSwKICAgICAgICBOYW1lICAgICAgICAgPT4gJFBhcmFte1ByZWZpeH0gLiAkUGFyYW17SXRlbX0tPntLZXl9LAogICAgICAgIERhdGEgICAgICAgICA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntEYXRhfSB8fCB7fSwKICAgICAgICBTZWxlY3RlZElEICAgPT4gJFBhcmFte1ZhbHVlfSwKICAgICAgICBUcmFuc2xhdGlvbiAgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VHJhbnNsYXRpb259LAogICAgICAgIFBvc3NpYmxlTm9uZSA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntQb3NzaWJsZU5vbmV9LAogICAgICAgIE11bHRpcGxlICAgICA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntNdWx0aXBsZX0sCiAgICAgICAgU2l6ZSAgICAgICAgID0+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1NpemV9LAogICAgKTsKCiAgICByZXR1cm4gJFN0cmluZzsKfQoKPWl0ZW0gRm9ybURhdGFHZXQoKQoKZ2V0IGZvcm0gZGF0YQoKICAgIG15ICRGb3JtRGF0YSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtRGF0YUdldCgKICAgICAgICBJdGVtICAgPT4gJEl0ZW1SZWYsCiAgICAgICAgUHJlZml4ID0+ICdQcmVmaXg6OicsICAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgRm9ybURhdGFHZXQgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgaWYgKCAhJFBhcmFte0l0ZW19ICkgewogICAgICAgICRTZWxmLT57TG9nT2JqZWN0fS0+TG9nKCBQcmlvcml0eSA9PiAnZXJyb3InLCBNZXNzYWdlID0+ICdOZWVkIEl0ZW0hJyApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAkUGFyYW17UHJlZml4fSB8fD0gJyc7CgogICAgIyBnZXQgZm9ybSBkYXRhCiAgICBteSBARm9ybURhdGFzID0gJFNlbGYtPntQYXJhbU9iamVjdH0tPkdldEFycmF5KAogICAgICAgIFBhcmFtID0+ICRQYXJhbXtQcmVmaXh9IC4gJFBhcmFte0l0ZW19LT57S2V5fSwKICAgICk7CgogICAgbXkgJEZvcm1EYXRhID0gam9pbiAnIyMjIyMnLCBARm9ybURhdGFzOwoKICAgIHJldHVybiAkRm9ybURhdGEgaWYgJEZvcm1EYXRhOwogICAgcmV0dXJuICRGb3JtRGF0YSBpZiAhJFBhcmFte0l0ZW19LT57SW5wdXR9LT57UmVxdWlyZWR9OwoKICAgICMgc2V0IGludmFsaWQgcGFyYW0KICAgICRQYXJhbXtJdGVtfS0+e0Zvcm19LT57SW52YWxpZH0gPSAxOwoKICAgIHJldHVybiAkRm9ybURhdGE7Cn0KCjE7Cgo9YmFjawoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIHNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoTDxodHRwOi8vb3Rycy5vcmcvPikuCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dD4uCgo9Y3V0Cgo9aGVhZDEgVkVSU0lPTgoKJFJldmlzaW9uOiAxLjkgJCAkRGF0ZTogMjAxMC8wOS8xNCAyMDo1ODoyMCAkCgo9Y3V0Cg==
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnRMYXlvdXRUZXh0LnBtIC0gbGF5b3V0IGJhY2tlbmQgbW9kdWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTAgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBJbXBvcnRFeHBvcnRMYXlvdXRUZXh0LnBtLHYgMS4xMSAyMDEwLzEwLzIwIDE5OjI2OjA1IGR6IEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRUZXh0OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS4xMSAkKSBbMV07Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydExheW91dFRleHQgLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKQWxsIGxheW91dCBmdW5jdGlvbnMgZm9yIHRleHQgZWxlbWVudHMKCj1vdmVyIDQKCj1jdXQKCj1pdGVtIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgJEJhY2tlbmRPYmplY3QgPSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0TGF5b3V0VGV4dC0+bmV3KAogICAgICAgICVQYXJhbSwKICAgICk7Cgo9Y3V0CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHt9OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgICMgY2hlY2sgbmVlZGVkIG9iamVjdHMKICAgIGZvciBteSAkT2JqZWN0IChxdyhDb25maWdPYmplY3QgTG9nT2JqZWN0IE1haW5PYmplY3QgUGFyYW1PYmplY3QgTGF5b3V0T2JqZWN0KSkgewogICAgICAgICRTZWxmLT57JE9iamVjdH0gPSAkUGFyYW17JE9iamVjdH0gfHwgZGllICJHb3Qgbm8gJE9iamVjdCEiOwogICAgfQoKICAgIHJldHVybiAkU2VsZjsKfQoKPWl0ZW0gRm9ybUlucHV0Q3JlYXRlKCkKCmNyZWF0ZSBhIGlucHV0IHN0cmluZwoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtSW5wdXRDcmVhdGUoCiAgICAgICAgSXRlbSAgID0+ICRJdGVtUmVmLAogICAgICAgIFByZWZpeCA9PiAnUHJlZml4OjonLCAgIyAob3B0aW9uYWwpCiAgICAgICAgVmFsdWUgID0+ICdWYWx1ZScsICAgICAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgRm9ybUlucHV0Q3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZyggUHJpb3JpdHkgPT4gJ2Vycm9yJywgTWVzc2FnZSA9PiAnTmVlZCBJdGVtIScgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgJFBhcmFte1ByZWZpeH0gfHw9ICcnOwoKICAgIG15ICRWYWx1ZSA9ICRQYXJhbXtWYWx1ZX0gfHwgJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VmFsdWVEZWZhdWx0fTsKICAgIG15ICRTaXplID0gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57U2l6ZX0gfHwgNDA7CiAgICBteSAkU2l6ZUNsYXNzOwogICAgaWYgKCAkU2l6ZSA8IDE1ICkgewogICAgICAgICRTaXplQ2xhc3MgPSAnVzEwcGMnOwogICAgfQogICAgZWxzaWYgKCAkU2l6ZSA8IDM1ICkgewogICAgICAgICRTaXplQ2xhc3MgPSAnVzMzcGMnOwogICAgfQogICAgZWxzaWYgKCAkU2l6ZSA8IDUwICkgewogICAgICAgICRTaXplQ2xhc3MgPSAnVzUwcGMnOwogICAgfQogICAgZWxzZSB7CiAgICAgICAgJFNpemVDbGFzcyA9ICdXNzVwYyc7CiAgICB9CgogICAgIyBwcmVwYXJlIGRhdGEKICAgIG15ICRJRCA9ICggJFBhcmFte1ByZWZpeH0gfHwgJycgKSAuICggJFBhcmFte0l0ZW19LT57S2V5fSApOwogICAgbXkgJE5hbWUgPSAoICRQYXJhbXtQcmVmaXh9IHx8ICcnICkgLiAoICRQYXJhbXtOYW1lfSB8fCAkSUQgKTsKICAgIG15ICRDbGFzcyA9ICggJFNpemVDbGFzcyB8fCAnJyApIC4gKCAkUGFyYW17Q2xhc3N9IHx8ICcnICk7CgogICAgbXkgJFN0cmluZwogICAgICAgID0gIjxpbnB1dCBpZD1cIiRJRFwiIHR5cGU9XCJ0ZXh0XCIgbmFtZT1cIiROYW1lXCIgY2xhc3M9XCIkQ2xhc3NcIiAiOwoKICAgIGlmICgkVmFsdWUpIHsKCiAgICAgICAgIyB0cmFuc2xhdGUKICAgICAgICBpZiAoICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1RyYW5zbGF0aW9ufSApIHsKICAgICAgICAgICAgJFZhbHVlID0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT57TGFuZ3VhZ2VPYmplY3R9LT5HZXQoJFZhbHVlKTsKICAgICAgICB9CgogICAgICAgICMgdHJhbnNmb3JtIGFzY2lpIHRvIGh0bWwKICAgICAgICAkVmFsdWUgPSAkU2VsZi0+e0xheW91dE9iamVjdH0tPkFzY2lpMkh0bWwoCiAgICAgICAgICAgIFRleHQgICAgICAgICAgID0+ICRWYWx1ZSwKICAgICAgICAgICAgSFRNTFJlc3VsdE1vZGUgPT4gMSwKICAgICAgICApOwoKICAgICAgICAkU3RyaW5nIC49ICJ2YWx1ZT1cIiRWYWx1ZVwiICI7CiAgICB9CgogICAgIyBhZGQgbWF4aW11bSBsZW5ndGgKICAgIGlmICggJFBhcmFte0l0ZW19LT57SW5wdXR9LT57TWF4TGVuZ3RofSApIHsKICAgICAgICAkU3RyaW5nIC49ICJtYXhsZW5ndGg9XCIkUGFyYW17SXRlbX0tPntJbnB1dH0tPntNYXhMZW5ndGh9XCIgIjsKICAgIH0KCiAgICAkU3RyaW5nIC49ICIvPiAiOwoKICAgIHJldHVybiAkU3RyaW5nOwp9Cgo9aXRlbSBGb3JtRGF0YUdldCgpCgpnZXQgZm9ybSBkYXRhCgogICAgbXkgJEZvcm1EYXRhID0gJEJhY2tlbmRPYmplY3QtPkZvcm1EYXRhR2V0KAogICAgICAgIEl0ZW0gICA9PiAkSXRlbVJlZiwKICAgICAgICBQcmVmaXggPT4gJ1ByZWZpeDo6JywgICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBGb3JtRGF0YUdldCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coIFByaW9yaXR5ID0+ICdlcnJvcicsIE1lc3NhZ2UgPT4gJ05lZWQgSXRlbSEnICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICRQYXJhbXtQcmVmaXh9IHx8PSAnJzsKCiAgICAjIGdldCBmb3JtIGRhdGEKICAgIG15ICRGb3JtRGF0YSA9ICRTZWxmLT57UGFyYW1PYmplY3R9LT5HZXRQYXJhbSgKICAgICAgICBQYXJhbSA9PiAkUGFyYW17UHJlZml4fSAuICRQYXJhbXtJdGVtfS0+e0tleX0sCiAgICApOwoKICAgICMgcmVnZXggY2hlY2sKICAgIGlmICggJFBhcmFte0l0ZW19LT57SW5wdXR9LT57UmVnZXh9ICYmICRGb3JtRGF0YSAhfiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntSZWdleH0gKSB7CgogICAgICAgICRQYXJhbXtJdGVtfS0+e0Zvcm19LT57SW52YWxpZH0gPSAxOwogICAgICAgIHJldHVybiAkRm9ybURhdGE7CiAgICB9CgogICAgcmV0dXJuICRGb3JtRGF0YSBpZiAkRm9ybURhdGE7CiAgICByZXR1cm4gJEZvcm1EYXRhIGlmICEkUGFyYW17SXRlbX0tPntJbnB1dH0tPntSZXF1aXJlZH07CgogICAgIyBzZXQgaW52YWxpZCBwYXJhbQogICAgJFBhcmFte0l0ZW19LT57Rm9ybX0tPntJbnZhbGlkfSA9IDE7CgogICAgcmV0dXJuICRGb3JtRGF0YTsKfQoKMTsKCj1iYWNrCgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKClRoaXMgc29mdHdhcmUgaXMgcGFydCBvZiB0aGUgT1RSUyBwcm9qZWN0IChMPGh0dHA6Ly9vdHJzLm9yZy8+KS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIEw8aHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0Pi4KCj1jdXQKCj1oZWFkMSBWRVJTSU9OCgokUmV2aXNpb246IDEuMTEgJCAkRGF0ZTogMjAxMC8xMC8yMCAxOToyNjowNSAkCgo9Y3V0Cg==
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9MYXlvdXRJbXBvcnRFeHBvcnQucG0gLSBwcm92aWRlcyBnZW5lcmljIEhUTUwgb3V0cHV0IGZvciBJbXBvcnRFeHBvcnQKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxMCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IExheW91dEltcG9ydEV4cG9ydC5wbSx2IDEuNiAyMDEwLzAyLzI0IDEyOjU3OjM5IGJlcyBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0SW1wb3J0RXhwb3J0OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS42ICQpIFsxXTsKCj1pdGVtIEltcG9ydEV4cG9ydEZvcm1JbnB1dENyZWF0ZSgpCgpyZXR1cm5zIGEgaW5wdXQgZmllbGQgaHRtbCBzdHJpbmcKCiAgICBteSAkU3RyaW5nID0gJExheW91dE9iamVjdC0+SW1wb3J0RXhwb3J0Rm9ybUlucHV0Q3JlYXRlKAogICAgICAgIEl0ZW0gID0+ICRJdGVtUmVmLAogICAgICAgIFZhbHVlID0+ICdWYWx1ZScsICAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEltcG9ydEV4cG9ydEZvcm1JbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIEl0ZW0hJwogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICMgbG9hZCBiYWNrZW5kCiAgICBteSAkQmFja2VuZE9iamVjdCA9ICRTZWxmLT5fSW1wb3J0RXhwb3J0TG9hZExheW91dEJhY2tlbmQoCiAgICAgICAgVHlwZSA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntUeXBlfSwKICAgICk7CgogICAgcmV0dXJuICcnIGlmICEkQmFja2VuZE9iamVjdDsKCiAgICAjIGxvb2t1cCBpdGVtIHZhbHVlCiAgICBteSAkU3RyaW5nID0gJEJhY2tlbmRPYmplY3QtPkZvcm1JbnB1dENyZWF0ZSglUGFyYW0pOwoKICAgIHJldHVybiAkU3RyaW5nOwp9Cgo9aXRlbSBJbXBvcnRFeHBvcnRGb3JtRGF0YUdldCgpCgpyZXR1cm5zIHRoZSB2YWx1ZXMgZnJvbSB0aGUgaHRtbCBmb3JtIGFzIGhhc2ggcmVmZXJlbmNlCgogICAgbXkgJEZvcm1EYXRhID0gJExheW91dE9iamVjdC0+SW1wb3J0RXhwb3J0Rm9ybURhdGFHZXQoCiAgICAgICAgSXRlbSA9PiAkSXRlbVJlZiwKICAgICk7Cgo9Y3V0CgpzdWIgSW1wb3J0RXhwb3J0Rm9ybURhdGFHZXQgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgaWYgKCAhJFBhcmFte0l0ZW19ICkgewogICAgICAgICRTZWxmLT57TG9nT2JqZWN0fS0+TG9nKCBQcmlvcml0eSA9PiAnZXJyb3InLCBNZXNzYWdlID0+ICdOZWVkIEl0ZW0hJyApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAjIGxvYWQgYmFja2VuZAogICAgbXkgJEJhY2tlbmRPYmplY3QgPSAkU2VsZi0+X0ltcG9ydEV4cG9ydExvYWRMYXlvdXRCYWNrZW5kKAogICAgICAgIFR5cGUgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VHlwZX0sCiAgICApOwoKICAgIHJldHVybiBpZiAhJEJhY2tlbmRPYmplY3Q7CgogICAgIyBnZXQgZm9ybSBkYXRhCiAgICBteSAkRm9ybURhdGEgPSAkQmFja2VuZE9iamVjdC0+Rm9ybURhdGFHZXQoJVBhcmFtKTsKCiAgICByZXR1cm4gJEZvcm1EYXRhOwp9Cgo9aXRlbSBfSW1wb3J0RXhwb3J0TG9hZExheW91dEJhY2tlbmQoKQoKdG8gbG9hZCBhIGltcG9ydC9leHBvcnQgbGF5b3V0IGJhY2tlbmQgbW9kdWxlCgogICAgbXkgJEJhY2tlbmQgPSAkTGF5b3V0T2JqZWN0LT5fSW1wb3J0RXhwb3J0TG9hZExheW91dEJhY2tlbmQoCiAgICAgICAgVHlwZSA9PiAnU2VsZWN0aW9uJywKICAgICk7CgpBbiBpbnN0YW5jZSBvZiB0aGUgbG9hZGVkIGJhY2tlbmQgbW9kdWxlIGlzIHJldHVybmVkLgoKPWN1dAoKc3ViIF9JbXBvcnRFeHBvcnRMb2FkTGF5b3V0QmFja2VuZCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIGlmICggISRQYXJhbXtUeXBlfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgVHlwZSEnLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIG15ICRHZW5lcmljTW9kdWxlID0gIktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXQkUGFyYW17VHlwZX0iOwoKICAgICMgbG9hZCB0aGUgYmFja2VuZCBtb2R1bGUKICAgIGlmICggISRTZWxmLT57TWFpbk9iamVjdH0tPlJlcXVpcmUoJEdlbmVyaWNNb2R1bGUpICkgewogICAgICAgICRTZWxmLT57TG9nT2JqZWN0fS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiQ2FuJ3QgbG9hZCBiYWNrZW5kIG1vZHVsZSAkUGFyYW17VHlwZX0hIgogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICMgY3JlYXRlIG5ldyBpbnN0YW5jZQogICAgbXkgJEJhY2tlbmRPYmplY3QgPSAkR2VuZXJpY01vZHVsZS0+bmV3KAogICAgICAgICV7JFNlbGZ9LAogICAgICAgICVQYXJhbSwKICAgICAgICBMYXlvdXRPYmplY3QgPT4gJFNlbGYsCiAgICApOwoKICAgIGlmICggISRCYWNrZW5kT2JqZWN0ICkgewogICAgICAgICRTZWxmLT57TG9nT2JqZWN0fS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiQ2FuJ3QgY3JlYXRlIGEgbmV3IGluc3RhbmNlIG9mIGJhY2tlbmQgbW9kdWxlICRQYXJhbXtUeXBlfSEiLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIHJldHVybiAkQmFja2VuZE9iamVjdDsKfQoKMTsK
# --
# AdminImportExport.dtl - provides HTML form for AdminImportExport
# Copyright (C) 2001-2010 OTRS AG, http://otrs.org/
# --
# $Id: AdminImportExport.dtl,v 1.25 2010/12/09 22:33:54 ub Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

<!-- dtl:block:Overview -->
<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 in order to can 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 Right">
                        <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: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 -->

# --
# Kernel/System/ImportExport.pm - all import and export functions
# Copyright (C) 2001-2010 OTRS AG, http://otrs.org/
# --
# $Id: ImportExport.pm,v 1.40 2010/09/08 18:51:10 en Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::System::ImportExport;

use strict;
use warnings;

use Kernel::System::CheckItem;

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

=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 ( 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 ( 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 ( 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 ( 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 ( 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},
    );

    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;

    # 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 ) {
            $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 (GPL). If you
did not receive this file, see L<http://www.gnu.org/licenses/gpl-2.0.txt>.

=cut

=head1 VERSION

$Revision: 1.40 $ $Date: 2010/09/08 18:51:10 $

=cut

# --
# Kernel/System/ImportExport/FormatBackend/CSV.pm - import/export backend for CSV
# Copyright (C) 2001-2010 OTRS AG, http://otrs.org/
# --
# $Id: CSV.pm,v 1.28 2010/04/13 12:18:28 ub Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::System::ImportExport::FormatBackend::CSV;

use strict;
use warnings;

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

=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,
            },
        },
    ];

    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;

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

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

    # parse the content
    my $LineCount = 1;
    my @ImportData;
    while ( 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 http://www.gnu.org/licenses/agpl.txt.

=cut

=head1 VERSION

$Revision: 1.28 $ $Date: 2010/04/13 12:18:28 $

=cut

# --
# ImportExport.t - all general import export tests
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: ImportExport.t,v 1.15 2009/05/18 09:42:53 mh Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

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 ( 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 ( 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 ( 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 ( 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-2010 OTRS AG, http://otrs.org/
# --
# $Id: ImportExportFormatCSV.t,v 1.11 2010/09/08 15:31:04 dz Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

use Data::Dumper;
use Kernel::System::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,
        },
    },
];

# turn off all pretty print
$Data::Dumper::Indent = 0;

# dump the list from FormatAttributesGet()
my $FormatAttributesGetDump1 = Data::Dumper::Dumper($FormatAttributesGet1);

# dump the reference table
my $FormatAttributesRefDump1 = Data::Dumper::Dumper($FormatAttributesGet1Reference);

$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()
my $MappingFormatAttributesGetDump1 = Data::Dumper::Dumper($MappingFormatAttributesGet1);

# dump the reference table
my $MappingFormatAttributesRefDump1 = Data::Dumper::Dumper($MappingFormatAttributesGet1Reference);

$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==
IyAtLQojIEltcG9ydEV4cG9ydC5wbSAtIGNvZGUgdG8gZXhjZWN1dGUgZHVyaW5nIHBhY2thZ2UgaW5zdGFsbGF0aW9uCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTAgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBJbXBvcnRFeHBvcnQucG0sdiAxLjMgMjAxMC8wNS8yMCAxOToyNDowMCB1YiBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSB2YXI6OnBhY2thZ2VzZXR1cDo6SW1wb3J0RXhwb3J0OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoQElTQSAkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjMgJCkgWzFdOwoKPWhlYWQxIE5BTUUKCkltcG9ydEV4cG9ydC5wbSAtIGNvZGUgdG8gZXhjZWN1dGUgZHVyaW5nIHBhY2thZ2UgaW5zdGFsbGF0aW9uCgo9aGVhZDEgU1lOT1BTSVMKCkFsbCBmdW5jdGlvbnMKCj1oZWFkMSBQVUJMSUMgSU5URVJGQUNFCgo9b3ZlciA0Cgo9Y3V0Cgo9aXRlbSBuZXcoKQoKY3JlYXRlIGFuIG9iamVjdAoKICAgIHVzZSBLZXJuZWw6OkNvbmZpZzsKICAgIHVzZSBLZXJuZWw6OlN5c3RlbTo6RW5jb2RlOwogICAgdXNlIEtlcm5lbDo6U3lzdGVtOjpMb2c7CiAgICB1c2UgS2VybmVsOjpTeXN0ZW06Ok1haW47CiAgICB1c2UgS2VybmVsOjpTeXN0ZW06OlRpbWU7CiAgICB1c2UgS2VybmVsOjpTeXN0ZW06OkRCOwogICAgdXNlIEtlcm5lbDo6U3lzdGVtOjpYTUw7CiAgICB1c2UgdmFyOjpwYWNrYWdlc2V0dXA6OkltcG9ydEV4cG9ydDsKCiAgICBteSAkQ29uZmlnT2JqZWN0ID0gS2VybmVsOjpDb25maWctPm5ldygpOwogICAgbXkgJEVuY29kZU9iamVjdCA9IEtlcm5lbDo6U3lzdGVtOjpFbmNvZGUtPm5ldygKICAgICAgICBDb25maWdPYmplY3QgPT4gJENvbmZpZ09iamVjdCwKICAgICk7CiAgICBteSAkTG9nT2JqZWN0ICAgID0gS2VybmVsOjpTeXN0ZW06OkxvZy0+bmV3KAogICAgICAgIENvbmZpZ09iamVjdCA9PiAkQ29uZmlnT2JqZWN0LAogICAgICAgIEVuY29kZU9iamVjdCA9PiAkRW5jb2RlT2JqZWN0LAogICAgKTsKICAgIG15ICRNYWluT2JqZWN0ID0gS2VybmVsOjpTeXN0ZW06Ok1haW4tPm5ldygKICAgICAgICBDb25maWdPYmplY3QgPT4gJENvbmZpZ09iamVjdCwKICAgICAgICBFbmNvZGVPYmplY3QgPT4gJEVuY29kZU9iamVjdCwKICAgICAgICBMb2dPYmplY3QgICAgPT4gJExvZ09iamVjdCwKICAgICk7CiAgICBteSAkVGltZU9iamVjdCA9IEtlcm5lbDo6U3lzdGVtOjpUaW1lLT5uZXcoCiAgICAgICAgQ29uZmlnT2JqZWN0ID0+ICRDb25maWdPYmplY3QsCiAgICAgICAgTG9nT2JqZWN0ICAgID0+ICRMb2dPYmplY3QsCiAgICApOwogICAgbXkgJERCT2JqZWN0ID0gS2VybmVsOjpTeXN0ZW06OkRCLT5uZXcoCiAgICAgICAgQ29uZmlnT2JqZWN0ID0+ICRDb25maWdPYmplY3QsCiAgICAgICAgRW5jb2RlT2JqZWN0ID0+ICRFbmNvZGVPYmplY3QsCiAgICAgICAgTG9nT2JqZWN0ICAgID0+ICRMb2dPYmplY3QsCiAgICAgICAgTWFpbk9iamVjdCAgID0+ICRNYWluT2JqZWN0LAogICAgKTsKICAgIG15ICRYTUxPYmplY3QgPSBLZXJuZWw6OlN5c3RlbTo6WE1MLT5uZXcoCiAgICAgICAgQ29uZmlnT2JqZWN0ID0+ICRDb25maWdPYmplY3QsCiAgICAgICAgRW5jb2RlT2JqZWN0ID0+ICRFbmNvZGVPYmplY3QsCiAgICAgICAgTG9nT2JqZWN0ICAgID0+ICRMb2dPYmplY3QsCiAgICAgICAgREJPYmplY3QgICAgID0+ICREQk9iamVjdCwKICAgICAgICBNYWluT2JqZWN0ICAgPT4gJE1haW5PYmplY3QsCiAgICApOwogICAgbXkgJENvZGVPYmplY3QgPSB2YXI6OnBhY2thZ2VzZXR1cDo6SW1wb3J0RXhwb3J0LT5uZXcoCiAgICAgICAgQ29uZmlnT2JqZWN0ID0+ICRDb25maWdPYmplY3QsCiAgICAgICAgRW5jb2RlT2JqZWN0ID0+ICRFbmNvZGVPYmplY3QsCiAgICAgICAgTG9nT2JqZWN0ICAgID0+ICRMb2dPYmplY3QsCiAgICAgICAgTWFpbk9iamVjdCAgID0+ICRNYWluT2JqZWN0LAogICAgICAgIFRpbWVPYmplY3QgICA9PiAkVGltZU9iamVjdCwKICAgICAgICBEQk9iamVjdCAgICAgPT4gJERCT2JqZWN0LAogICAgICAgIFhNTE9iamVjdCAgICA9PiAkWE1MT2JqZWN0LAogICAgKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgIyBjaGVjayBuZWVkZWQgb2JqZWN0cwogICAgZm9yIG15ICRPYmplY3QgKAogICAgICAgIHF3KENvbmZpZ09iamVjdCBMb2dPYmplY3QgRW5jb2RlT2JqZWN0IE1haW5PYmplY3QgVGltZU9iamVjdCBEQk9iamVjdCBYTUxPYmplY3QpCiAgICAgICAgKQogICAgewogICAgICAgICRTZWxmLT57JE9iamVjdH0gPSAkUGFyYW17JE9iamVjdH0gfHwgZGllICJHb3Qgbm8gJE9iamVjdCEiOwogICAgfQoKICAgIHJldHVybiAkU2VsZjsKfQoKPWl0ZW0gQ29kZUluc3RhbGwoKQoKcnVuIHRoZSBjb2RlIGluc3RhbGwgcGFydAoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+Q29kZUluc3RhbGwoKTsKCj1jdXQKCnN1YiBDb2RlSW5zdGFsbCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiAxOwp9Cgo9aXRlbSBDb2RlUmVpbnN0YWxsKCkKCnJ1biB0aGUgY29kZSByZWluc3RhbGwgcGFydAoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+Q29kZVJlaW5zdGFsbCgpOwoKPWN1dAoKc3ViIENvZGVSZWluc3RhbGwgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gMTsKfQoKPWl0ZW0gQ29kZVVwZ3JhZGUoKQoKcnVuIHRoZSBjb2RlIHVwZ3JhZGUgcGFydAoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+Q29kZVVwZ3JhZGUoKTsKCj1jdXQKCnN1YiBDb2RlVXBncmFkZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiAxOwp9Cgo9aXRlbSBDb2RlVXBncmFkZUZyb21CZWZvcmVfMl8wXzMoKQoKVGhpcyBmdW5jdGlvbiBpcyBvbmx5IGV4ZWN1dGVkIGlmIHRoZSBpbnN0YWxsZWQgbW9kdWxlIHZlcnNpb24gaXMgc21hbGxlciB0aGFuIDIuMC4zLgoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+Q29kZVVwZ3JhZGVGcm9tQmVmb3JlXzJfMF8zKCk7Cgo9Y3V0CgpzdWIgQ29kZVVwZ3JhZGVGcm9tQmVmb3JlXzJfMF8zIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBmaXggYSB0eXBvIGluIHRoZSBkYXRhYmFzZQogICAgJFNlbGYtPl9GaXhEYXRhYmFzZVR5cG8oKTsKCiAgICByZXR1cm4gMTsKfQoKPWl0ZW0gQ29kZVVuaW5zdGFsbCgpCgpydW4gdGhlIGNvZGUgdW5pbnN0YWxsIHBhcnQKCiAgICBteSAkUmVzdWx0ID0gJENvZGVPYmplY3QtPkNvZGVVbmluc3RhbGwoKTsKCj1jdXQKCnN1YiBDb2RlVW5pbnN0YWxsIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIDE7Cn0KCj1iZWdpbiBJbnRlcm5hbDoKCj1pdGVtIF9GaXhEYXRhYmFzZVR5cG8oKQoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+X0ZpeERhdGFiYXNlVHlwbygpOwoKPWN1dAoKc3ViIF9GaXhEYXRhYmFzZVR5cG8gewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGZpeCB0aGUgQ29sdW1uU2VwZXJhdG9yIHR5cG8gKGNvcnJlY3QgaXMgQ29sdW1uU2VwYXJhdG9yKQogICAgcmV0dXJuIGlmICEkU2VsZi0+e0RCT2JqZWN0fS0+RG8oCiAgICAgICAgU1FMID0+ICJVUERBVEUgaW1leHBvcnRfZm9ybWF0ICIKICAgICAgICAgICAgLiAiU0VUIGRhdGFfa2V5ID0gJ0NvbHVtblNlcGFyYXRvcicgIgogICAgICAgICAgICAuICJXSEVSRSBkYXRhX2tleSA9ICdDb2x1bW5TZXBlcmF0b3InIiwKICAgICk7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cgo9ZW5kIEludGVybmFsOgoKPWJhY2sKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBTb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKGh0dHA6Ly9vdHJzLm9yZy8pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTIuMC50eHQuCgo9Y3V0Cgo9aGVhZDEgVkVSU0lPTgoKJFJldmlzaW9uOiAxLjMgJCAkRGF0ZTogMjAxMC8wNS8yMCAxOToyNDowMCAkCgo9Y3V0Cg==
LyoqCiAqIEBwcm9qZWN0ICAgICBPVFJTIChodHRwOi8vd3d3Lm90cnMub3JnKSAtIEFnZW50IEZyb250ZW5kCiAqIEB2ZXJzaW9uICAgICAkUmV2aXNpb246IDEuNCAkCiAqIEBjb3B5cmlnaHQgICBPVFJTIEFHCiAqIEBsaWNlbnNlICAgICBBR1BMIChodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQpCiAqLwoKLyoqCiAqIEBwYWNrYWdlICAgICBTa2luICJEZWZhdWx0IgogKiBAc2VjdGlvbiAgICAgSW1wb3J0IEV4cG9ydCBTY3JlZW4KICovCgpAbWVkaWEgc2NyZWVuLHByb2plY3Rpb24sdHYsaGFuZGhlbGQgewoKLyoqCiAqIEBzdWJzZWN0aW9uCiAqLwoKCi5NYXBIZWFkZXJSb3cgbGFiZWwgewogICAgY29sb3I6ICM5MjkyOTI7Cn0KCi5NYXBIZWFkZXJSb3cgLkhlYWRlciwKLk1hcEhlYWRlclJvdyAuRmllbGQgewogICAgZGlzcGxheTogaW5saW5lOwogICAgbWFyZ2luLXJpZ2h0OiAxNXB4OwogICAgcGFkZGluZy1sZWZ0OiAycHg7Cn0KCmJ1dHRvbi5BcnJvd1VwLApidXR0b24uQXJyb3dEb3duIHsKICAgIGhlaWdodDogMTZweDsKICAgIHdpZHRoOiAxNnB4OwogICAgcGFkZGluZzogMXB4OwogICAgbWFyZ2luLXRvcDogMDsKICAgIG1hcmdpbi1ib3R0b206IDFweDsKICAgIGJvcmRlci1zdHlsZTogbm9uZTsKICAgIHRleHQtaW5kZW50OiAtOTk5OXB4OwogICAgZGlzcGxheTogaW5saW5lLWJsb2NrOwogICAgdmVydGljYWwtYWxpZ246IG1pZGRsZTsKICAgIGN1cnNvcjogcG9pbnRlcjsKfQoKYnV0dG9uLkFycm93VXA6YWN0aXZlLApidXR0b24uQXJyb3dEb3duOmFjdGl2ZSB7CiAgICBtYXJnaW4tdG9wOiAxcHg7CiAgICBtYXJnaW4tYm90dG9tOiAwOwp9CgpidXR0b24uQXJyb3dVcFtkaXNhYmxlZD0iZGlzYWJsZWQiXTphY3RpdmUsCmJ1dHRvbi5BcnJvd0Rvd25bZGlzYWJsZWQ9ImRpc2FibGVkIl06YWN0aXZlIHsKICAgIG1hcmdpbi10b3A6IDBweDsKICAgIG1hcmdpbi1ib3R0b206IDFweDsKfQoKYnV0dG9uLkFycm93VXAgewogICAgYmFja2dyb3VuZDogdXJsKC4uL2ltZy9pY29ucy9hcnJvd191cC5wbmcpOwp9CgpidXR0b24uQXJyb3dVcFtkaXNhYmxlZD0iZGlzYWJsZWQiXSB7CiAgICBiYWNrZ3JvdW5kOiB1cmwoLi4vaW1nL2ljb25zL2Fycm93X3VwX2Rpc2FibGVkLnBuZyk7CiAgICBjdXJzb3I6IGRlZmF1bHQ7Cn0KCmJ1dHRvbi5BcnJvd0Rvd24gewogICAgYmFja2dyb3VuZDogdXJsKC4uL2ltZy9pY29ucy9hcnJvd19kb3duLnBuZyk7Cn0KCmJ1dHRvbi5BcnJvd0Rvd25bZGlzYWJsZWQ9ImRpc2FibGVkIl17CiAgICBiYWNrZ3JvdW5kOiB1cmwoLi4vaW1nL2ljb25zL2Fycm93X2Rvd25fZGlzYWJsZWQucG5nKTsKICAgIGN1cnNvcjogZGVmYXVsdDsKfQoKfSAvKiBlbmQgQG1lZGlhICov
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==