ImportExport
1.3.1
OTRS AG
http://otrs.org/
GNU AFFERO GENERAL PUBLIC LICENSE Version 3, November 2007
Build for OTRS::ITSM 1.3.1.
The ImportExport package.
Das ImportExport Paket.
2.4.x
<br>
<b>ATTENTION</b>
<br>
<br>
If you uninstall this package, all database tables that were created during installation will be deleted.
All data from these tables will be irrevocably lost!
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>ACHTUNG</b>
<br>
<br>
Bei der Deinstallation werden die von diesem Paket angelegten Datenbank-Tabellen gelöscht.
Alle darin enthaltenen Daten gehen unwiderruflich verloren!
<br>
<br>
((enjoy))<br>
<br>
2009-07-27 10:09:16
opms.otrs.com
IyEvdXNyL2Jpbi9wZXJsIC13CiMgLS0KIyBJbXBvcnRFeHBvcnQucGwgLSBpbXBvcnQvZXhwb3J0IHNjcmlwdAojIENvcHlyaWdodCAoQykgMjAwMS0yMDA5IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogSW1wb3J0RXhwb3J0LnBsLHYgMS4xMCAyMDA5LzA1LzE4IDA5OjQyOjUyIG1oIEV4cCAkCiMgLS0KIyBUaGlzIHByb2dyYW0gaXMgZnJlZSBzb2Z0d2FyZTsgeW91IGNhbiByZWRpc3RyaWJ1dGUgaXQgYW5kL29yIG1vZGlmeQojIGl0IHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgR05VIEFGRkVSTyBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGFzIHB1Ymxpc2hlZCBieQojIHRoZSBGcmVlIFNvZnR3YXJlIEZvdW5kYXRpb247IGVpdGhlciB2ZXJzaW9uIDMgb2YgdGhlIExpY2Vuc2UsIG9yCiMgYW55IGxhdGVyIHZlcnNpb24uCiMKIyBUaGlzIHByb2dyYW0gaXMgZGlzdHJpYnV0ZWQgaW4gdGhlIGhvcGUgdGhhdCBpdCB3aWxsIGJlIHVzZWZ1bCwKIyBidXQgV0lUSE9VVCBBTlkgV0FSUkFOVFk7IHdpdGhvdXQgZXZlbiB0aGUgaW1wbGllZCB3YXJyYW50eSBvZgojIE1FUkNIQU5UQUJJTElUWSBvciBGSVRORVNTIEZPUiBBIFBBUlRJQ1VMQVIgUFVSUE9TRS4gIFNlZSB0aGUKIyBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBmb3IgbW9yZSBkZXRhaWxzLgojCiMgWW91IHNob3VsZCBoYXZlIHJlY2VpdmVkIGEgY29weSBvZiB0aGUgR05VIEFmZmVybyBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlCiMgYWxvbmcgd2l0aCB0aGlzIHByb2dyYW07IGlmIG5vdCwgd3JpdGUgdG8gdGhlIEZyZWUgU29mdHdhcmUKIyBGb3VuZGF0aW9uLCBJbmMuLCA1OSBUZW1wbGUgUGxhY2UsIFN1aXRlIDMzMCwgQm9zdG9uLCBNQSAgMDIxMTEtMTMwNyAgVVNBCiMgb3Igc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIEZpbGU6OkJhc2VuYW1lOwp1c2UgRmluZEJpbiBxdygkUmVhbEJpbik7CnVzZSBsaWIgZGlybmFtZSgkUmVhbEJpbik7CnVzZSBsaWIgZGlybmFtZSgkUmVhbEJpbikgLiAnL0tlcm5lbC9jcGFuLWxpYic7Cgp1c2UgR2V0b3B0OjpTdGQ7CnVzZSBLZXJuZWw6OkNvbmZpZzsKdXNlIEtlcm5lbDo6U3lzdGVtOjpEQjsKdXNlIEtlcm5lbDo6U3lzdGVtOjpFbmNvZGU7CnVzZSBLZXJuZWw6OlN5c3RlbTo6SW1wb3J0RXhwb3J0Owp1c2UgS2VybmVsOjpTeXN0ZW06OkxvZzsKdXNlIEtlcm5lbDo6U3lzdGVtOjpNYWluOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04gJFJlYWxCaW4pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS4xMCAkKSBbMV07CgojIGdldCBvcHRpb25zCm15ICVPcHRzOwpnZXRvcHQoICdobmFpbycsIFwlT3B0cyApOwoKaWYgKCAkT3B0c3tofSApIHsKCiAgICBwcmludCBTVERPVVQgIkltcG9ydEV4cG9ydC5wbCA8UmV2aXNpb24gJFZFUlNJT04+IC0gYSBpbXBvcnQvZXhwb3J0IHRvb2xcbiI7CiAgICBwcmludCBTVERPVVQgIkNvcHlyaWdodCAoQykgMjAwMS0yMDA5IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy9cbiI7CiAgICBwcmludCBTVERPVVQgInVzYWdlOkltcG9ydEV4cG9ydC5wbCAtbiA8VGVtcGxhdGVOdW1iZXI+IC1hIGltcG9ydHxleHBvcnQgIjsKICAgIHByaW50IFNURE9VVCAiWy1pIDxTb3VyY2VGaWxlPl0gWy1vIDxEZXN0aW5hdGlvbkZpbGU+XVxuIjsKICAgIHByaW50IFNURE9VVCAiXG4iOwogICAgcHJpbnQgU1RET1VUICIgICBleGFtcGxlczpcbiI7CiAgICBwcmludCBTVERPVVQgIiAgICAgICBJbXBvcnRFeHBvcnQucGwgLW4gMDAwMDQgLWEgaW1wb3J0IC1pIC90bXAvaW1wb3J0LmNzdlxuIjsKICAgIHByaW50IFNURE9VVCAiICAgICAgIEltcG9ydEV4cG9ydC5wbCAtbiAwMDAwNCAtYSBleHBvcnQgLW8gL3RtcC9leHBvcnQuY3N2XG4iOwoKICAgIGV4aXQgMTsKfQoKIyBjaGVjayB0ZW1wYWx0ZSBudW1iZXIKaWYgKCAhJE9wdHN7bn0gKSB7CiAgICBwcmludCBTVERFUlIgIkVSUk9SOiBOZWVkIC1uIFRlbXBsYXRlTnVtYmVyXG4iOwogICAgZXhpdCAxOwp9CmlmICggJE9wdHN7bn0gIX4gbXsgXEEgXGQrIFx6IH14bXMgKSB7CiAgICBwcmludCBTVERFUlIgIkVSUk9SOiBJbnZhbGlkIFRlbXBsYXRlTnVtYmVyXG4iOwogICAgZXhpdCAxOwp9Cm15ICRUZW1wbGF0ZUlEID0gaW50ICRPcHRze259OwoKIyBjaGVjayBhY3Rpb24gbW9kZQppZiAoICEkT3B0c3thfSApIHsKICAgIHByaW50IFNUREVSUiAiRVJST1I6IE5lZWQgLWEgaW1wb3J0fGV4cG9ydFxuIjsKICAgIGV4aXQgMTsKfQppZiAoIGxjICRPcHRze2F9IG5lICdpbXBvcnQnICYmIGxjICRPcHRze2F9IG5lICdleHBvcnQnICkgewogICAgcHJpbnQgU1RERVJSICJFUlJPUjogSW52YWxpZCBhY3Rpb25cbiI7CiAgICBleGl0IDE7Cn0KCiMgY3JlYXRlIGNvbW1vbiBvYmplY3RzCm15ICVDb21tb25PYmplY3Q7CiRDb21tb25PYmplY3R7Q29uZmlnT2JqZWN0fSA9IEtlcm5lbDo6Q29uZmlnLT5uZXcoKTsKJENvbW1vbk9iamVjdHtMb2dPYmplY3R9ICAgID0gS2VybmVsOjpTeXN0ZW06OkxvZy0+bmV3KAogICAgTG9nUHJlZml4ID0+ICdPVFJTLUltcG9ydEV4cG9ydCcsCiAgICAlQ29tbW9uT2JqZWN0LAopOwokQ29tbW9uT2JqZWN0e0VuY29kZU9iamVjdH0gICAgICAgPSBLZXJuZWw6OlN5c3RlbTo6RW5jb2RlLT5uZXcoJUNvbW1vbk9iamVjdCk7CiRDb21tb25PYmplY3R7TWFpbk9iamVjdH0gICAgICAgICA9IEtlcm5lbDo6U3lzdGVtOjpNYWluLT5uZXcoJUNvbW1vbk9iamVjdCk7CiRDb21tb25PYmplY3R7REJPYmplY3R9ICAgICAgICAgICA9IEtlcm5lbDo6U3lzdGVtOjpEQi0+bmV3KCVDb21tb25PYmplY3QpOwokQ29tbW9uT2JqZWN0e0ltcG9ydEV4cG9ydE9iamVjdH0gPSBLZXJuZWw6OlN5c3RlbTo6SW1wb3J0RXhwb3J0LT5uZXcoJUNvbW1vbk9iamVjdCk7CgojIGdldCB0ZW1wbGF0ZSBkYXRhCm15ICRUZW1wbGF0ZURhdGEgPSAkQ29tbW9uT2JqZWN0e0ltcG9ydEV4cG9ydE9iamVjdH0tPlRlbXBsYXRlR2V0KAogICAgVGVtcGxhdGVJRCA9PiAkVGVtcGxhdGVJRCwKICAgIFVzZXJJRCAgICAgPT4gMSwKKTsKCmlmICggISRUZW1wbGF0ZURhdGEtPntUZW1wbGF0ZUlEfSApIHsKICAgIHByaW50IFNUREVSUiAiRVJST1I6IFRlbXBsYXRlICRPcHRze259IG5vdCBmb3VuZCFcbiI7CiAgICBwcmludCBTVERFUlIgIkV4cG9ydCBhYm9ydGVkLlxuIjsKICAgIGV4aXQgMTsKfQoKIyB0aW1lIHRvIHN0YXJ0CmlmICggbGMgJE9wdHN7YX0gZXEgJ2ltcG9ydCcgKSB7CgogICAgbXkgJFNvdXJjZUNvbnRlbnQgPSBcZG8geycnfTsKICAgIGlmICggJE9wdHN7aX0gKSB7CgogICAgICAgIHByaW50IFNURE9VVCAiUmVhZCBGaWxlICRPcHRze2l9LlxuIjsKCiAgICAgICAgIyByZWFkIHNvdXJjZSBmaWxlCiAgICAgICAgJFNvdXJjZUNvbnRlbnQgPSAkQ29tbW9uT2JqZWN0e01haW5PYmplY3R9LT5GaWxlUmVhZCgKICAgICAgICAgICAgTG9jYXRpb24gPT4gJE9wdHN7aX0sCiAgICAgICAgICAgIFJlc3VsdCAgID0+ICdTQ0FMQVInLAogICAgICAgICAgICBNb2RlICAgICA9PiAnYmlubW9kZScsCiAgICAgICAgKTsKCiAgICAgICAgZGllICJDYW4ndCByZWFkIGZpbGUgJE9wdHN7aX0uXG5JbXBvcnQgYWJvcnRlZC5cbiIgaWYgISRTb3VyY2VDb250ZW50OwogICAgfQoKICAgIHByaW50IFNURE9VVCAiSW1wb3J0IGluIHByb2Nlc3MuLi5cbiI7CgogICAgIyBpbXBvcnQgZGF0YQogICAgbXkgJFJlc3VsdCA9ICRDb21tb25PYmplY3R7SW1wb3J0RXhwb3J0T2JqZWN0fS0+SW1wb3J0KAogICAgICAgIFRlbXBsYXRlSUQgICAgPT4gJFRlbXBsYXRlSUQsCiAgICAgICAgU291cmNlQ29udGVudCA9PiAkU291cmNlQ29udGVudCwKICAgICAgICBVc2VySUQgICAgICAgID0+IDEsCiAgICApOwoKICAgIGRpZSAiXG5FcnJvciBvY2N1cnJlZC4gSW1wb3J0IGltcG9zc2libGUhIFNlZSBTeXNsb2cgZm9yIGRldGFpbHMuXG4iIGlmICFkZWZpbmVkICRSZXN1bHQ7CgogICAgcHJpbnQgU1RET1VUICJcbiI7CiAgICBwcmludCBTVERPVVQgIlN1Y2Nlc3M6ICRSZXN1bHQtPntTdWNjZXNzfVxuIjsKICAgIHByaW50IFNURE9VVCAiRmFpbGVkIDogJFJlc3VsdC0+e0ZhaWxlZH1cbiI7CiAgICBwcmludCBTVERPVVQgIlxuIjsKICAgIHByaW50IFNURE9VVCAiSW1wb3J0IGNvbXBsZXRlLlxuIjsKfQplbHNpZiAoIGxjICRPcHRze2F9IGVxICdleHBvcnQnICkgewoKICAgIHByaW50IFNURE9VVCAiRXhwb3J0IGluIHByb2Nlc3MuLi5cbiI7CgogICAgIyBleHBvcnQgZGF0YQogICAgbXkgJFJlc3VsdCA9ICRDb21tb25PYmplY3R7SW1wb3J0RXhwb3J0T2JqZWN0fS0+RXhwb3J0KAogICAgICAgIFRlbXBsYXRlSUQgPT4gJFRlbXBsYXRlSUQsCiAgICAgICAgVXNlcklEICAgICA9PiAxLAogICAgKTsKCiAgICBkaWUgIlxuRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLlxuIiBpZiAhZGVmaW5lZCAkUmVzdWx0OwoKICAgIHByaW50IFNURE9VVCAiXG4iOwogICAgcHJpbnQgU1RET1VUICJTdWNjZXNzOiAkUmVzdWx0LT57U3VjY2Vzc31cbiI7CiAgICBwcmludCBTVERPVVQgIkZhaWxlZCA6ICRSZXN1bHQtPntGYWlsZWR9XG4iOwogICAgcHJpbnQgU1RET1VUICJcbiI7CgogICAgaWYgKCAkT3B0c3tvfSApIHsKCiAgICAgICAgbXkgJEZpbGVDb250ZW50ID0gam9pbiAiXG4iLCBAeyAkUmVzdWx0LT57RGVzdGluYXRpb25Db250ZW50fSB9OwoKICAgICAgICAjIHNhdmUgZGVzdGluYXRpb24gY29udGVudCB0byBmaWxlCiAgICAgICAgbXkgJFN1Y2Nlc3MgPSAkQ29tbW9uT2JqZWN0e01haW5PYmplY3R9LT5GaWxlV3JpdGUoCiAgICAgICAgICAgIExvY2F0aW9uID0+ICRPcHRze299LAogICAgICAgICAgICBDb250ZW50ICA9PiBcJEZpbGVDb250ZW50LAogICAgICAgICk7CgogICAgICAgIGRpZSAiQ2FuJ3Qgd3JpdGUgZmlsZSAkT3B0c3tvfS5cbkV4cG9ydCBhYm9ydGVkLlxuIiBpZiAhJFN1Y2Nlc3M7CgogICAgICAgIHByaW50IFNURE9VVCAiRmlsZSAkT3B0c3tvfSBzYXZlZC5cbiI7CiAgICB9CgogICAgcHJpbnQgU1RET1VUICJFeHBvcnQgY29tcGxldGUuXG4iOwp9CgpleGl0IDA7Cgo9YmFjawoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIFNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoaHR0cDovL290cnMub3JnLykuCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251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LZP2rUDD3ZS6XUKFHKcZM3zOMAwanS5cRlX0LgEAAABMtZTE55z0LgEAAAAcTW993P+BdAkAAADgaIa7cb950iUAAAAA4qRLAAAAwJvV7SHqLljrtRcday3b/jnD9qXR6U5UxpnZ1RsAAAB4s7rxzfBP3HKXetph7Zjckrfhdku9M7npYmVUZZGRGAAAAMBB9bQC1VPJKVgZBwAAAECcdAkAAACAuLfclwUAAADAqeldAgAAACBOugQAAABAnHQJAAAAgDjpEgAAAABx0iUAAAAA4qRLAAAAAMRJlwAAAACIky4BAAAAECddAgAAACBOugQAAABAnHQJAAAAgLh3ly4AAAAAYAGapmkPUkq9j4VL0x9SnrE88rL0LgEAAAAcllLqJTvdj4VLw+cUnpmbsTzysvQuAQAAABzZvtXo6lCD0uhd1QZJo6RLAAAAAIe10c+U3GfKqrfuY7sNSoUZuyOrYmUcAAAAwFSjTUnD4+7HwhPmzlinSkMvAAAAABZB7xIAAAAAcdIlAAAAAOKkSwAAAADESZcAAAAAiJMuAQAAABAnXQIAAAAgTroEAAAAQNw/LkNDE7BQzoUAAAAASUVORK5CYII=
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSIgPz4KPG90cnNfY29uZmlnIHZlcnNpb249IjEuMCIgaW5pdD0iRnJhbWV3b3JrIj4KICAgIDxDVlM+JElkOiBJbXBvcnRFeHBvcnQueG1sLHYgMS4zIDIwMDgvMDEvMjQgMTY6MzM6NTYgbWggRXhwICQ8L0NWUz4KICAgIDxDb25maWdJdGVtIE5hbWU9IkZyb250ZW5kOjpNb2R1bGUjIyNBZG1pbkltcG9ydEV4cG9ydCIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkZyb250ZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBBZG1pbkltcG9ydEV4cG9ydCBpbiB0aGUgYWRtaW4gYXJlYS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+RnJvbnRlbmRtb2R1bC1SZWdpc3RyYXRpb24gZGVyIEFkbWluSW1wb3J0RXhwb3J0IGltIEFkbWluLUJlcmVpY2guPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+SW1wb3J0RXhwb3J0PC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFkbWluOjpNb2R1bGVSZWdpc3RyYXRpb248L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8RnJvbnRlbmRNb2R1bGVSZWc+CiAgICAgICAgICAgICAgICA8R3JvdXA+YWRtaW48L0dyb3VwPgogICAgICAgICAgICAgICAgPERlc2NyaXB0aW9uPkFkbWluPC9EZXNjcmlwdGlvbj4KICAgICAgICAgICAgICAgIDxUaXRsZT5JbXBvcnQvRXhwb3J0PC9UaXRsZT4KICAgICAgICAgICAgICAgIDxOYXZCYXJOYW1lPkFkbWluPC9OYXZCYXJOYW1lPgogICAgICAgICAgICAgICAgPE5hdkJhck1vZHVsZT4KICAgICAgICAgICAgICAgICAgICA8TW9kdWxlPktlcm5lbDo6T3V0cHV0OjpIVE1MOjpOYXZCYXJNb2R1bGVBZG1pbjwvTW9kdWxlPgogICAgICAgICAgICAgICAgICAgIDxOYW1lPkltcG9ydC9FeHBvcnQ8L05hbWU+CiAgICAgICAgICAgICAgICAgICAgPEJsb2NrPkJsb2NrNDwvQmxvY2s+CiAgICAgICAgICAgICAgICAgICAgPFByaW8+NzEwPC9QcmlvPgogICAgICAgICAgICAgICAgPC9OYXZCYXJNb2R1bGU+CiAgICAgICAgICAgIDwvRnJvbnRlbmRNb2R1bGVSZWc+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iSW1wb3J0RXhwb3J0OjpGb3JtYXRCYWNrZW5kUmVnaXN0cmF0aW9uIyMjQ1NWIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+Rm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWwuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkZvcm1hdC1CYWNrZW5kIE1vZHVsIFJlZ2lzdHJhdGlvbiBkZXMgSW1wb3J0L0V4cG9ydCBNb2R1bHMuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+SW1wb3J0RXhwb3J0PC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Rm9ybWF0QmFja2VuZDo6TW9kdWxlUmVnaXN0cmF0aW9uPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik1vZHVsZSI+S2VybmVsOjpTeXN0ZW06OkltcG9ydEV4cG9ydDo6Rm9ybWF0QmFja2VuZDo6Q1NWPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJOYW1lIj5DU1Y8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+Cjwvb3Ryc19jb25maWc+
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9iZ19JbXBvcnRFeHBvcnQucG0gLSB0aGUgYnVsZ2FyaWFuIHRyYW5zbGF0aW9uIG9mIEltcG9ydEV4cG9ydAojIENvcHlyaWdodCAoQykgMjAwMS0yMDA5IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyBDb3B5cmlnaHQgKEMpIDIwMDctMjAwOCBNaWxlbiBLb3V0ZXYKIyAtLQojICRJZDogYmdfSW1wb3J0RXhwb3J0LnBtLHYgMS44IDIwMDkvMDUvMTggMDk6NDI6NTIgbWggRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6YmdfSW1wb3J0RXhwb3J0OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS44ICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAkTGFuZy0+eydJbXBvcnQvRXhwb3J0J30gICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSAgID0gJyc7CiAgICAkTGFuZy0+eydBZGQgbWFwcGluZyB0ZW1wbGF0ZSd9ICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydTdGFydCBJbXBvcnQnfSAgICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydTdGFydCBFeHBvcnQnfSAgICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydTdGVwJ30gICAgICAgICAgICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ICAgID0gJyc7CiAgICAkTGFuZy0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ICAgID0gJyc7CiAgICAkTGFuZy0+eydFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ICAgID0gJyc7CiAgICAkTGFuZy0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSAgID0gJyc7CiAgICAkTGFuZy0+eydFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ICAgID0gJyc7CiAgICAkTGFuZy0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydDb2x1bW4nfSAgICAgICAgICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJyc7CiAgICAkTGFuZy0+eydTb3VyY2UgRmlsZSd9ICAgICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydDb2x1bW4gU2VwZXJhdG9yJ30gICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydUYWJ1bGF0b3IgKFRBQiknfSAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydTZW1pY29sb24gKDspJ30gICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydDb2xvbiAoOiknfSAgICAgICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydEb3QgKC4pJ30gICAgICAgICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydDaGFyc2V0J30gICAgICAgICAgICAgICAgICAgID0gJyc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9jdF9JbXBvcnRFeHBvcnQucG0gLSB0aGUgY2F0YWxhbiB0cmFuc2xhdGlvbiBvZiBJbXBvcnRFeHBvcnQKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgQ29weXJpZ2h0IChDKSAyMDA4IFNpc3RlbWVzIE9USUMgKGlic2FsdXQpIC0gQW50b25pbyBMaW5kZQojIC0tCiMgJElkOiBjdF9JbXBvcnRFeHBvcnQucG0sdiAxLjMgMjAwOS8wNS8xOCAwOTo0Mjo1MiBtaCBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpjdF9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjMgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQnfSAgICAgICAgICAgICAgPSAnSW1wb3J0YXIvRXhwb3J0YXInOwogICAgJExhbmctPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gICA9ICdHZXN0afIgZGUgSW1wb3J0YXIvRXhwb3J0YXInOwogICAgJExhbmctPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSAgICAgICA9ICdBZmVnaXIgcGxhbnRpbGxhIGRlIG1hcGF0Z2UnOwogICAgJExhbmctPnsnU3RhcnQgSW1wb3J0J30gICAgICAgICAgICAgICA9ICdDb21lbudhciBpbXBvcnRhY2nzJzsKICAgICRMYW5nLT57J1N0YXJ0IEV4cG9ydCd9ICAgICAgICAgICAgICAgPSAnQ29tZW7nYXIgZXhwb3J0YWNp8yc7CiAgICAkTGFuZy0+eydTdGVwJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ1Bhcyc7CiAgICAkTGFuZy0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ICAgID0gJ0VkaXRhciBpbmZvcm1hY2nzIGNvbXVuYSc7CiAgICAkTGFuZy0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ICAgID0gJ0VkaXRhciBpbmZvcm1hY2nzIGRcJ29iamVjdGUnOwogICAgJExhbmctPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSAgICA9ICdFZGl0YXIgaW5mb3JtYWNp8yBkZSBmb3JtYXQnOwogICAgJExhbmctPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gICA9ICdFZGl0YXIgaW5mb3JtYWNp8yBkZSBtYXBhdGdlJzsKICAgICRMYW5nLT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gICAgPSAnRWRpdGFyIGluZm9ybWFjafMgZGUgcmVjZXJjYSc7CiAgICAkTGFuZy0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSAgICAgICAgID0gJ0ltcG9ydGFyIGluZm9ybWFjafMnOwogICAgJExhbmctPnsnQ29sdW1uJ30gICAgICAgICAgICAgICAgICAgICA9ICdDb2x1bW5hJzsKICAgICRMYW5nLT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnUmVzdHJpbmdpciBleHBvcnRhY2nzIHBlciByZWNlcmNhJzsKICAgICRMYW5nLT57J1NvdXJjZSBGaWxlJ30gICAgICAgICAgICAgICAgPSAnRml0eGVyIGZvbnQnOwogICAgJExhbmctPnsnQ29sdW1uIFNlcGVyYXRvcid9ICAgICAgICAgICA9ICdTZXBhcmFkb3IgZGUgY29sdW1uYSc7CiAgICAkTGFuZy0+eydUYWJ1bGF0b3IgKFRBQiknfSAgICAgICAgICAgID0gJ1RhYnVsYWRvciAoVEFCKSc7CiAgICAkTGFuZy0+eydTZW1pY29sb24gKDspJ30gICAgICAgICAgICAgID0gJ1B1bnQgaSBjb21hICg7KSc7CiAgICAkTGFuZy0+eydDb2xvbiAoOiknfSAgICAgICAgICAgICAgICAgID0gJ0RvcyBwdW50cyAoOiknOwogICAgJExhbmctPnsnRG90ICguKSd9ICAgICAgICAgICAgICAgICAgICA9ICdQdW50ICguKSc7CiAgICAkTGFuZy0+eydDaGFyc2V0J30gICAgICAgICAgICAgICAgICAgID0gJ0Nvbmp1bnQgZGUgY2Fy4GN0ZXJzJzsKCiAgICByZXR1cm4gMTsKfQoKMTsK
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9jel9JbXBvcnRFeHBvcnQucG0gLSB0aGUgY3plY2ggdHJhbnNsYXRpb24gb2YgSW1wb3J0RXhwb3J0CiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDkgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBjel9JbXBvcnRFeHBvcnQucG0sdiAxLjcgMjAwOS8wNS8xOCAwOTo0Mjo1MiBtaCBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpjel9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjcgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQnfSAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQgTWFuYWdlbWVudCd9ICAgPSAnJzsKICAgICRMYW5nLT57J0FkZCBtYXBwaW5nIHRlbXBsYXRlJ30gICAgICAgPSAnJzsKICAgICRMYW5nLT57J1N0YXJ0IEltcG9ydCd9ICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1N0YXJ0IEV4cG9ydCd9ICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1N0ZXAnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gICAgPSAnJzsKICAgICRMYW5nLT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gICAgPSAnJzsKICAgICRMYW5nLT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gICAgPSAnJzsKICAgICRMYW5nLT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ICAgPSAnJzsKICAgICRMYW5nLT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gICAgPSAnJzsKICAgICRMYW5nLT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0NvbHVtbid9ICAgICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnJzsKICAgICRMYW5nLT57J1NvdXJjZSBGaWxlJ30gICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0NvbHVtbiBTZXBlcmF0b3InfSAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1RhYnVsYXRvciAoVEFCKSd9ICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J1NlbWljb2xvbiAoOyknfSAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0NvbG9uICg6KSd9ICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0RvdCAoLiknfSAgICAgICAgICAgICAgICAgICAgPSAnJzsKICAgICRMYW5nLT57J0NoYXJzZXQnfSAgICAgICAgICAgICAgICAgICAgPSAnJzsKCiAgICByZXR1cm4gMTsKfQoKMTsK
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9kZV9JbXBvcnRFeHBvcnQucG0gLSB0aGUgZ2VybWFuIHRyYW5zbGF0aW9uIG9mIEltcG9ydEV4cG9ydAojIENvcHlyaWdodCAoQykgMjAwMS0yMDA5IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogZGVfSW1wb3J0RXhwb3J0LnBtLHYgMS4xOSAyMDA5LzA3LzIxIDAwOjU5OjQyIHViIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmRlX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuMTkgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQnfSAgICAgICAgICAgICAgPSAnSW1wb3J0L0V4cG9ydCc7CiAgICAkTGFuZy0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSAgID0gJ0ltcG9ydC9FeHBvcnQgVmVyd2FsdHVuZyc7CiAgICAkTGFuZy0+eydBZGQgbWFwcGluZyB0ZW1wbGF0ZSd9ICAgICAgID0gJ01hcHBpbmctVGVtcGxhdGUgaGluenVm/Gdlbic7CiAgICAkTGFuZy0+eydTdGFydCBJbXBvcnQnfSAgICAgICAgICAgICAgID0gJ0ltcG9ydCBzdGFydGVuJzsKICAgICRMYW5nLT57J1N0YXJ0IEV4cG9ydCd9ICAgICAgICAgICAgICAgPSAnRXhwb3J0IHN0YXJ0ZW4nOwogICAgJExhbmctPnsnU3RlcCd9ICAgICAgICAgICAgICAgICAgICAgICA9ICdTY2hyaXR0JzsKICAgICRMYW5nLT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gICAgPSAnQWxsZ2VtZWluZSBJbmZvcm1hdGlvbmVuIGJlYXJiZWl0ZW4nOwogICAgJExhbmctPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSAgICA9ICdPYmpla3QtSW5mb3JtYXRpb25lbiBiZWFyYmVpdGVuJzsKICAgICRMYW5nLT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gICAgPSAnRm9ybWF0LUluZm9ybWF0aW9uZW4gYmVhcmJlaXRlbic7CiAgICAkTGFuZy0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSAgID0gJ01hcHBpbmctSW5mb3JtYXRpb25lbiBiZWFyYmVpdGVuJzsKICAgICRMYW5nLT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gICAgPSAnU3VjaC1JbmZvcm1hdGlvbmVuIGJlYXJiZWl0ZW4nOwogICAgJExhbmctPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gICAgICAgICA9ICdJbXBvcnQgSW5mb3JtYXRpb25lbic7CiAgICAkTGFuZy0+eydDb2x1bW4nfSAgICAgICAgICAgICAgICAgICAgID0gJ1NwYWx0ZSc7CiAgICAkTGFuZy0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ0V4cG9ydCBwZXIgU3VjaGUgZWluc2NocuRua2VuJzsKICAgICRMYW5nLT57J1NvdXJjZSBGaWxlJ30gICAgICAgICAgICAgICAgPSAnUXVlbGwtRGF0ZWknOwogICAgJExhbmctPnsnQ29sdW1uIFNlcGVyYXRvcid9ICAgICAgICAgICA9ICdTcGFsdGVudHJlbm5lcic7CiAgICAkTGFuZy0+eydUYWJ1bGF0b3IgKFRBQiknfSAgICAgICAgICAgID0gJ1RhYnVsYXRvciAoVEFCKSc7CiAgICAkTGFuZy0+eydTZW1pY29sb24gKDspJ30gICAgICAgICAgICAgID0gJ1NlbWljb2xvbiAoOyknOwogICAgJExhbmctPnsnQ29sb24gKDopJ30gICAgICAgICAgICAgICAgICA9ICdEb3BwZWxwdW5rdCAoOiknOwogICAgJExhbmctPnsnRG90ICguKSd9ICAgICAgICAgICAgICAgICAgICA9ICdQdW5rdCAoLiknOwogICAgJExhbmctPnsnQ2hhcnNldCd9ICAgICAgICAgICAgICAgICAgICA9ICdaZWljaGVuc2F0eic7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9lc19JbXBvcnRFeHBvcnQucG0gLSB0aGUgc3BhbmlzaCB0cmFuc2xhdGlvbiBvZiBJbXBvcnRFeHBvcnQKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgQ29weXJpZ2h0IChDKSAyMDA4IEFxdWlsZXMgQ29oZW4KIyAtLQojICRJZDogZXNfSW1wb3J0RXhwb3J0LnBtLHYgMS41IDIwMDkvMDUvMTggMDk6NDI6NTIgbWggRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6ZXNfSW1wb3J0RXhwb3J0OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS41ICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAkTGFuZy0+eydJbXBvcnQvRXhwb3J0J30gICAgICAgICAgICAgID0gJ0ltcG9ydGFyL0V4cG9ydGFyJzsKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQgTWFuYWdlbWVudCd9ICAgPSAnR2VzdGnzbiBkZSBJbXBvcnRhY2nzbi9FeHBvcnRhY2nzbic7CiAgICAkTGFuZy0+eydBZGQgbWFwcGluZyB0ZW1wbGF0ZSd9ICAgICAgID0gJ0HxYWRpciBtYXBlYWRvIGRlIHBsYW50aWxsYSc7CiAgICAkTGFuZy0+eydTdGFydCBJbXBvcnQnfSAgICAgICAgICAgICAgID0gJ0luaWNpYXIgSW1wb3J0YWNp824nOwogICAgJExhbmctPnsnU3RhcnQgRXhwb3J0J30gICAgICAgICAgICAgICA9ICdJbmljaWFyIEV4cG9ydGFjafNuJzsKICAgICRMYW5nLT57J1N0ZXAnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnUGFzbyc7CiAgICAkTGFuZy0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ICAgID0gJ0VkaXRhciBpbmZvcm1hY2nzbiBjb236bic7CiAgICAkTGFuZy0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ICAgID0gJ0VkaXRhciBpbmZvcm1hY2nzbiBkZSBvYmpldG8nOwogICAgJExhbmctPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSAgICA9ICdFZGl0YXIgZm9ybWF0byBkZSBpbmZvcm1hY2nzbic7CiAgICAkTGFuZy0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSAgID0gJ0VkaXRhciBtYXBlYWRvIGRlIGluZm9ybWFjafNuJzsKICAgICRMYW5nLT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gICAgPSAnRWRpdGFyIGluZm9ybWFjafNuIGRlIGJ1c3F1ZWRhJzsKICAgICRMYW5nLT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ICAgICAgICAgPSAnSW1wb3J0YXIgaW5mb3JtYWNp824nOwogICAgJExhbmctPnsnQ29sdW1uJ30gICAgICAgICAgICAgICAgICAgICA9ICdDb2x1bW5hJzsKICAgICRMYW5nLT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnUmVzdHJpbmdpciBleHBvcnRhY2nzbiBwb3IgY29uc3VsdGEnOwogICAgJExhbmctPnsnU291cmNlIEZpbGUnfSAgICAgICAgICAgICAgICA9ICdBcmNoaXZvIG9yaWdlbic7CiAgICAkTGFuZy0+eydDb2x1bW4gU2VwZXJhdG9yJ30gICAgICAgICAgID0gJ1NlcGFyYWRvciBkZSBDb2x1bW5hJzsKICAgICRMYW5nLT57J1RhYnVsYXRvciAoVEFCKSd9ICAgICAgICAgICAgPSAnVGFidWxhZG9yIChUQUIpJzsKICAgICRMYW5nLT57J1NlbWljb2xvbiAoOyknfSAgICAgICAgICAgICAgPSAnUHVudG8geSBDb21hICg7KSc7CiAgICAkTGFuZy0+eydDb2xvbiAoOiknfSAgICAgICAgICAgICAgICAgID0gJ0RvcyBwdW50b3MgKDopJzsKICAgICRMYW5nLT57J0RvdCAoLiknfSAgICAgICAgICAgICAgICAgICAgPSAnUHVudG8gKC4pJzsKICAgICRMYW5nLT57J0NoYXJzZXQnfSAgICAgICAgICAgICAgICAgICAgPSAnJzsKCiAgICByZXR1cm4gMTsKfQoKMTsK
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9mYV9JbXBvcnRFeHBvcnQucG0gLSB0aGUgcGVyc2lhbiAoZmFyc2kpIHRyYW5zbGF0aW9uIG9mIGZhX0ltcG9ydEV4cG9ydAojIENvcHlyaWdodCAoQykgMjAwMS0yMDA5IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyBDb3B5cmlnaHQgKEMpIDIwMDMtMjAwOSBBZnNoYXIgTW9oZWJiaSA8YWZzaGFyLm1vaGViYmkgYXQgZ21haWwuY29tPgojIC0tCiMgJElkOiBmYV9JbXBvcnRFeHBvcnQucG0sdiAxLjEgMjAwOS8wNy8yMCAxMDo0NjowOCB1YiBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpmYV9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjEgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQnfSAgICAgICAgICAgICAgPSAn2YjYsdmI2K8v2LXYr9mI2LEnOwogICAgJExhbmctPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gICA9ICfZhdiv24zYsduM2Kog2YjYsdmI2K8v2LXYr9mI2LEnOwogICAgJExhbmctPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSAgICAgICA9ICfYp9i22KfZgdmHINqp2LHYr9mGINuM2qkg2YLYp9mE2Kgg2Ybar9in2LTYqic7CiAgICAkTGFuZy0+eydTdGFydCBJbXBvcnQnfSAgICAgICAgICAgICAgID0gJ9i02LHZiNi5INi52YXZhNuM2KfYqiDZiNix2YjYryc7CiAgICAkTGFuZy0+eydTdGFydCBFeHBvcnQnfSAgICAgICAgICAgICAgID0gJ9i02LHZiNi5INi52YXZhNuM2KfYqiDYtdiv2YjYsSc7CiAgICAkTGFuZy0+eydTdGVwJ30gICAgICAgICAgICAgICAgICAgICAgID0gJ9qv2KfZhSc7CiAgICAkTGFuZy0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ICAgID0gJ9mI24zYsdin24zYtCDYp9i32YTYp9i52KfYqiDYudmF2YjZhduMJzsKICAgICRMYW5nLT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gICAgPSAn2YjbjNix2KfbjNi0INin2LfZhNin2LnYp9iqINii2KjYrNqp2KrbjCc7CiAgICAkTGFuZy0+eydFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ICAgID0gJ9mI24zYsdin24zYtCDYp9i32YTYp9i52KfYqiDZgtin2YTYqOKAjNio2YbYr9uMJzsKICAgICRMYW5nLT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ICAgPSAn2YjbjNix2KfbjNi0INin2LfZhNin2LnYp9iqINmG2q/Yp9i02KonOwogICAgJExhbmctPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSAgICA9ICfZiNuM2LHYp9uM2LQg2KfYt9mE2KfYudin2Kog2KzYs9iq2KzZiCc7CiAgICAkTGFuZy0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSAgICAgICAgID0gJ9mI2LHZiNivINin2LfZhNin2LnYp9iqJzsKICAgICRMYW5nLT57J0NvbHVtbid9ICAgICAgICAgICAgICAgICAgICAgPSAn2LPYqtmI2YYnOwogICAgJExhbmctPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICfZhdit2K/ZiNiv2LPYp9iy24wg2LnZhdmE24zYp9iqINi12K/ZiNixINio2Ycg2KfYstin24wg2KzYs9iq2KzZiCc7CiAgICAkTGFuZy0+eydTb3VyY2UgRmlsZSd9ICAgICAgICAgICAgICAgID0gJ9mB2KfbjNmEINmF2YbYqNi5JzsKICAgICRMYW5nLT57J0NvbHVtbiBTZXBlcmF0b3InfSAgICAgICAgICAgPSAn2KzYr9in2qnZhtmG2K/ZhyDYs9iq2YjZhuKAjNmH2KcnOwogICAgJExhbmctPnsnVGFidWxhdG9yIChUQUIpJ30gICAgICAgICAgICA9ICfYrNiv2YjZhCDYs9in2LIgKFRBQiknOwogICAgJExhbmctPnsnU2VtaWNvbG9uICg7KSd9ICAgICAgICAgICAgICA9ICfYs9mF24wg2qnYp9mE2YYgKDspJzsKICAgICRMYW5nLT57J0NvbG9uICg6KSd9ICAgICAgICAgICAgICAgICAgPSAn2K/ZiNmG2YLYt9mHICg6KSc7CiAgICAkTGFuZy0+eydEb3QgKC4pJ30gICAgICAgICAgICAgICAgICAgID0gJ9mG2YLYt9mHICguKSc7CiAgICAkTGFuZy0+eydDaGFyc2V0J30gICAgICAgICAgICAgICAgICAgID0gJ9qp2K/YqNmG2K/bjCDYp9i32YTYp9i52KfYqic7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9pdF9JbXBvcnRFeHBvcnQucG0gLSB0aGUgaXRhbGlhbiB0cmFuc2xhdGlvbiBvZiBJbXBvcnRFeHBvcnQKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IGl0X0ltcG9ydEV4cG9ydC5wbSx2IDEuMSAyMDA5LzA1LzE4IDA5OjQyOjM0IG1oIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6Oml0X0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuMSAkKSBbMV07CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgIG15ICRMYW5nID0gJFNlbGYtPntUcmFuc2xhdGlvbn07CgogICAgcmV0dXJuIGlmIHJlZiAkTGFuZyBuZSAnSEFTSCc7CgogICAgJExhbmctPnsnSW1wb3J0L0V4cG9ydCd9ICAgICAgICAgICAgICA9ICdJbXBvcnRhcmUvRXNwb3J0YXJlJzsKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQgTWFuYWdlbWVudCd9ICAgPSAnR2VzdGlvbmUgSW1wb3J0YXppb25lL0VzcG9ydGF6aW9uZSc7CiAgICAkTGFuZy0+eydBZGQgbWFwcGluZyB0ZW1wbGF0ZSd9ICAgICAgID0gJ0FnZ2l1bmdpIG1hcHBhdHVyYSBtb2RlbGxvJzsKICAgICRMYW5nLT57J1N0YXJ0IEltcG9ydCd9ICAgICAgICAgICAgICAgPSAnSW5pemlhcmUgSW1wb3J0YXppb25lJzsKICAgICRMYW5nLT57J1N0YXJ0IEV4cG9ydCd9ICAgICAgICAgICAgICAgPSAnSW5pemlhcmUgRXNwb3J0YXppb25lJzsKICAgICRMYW5nLT57J1N0ZXAnfSAgICAgICAgICAgICAgICAgICAgICAgPSAnUGFzc28nOwogICAgJExhbmctPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSAgICA9ICdNb2RpZmljYSBpbmZvcm1hemlvbmkgY29tdW5pJzsKICAgICRMYW5nLT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gICAgPSAnTW9kaWZpY2EgaW5mb3JtYXppb25pIG9nZ2V0dG8nOwogICAgJExhbmctPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSAgICA9ICdNb2RpZmljYSBmb3JtYXRvIGluZm9ybWF6aW9uZSc7CiAgICAkTGFuZy0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSAgID0gJ01vZGlmaWNhIG1hcHBhdHVyYSBpbmZvcm1hemlvbmknOwogICAgJExhbmctPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSAgICA9ICdNb2RpZmljYSBpbmZvcm1hemlvbmkgZGkgcmljZXJjYSc7CiAgICAkTGFuZy0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSAgICAgICAgID0gJ0ltcG9ydGFyZSBpbmZvcm1hemlvbmUnOwogICAgJExhbmctPnsnQ29sdW1uJ30gICAgICAgICAgICAgICAgICAgICA9ICdDb2xvbm5hJzsKICAgICRMYW5nLT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnUmVzdHJpbmdlcmUgZXNwb3J0YXppb25lIHBlciByaWNlcmNhJzsKICAgICRMYW5nLT57J1NvdXJjZSBGaWxlJ30gICAgICAgICAgICAgICAgPSAnQXJjaGl2aW8gb3JpZ2luZSc7CiAgICAkTGFuZy0+eydDb2x1bW4gU2VwZXJhdG9yJ30gICAgICAgICAgID0gJ1NlcGFyYXRvcmUgZGkgY29sb25uYSc7CiAgICAkTGFuZy0+eydUYWJ1bGF0b3IgKFRBQiknfSAgICAgICAgICAgID0gJ1RhYnVsYXRvcmUgKFRBQiknOwogICAgJExhbmctPnsnU2VtaWNvbG9uICg7KSd9ICAgICAgICAgICAgICA9ICdQdW50byBlIHZpcmdvbGEgKDspJzsKICAgICRMYW5nLT57J0NvbG9uICg6KSd9ICAgICAgICAgICAgICAgICAgPSAnRHVlIHB1bnRpICg6KSc7CiAgICAkTGFuZy0+eydEb3QgKC4pJ30gICAgICAgICAgICAgICAgICAgID0gJ1B1bnRvICguKSc7CiAgICAkTGFuZy0+eydDaGFyc2V0J30gICAgICAgICAgICAgICAgICAgID0gJ0NoYXJzZXQnOwoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9ubF9JbXBvcnRFeHBvcnQucG0gLSB0aGUgRHV0Y2ggdHJhbnNsYXRpb24gb2YgSW1wb3J0RXhwb3J0CiMgQ29weXJpZ2h0IChDKSAyMDA5IE1pY2hpZWwgQmVpamVuIDxtaWNoaWVsICdhdCcgYmVlZnJlZWl0Lm5sPgojIC0tCiMgJElkOiBubF9JbXBvcnRFeHBvcnQucG0sdiAxLjIgMjAwOS8wNy8yMCAxMzo1NTo1MCB1YiBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpubF9JbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjIgJCkgWzFdOwoKc3ViIERhdGEgewogICAgbXkgJFNlbGYgPSBzaGlmdDsKCiAgICBteSAkTGFuZyA9ICRTZWxmLT57VHJhbnNsYXRpb259OwoKICAgIHJldHVybiBpZiByZWYgJExhbmcgbmUgJ0hBU0gnOwoKICAgICRMYW5nLT57J0ltcG9ydC9FeHBvcnQnfSAgICAgICAgICAgICAgPSAnSW1wb3J0L0V4cG9ydCc7CiAgICAkTGFuZy0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSAgID0gJ0ltcG9ydC9FeHBvcnQgYmVoZWVyJzsKICAgICRMYW5nLT57J0FkZCBtYXBwaW5nIHRlbXBsYXRlJ30gICAgICAgPSAnTWFwcGluZ3RlbXBsYXRlIHRvZXZvZWdlbic7CiAgICAkTGFuZy0+eydTdGFydCBJbXBvcnQnfSAgICAgICAgICAgICAgID0gJ0ltcG9ydCBzdGFydGVuJzsKICAgICRMYW5nLT57J1N0YXJ0IEV4cG9ydCd9ICAgICAgICAgICAgICAgPSAnRXhwb3J0IHN0YXJ0ZW4nOwogICAgJExhbmctPnsnU3RlcCd9ICAgICAgICAgICAgICAgICAgICAgICA9ICdTdGFwJzsKICAgICRMYW5nLT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gICAgPSAnQWxnZW1lbmUgaW5mb3JtYXRpZSBiZXdlcmtlbic7CiAgICAkTGFuZy0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ICAgID0gJ09iamVjdC1pbmZvcm1hdGllIGJld2Vya2VuJzsKICAgICRMYW5nLT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gICAgPSAnRm9ybWF0LWluZm9ybWF0aW9uZW4gYmV3ZXJrZW4nOwogICAgJExhbmctPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gICA9ICdNYXBwaW5nLWluZm9ybWF0aWUgYmV3ZXJrZW4nOwogICAgJExhbmctPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSAgICA9ICdab2VrLWluZm9ybWF0aWUgYmV3ZXJrZW4nOwogICAgJExhbmctPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gICAgICAgICA9ICdJbXBvcnQtaW5mb3JtYXRpZSc7CiAgICAkTGFuZy0+eydDb2x1bW4nfSAgICAgICAgICAgICAgICAgICAgID0gJ0tvbG9tJzsKICAgICRMYW5nLT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnQmVwZXJrIGV4cG9ydCB0b3Qgem9la29wZHJhY2h0JzsKICAgICRMYW5nLT57J1NvdXJjZSBGaWxlJ30gICAgICAgICAgICAgICAgPSAnQnJvbmJlc3RhbmQnOwogICAgJExhbmctPnsnQ29sdW1uIFNlcGVyYXRvcid9ICAgICAgICAgICA9ICdLb2xvbXNjaGVpZGluZ3N0ZWtlbic7CiAgICAkTGFuZy0+eydUYWJ1bGF0b3IgKFRBQiknfSAgICAgICAgICAgID0gJ1RhYic7CiAgICAkTGFuZy0+eydTZW1pY29sb24gKDspJ30gICAgICAgICAgICAgID0gJ1B1bnRrb21tYSAoOyknOwogICAgJExhbmctPnsnQ29sb24gKDopJ30gICAgICAgICAgICAgICAgICA9ICdEdWJiZWxlIHB1bnQgKDopJzsKICAgICRMYW5nLT57J0RvdCAoLiknfSAgICAgICAgICAgICAgICAgICAgPSAnUHVudCAoLiknOwogICAgJExhbmctPnsnQ2hhcnNldCd9ICAgICAgICAgICAgICAgICAgICA9ICdLYXJha3RlcnNldCc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9wbF9JbXBvcnRFeHBvcnQucG0gLSB0aGUgcG9saXNoIHRyYW5zbGF0aW9uIG9mIEltcG9ydEV4cG9ydAojIENvcHlyaWdodCAoQykgMjAwMS0yMDA5IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyBDb3B5cmlnaHQgKEMpIDIwMDggTWFjaWVqIExvc3phamMKIyAtLQojICRJZDogcGxfSW1wb3J0RXhwb3J0LnBtLHYgMS40IDIwMDkvMDUvMTggMDk6NDI6NTIgbWggRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6cGxfSW1wb3J0RXhwb3J0OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS40ICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAkTGFuZy0+eydJbXBvcnQvRXhwb3J0J30gICAgICAgICAgICAgID0gJ0ltcG9ydC9FeHBvcnQnOwogICAgJExhbmctPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gICA9ICdaYXJ6MWR6YW5pZSBJbXBvcnRlbS9FeHBvcnRlbSc7CiAgICAkTGFuZy0+eydBZGQgbWFwcGluZyB0ZW1wbGF0ZSd9ICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydTdGFydCBJbXBvcnQnfSAgICAgICAgICAgICAgID0gJ1JvenBvY3puaWogSW1wb3J0JzsKICAgICRMYW5nLT57J1N0YXJ0IEV4cG9ydCd9ICAgICAgICAgICAgICAgPSAnUm96cG9jem5paiBFeHBvcnQnOwogICAgJExhbmctPnsnU3RlcCd9ICAgICAgICAgICAgICAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSAgICA9ICcnOwogICAgJExhbmctPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSAgICA9ICcnOwogICAgJExhbmctPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSAgICA9ICcnOwogICAgJExhbmctPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gICA9ICcnOwogICAgJExhbmctPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSAgICA9ICcnOwogICAgJExhbmctPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gICAgICAgICA9ICcnOwogICAgJExhbmctPnsnQ29sdW1uJ30gICAgICAgICAgICAgICAgICAgICA9ICdLb2x1bW5hJzsKICAgICRMYW5nLT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnJzsKICAgICRMYW5nLT57J1NvdXJjZSBGaWxlJ30gICAgICAgICAgICAgICAgPSAnUGxpayBZcvNkM293eSc7CiAgICAkTGFuZy0+eydDb2x1bW4gU2VwZXJhdG9yJ30gICAgICAgICAgID0gJ1NlcGFyYXRvciBrb2x1bW55JzsKICAgICRMYW5nLT57J1RhYnVsYXRvciAoVEFCKSd9ICAgICAgICAgICAgPSAnVGFidWxhdG9yIChUQUIpJzsKICAgICRMYW5nLT57J1NlbWljb2xvbiAoOyknfSAgICAgICAgICAgICAgPSAnT3JlZG5payAoOyknOwogICAgJExhbmctPnsnQ29sb24gKDopJ30gICAgICAgICAgICAgICAgICA9ICdEd3Vrcm9wZWsgKDopJzsKICAgICRMYW5nLT57J0RvdCAoLiknfSAgICAgICAgICAgICAgICAgICAgPSAnS3JvcGthICguKSc7CiAgICAkTGFuZy0+eydDaGFyc2V0J30gICAgICAgICAgICAgICAgICAgID0gJ0tvZG93YW5pZSc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9ydV9JbXBvcnRFeHBvcnQucG0gLSB0aGUgcnVzc2lhbiB0cmFuc2xhdGlvbiBvZiBJbXBvcnRFeHBvcnQKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgQ29weXJpZ2h0IChDKSAyMDA4IEVnb3IgVHNpbGVua28gPGJnOHMgYXQgc3ltbGluay5ydT4KIyAtLQojICRJZDogcnVfSW1wb3J0RXhwb3J0LnBtLHYgMS4yIDIwMDkvMDUvMTggMDk6NDI6NTIgbWggRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6cnVfSW1wb3J0RXhwb3J0OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS4yICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+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+eydTb3VyY2UgRmlsZSd9ICAgICAgICAgICAgICAgID0gJ8jx9e7k7fvpIPTg6esnOwogICAgJExhbmctPnsnQ29sdW1uIFNlcGVyYXRvcid9ICAgICAgICAgICA9ICfQ4Ofk5evo8uXr/Cc7CiAgICAkTGFuZy0+eydUYWJ1bGF0b3IgKFRBQiknfSAgICAgICAgICAgID0gJ9Lg4fPr//bo/yAoVEFCKSc7CiAgICAkTGFuZy0+eydTZW1pY29sb24gKDspJ30gICAgICAgICAgICAgID0gJ9Lu9+rgIPEg5+Dv//Lu6SAoOyknOwogICAgJExhbmctPnsnQ29sb24gKDopJ30gICAgICAgICAgICAgICAgICA9ICfE4u7l8u736OUgKDopJzsKICAgICRMYW5nLT57J0RvdCAoLiknfSAgICAgICAgICAgICAgICAgICAgPSAn0u736uAgKC4pJzsKICAgICRMYW5nLT57J0NoYXJzZXQnfSAgICAgICAgICAgICAgICAgICAgPSAnyu7k6PDu4urgJzsKCiAgICByZXR1cm4gMTsKfQoKMTsK
# --
# Kernel/Modules/AdminImportExport.pm - admin frontend of import export module
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: AdminImportExport.pm,v 1.20 2009/05/18 09:42:52 mh Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Modules::AdminImportExport;

use strict;
use warnings;

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

use vars qw($VERSION);
$VERSION = qw($Revision: 1.20 $) [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();

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

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

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

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

            # get needed data
            $TemplateData->{Object} = $Self->{ParamObject}->GetParam( Param => 'Object' );
            $TemplateData->{Format} = $Self->{ParamObject}->GetParam( Param => 'Format' );

            # redirect to overview
            return $Self->{LayoutObject}->Redirect( OP => "Action=$Self->{Action}" )
                if !$TemplateData->{Object} || !$TemplateData->{Format};
        }
        else {

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

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

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

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

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

        # 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
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'TemplateSave1' ) {
        my $TemplateData = {};

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

        my %Submit = (
            SubmitNext => 'TemplateEdit2',
            Reload     => 'TemplateEdit1',
        );

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

        # save to database
        my $Success;
        if ( $TemplateData->{TemplateID} eq 'NEW' ) {
            $TemplateData->{TemplateID} = $Self->{ImportExportObject}->TemplateAdd(
                %{$TemplateData},
                UserID => $Self->{UserID},
            );

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

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

        # redirect to overview object list
        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();

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

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

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

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

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

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

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

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

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

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

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

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

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

        # 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 (object)
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'TemplateSave2' ) {

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

        my %Submit = (
            SubmitNext => 'TemplateEdit3',
            SubmitBack => 'TemplateEdit1',
            Reload     => 'TemplateEdit2',
        );

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

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

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

            # 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 object data
        $Self->{ImportExportObject}->ObjectDataSave(
            TemplateID => $TemplateID,
            ObjectData => \%AttributeValues,
            UserID     => $Self->{UserID},
        );

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

        # 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 (format)
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'TemplateSave3' ) {

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

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

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

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

        # 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} ) {
                $Subaction = $Submit{Reload};
            }
        }

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

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

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

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

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

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

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

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

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

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

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

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

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

            # 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::' . $Counter . '::',
                    Value  => $MappingObjectData->{ $Item->{Key} },
                );

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

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

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

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

            $Counter++;
        }

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

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

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

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

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

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

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

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

        # 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 => 'TemplateEdit5Row',
                Data => {
                    Name => $Item->{Name} || '',
                    InputStrg => $InputString,
                },
            );
        }

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

        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();

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

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

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

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

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

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

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

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

            $EmptyDatabase = 0;

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

            my $CssClass = '';
            for my $TemplateID ( @{$TemplateList} ) {

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

                # 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} },
                        CssClass   => $CssClass,
                        Valid      => $ValidList{ $TemplateData->{ValidID} },
                    },
                );
            }
        }

        # output an empty list
        if ($EmptyDatabase) {

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

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

IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnRMYXlvdXRDaGVja2JveC5wbSAtIGxheW91dCBiYWNrZW5kIG1vZHVsZQojIENvcHlyaWdodCAoQykgMjAwMS0yMDA5IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyAtLQojICRJZDogSW1wb3J0RXhwb3J0TGF5b3V0Q2hlY2tib3gucG0sdiAxLjQgMjAwOS8wNS8xOCAwOTo0Mjo1MiBtaCBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0TGF5b3V0Q2hlY2tib3g7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjQgJCkgWzFdOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRDaGVja2JveCAtIGxheW91dCBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIFNZTk9QU0lTCgpBbGwgbGF5b3V0IGZ1bmN0aW9ucyBmb3IgY2hlY2tib3ggZWxlbWVudHMKCj1vdmVyIDQKCj1jdXQKCj1pdGVtIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgJEJhY2tlbmRPYmplY3QgPSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0TGF5b3V0Q2hlY2tib3gtPm5ldygKICAgICAgICAlUGFyYW0sCiAgICApOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBvYmplY3RzCiAgICBmb3IgbXkgJE9iamVjdCAocXcoQ29uZmlnT2JqZWN0IExvZ09iamVjdCBNYWluT2JqZWN0IFBhcmFtT2JqZWN0IExheW91dE9iamVjdCkpIHsKICAgICAgICAkU2VsZi0+eyRPYmplY3R9ID0gJFBhcmFteyRPYmplY3R9IHx8IGRpZSAiR290IG5vICRPYmplY3QhIjsKICAgIH0KCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1pdGVtIEZvcm1JbnB1dENyZWF0ZSgpCgpjcmVhdGUgYSBpbnB1dCBzdHJpbmcKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+Rm9ybUlucHV0Q3JlYXRlKAogICAgICAgIEl0ZW0gICA9PiAkSXRlbVJlZiwKICAgICAgICBQcmVmaXggPT4gJ1ByZWZpeDo6JywgICMgKG9wdGlvbmFsKQogICAgICAgIFZhbHVlICA9PiAnVmFsdWUnLCAgICAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1JbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coIFByaW9yaXR5ID0+ICdlcnJvcicsIE1lc3NhZ2UgPT4gJ05lZWQgSXRlbSEnICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICRQYXJhbXtQcmVmaXh9IHx8PSAnJzsKCiAgICBteSAkU3RyaW5nID0gIjxpbnB1dCB0eXBlPVwiY2hlY2tib3hcIiBuYW1lPVwiJFBhcmFte1ByZWZpeH0kUGFyYW17SXRlbX0tPntLZXl9XCIgIjsKCiAgICBpZiAoICRQYXJhbXtWYWx1ZX0gKSB7CiAgICAgICAgJFN0cmluZyAuPSAiY2hlY2tlZCAiOwogICAgfQoKICAgICRTdHJpbmcgLj0gIj4gIjsKCiAgICByZXR1cm4gJFN0cmluZzsKfQoKPWl0ZW0gRm9ybURhdGFHZXQoKQoKZ2V0IGZvcm0gZGF0YQoKICAgIG15ICRGb3JtRGF0YSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtRGF0YUdldCgKICAgICAgICBJdGVtICAgPT4gJEl0ZW1SZWYsCiAgICAgICAgUHJlZml4ID0+ICdQcmVmaXg6OicsICAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgRm9ybURhdGFHZXQgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgaWYgKCAhJFBhcmFte0l0ZW19ICkgewogICAgICAgICRTZWxmLT57TG9nT2JqZWN0fS0+TG9nKCBQcmlvcml0eSA9PiAnZXJyb3InLCBNZXNzYWdlID0+ICdOZWVkIEl0ZW0hJyApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAkUGFyYW17UHJlZml4fSB8fD0gJyc7CgogICAgIyBnZXQgZm9ybSBkYXRhCiAgICBteSAkRm9ybURhdGEgPSAkU2VsZi0+e1BhcmFtT2JqZWN0fS0+R2V0UGFyYW0oCiAgICAgICAgUGFyYW0gPT4gJFBhcmFte1ByZWZpeH0gLiAkUGFyYW17SXRlbX0tPntLZXl9LAogICAgKTsKCiAgICByZXR1cm4gJEZvcm1EYXRhOwp9CgoxOwoKPWJhY2sKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKGh0dHA6Ly9vdHJzLm9yZy8pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgoKPWN1dAoKPWhlYWQxIFZFUlNJT04KCiRSZXZpc2lvbjogMS40ICQgJERhdGU6IDIwMDkvMDUvMTggMDk6NDI6NTIgJAoKPWN1dAo=
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnRMYXlvdXREVEwucG0gLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IEltcG9ydEV4cG9ydExheW91dERUTC5wbSx2IDEuMyAyMDA5LzA1LzE4IDA5OjQyOjUyIG1oIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXREVEw7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjMgJCkgWzFdOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXREVEwgLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKQWxsIGxheW91dCBmdW5jdGlvbnMgZm9yIGRpc3BsYXkgRFRMIGNvZGUKCj1vdmVyIDQKCj1jdXQKCj1pdGVtIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgJEJhY2tlbmRPYmplY3QgPSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0TGF5b3V0RFRMLT5uZXcoCiAgICAgICAgJVBhcmFtLAogICAgKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgIyBjaGVjayBuZWVkZWQgb2JqZWN0cwogICAgZm9yIG15ICRPYmplY3QgKHF3KENvbmZpZ09iamVjdCBMb2dPYmplY3QgTWFpbk9iamVjdCBQYXJhbU9iamVjdCBMYXlvdXRPYmplY3QpKSB7CiAgICAgICAgJFNlbGYtPnskT2JqZWN0fSA9ICRQYXJhbXskT2JqZWN0fSB8fCBkaWUgIkdvdCBubyAkT2JqZWN0ISI7CiAgICB9CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aXRlbSBGb3JtSW5wdXRDcmVhdGUoKQoKY3JlYXRlIGEgaW5wdXQgc3RyaW5nCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkZvcm1JbnB1dENyZWF0ZSgKICAgICAgICBJdGVtICAgPT4gJEl0ZW1SZWYsCiAgICApOwoKPWN1dAoKc3ViIEZvcm1JbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coIFByaW9yaXR5ID0+ICdlcnJvcicsIE1lc3NhZ2UgPT4gJ05lZWQgSXRlbSEnICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIHJldHVybiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntEYXRhfTsKfQoKPWl0ZW0gRm9ybURhdGFHZXQoKQoKZ2V0IGZvcm0gZGF0YQoKICAgIG15ICRGb3JtRGF0YSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtRGF0YUdldCgpOwoKPWN1dAoKc3ViIEZvcm1EYXRhR2V0IHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuOwp9CgoxOwoKPWJhY2sKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKGh0dHA6Ly9vdHJzLm9yZy8pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgoKPWN1dAoKPWhlYWQxIFZFUlNJT04KCiRSZXZpc2lvbjogMS4zICQgJERhdGU6IDIwMDkvMDUvMTggMDk6NDI6NTIgJAoKPWN1dAo=
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnRMYXlvdXRTZWxlY3Rpb24ucG0gLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IEltcG9ydEV4cG9ydExheW91dFNlbGVjdGlvbi5wbSx2IDEuOCAyMDA5LzA1LzE4IDA5OjQyOjUyIG1oIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRTZWxlY3Rpb247Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjggJCkgWzFdOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRTZWxlY3Rpb24gLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKQWxsIGxheW91dCBmdW5jdGlvbnMgZm9yIHNlbGVjdGlvbiBlbGVtZW50cwoKPW92ZXIgNAoKPWN1dAoKPWl0ZW0gbmV3KCkKCmNyZWF0ZSBhbiBvYmplY3QKCiAgICAkQmFja2VuZE9iamVjdCA9IEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRTZWxlY3Rpb24tPm5ldygKICAgICAgICAlUGFyYW0sCiAgICApOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBvYmplY3RzCiAgICBmb3IgbXkgJE9iamVjdCAocXcoQ29uZmlnT2JqZWN0IExvZ09iamVjdCBNYWluT2JqZWN0IFBhcmFtT2JqZWN0IExheW91dE9iamVjdCkpIHsKICAgICAgICAkU2VsZi0+eyRPYmplY3R9ID0gJFBhcmFteyRPYmplY3R9IHx8IGRpZSAiR290IG5vICRPYmplY3QhIjsKICAgIH0KCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1pdGVtIEZvcm1JbnB1dENyZWF0ZSgpCgpjcmVhdGUgYSBpbnB1dCBzdHJpbmcKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+Rm9ybUlucHV0Q3JlYXRlKAogICAgICAgIEl0ZW0gICA9PiAkSXRlbVJlZiwKICAgICAgICBQcmVmaXggPT4gJ1ByZWZpeDo6JywgICMgKG9wdGlvbmFsKQogICAgICAgIFZhbHVlICA9PiAnVmFsdWUnLCAgICAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1JbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coIFByaW9yaXR5ID0+ICdlcnJvcicsIE1lc3NhZ2UgPT4gJ05lZWQgSXRlbSEnICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICMgc2V0IGRlZmF1bHQgdmFsdWUKICAgICRQYXJhbXtQcmVmaXh9IHx8PSAnJzsKICAgICRQYXJhbXtWYWx1ZX0gIHx8PSAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntWYWx1ZURlZmF1bHR9OwoKICAgIGlmICggJFBhcmFte1ZhbHVlfSAmJiAkUGFyYW17VmFsdWV9ID1+IG17ICMjIyMjIH14bXMgKSB7CiAgICAgICAgbXkgQFZhbHVlcyA9IHNwbGl0ICcjIyMjIycsICRQYXJhbXtWYWx1ZX07CiAgICAgICAgJFBhcmFte1ZhbHVlfSA9IFxAVmFsdWVzOwogICAgfQoKICAgICMgZ2VuZXJhdGUgb3B0aW9uIHN0cmluZwogICAgbXkgJFN0cmluZyA9ICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+QnVpbGRTZWxlY3Rpb24oCiAgICAgICAgTmFtZSAgICAgICAgID0+ICRQYXJhbXtQcmVmaXh9IC4gJFBhcmFte0l0ZW19LT57S2V5fSwKICAgICAgICBEYXRhICAgICAgICAgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57RGF0YX0gfHwge30sCiAgICAgICAgU2VsZWN0ZWRJRCAgID0+ICRQYXJhbXtWYWx1ZX0sCiAgICAgICAgVHJhbnNsYXRpb24gID0+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1RyYW5zbGF0aW9ufSwKICAgICAgICBQb3NzaWJsZU5vbmUgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57UG9zc2libGVOb25lfSwKICAgICAgICBNdWx0aXBsZSAgICAgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57TXVsdGlwbGV9LAogICAgICAgIFNpemUgICAgICAgICA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntTaXplfSwKICAgICk7CgogICAgcmV0dXJuICRTdHJpbmc7Cn0KCj1pdGVtIEZvcm1EYXRhR2V0KCkKCmdldCBmb3JtIGRhdGEKCiAgICBteSAkRm9ybURhdGEgPSAkQmFja2VuZE9iamVjdC0+Rm9ybURhdGFHZXQoCiAgICAgICAgSXRlbSAgID0+ICRJdGVtUmVmLAogICAgICAgIFByZWZpeCA9PiAnUHJlZml4OjonLCAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1EYXRhR2V0IHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZyggUHJpb3JpdHkgPT4gJ2Vycm9yJywgTWVzc2FnZSA9PiAnTmVlZCBJdGVtIScgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgJFBhcmFte1ByZWZpeH0gfHw9ICcnOwoKICAgICMgZ2V0IGZvcm0gZGF0YQogICAgbXkgQEZvcm1EYXRhcyA9ICRTZWxmLT57UGFyYW1PYmplY3R9LT5HZXRBcnJheSgKICAgICAgICBQYXJhbSA9PiAkUGFyYW17UHJlZml4fSAuICRQYXJhbXtJdGVtfS0+e0tleX0sCiAgICApOwoKICAgIG15ICRGb3JtRGF0YSA9IGpvaW4gJyMjIyMjJywgQEZvcm1EYXRhczsKCiAgICByZXR1cm4gJEZvcm1EYXRhIGlmICRGb3JtRGF0YTsKICAgIHJldHVybiAkRm9ybURhdGEgaWYgISRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1JlcXVpcmVkfTsKCiAgICAjIHNldCBpbnZhbGlkIHBhcmFtCiAgICAkUGFyYW17SXRlbX0tPntGb3JtfS0+e0ludmFsaWR9ID0gMTsKCiAgICByZXR1cm4gJEZvcm1EYXRhOwp9CgoxOwoKPWJhY2sKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKGh0dHA6Ly9vdHJzLm9yZy8pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgoKPWN1dAoKPWhlYWQxIFZFUlNJT04KCiRSZXZpc2lvbjogMS44ICQgJERhdGU6IDIwMDkvMDUvMTggMDk6NDI6NTIgJAoKPWN1dAo=
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnRMYXlvdXRUZXh0LnBtIC0gbGF5b3V0IGJhY2tlbmQgbW9kdWxlCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDkgT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIC0tCiMgJElkOiBJbXBvcnRFeHBvcnRMYXlvdXRUZXh0LnBtLHYgMS43IDIwMDkvMDUvMTggMDk6NDI6NTIgbWggRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydExheW91dFRleHQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjcgJCkgWzFdOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRUZXh0IC0gbGF5b3V0IGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgU1lOT1BTSVMKCkFsbCBsYXlvdXQgZnVuY3Rpb25zIGZvciB0ZXh0IGVsZW1lbnRzCgo9b3ZlciA0Cgo9Y3V0Cgo9aXRlbSBuZXcoKQoKY3JlYXRlIGFuIG9iamVjdAoKICAgICRCYWNrZW5kT2JqZWN0ID0gS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydExheW91dFRleHQtPm5ldygKICAgICAgICAlUGFyYW0sCiAgICApOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBvYmplY3RzCiAgICBmb3IgbXkgJE9iamVjdCAocXcoQ29uZmlnT2JqZWN0IExvZ09iamVjdCBNYWluT2JqZWN0IFBhcmFtT2JqZWN0IExheW91dE9iamVjdCkpIHsKICAgICAgICAkU2VsZi0+eyRPYmplY3R9ID0gJFBhcmFteyRPYmplY3R9IHx8IGRpZSAiR290IG5vICRPYmplY3QhIjsKICAgIH0KCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1pdGVtIEZvcm1JbnB1dENyZWF0ZSgpCgpjcmVhdGUgYSBpbnB1dCBzdHJpbmcKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+Rm9ybUlucHV0Q3JlYXRlKAogICAgICAgIEl0ZW0gICA9PiAkSXRlbVJlZiwKICAgICAgICBQcmVmaXggPT4gJ1ByZWZpeDo6JywgICMgKG9wdGlvbmFsKQogICAgICAgIFZhbHVlICA9PiAnVmFsdWUnLCAgICAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1JbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coIFByaW9yaXR5ID0+ICdlcnJvcicsIE1lc3NhZ2UgPT4gJ05lZWQgSXRlbSEnICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICRQYXJhbXtQcmVmaXh9IHx8PSAnJzsKCiAgICBteSAkVmFsdWUgPSAkUGFyYW17VmFsdWV9IHx8ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1ZhbHVlRGVmYXVsdH07CiAgICBteSAkU2l6ZSA9ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1NpemV9IHx8IDQwOwogICAgbXkgJFN0cmluZyA9ICI8aW5wdXQgdHlwZT1cIlRleHRcIiBuYW1lPVwiJFBhcmFte1ByZWZpeH0kUGFyYW17SXRlbX0tPntLZXl9XCIgc2l6ZT1cIiRTaXplXCIgIjsKCiAgICBpZiAoJFZhbHVlKSB7CgogICAgICAgICMgdHJhbnNsYXRlCiAgICAgICAgaWYgKCAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntUcmFuc2xhdGlvbn0gKSB7CiAgICAgICAgICAgICRWYWx1ZSA9ICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+e0xhbmd1YWdlT2JqZWN0fS0+R2V0KCRWYWx1ZSk7CiAgICAgICAgfQoKICAgICAgICAjIHRyYW5zZm9ybSBhc2NpaSB0byBodG1sCiAgICAgICAgJFZhbHVlID0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5Bc2NpaTJIdG1sKAogICAgICAgICAgICBUZXh0ICAgICAgICAgICA9PiAkVmFsdWUsCiAgICAgICAgICAgIEhUTUxSZXN1bHRNb2RlID0+IDEsCiAgICAgICAgKTsKCiAgICAgICAgJFN0cmluZyAuPSAidmFsdWU9XCIkVmFsdWVcIiAiOwogICAgfQoKICAgICMgYWRkIG1heGltdW0gbGVuZ3RoCiAgICBpZiAoICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e01heExlbmd0aH0gKSB7CiAgICAgICAgJFN0cmluZyAuPSAibWF4bGVuZ3RoPVwiJFBhcmFte0l0ZW19LT57SW5wdXR9LT57TWF4TGVuZ3RofVwiICI7CiAgICB9CgogICAgJFN0cmluZyAuPSAiPiAiOwoKICAgIHJldHVybiAkU3RyaW5nOwp9Cgo9aXRlbSBGb3JtRGF0YUdldCgpCgpnZXQgZm9ybSBkYXRhCgogICAgbXkgJEZvcm1EYXRhID0gJEJhY2tlbmRPYmplY3QtPkZvcm1EYXRhR2V0KAogICAgICAgIEl0ZW0gICA9PiAkSXRlbVJlZiwKICAgICAgICBQcmVmaXggPT4gJ1ByZWZpeDo6JywgICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBGb3JtRGF0YUdldCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coIFByaW9yaXR5ID0+ICdlcnJvcicsIE1lc3NhZ2UgPT4gJ05lZWQgSXRlbSEnICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICRQYXJhbXtQcmVmaXh9IHx8PSAnJzsKCiAgICAjIGdldCBmb3JtIGRhdGEKICAgIG15ICRGb3JtRGF0YSA9ICRTZWxmLT57UGFyYW1PYmplY3R9LT5HZXRQYXJhbSgKICAgICAgICBQYXJhbSA9PiAkUGFyYW17UHJlZml4fSAuICRQYXJhbXtJdGVtfS0+e0tleX0sCiAgICApOwoKICAgICMgcmVnZXggY2hlY2sKICAgIGlmICggJFBhcmFte0l0ZW19LT57SW5wdXR9LT57UmVnZXh9ICYmICRGb3JtRGF0YSAhfiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntSZWdleH0gKSB7CgogICAgICAgICRQYXJhbXtJdGVtfS0+e0Zvcm19LT57SW52YWxpZH0gPSAxOwogICAgICAgIHJldHVybiAkRm9ybURhdGE7CiAgICB9CgogICAgcmV0dXJuICRGb3JtRGF0YSBpZiAkRm9ybURhdGE7CiAgICByZXR1cm4gJEZvcm1EYXRhIGlmICEkUGFyYW17SXRlbX0tPntJbnB1dH0tPntSZXF1aXJlZH07CgogICAgIyBzZXQgaW52YWxpZCBwYXJhbQogICAgJFBhcmFte0l0ZW19LT57Rm9ybX0tPntJbnZhbGlkfSA9IDE7CgogICAgcmV0dXJuICRGb3JtRGF0YTsKfQoKMTsKCj1iYWNrCgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKClRoaXMgc29mdHdhcmUgaXMgcGFydCBvZiB0aGUgT1RSUyBwcm9qZWN0IChodHRwOi8vb3Rycy5vcmcvKS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KCj1jdXQKCj1oZWFkMSBWRVJTSU9OCgokUmV2aXNpb246IDEuNyAkICREYXRlOiAyMDA5LzA1LzE4IDA5OjQyOjUyICQKCj1jdXQK
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9MYXlvdXRJbXBvcnRFeHBvcnQucG0gLSBwcm92aWRlcyBnZW5lcmljIEhUTUwgb3V0cHV0IGZvciBJbXBvcnRFeHBvcnQKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOSBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IExheW91dEltcG9ydEV4cG9ydC5wbSx2IDEuNCAyMDA5LzA1LzE4IDA5OjQyOjUyIG1oIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXRJbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjQgJCkgWzFdOwoKPWl0ZW0gSW1wb3J0RXhwb3J0Rm9ybUlucHV0Q3JlYXRlKCkKCnJldHVybnMgYSBpbnB1dCBmaWVsZCBodG1sIHN0cmluZwoKICAgIG15ICRTdHJpbmcgPSAkTGF5b3V0T2JqZWN0LT5JbXBvcnRFeHBvcnRGb3JtSW5wdXRDcmVhdGUoCiAgICAgICAgSXRlbSAgPT4gJEl0ZW1SZWYsCiAgICAgICAgVmFsdWUgPT4gJ1ZhbHVlJywgICAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgSW1wb3J0RXhwb3J0Rm9ybUlucHV0Q3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZyggUHJpb3JpdHkgPT4gJ2Vycm9yJywgTWVzc2FnZSA9PiAnTmVlZCBJdGVtIScgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBsb2FkIGJhY2tlbmQKICAgIG15ICRCYWNrZW5kT2JqZWN0ID0gJFNlbGYtPl9JbXBvcnRFeHBvcnRMb2FkTGF5b3V0QmFja2VuZCgKICAgICAgICBUeXBlID0+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1R5cGV9LAogICAgKTsKCiAgICByZXR1cm4gJycgaWYgISRCYWNrZW5kT2JqZWN0OwoKICAgICMgbG9va3VwIGl0ZW0gdmFsdWUKICAgIG15ICRTdHJpbmcgPSAkQmFja2VuZE9iamVjdC0+Rm9ybUlucHV0Q3JlYXRlKCVQYXJhbSk7CgogICAgcmV0dXJuICRTdHJpbmc7Cn0KCj1pdGVtIEltcG9ydEV4cG9ydEZvcm1EYXRhR2V0KCkKCnJldHVybnMgdGhlIHZhbHVlcyBmcm9tIHRoZSBodG1sIGZvcm0gYXMgaGFzaCByZWZlcmVuY2UKCiAgICBteSAkRm9ybURhdGEgPSAkTGF5b3V0T2JqZWN0LT5JbXBvcnRFeHBvcnRGb3JtRGF0YUdldCgKICAgICAgICBJdGVtID0+ICRJdGVtUmVmLAogICAgKTsKCj1jdXQKCnN1YiBJbXBvcnRFeHBvcnRGb3JtRGF0YUdldCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coIFByaW9yaXR5ID0+ICdlcnJvcicsIE1lc3NhZ2UgPT4gJ05lZWQgSXRlbSEnICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICMgbG9hZCBiYWNrZW5kCiAgICBteSAkQmFja2VuZE9iamVjdCA9ICRTZWxmLT5fSW1wb3J0RXhwb3J0TG9hZExheW91dEJhY2tlbmQoCiAgICAgICAgVHlwZSA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntUeXBlfSwKICAgICk7CgogICAgcmV0dXJuIGlmICEkQmFja2VuZE9iamVjdDsKCiAgICAjIGdldCBmb3JtIGRhdGEKICAgIG15ICRGb3JtRGF0YSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtRGF0YUdldCglUGFyYW0pOwoKICAgIHJldHVybiAkRm9ybURhdGE7Cn0KCj1pdGVtIF9JbXBvcnRFeHBvcnRMb2FkTGF5b3V0QmFja2VuZCgpCgp0byBsb2FkIGEgaW1wb3J0L2V4cG9ydCBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCiAgICAkSGFzaFJlZiA9ICRMYXlvdXRPYmplY3QtPl9JbXBvcnRFeHBvcnRMb2FkTGF5b3V0QmFja2VuZCgKICAgICAgICBUeXBlID0+ICdTZWxlY3Rpb24nLAogICAgKTsKCj1jdXQKCnN1YiBfSW1wb3J0RXhwb3J0TG9hZExheW91dEJhY2tlbmQgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICBpZiAoICEkUGFyYW17VHlwZX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIFR5cGUhJywKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAjIGNoZWNrIGlmIG9iamVjdCBpcyBhbHJlYWR5IGNhY2hlZAogICAgcmV0dXJuICRTZWxmLT57Q2FjaGV9LT57SW1wb3J0RXhwb3J0TG9hZExheW91dEJhY2tlbmR9LT57ICRQYXJhbXtUeXBlfSB9CiAgICAgICAgaWYgJFNlbGYtPntDYWNoZX0tPntJbXBvcnRFeHBvcnRMb2FkTGF5b3V0QmFja2VuZH0tPnsgJFBhcmFte1R5cGV9IH07CgogICAgbXkgJEdlbmVyaWNNb2R1bGUgPSAiS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydExheW91dCRQYXJhbXtUeXBlfSI7CgogICAgIyBsb2FkIHRoZSBiYWNrZW5kIG1vZHVsZQogICAgaWYgKCAhJFNlbGYtPntNYWluT2JqZWN0fS0+UmVxdWlyZSgkR2VuZXJpY01vZHVsZSkgKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJDYW4ndCBsb2FkIGJhY2tlbmQgbW9kdWxlICRQYXJhbXtUeXBlfSEiCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBjcmVhdGUgbmV3IGluc3RhbmNlCiAgICBteSAkQmFja2VuZE9iamVjdCA9ICRHZW5lcmljTW9kdWxlLT5uZXcoCiAgICAgICAgJXskU2VsZn0sCiAgICAgICAgJVBhcmFtLAogICAgICAgIExheW91dE9iamVjdCA9PiAkU2VsZiwKICAgICk7CgogICAgaWYgKCAhJEJhY2tlbmRPYmplY3QgKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJDYW4ndCBjcmVhdGUgYSBuZXcgaW5zdGFuY2Ugb2YgYmFja2VuZCBtb2R1bGUgJFBhcmFte1R5cGV9ISIsCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBjYWNoZSB0aGUgb2JqZWN0CiAgICAkU2VsZi0+e0NhY2hlfS0+e0ltcG9ydEV4cG9ydExvYWRMYXlvdXRCYWNrZW5kfS0+eyAkUGFyYW17VHlwZX0gfSA9ICRCYWNrZW5kT2JqZWN0OwoKICAgIHJldHVybiAkQmFja2VuZE9iamVjdDsKfQoKMTsK
# --
# AdminImportExport.dtl - provides HTML form for AdminImportExport
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: AdminImportExport.dtl,v 1.16 2009/05/18 09:42:52 mh Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

<!-- dtl:block:Overview -->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td colspan="2" class="mainhead">
      $Env{"Box0"}$Text{"Import/Export Management"}$Env{"Box1"}
    </td>
  </tr>
  <tr>
    <td width="30%" class="mainbody">
      <form action="$Env{"CGIHandle"}" method="get">
        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="TemplateEdit1">
        <input type="hidden" name="TemplateID" value="NEW">
        <table width="100%" cellspacing="0" cellpadding="4">
          <tr>
            <td class="contenthead">$Text{"Add mapping template"}:</td>
          </tr>
          <tr>
            <td class="contentbody">
              <table>
                <tr>
                  <td class="contentkey" width="100">$Text{"Object"}: </td>
                  <td class="contentvalue">$Data{"ObjectOptionStrg"}</td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Format"}: </td>
                  <td class="contentvalue">$Data{"FormatOptionStrg"}</td>
                </tr>
              </table>
            </td>
          </tr>
          <tr>
            <td class="contentfooter">
              <input class="button" type="submit" value="$Text{"Add"}">
            </td>
          </tr>
        </table>
      </form>
    </td>
    <td width="70%" class="mainbody">
<!-- dtl:block:OverviewList -->
      <table width="100%" cellspacing="0" cellpadding="4">
        <tr>
          <td class="contenthead">$Text{"$QData{"ObjectName"}"}:</td>
        </tr>
        <tr>
          <td class="contentbody">
            <table width="100%" border="0" cellspacing="0" cellpadding="3">
              <tr>
                <td class="contentkey">$Text{"Number"}</td>
                <td class="contentkey">$Text{"Name"}</td>
                <td class="contentkey">$Text{"Format"}</td>
                <td align="center" class="contentkey">$Text{"valid"}/$Text{"invalid"}</td>
                <td align="center" class="contentkey" width="90">$Text{"Delete"}</td>
                <td align="center" class="contentkey" width="90">$Text{"Start Import"}</td>
                <td align="center" class="contentkey" width="90">$Text{"Start Export"}</td>
              </tr>
<!-- dtl:block:OverviewListRow -->
              <tr>
                <td class="$QData{"CssClass"}">
                  <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=TemplateEdit1&TemplateID=$QData{"TemplateID"}">
                  $QData{"Number"}
                  </a>
                </td>
                <td class="$QData{"CssClass"}">$QData{"Name"}</td>
                <td class="$QData{"CssClass"}">$Text{"$QData{"FormatName"}"}</td>
                <td align="center" class="$QData{"CssClass"}">$Text{"$QData{"Valid"}"}</td>
                <td align="center" class="$QData{"CssClass"}">
                  <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=TemplateDelete&TemplateID=$QData{"TemplateID"}">
                  $Text{"x"}
                  </a>
                </td>
                <td align="center" class="$QData{"CssClass"}">
                  <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=ImportInformation&TemplateID=$QData{"TemplateID"}">
                  $Text{"x"}
                  </a>
                </td>
                <td align="center" class="$QData{"CssClass"}">
                  <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=Export&TemplateID=$QData{"TemplateID"}">
                  $Text{"x"}
                  </a>
                </td>
              </tr>
<!-- dtl:block:OverviewListRow -->
            </table>
            <br>
          </td>
        </tr>
        <tr>
          <td class="contentfooter">
            &nbsp;
          </td>
        </tr>
      </table>
      <br>
<!-- dtl:block:OverviewList -->
<!-- dtl:block:TemplateEdit1 -->
      <form action="$Env{"CGIHandle"}" method="get">
        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="TemplateSave1">
        <input type="hidden" name="TemplateID" value="$QData{"TemplateID"}">
        <table width="100%" cellspacing="0" cellpadding="4">
          <tr>
            <td class="contenthead">$Text{"Step"} 1 $Text{"of"} 5 - $Text{"Edit common information"}:</td>
          </tr>
          <tr>
            <td class="contentbody">
              <table border="0" cellspacing="0" cellpadding="3">
                <tr>
                  <td class="contentkey" width="150">$Text{"Name"}: </td>
                  <td class="contentvalue">
                    <input type="text" name="Name" value="$QData{"Name"}" size="45" maxlength="200">
                  </td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Object"}: </td>
                  <td class="contentvalue">
                    $QData{"ObjectName"}
                    <input type="hidden" name="Object" value="$QData{"Object"}">
                  </td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Format"}: </td>
                  <td class="contentvalue">
                    $QData{"FormatName"}
                    <input type="hidden" name="Format" value="$QData{"Format"}">
                  </td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Valid"}: </td>
                  <td class="contentvalue">$Data{"ValidOptionStrg"}</td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Comment"}: </td>
                  <td class="contentvalue">
                    <input type="text" name="Comment" value="$QData{"Comment"}" size="50" maxlength="200">
                  </td>
                </tr>
              </table>
            </td>
          </tr>
          <tr>
            <td class="contentfooter">
              <input class="button" type="submit" name="SubmitNext" value="$Text{"Next..."}">
            </td>
          </tr>
        </table>
      </form>
<!-- dtl:block:TemplateEdit1 -->
<!-- dtl:block:TemplateEdit2 -->
      <form action="$Env{"CGIHandle"}" method="get">
        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="TemplateSave2">
        <input type="hidden" name="TemplateID" value="$QData{"TemplateID"}">
        <table width="100%" cellspacing="0" cellpadding="4">
          <tr>
            <td class="contenthead">$Text{"Step"} 2 $Text{"of"} 5 - $Text{"Edit object information"}:</td>
          </tr>
          <tr>
            <td class="contentbody">
              <table border="0" cellspacing="0" cellpadding="3">
                <tr>
                  <td class="contentkey" width="150">$Text{"Name"}: </td>
                  <td class="contentvalue">$QData{"Name"}</td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Object"}: </td>
                  <td class="contentvalue">$QData{"ObjectName"}</td>
                </tr>
<!-- dtl:block:TemplateEdit2Row -->
                <tr>
                  <td class="contentkey">$Text{"$QData{"Name"}"}:
<!-- dtl:block:TemplateEdit2RowRequired -->
                    <font color="red" title="$Text{"Required"}">*</font>
<!-- dtl:block:TemplateEdit2RowRequired -->
                    &nbsp;
                  </td>
                  <td class="contentvalue">$Data{"InputStrg"}</td>
                </tr>
<!-- dtl:block:TemplateEdit2Row -->
              </table>
            </td>
          </tr>
          <tr>
            <td class="contentfooter">
              <input class="button" type="submit" name="SubmitBack" value="$Text{"...Back"}">&nbsp;
              <input class="button" type="submit" name="SubmitNext" value="$Text{"Next..."}">
            </td>
          </tr>
        </table>
      </form>
<!-- dtl:block:TemplateEdit2 -->
<!-- dtl:block:TemplateEdit3 -->
      <form action="$Env{"CGIHandle"}" method="get">
        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="TemplateSave3">
        <input type="hidden" name="TemplateID" value="$QData{"TemplateID"}">
        <table width="100%" cellspacing="0" cellpadding="4">
          <tr>
            <td class="contenthead">$Text{"Step"} 3 $Text{"of"} 5 - $Text{"Edit format information"}:</td>
          </tr>
          <tr>
            <td class="contentbody">
              <table border="0" cellspacing="0" cellpadding="3">
                <tr>
                  <td class="contentkey" width="150">$Text{"Name"}: </td>
                  <td class="contentvalue">$QData{"Name"}</td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Format"}: </td>
                  <td class="contentvalue">$QData{"FormatName"}</td>
                </tr>
<!-- dtl:block:TemplateEdit3Row -->
                <tr>
                  <td class="contentkey">$Text{"$QData{"Name"}"}:
<!-- dtl:block:TemplateEdit3RowRequired -->
                    <font color="red" title="$Text{"Required"}">*</font>
<!-- dtl:block:TemplateEdit3RowRequired -->
                    &nbsp;
                  </td>
                  <td class="contentvalue">$Data{"InputStrg"}</td>
                </tr>
<!-- dtl:block:TemplateEdit3Row -->
              </table>
            </td>
          </tr>
          <tr>
            <td class="contentfooter">
              <input class="button" type="submit" name="SubmitBack" value="$Text{"...Back"}">&nbsp;
              <input class="button" type="submit" name="SubmitNext" value="$Text{"Next..."}">
            </td>
          </tr>
        </table>
      </form>
<!-- dtl:block:TemplateEdit3 -->
<!-- dtl:block:TemplateEdit4 -->
      <form action="$Env{"CGIHandle"}" method="get">
        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="TemplateSave4">
        <input type="hidden" name="TemplateID" value="$QData{"TemplateID"}">
        <table width="100%" cellspacing="0" cellpadding="4">
          <tr>
            <td class="contenthead">$Text{"Step"} 4 $Text{"of"} 5 - $Text{"Edit mapping information"}:</td>
          </tr>
          <tr>
            <td class="contentbody">
              <table border="0" cellspacing="0" cellpadding="3">
                <tr>
                  <td class="contentkey" width="150">$Text{"Name"}: </td>
                  <td class="contentvalue">$QData{"Name"}</td>
                </tr>
              </table>
              <br>
              <table border="0" cellspacing="0" cellpadding="2" width="100%">
                <tr>
                  <td class="contentkey">$QData{"ObjectName"}</td>
                  <td class="contentkey">$QData{"FormatName"}</td>
                  <td class="contentkey">&nbsp;</td>
                </tr>
                <br>
                <tr>
                  <td class="contentvalue" colspan="3"><hr width="100%"></td>
                </tr>
<!-- dtl:block:TemplateEdit4Row -->
                <tr>
                  <td>
                    <table border="0" cellspacing="0" cellpadding="3">
<!-- dtl:block:TemplateEdit4RowObject -->
                      <tr>
                        <td class="contentkey">$Text{"$QData{"Name"}"}: </td>
                        <td class="contentvalue">$Data{"InputStrg"}</td>
                      </tr>
<!-- dtl:block:TemplateEdit4RowObject -->
                    </table>
                  </td>
                  <td>
                    <table border="0" cellspacing="0" cellpadding="3">
<!-- dtl:block:TemplateEdit4RowFormat -->
                      <tr>
                        <td class="contentkey">$Text{"$QData{"Name"}"}: </td>
                        <td class="contentvalue">$Data{"InputStrg"}</td>
                      </tr>
<!-- dtl:block:TemplateEdit4RowFormat -->
                    </table>
                  </td>
                  <td align="right">
                    <input class="button" type="submit" name="MappingUp::$QData{"MappingID"}" value="$Text{"Up"}">
                    <input class="button" type="submit" name="MappingDown::$QData{"MappingID"}" value="$Text{"Down"}">
                    &nbsp;&nbsp;&nbsp;
                    <input class="button" type="submit" name="MappingDelete::$QData{"MappingID"}" value="$Text{"Delete"}">
                  </td>
                </tr>
                <tr>
                  <td class="contentvalue" colspan="3"><hr width="100%"></td>
                </tr>
<!-- dtl:block:TemplateEdit4Row -->
                <tr>
                  <td class="contentvalue" colspan="3">
                    <input class="button" type="submit" name="MappingAdd" value="$Text{"Add"}">
                  </td>
                </tr>
              </table>
              <br>
            </td>
          </tr>
          <tr>
            <td class="contentfooter">
              <input class="button" type="submit" name="SubmitBack" value="$Text{"...Back"}">&nbsp;
              <input class="button" type="submit" name="SubmitNext" value="$Text{"Next..."}">
            </td>
          </tr>
        </table>
      </form>
<!-- dtl:block:TemplateEdit4 -->
<!-- dtl:block:TemplateEdit5 -->
      <form action="$Env{"CGIHandle"}" method="get">
        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="TemplateSave5">
        <input type="hidden" name="TemplateID" value="$QData{"TemplateID"}">
        <table width="100%" cellspacing="0" cellpadding="4">
          <tr>
            <td class="contenthead">$Text{"Step"} 5 $Text{"of"} 5 - $Text{"Edit search information"}:</td>
          </tr>
          <tr>
            <td class="contentbody">
              <table border="0" cellspacing="0" cellpadding="3">
                <tr>
                  <td class="contentkey" width="150">$Text{"Name"}: </td>
                  <td class="contentvalue">$QData{"Name"}</td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Restrict export per search"}: </td>
                  <td class="contentvalue">$Data{"RestrictExportStrg"}</td>
                </tr>
              </table>
            </td>
          </tr>
          <tr>
            <td class="contenthead">&nbsp;</td>
          </tr>
          <tr>
            <td class="contentbody">
              <table border="0" cellspacing="0" cellpadding="3">
<!-- dtl:block:TemplateEdit5Row -->
                <tr>
                  <td class="contentkey">$Text{"$QData{"Name"}"}:
                    &nbsp;
                  </td>
                  <td class="contentvalue">$Data{"InputStrg"}</td>
                </tr>
<!-- dtl:block:TemplateEdit5Row -->
              </table>
            </td>
          </tr>
          <tr>
            <td class="contentfooter">
              <input class="button" type="submit" name="SubmitBack" value="$Text{"...Back"}">&nbsp;
              <input class="button" type="submit" name="SubmitNext" value="$Text{"Finish"}">
            </td>
          </tr>
        </table>
      </form>
<!-- dtl:block:TemplateEdit5 -->
<!-- dtl:block:ImportInformation -->
      <form action="$Env{"CGIHandle"}" method="post" enctype="multipart/form-data">
        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="Import">
        <input type="hidden" name="TemplateID" value="$QData{"TemplateID"}">
        <table width="100%" cellspacing="0" cellpadding="4">
          <tr>
            <td class="contenthead">$Text{"Import information"}:</td>
          </tr>
          <tr>
            <td class="contentbody">
              <table border="0" cellspacing="0" cellpadding="3">
                <tr>
                  <td class="contentkey" width="150">$Text{"Name"}: </td>
                  <td class="contentvalue">$QData{"Name"}</td>
                </tr>
                <tr>
                  <td class="contentkey" width="150">$Text{"Source File"}: </td>
                  <td class="contentvalue">
                    <input type="file" name="SourceFile" size="40" class="fixed">
                  </td>
                </tr>
              </table>
            </td>
          </tr>
          <tr>
            <td class="contentfooter">
              <input class="button" type="submit" value="$Text{"Start Import"}">
            </td>
          </tr>
        </table>
      </form>
<!-- dtl:block:ImportInformation -->
    </td>
  </tr>
</table>
<!-- dtl:block:Overview -->

# --
# Kernel/System/ImportExport.pm - all import and export functions
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: ImportExport.pm,v 1.32 2009/07/20 22:55:23 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;

use strict;
use warnings;

use Kernel::System::CheckItem;

use vars qw($VERSION);
$VERSION = qw($Revision: 1.32 $) [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::DB;
    use Kernel::System::ImportExport;
    use Kernel::System::Log;
    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,
        LogObject    => $LogObject,
        EncodeObject => $EncodeObject,
    );
    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! I can't find the template.",
        );
        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} not complete!",
        );
        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(
        %Param,
        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 an 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} not complete!",
        );
        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(
        %Param,
        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 an 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} not complete!",
        );
        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(
        %Param,
        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 an 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} not complete!",
        );
        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(
        %Param,
        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 an 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} not complete!",
        );
        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(
        %Param,
        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 an 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,
    );

=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} not complete!",
        );
        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;
    $Result{Success}            = 0;
    $Result{Failed}             = 0;
    $Result{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}): success!",
    );

    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} not complete!",
        );
        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;

    my %Result;
    $Result{Success} = 0;
    $Result{Failed}  = 0;

    IMPORTDATAROW:
    for my $ImportDataRow ( @{$ImportData} ) {

        # import one row
        my $Success = $ObjectBackend->ImportDataSave(
            TemplateID    => $Param{TemplateID},
            ImportDataRow => $ImportDataRow,
            UserID        => $Param{UserID},
        );

        if ( !$Success ) {
            $Result{Failed}++;
            next IMPORTDATAROW;
        }

        $Result{Success}++;
    }

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

    return \%Result;
}

=item _LoadBackend()

to load a import/export backend module

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

=cut

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

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

    # check if object is already cached
    return $Self->{Cache}->{LoadBackend}->{ $Param{Module} }
        if $Self->{Cache}->{LoadBackend}->{ $Param{Module} };

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

    # cache the object
    $Self->{Cache}->{LoadBackend}->{ $Param{Module} } = $BackendObject;

    return $BackendObject;
}

1;

=back

=head1 TERMS AND CONDITIONS

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

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

=cut

=head1 VERSION

$Revision: 1.32 $ $Date: 2009/07/20 22:55:23 $

=cut

# --
# Kernel/System/ImportExport/FormatBackend/CSV.pm - import/export backend for CSV
# Copyright (C) 2001-2009 OTRS AG, http://otrs.org/
# --
# $Id: CSV.pm,v 1.25 2009/07/20 22:55:23 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.25 $) [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::DB;
    use Kernel::System::Log;
    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,
        LogObject    => $LogObject,
        EncodeObject => $EncodeObject,
    );
    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 seperators
    $Self->{AvailableSeperators} = {
        Tabulator => "\t",
        Semicolon => ';',
        Colon     => ':',
        Dot       => '.',
    };

    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   => 'ColumnSeperator',
            Name  => 'Column Seperator',
            Input => {
                Type => 'Selection',
                Data => {
                    Tabulator => 'Tabulator (TAB)',
                    Semicolon => 'Semicolon (;)',
                    Colon     => 'Colon (:)',
                    Dot       => 'Dot (.)',
                },
                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 charset
    $FormatData->{ColumnSeperator} ||= '';
    my $Seperator = $Self->{AvailableSeperators}->{ $FormatData->{ColumnSeperator} } || '';

    # check the seperator
    if ( !$Seperator ) {

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

    # create the parser object
    my $ParseObject = Text::CSV->new(
        {
            quote_char          => '"',
            escape_char         => '"',
            sep_char            => $Seperator,
            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 @ImportData;
    while ( my $Column = $ParseObject->getline($FH) ) {
        push @ImportData, $Column;
    }

    # 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->{ColumnSeperator} ||= '';
    my $Seperator = $Self->{AvailableSeperators}->{ $FormatData->{ColumnSeperator} } || '';

    # check the seperator
    if ( !$Seperator ) {

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

    # create the parser object
    my $ParseObject = Text::CSV->new(
        {
            quote_char          => '"',
            escape_char         => '"',
            sep_char            => $Seperator,
            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.25 $ $Date: 2009/07/20 22:55:23 $

=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-2009 OTRS AG, http://otrs.org/
# --
# $Id: ImportExportFormatCSV.t,v 1.8 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 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   => 'ColumnSeperator',
        Name  => 'Column Seperator',
        Input => {
            Type => 'Selection',
            Data => {
                Tabulator => 'Tabulator (TAB)',
                Semicolon => 'Semicolon (;)',
                Colon     => 'Colon (:)',
                Dot       => 'Dot (.)',
            },
            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 seperator and charset are given (check return false)
    {
        SourceImportData => {
            ImportDataGet => {
                TemplateID    => $TemplateIDs[2],
                SourceContent => \do {'Dummy'},
                UserID        => 1,
            },
        },
    },

    # no column seperator 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 => {
                ColumnSeperator => 'Dummy',
            },
            ImportDataGet => {
                TemplateID    => $TemplateIDs[2],
                SourceContent => \do {'Dummy'},
                UserID        => 1,
            },
        },
    },

    # invalid column seperator is given (check return false)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => 'Semicolon',
                Charset         => 'UTF-8',
            },
            ImportDataGet => {
                TemplateID    => $TemplateIDs[4],
                SourceContent => \do {'  '},
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            ['  '],
        ],
    },

    # all required values are given (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeperator => '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 seperator (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 seperator (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => 'Colon',
                Charset         => 'UTF-8',
            },
            SourceFile    => 'ImportExportFormatCSV006-OpenOffice-Colon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[10],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'ʩ ʬ ʮ',     ' ʡ ˤ Ό ' ],
            [ '  Η ϗ Ϡ  ', 'Ά Λ Ξ' ],
        ],
    },
];

# ------------------------------------------------------------ #
# 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/' . $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 seperator and charset are given (check return false)
    {
        SourceExportData => {
            ExportDataSave => {
                TemplateID    => $TemplateIDs[21],
                ExportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # no column seperator 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 => {
                ColumnSeperator => 'Dummy',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[21],
                ExportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # invalid column seperator is given (check return false)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => '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 => {
                ColumnSeperator => 'Semicolon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[26],
                ExportDataRow => [ ' Ѫ Ѭ Ѳ', 'ѯ Ѵ ѿ', '҂ Ҋ Җ ' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '" Ѫ Ѭ Ѳ";"ѯ Ѵ ѿ";"҂ Ҋ Җ "',
    },

    # all required values are given (UTF-8 checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeperator => 'Tabulator',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[26],
                ExportDataRow => [ ' Ѫ Ѭ Ѳ', 'ѯ Ѵ ѿ', '҂ Ҋ Җ ' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => "\" Ѫ Ѭ Ѳ\"\t\"ѯ Ѵ ѿ\"\t\"҂ Ҋ Җ \"",
    },

    # all required values are given (UTF-8 checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeperator => 'Colon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[26],
                ExportDataRow => [ ' Ѫ Ѭ Ѳ', 'ѯ Ѵ ѿ', '҂ Ҋ Җ ' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '" Ѫ Ѭ Ѳ":"ѯ Ѵ ѿ":"҂ Ҋ Җ "',
    },

    # all required values are given (UTF-8 checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeperator => '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==