ImportExport
4.0.32
OTRS AG
https://otrs.com/
GNU GENERAL PUBLIC LICENSE Version 3, 29 June 2007
Build for OTRS::ITSM 4 patch level 32.
Build for OTRS::ITSM 4 patch level 31.
Build for OTRS::ITSM 4 patch level 30.
Build for OTRS::ITSM 4 patch level 29.
Build for OTRS::ITSM 4 patch level 28.
Build for OTRS::ITSM 4 patch level 27.
Build for OTRS::ITSM 4 patch level 26.
Build for OTRS::ITSM 4 patch level 25.
Build for OTRS::ITSM 4 patch level 24.
Build for OTRS::ITSM 4 patch level 23.
Build for OTRS::ITSM 4 patch level 22.
Build for OTRS::ITSM 4 patch level 21.
Build for OTRS::ITSM 4 patch level 20.
Build for OTRS::ITSM 4 patch level 19.
Build for OTRS::ITSM 4 patch level 18.
Build for OTRS::ITSM 4 patch level 17.
Build for OTRS::ITSM 4 patch level 16.
Build for OTRS::ITSM 4 patch level 15.
Build for OTRS::ITSM 4 patch level 14.
Build for OTRS::ITSM 4 patch level 13.
Build for OTRS::ITSM 4 patch level 12.
Build for OTRS::ITSM 4 patch level 11.
Build for OTRS::ITSM 4 patch level 10.
Build for OTRS::ITSM 4 patch level 9.
Build for OTRS::ITSM 4 patch level 8.
Build for OTRS::ITSM 4 patch level 7.
Build for OTRS::ITSM 4 patch level 6.
Build for OTRS::ITSM 4 patch level 5.
Build for OTRS::ITSM 4 patch level 4.
Build for OTRS::ITSM 4 patch level 3.
Build for OTRS::ITSM 4 patch level 2.
Build for OTRS::ITSM 4.
Build for OTRS::ITSM 4 rc1.
Build for OTRS::ITSM 4 beta5.
Build for OTRS::ITSM 4 beta4.
Build for OTRS::ITSM 4 beta3.
Build for OTRS::ITSM 4 beta2.
Build for OTRS::ITSM 4 beta1.
The ImportExport package.
Das ImportExport Paket.
El paquete ImportExport.
4.0.x
</br>
<strong>ATTENTION</strong>
</br>
</br>
If you uninstall this package, all database tables that were created during installation will be deleted.
All data from these tables will be irrevocably lost!
</br>
</br>
((enjoy))</br>
</br>
</br>
<strong>ACHTUNG</strong>
</br>
</br>
Bei der Deinstallation werden die von diesem Paket angelegten Datenbank-Tabellen gelöscht.
Alle darin enthaltenen Daten gehen unwiderruflich verloren!
</br>
</br>
((enjoy))</br>
</br>
<br/>
<strong>ATENCIÓN</strong>
<br/>
<br/>
Si usted desinstala este paquete, todas las tablas de la base de datos que se crearon durante la instalación
del mismo, se eliminarán. ¡La totalidad de los datos que contenían dichas tablas, se perderá irrevocablemente!
<br/>
<br/>
((enjoy))<br/>
<br/>
# create the package name
my $CodeModule = 'var::packagesetup::' . $Param{Structure}->{Name}->{Content};
# discard internally stored object, so that the next access to object creates them newly
$Kernel::OM->ObjectsDiscard(
Objects => [$CodeModule],
ForcePackageReload => 1,
);
$Kernel::OM->Get($CodeModule)->CodeUpgradeFromBefore_2_0_3();
2018-09-14 17:58:39
opms.otrs.com
IyEvdXNyL2Jpbi9wZXJsCiMgLS0KIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxOCBPVFJTIEFHLCBodHRwczovL290cnMuY29tLwojIC0tCiMgVGhpcyBwcm9ncmFtIGlzIGZyZWUgc29mdHdhcmU6IHlvdSBjYW4gcmVkaXN0cmlidXRlIGl0IGFuZC9vciBtb2RpZnkKIyBpdCB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGFzIHB1Ymxpc2hlZCBieQojIHRoZSBGcmVlIFNvZnR3YXJlIEZvdW5kYXRpb24sIGVpdGhlciB2ZXJzaW9uIDMgb2YgdGhlIExpY2Vuc2UsIG9yCiMgKGF0IHlvdXIgb3B0aW9uKSBhbnkgbGF0ZXIgdmVyc2lvbi4KIwojIFRoaXMgcHJvZ3JhbSBpcyBkaXN0cmlidXRlZCBpbiB0aGUgaG9wZSB0aGF0IGl0IHdpbGwgYmUgdXNlZnVsLAojIGJ1dCBXSVRIT1VUIEFOWSBXQVJSQU5UWTsgd2l0aG91dCBldmVuIHRoZSBpbXBsaWVkIHdhcnJhbnR5IG9mCiMgTUVSQ0hBTlRBQklMSVRZIG9yIEZJVE5FU1MgRk9SIEEgUEFSVElDVUxBUiBQVVJQT1NFLiBTZWUgdGhlCiMgR05VIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgZm9yIG1vcmUgZGV0YWlscy4KIwojIFlvdSBzaG91bGQgaGF2ZSByZWNlaXZlZCBhIGNvcHkgb2YgdGhlIEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlCiMgYWxvbmcgd2l0aCB0aGlzIHByb2dyYW0uIElmIG5vdCwgc2VlIGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQuCiMgLS0KCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSBGaWxlOjpCYXNlbmFtZTsKdXNlIEZpbmRCaW4gcXcoJFJlYWxCaW4pOwp1c2UgbGliIGRpcm5hbWUoJFJlYWxCaW4pOwp1c2UgbGliIGRpcm5hbWUoJFJlYWxCaW4pIC4gJy9LZXJuZWwvY3Bhbi1saWInOwp1c2UgdmFycyBxdygkUmVhbEJpbik7Cgp1c2UgR2V0b3B0OjpTdGQ7Cgp1c2UgS2VybmVsOjpTeXN0ZW06Ok9iamVjdE1hbmFnZXI7Cgpsb2NhbCAkS2VybmVsOjpPTSA9IEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyLT5uZXcoCiAgICAnS2VybmVsOjpTeXN0ZW06OkxvZycgPT4gewogICAgICAgIExvZ1ByZWZpeCA9PiAnT1RSUy1JbXBvcnRFeHBvcnQnLAogICAgfSwKKTsKCiMgZ2V0IG9wdGlvbnMKbXkgJU9wdHM7CmdldG9wdHMoICdobjphOmk6bzonLCBcJU9wdHMgKTsKCmlmICggJE9wdHN7aH0gKSB7CgogICAgcHJpbnQgU1RET1VUICJvdHJzLkltcG9ydEV4cG9ydC5wbCAtIGFuIGltcG9ydC9leHBvcnQgdG9vbFxuIjsKICAgIHByaW50IFNURE9VVCAiQ29weXJpZ2h0IChDKSAyMDAxLTIwMTggT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS9cbiI7CiAgICBwcmludCBTVERPVVQgIlxuIjsKICAgIHByaW50IFNURE9VVCAidXNhZ2U6IG90cnMuSW1wb3J0RXhwb3J0LnBsIC1uIDxUZW1wbGF0ZU51bWJlcj4gLWEgaW1wb3J0fGV4cG9ydCAiOwogICAgcHJpbnQgU1RET1VUICJbLWkgPFNvdXJjZUZpbGU+XSBbLW8gPERlc3RpbmF0aW9uRmlsZT5dXG4iOwogICAgcHJpbnQgU1RET1VUICJcbiI7CiAgICBwcmludCBTVERPVVQgIiAgIGV4YW1wbGVzOlxuIjsKICAgIHByaW50IFNURE9VVCAiICAgICAgIG90cnMuSW1wb3J0RXhwb3J0LnBsIC1uIDAwMDA0IC1hIGltcG9ydCAtaSAvdG1wL2ltcG9ydC5jc3ZcbiI7CiAgICBwcmludCBTVERPVVQgIiAgICAgICBvdHJzLkltcG9ydEV4cG9ydC5wbCAtbiAwMDAwNCAtYSBleHBvcnQgLW8gL3RtcC9leHBvcnQuY3N2XG4iOwoKICAgIGV4aXQgMTsKfQoKIyBjaGVjayB0ZW1wbGF0ZSBudW1iZXIKaWYgKCAhJE9wdHN7bn0gKSB7CiAgICBwcmludCBTVERFUlIgIkVSUk9SOiBOZWVkIC1uIFRlbXBsYXRlTnVtYmVyXG4iOwogICAgZXhpdCAxOwp9CmlmICggJE9wdHN7bn0gIX4gbXsgXEEgXGQrIFx6IH14bXMgKSB7CiAgICBwcmludCBTVERFUlIgIkVSUk9SOiBJbnZhbGlkIFRlbXBsYXRlTnVtYmVyXG4iOwogICAgZXhpdCAxOwp9Cm15ICRUZW1wbGF0ZUlEID0gaW50ICRPcHRze259OwoKIyBjaGVjayBhY3Rpb24gbW9kZQppZiAoICEkT3B0c3thfSApIHsKICAgIHByaW50IFNUREVSUiAiRVJST1I6IE5lZWQgLWEgaW1wb3J0fGV4cG9ydFxuIjsKICAgIGV4aXQgMTsKfQppZiAoIGxjICRPcHRze2F9IG5lICdpbXBvcnQnICYmIGxjICRPcHRze2F9IG5lICdleHBvcnQnICkgewogICAgcHJpbnQgU1RERVJSICJFUlJPUjogSW52YWxpZCBhY3Rpb25cbiI7CiAgICBleGl0IDE7Cn0KCiMgZ2V0IHRlbXBsYXRlIGRhdGEKbXkgJFRlbXBsYXRlRGF0YSA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJbXBvcnRFeHBvcnQnKS0+VGVtcGxhdGVHZXQoCiAgICBUZW1wbGF0ZUlEID0+ICRUZW1wbGF0ZUlELAogICAgVXNlcklEICAgICA9PiAxLAopOwoKaWYgKCAhJFRlbXBsYXRlRGF0YS0+e1RlbXBsYXRlSUR9ICkgewogICAgcHJpbnQgU1RERVJSICJFUlJPUjogVGVtcGxhdGUgJE9wdHN7bn0gbm90IGZvdW5kIVxuIjsKICAgIHByaW50IFNUREVSUiAiRXhwb3J0IGFib3J0ZWQuXG4iOwogICAgZXhpdCAxOwp9CgojIHRpbWUgdG8gc3RhcnQKaWYgKCBsYyAkT3B0c3thfSBlcSAnaW1wb3J0JyApIHsKCiAgICBteSAkU291cmNlQ29udGVudCA9IFxkbyB7Jyd9OwogICAgaWYgKCAkT3B0c3tpfSApIHsKCiAgICAgICAgcHJpbnQgU1RET1VUICJSZWFkIEZpbGUgJE9wdHN7aX0uXG4iOwoKICAgICAgICAjIHJlYWQgc291cmNlIGZpbGUKICAgICAgICAkU291cmNlQ29udGVudCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpNYWluJyktPkZpbGVSZWFkKAogICAgICAgICAgICBMb2NhdGlvbiA9PiAkT3B0c3tpfSwKICAgICAgICAgICAgUmVzdWx0ICAgPT4gJ1NDQUxBUicsCiAgICAgICAgICAgIE1vZGUgICAgID0+ICdiaW5tb2RlJywKICAgICAgICApOwoKICAgICAgICBkaWUgIkNhbid0IHJlYWQgZmlsZSAkT3B0c3tpfS5cbkltcG9ydCBhYm9ydGVkLlxuIiBpZiAhJFNvdXJjZUNvbnRlbnQ7CiAgICB9CgogICAgcHJpbnQgU1RET1VUICJJbXBvcnQgaW4gcHJvY2Vzcy4uLlxuIjsKCiAgICAjIGltcG9ydCBkYXRhCiAgICBteSAkUmVzdWx0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkltcG9ydEV4cG9ydCcpLT5JbXBvcnQoCiAgICAgICAgVGVtcGxhdGVJRCAgICA9PiAkVGVtcGxhdGVJRCwKICAgICAgICBTb3VyY2VDb250ZW50ID0+ICRTb3VyY2VDb250ZW50LAogICAgICAgIFVzZXJJRCAgICAgICAgPT4gMSwKICAgICk7CgogICAgZGllICJcbkVycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIHRoZSBPVFJTIGxvZyBmb3IgZGV0YWlscy5cbiIgaWYgIWRlZmluZWQgJFJlc3VsdDsKCiAgICAjIFByaW50IHJlc3VsdAogICAgcHJpbnQgU1RET1VUICJcbiI7CiAgICBwcmludCBTVERPVVQKICAgICAgICAiSW1wb3J0IG9mICRSZXN1bHQtPntDb3VudGVyfSAkUmVzdWx0LT57T2JqZWN0fSByZWNvcmRzOiAiCiAgICAgICAgLiAiJFJlc3VsdC0+e0ZhaWxlZH0gZmFpbGVkLCAkUmVzdWx0LT57U3VjY2Vzc30gc3VjY2VlZGVkXG4iOwogICAgZm9yIG15ICRSZXRDb2RlICggc29ydCBrZXlzICV7ICRSZXN1bHQtPntSZXRDb2RlfSB9ICkgewogICAgICAgIG15ICRDb3VudCA9ICRSZXN1bHQtPntSZXRDb2RlfS0+eyRSZXRDb2RlfSB8fCAwOwogICAgICAgIHByaW50IFNURE9VVAogICAgICAgICAgICAiSW1wb3J0IG9mICRSZXN1bHQtPntDb3VudGVyfSAkUmVzdWx0LT57T2JqZWN0fSByZWNvcmRzOiAkQ291bnQgJFJldENvZGVcbiI7CiAgICB9CiAgICBpZiAoICRSZXN1bHQtPntGYWlsZWR9ICkgewogICAgICAgIHByaW50IFNURE9VVAogICAgICAgICAgICAiTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGU6ICRSZXN1bHQtPntDb3VudGVyfVxuIjsKICAgIH0KfQplbHNpZiAoIGxjICRPcHRze2F9IGVxICdleHBvcnQnICkgewoKICAgIHByaW50IFNURE9VVCAiRXhwb3J0IGluIHByb2Nlc3MuLi5cbiI7CgogICAgIyBleHBvcnQgZGF0YQogICAgbXkgJFJlc3VsdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJbXBvcnRFeHBvcnQnKS0+RXhwb3J0KAogICAgICAgIFRlbXBsYXRlSUQgPT4gJFRlbXBsYXRlSUQsCiAgICAgICAgVXNlcklEICAgICA9PiAxLAogICAgKTsKCiAgICBkaWUgIlxuRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLlxuIiBpZiAhZGVmaW5lZCAkUmVzdWx0OwoKICAgIHByaW50IFNURE9VVAogICAgICAgICJcbiIsCiAgICAgICAgIlN1Y2Nlc3M6ICRSZXN1bHQtPntTdWNjZXNzfVxuIiwKICAgICAgICAiRmFpbGVkIDogJFJlc3VsdC0+e0ZhaWxlZH1cbiIsCiAgICAgICAgIlxuIjsKCiAgICBpZiAoICRPcHRze299ICkgewoKICAgICAgICBteSAkRmlsZUNvbnRlbnQgPSBqb2luICJcbiIsIEB7ICRSZXN1bHQtPntEZXN0aW5hdGlvbkNvbnRlbnR9IH07CgogICAgICAgICMgc2F2ZSBkZXN0aW5hdGlvbiBjb250ZW50IHRvIGZpbGUKICAgICAgICBteSAkU3VjY2VzcyA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpNYWluJyktPkZpbGVXcml0ZSgKICAgICAgICAgICAgTG9jYXRpb24gPT4gJE9wdHN7b30sCiAgICAgICAgICAgIENvbnRlbnQgID0+IFwkRmlsZUNvbnRlbnQsCiAgICAgICAgKTsKCiAgICAgICAgZGllICJDYW4ndCB3cml0ZSBmaWxlICRPcHRze299LlxuRXhwb3J0IGFib3J0ZWQuXG4iIGlmICEkU3VjY2VzczsKCiAgICAgICAgcHJpbnQgU1RET1VUICJGaWxlICRPcHRze299IHNhdmVkLlxuIjsKICAgIH0KCiAgICBwcmludCBTVERPVVQgIkV4cG9ydCBjb21wbGV0ZS5cbiI7Cn0KCmV4aXQgMDsKCj1iYWNrCgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKClRoaXMgU29mdHdhcmUgaXMgcGFydCBvZiB0aGUgT1RSUyBwcm9qZWN0IChodHRwOi8vb3Rycy5vcmcvKS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0Pi4KCj1jdXQK
H4sIAAAAAAACA+1dUXPauBZ+76/wsC+7M0SxZMmyy7Y7d3d253amO72zzb70hRHgEN8am7FNUvbh/vYrGUiwsbGxLRqSQ6ckwUKfZPQdHenTOfz8y7dFYNx7ceJH4bsBRubA8MJpNPPD+bvB3zd/XDmDX96/+Xnmi7fy/zwWC0O+I0zUX+8Gd2m6fHt9/fDwgIJ1ItIoRoG/Qol3/T8RBOJaFroevH9jGPsVzEQq1GvbV0Waxv5klXpGKBbeu8FETL/O42gVzgabUtty0yiIYuNeBO8GP9xmj8H1tprrXD1H6l6KuTeJPfG1umpTPly3TdVLLy5Wu1hGiS+LpOvlQZGKetTzXpltqUQWCufvf/gX/mHTpO0LT3WVNbQSJF2IeO6Hhzjy3gSbG0GQQwiRN4O5jmtyyvHunpwONzkvXHBeuPi8cH4yXkZxGgs/PYScRFHgiXCDeiuCxGsPlExFIMfYsX6ZiBHLwcy0mKO6xy3SHu/WT9Oopy5tXn4k4Klknsf+7DiXcyUqannwZ+nd+FvNDWx/wzb1r7XVf+8n/iTwynrgh+kGAHevft1P9cVPKDPqT2/vOCI2tVXOGjPHYx47fdaYr/yZl9QMtXyZiprutsWu6+56sVzTG7N5qTCHZxCBWHvxtvpfn2ZvY/sBy+kmXnkDQ0xT/3731949iSb/9abptrOfUxHORDwzrowb71s62PNNBoY/ezf4ZObvVrF/sjZpHw9u1zJ6HFNsiEmx+4cGpKruyaRYdSxbL8J54D1WbyLpQ7ARochxHImGbIzbIabqHtQM9EKZKmueOQy5cgXf4tPNX58NavwhP17vIYq/VngaR3hYZdijMC0DVq8bt2LhB2vZPhEmAyNJ12rASC9089Z/e8G9l/pTsT+uT2/BnefP70rbsD9H22bugbth5gdh/UBsg5G3ShW2id+qf92ApCcwDxde+efohavFdrppiFJqc05jxn3WpCI3nppi1RBu8+LG9jQwRr96//he/NEPvT2TZG5NEu5qksyhjTWapCsTYemi2c4w+w1zPGJO9ovEVZfsduAT759x1ovmfSthARsy92gBjkzHtYeMIG6b9tGiztBs15VpFIdePFafe1I9qMwD8NJrpwBLF9sbF3l86F8I9a8DQuYrHg6TJz+RmJXW72SwzIpX38WWs+9MJHeBF86P9YP0y/ov/vyLmBdY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acqUKUKIixcv1tbWBgUFWSyW2traixcvDh48WFyJNMl/+srKyt27d1sslvHjx0+dOnXgwIF6vb6goOD777/ftWtXbGxsREREh+fB2k8//bR9+/bGxsaIiIhf//rX1ou4A4Armc1m6TLa5nMMAHpcZWWlTcmuXbukq1P5zkeKO7z00kurV6/+9NNPjx079h//8R9vvvnm3LlzdTrdzJkzDx06FBER8cADD7z44otSmCkiIqK1tbW6uloIoVKp/P39L1y4IFf12Wef/frXvx46dOj7778vXQ0ajcb/9//+X35+vsVimTNnztq1a6VnQEv719TUPPnkk9u3b//DH/6wfv16qc76+no5INJhLGzYsGHnzp0TQsybN8/6kE616+A8CCGWLVu2Z8+e8vLyKVOmrFy58o477uiwn4mJiUII67vxNtkcbrPZZrvtnbfm5uann376nXfe8fHxWbdu3V/+8pcTJ07IDe3bty8jI6Oqqso6Btep+mWbN29OTU2VN8eNGzdu3Dj7obW0tMiX3FOmTGloaHBcLoRITk7etGnTCy+84PikoVd1+fbQuoYJEybcdtttoaGhly9f/vLLLwsLCz/55JPhw4eHh4eLnrjXk58I352kHl9f3/T09Ndee+3cuXO7du269957hRAff/xxWVlZaGhoenq6fcqLtwzNBbxg3aXGxkZhl0lkQ3pX2rMLnr3ixRdf3LZtm16vv+mmm26//XbrfaZPn37zzTe3trYePnz4n//85/r161966SUp2ORkK999911ZWdntt9/enSjGs3bs95k0aVJqaurgwYOtQyqLFy++6aabpBWjAgMDR40atWDBAiGE9ReM7LbbbktJSRk0aJC/v39MTMzMmTOFEHq9vs1y62yjb775pqWl5eabb77//vvVanVAQEBoaOiUKVOkRabkr1IpYHT27FlpJrYcRZLKpU2TyXT27FlhFV369ttvzWZzXFzcfffdN2jQIF9f30GDBs2ZM+e6664zmUzffPONM+dBdujQoTfffLOxsTE+Pt7m+YAA4GLSh2GbH1YA0NtmzpxpEwSRNk0m05YtW2bNmlVUVLRly5b09HQhRFZWVk5OjsFgOH36dHJyclZWlnTId999N3ny5PPnz5eVlSUkJMhpDlJVly5dKisre/rpp//0pz9J5atWrYqOji4pKTlx4kR0dPSqVaus9583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AUCAyMzMlAYgIQUAQcH5R0YBAAgryjo1rOoNAKprbm5WOwQAgIf4qhYAENbILgEAAACXiewSACB8ybKs/HYqM+MAAAAAjzEzDgAQvuw/3+63sUvd3d29vb1JSUmJiYn+aREAAADwNb6qBQCEL/8v6d3X12cwGMxms99aBAAAAHyNsUsAgPDl/0WXkpKSYmNj4+Li/NYiAHhAkiQhhDJ3WHX+DEaSpAC5agAILoxdAgCEL2Xskj+zSwkJCampqWSXAKglKyvLncP8k2Hp7e115zB/pnvcbyspKcmnkQBAcCG7BAAIX8rYJZb0BhA+mpqaHD/29fXdd999mZmZWVlZe/bsmThxouPeo0eP5ubmxsTElJeX2wvXrl07adIkjUYzf/78d955RymUJEmSpB07djgd39vbe+edd2q12ry8vFOnTimHCSFaW1tnzJih0WgyMzPvueced9JM586dmzlzplarXbVqldFodGz38OHDQojDhw9LknTFFVfYTykoKCgoKHDzziinK+HZDXVdOp3OYDBIX3KzCQAIYTxPAwDCl/9nxgFAQNm4caPZbK6urj5z5oxGo+no6HDcW1FRcerUqX379hUXF9sLt2zZcurUqa6urg0bNixatEgpVIb86PX6mpoax+PXr1+v0+mampoqKyuPHDliP7KkpGTbtm1ms7m+vn7u3LklJSXDhrpnz54DBw7U19cnJib+8pe/VArff//9mTNnFhUVCSEKCwvz8/P//Oc/20+x/zCoO4qKigYePNR1KTdK/pKbTQBACGNeMQAgfLW3t3d0dKSkpGRkZKgdS9C44YYbPvvss+zsbLUDwVecPHmyt7d3+vTpageCAGIwGGRZPnXqlBAiJydHr9c77s3Ozm5sbMzKyjpx4oROpxt4uiRJFoslKipKOKxGVFdX98ADD1RVVZlMpry8vMrKSvvbhOOKRfbtjIyMzz//XKm/tbU1IyNDKR81apRjJis9Pb2lpcXFtUiS1NDQkJOTI4To6OiYPHmy/fglS5bcdtttS5YsOXDgwMGDB//0pz95dLf+1ZDT+9Gg1zXokfC/mTNnlpaWzp07V+1AALCqNwAgjDF2yQM9PT06ne7RRx9VOxB8xd13311fX8/fCxx99tlnu3fvVrYbGxuVDaecSH9/v4salNSSo6VLly5btmzXrl06na6trW3YVZwcZ405tjt9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bzcYDPYKe3p6li1bNn78+Nzc3MLCwosXLzoF09nZuXLlysTExPvuu08pv+qqq5SwJQfuXLKLtoQQ586dmzlzplarXbVqldFodIxhqFaGuhtCiIMHD+bl5SUkJHzjG9946aWXJk6ceJnBAwErDDuTMWPGOJ3ocRjD3ighxKpVqzIzMydNmnTo0CF7IZ1JUJIBAIBvnD9/vra29uLFi2oH4k2zZs269tpr1Y4CzoqKimJiYtSOYkiJiYkXLlzwf7sDn3WFEFu3bn3xxRfT0tKKi4v3798fHR1t37tixYp3333XbDb39PQcOnTo5z//uX3X6dOnS0pKenp6rFbr888/39DQ4Fjnf/zHfzQ1Ncmy/OSTT/70pz+177rnnns++eQTZfvtt99es2aNUzDXX3/9nj17uru7h418WC7aEkI88sgj58+f7+zs/NWvfuV4XS5aHOpu7N69++abb66pqenr62tra/vv//5vnU53mcF7y8KFC1944QW1WsewKisrr776ao9Pr6iomDNnjhfjcV9YdSYuTvQgDNc3SpZlo9Fos9k++OCDhIQEbwV/mZTvBc1msyqtBy+ySwAA+Mq5c+dqa2sNBoPagXgT2aXARHZpUIO+ENo3bDab0zE6nc7xW9j09HTHcz/44INp06YtXLiwurraqc7m5mZlu7OzMzU11b4rMzPT/n7S19eXmZnpdKLRaHQz8mG5aEsIYX+D7erqcrquoVoc6m6MGzfO9d8m2SUMJcSyS/aNEOtMXJzoQRiub9T7778/derU+Pj4GTNmuLjJfkZ2yTPMjAMAwFdsNpsQIiKC/22BQDRwnsXkyZMdH5RbWloc9yYkJGg0moaGhkuXLg1VlSzLMTEx9nJZlpV+QAyxKnBCQoI7oZpMpmGPGbYtRXR0tJudkou7YW/IHe4EDwS1EOtMXLjMMAbeqBUrVtx7773t7e2HDx8etnU6kwDH8y4AAL5itVoF2SUgeCxatOiZZ545f/682Wz+4osvVq9ebd915syZNWvWvPjii8eOHdu/f399fb3jia+//npzc7MQYs+ePbfeequ9fPHixe+9956y/corr9x8881uRhIdHa0sTXLhwoUdO3bMnj172FNct7Vz586GhobOzs69e/cuXbrUnRiGuhtbt2699957u7q6bDbb2bNn9+7du2LFissMHggxQd2ZuOBxGENpaWmZ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ukYoCMJLL700YcIEvV5/zTXXvPzyy2PGjAnd8KOPPvryl7+cmZmp0+lCTy+AVEF2CQAA9AudTldSUiKKosPhaGpqUjscAOjG2LFju9Zayi2BQODpp5/+9re//dRTT+3atWvRokVKhzVr1jz++ONer/fcuXPTp09fs2aN3F5VVXX27Nnvfve7TqczGAxu3Lixqqrq4sWLypgVFRXHjh1rbW294YYbHnjgAWXADRs2bNiw4eLFi/X19WvWrNmwYUNYMLfddtvXv/71mpqa7du3y+1nz56VV0kh4jnkJUuWzJ071+fz2e32CRMmNDQ0dHtcsieffLKuru6555674447up6urhE+88wzzz333PPPP2+3248fP26325ubm0M3fPTRR59//nm/319WVhZ6egGkCpFidQAA0H/a29vl2qXCwkKz2ax2OGlr3759e/fufe2119QOBOgbTz/99KFDh/bt2zfwuxbF8EskpUUUxWAwKJfnKH0KCgpaWlqUzlarNTRH89577/3whz8cOnTopk2brr322tAx6+vrbTabIAitra0jRoyw2+3yquLi4kuXLmm1WkEQPB7P8OHD6+rqQjd0uVx6vT6eyLt14MCB2267zefzFRYWWq3WvXv3Tpo0qdvj+utf//qjH/2osrLyuuuue++997qerq4Rjhw58sSJE8XFxRHDEEXR6/XKh5zYgSTss88+mzRpUmNj48DsDkhj1C4BAIB+lJubK18+NTU1tbW1qR0OAPRK1zvOrrrqqtCKodDUkiAIer3eYDBUVVV1dnZGG0qSJJ1Op7RLkqTMghQxyRIxtdRVPE/tnDdvntfrdTqdf//73++4445Zs2bFc1x33nnnqlWrmpubDx48GHHYiBFGm9pJpqSWAKQosksAAKB/5eXlWSwWQRAaGhpC5xYBgDQwf/78J5988tKlS16v99NPP12xYoWy6vz58/fee++LL75YUVGxb9++c+fOhW74xhtvyKWdzzzzzHe+8x2lfcGCBW+//ba8/Mc//vGWW26JMxKtVivPWFRbW7tjx46bbrqp202sVuu7776r0WiGDRt21VVXuVyueI6roaFh7NixDofjmWeeiTO2LVu2rFq1qrW1NRgMXrhw4fnnn7/zzjvj3BZASiC7BAAA+t3/b+/O46Oo8/yPV/WZTnc6SeciF3ggojgqp6DgunI46HiAKKis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6kiXAAAAAAAAkDrSJQAAAAAAAKSOdAkAAAAAAACpI10CAAAAAABA6kiXAAAAAAAAkDrSJQAAAAAAAKSOdAkAAAAAAACpI10CAAAAAABA6kiXAAAAAAAAkDrSJQAAAAAAAKSOdAkAAAAAAACpI10CAAAAAGBw+e677/71X/915MiRNpvNarWeeOKJN95443fffRfTTUxCvM5Wq3XUqFF33HFHQ0ND5wJ8Pt/q1asnTZqUl5dnMBgKCgrOOuuse++99+uvv07tjqKL6VHZSVainnjVVVd1vvpVV13VeeRkuN3uYcOGpXBifyPKsqx1DQAAAAAAoI+sWrXq3nvvDYfDMe16vX7FihV333232pJM5KGmCvE6l5SU7Nq169hjj1Vbmpqazj333D179iQesEeUqyvnJl928pUoY4qiaDab6+vr8/Ly1ENtbW2lpaWBQEDp36P6b7nllueeey4YDPb0xP6GuUsAAAAAAAwWjz322NKlSyORyKJFiz755BOXy+XxeD799NPf/e53kUhkyZIljz/+uNpZ/kcJGjuf0tTU9Oabbx533HGNjY333XdfdJ+77757z5495eXl69evr66uDgQCbW1tu3btWrFixamnnnr095h82T2t5Nxzz/X7/a+88kp048svv+z3+6dOndrTOnft2vWnP/0pOs7LXMxdAgAAAABgUKiurh4+fHgwGHz++ed/+9vfxhxdv379woULTSbTjz/+WFFR0fn06PlBSR7929/+Nnny5OLi4sbGRrWxoKCgtbV1165dEydOPMo7Sqa2BIeSr0QZ5JVXXrnyyivHjBnzxRdfqIfGjBnz5Zdfvvrqq1dccUW8C3UWDAZHjx4tSdI333xjsViSP7F/Yu4SAAAAAACDwjPPPBMMBufMmdM5WhIE4be//e2ll14aDAafeeaZdF3xtNNOEwShvb09utHj8QiCcMIJJ3R7epcbEqVxl6LkK1HMnj3b4XD8/e9//+qrr5SWL7/88ssvv3Q4HLNmzerRpVesWLF3796nn346KyurRyf2T6RLAAAAAAAMClu2bBEE4dprr43XQTn0wQcfpOuKf//73wVBiJkJNXr0aEEQFi1adODAgXRdKDU9rcRsNiu7eq9bt05pUV7Mnz/fbDYnf929e/euXLly/vz5Kayn659YGQcAAAAAwKDgcDicTmdLS4vD4eiyw5EjR4qKigoKCo4cOdL5aI9Wxh05cmTHjh133HHHoUOHli1bdv/996s9d+zYMWPGDL/fLwjCcccdd/rpp59yyilnn332Oeeco9fru71i8o3dHkqhkq+//vr000/Pz8+vq6sTBKG0tLStre3rr78+9dRTEz8fVSQSmTx58g8//PD9998XFRUlrjBTkC4BAAAAADAoGI3GUCgkSZLBYOiygyRJJpPJYDBIktT5aDLpUmdXXHHFiy++aDQaoxu///77FStWvPnmmy6XS22sqKh49NFHL7300iRvJ8naEpedZCXRg4wfP/7zzz9/5ZVXZFm+6qqrxo8f/+mnn3Z7IdUTTzxx6623rl27dtGiRclUmBFIlwAAAAAAGBT6YO5SjIqKirfeektZgNZZOBzevXv3nj17Pv300/fee2///v2iKFZVVV188cXJ3lIStSWT3XRbSfQgzzzzzI033jh16lRZlrdt2/bMM89cf/31SV7o8OHDo0aNGj169Mcff6w+MdIlAAAAAACQGSZMmPDZZ59t3rx5+vTpXXbYvHnzeeedN2HChE8++aTz0eRXxkmS9NNPPy1fvvyll14qKir69ttvS0pKEtcWiUTuvffeBx98MOYb2ZJ0lOlSt5VED9Le3l5WVubz+QRBsFgs9fX1drs9yQtdcMEFH3zwwVdffXXSSSelXGE/xK7eAAAAAAAMCtOmTRMEYc2aNfE6KIeUbkfDaDSOGDHihRdeOO+885qbm++9995uT9HpdEuWLBEEYc+ePUd59aPUbSW5ublz5syRZVmW5csuu0yJlpL03nvvBYPBk08+WYyiHErjd+H1PdIlAAAAAAAGhRtuuMFoNP75z3/+r//6r85HX3rppddff91kMt1www1puZwoio8//rjBYHjxxRf37dvXbX/li9tsNltarn40uq1E3TJJfTHIkS4BAAAAADAoDB06dOXKlYIgXHPNNdddd91nn33m9Xp9Pt/nn39+/fXXL1iwQBCEBx98sLKyMl1XHDFixMKFC0OhUPT0pVGjRt17773vv/9+dXV1IBDw+/0//fTTn/70J2WTo5kzZ6o9u5zOk8Y5PslXEuPss89W5i5NmTKlR1eUuxJ96GhuR0PsuwQAAAAAwCCyfPnyZcuWRSKRmHadTvfAAw/8/ve/j3di8vsuRaurqxs+fLjf7//000/HjRsnxP92OUEQTjrppG3btg0ZMiTBmMk3Jnko5UqSv1BiA2DfJdIlAAAAAAAGlz179jz55JNbt26tra2VZbmiouLcc8+95ZZbRo0aleCs1NIlQRCWLl26atWqadOmbdmyRRCEvXv3vvnmm9u3b9+zZ09TU5MsywUFBaNGjbrkkkt+97vfZWVlJR4zjenSUVaS/IUSI10CAAAAAADAoMa+SwAAAAAAAEgd6RIAAAAAAABSR7oEAAAAAACA1JEuAQAAAAAAIHWkSwAAAAAAAEgd6RIAAAAAAABSR7oEAAAAAACA1JEuAQAAAAAAIHUGrQsAAAAAAAB9QRRFWZa1ruL/iKLYubFfVYgkkS4BAAAAAIBe12W2pW2W1N/itszFyjgAAAAAAACkjnQJAAAAAIDBRRTFV155Zfz48dnZ2Tabbfr06bt37960adNZZ51ltVoLCwuvvPLKlpYWtfOzzz57wgknmEym448//oUXXoge6tlnnx0+fLjJZBo+fPiaNWtiLnHGGWfYbDa9Xq8sghN/0W2FN9xwwxVXXBHdMm/evBtuuEHDkpCIDAAAAAAABgE1BBAEYeTIkR988IHL5aqvr7/mmmscDsfJJ5+8ZcsWteXKK69UOw8bNuyjjz5yuVwffvjh0KFD33//feXQxo0bKyoqtm7d2tHRsXXr1oqKirfeeks9a8SIEVu3bvV4PDFX71xPZ8FgcPLkyU899ZTy9sknn5wyZUowGOztkpAaVhgCAAAAADAoqNsMiaL4+eefjx07Vmlvbm4uLi7+4osvxowZo7acdNJJR44cUTq/9dZbF154oXLozTfffPTRRz/66CNBEM4666zFixdfcsklyqGqqqqHH374r3/9q3LW9u3bp0yZ0vnq0S2di1T7NDU1nXXWWf/zP/8jCMK8efP+9re/FRcXqyf2UklIDc8RAAAAAIBBITpdCofDOp0u+lDnFrWz0+nMy8tT2p1O5/Dhw5V1cw6H48cff8zPz1cPHX/88a2trcpZPp8vKyur84AJWmJ88cUX8+bNMxgML7/8spp89WpJSA37LgEAAAAAMOhEB0nxWo5SdI6Tmk8++cRutwcCgcOHD/eTktAl0iUAAAAAAJDIjh071Ncff/zxKaecorw+6aSTtm/fHn3o5JNPjjeIwWAIh8PJX/Trr7/+wx/+8Nprr23YsOH666//8ccfNS8J8ZAuAQAAAACARG655ZaPP/7Y7XZ/9NFHt9122z333KO033XXXbfeeutHH32kHlqyZEm8QYYOHfrBBx9EIpFkrujxeObOnfvoo48ef/zx48eP/3//7//NmTPH7/drWBISYIUhAAAAAACDQvRWSt1ugRTd+emnn169evXPP/9cWVl53333LVy4UO32zDPPKIeGDRu2ZMmSa6+9Nt6AGzduvOuuuw4fPhwOh9WROxepHLrmmmv0ev3zzz+vtl955ZUWi2XdunW9WhJSQ7oEAAAAAADi6odbX/fDkgY5VsYBAAAAAAAgdaRLAAAAAAAASB1zyQAAAAAAAJA65i4BAAAAAAAgdaRLAAAAAAAASB3pEgAAAAAAAFJHugQAAAAAAIDUkS4BAAAAADDo/Pzzz9XV1cprt9t96NAh5XUgEGhpaVG7tbe3+3y+6BOjO0Sf2NLScvjw4ZirqJ0DgUBNTU1NTY3L5Ur7vUBzBq0LAAAAAAAAfeqtt9765ptvWltbjz322Dlz5tx+++2nnXaa2Wy+6KKL/uM//qO5ufnNN98UBMHlcp177rn33HPPnDlzlBP379+vdqivr1dPnDRp0nPPPXfCCSfk5ubefPPNnTvv2rXrscceGzt27LnnnnvmmWdqdufoHaIsy1rXAAAAAAAA+lpHR8eNN944adKkUaNG/fM///Ps2bM3btwoCMIll1yipEvLli1zOBwVFRVquqRQOjz55JPqiSNHjrzyyitPOeWUadOmffDBB507f/TRRxs2bLj44otnzJixZcuWN954w263L1u2zG639+Uto5cwdwkAAAAAgEFHluW777578eLFVVVVRUVFnTt89NFHJ554ojIl5eDBg62traNGjbJYLGqH5uZm9cQpU6Y8/fTTo0ePPnDgQJedR4wYcdlllx06dOj2228vKCg444wzZs2aRbQ0YJAuAQAAAAAw6Nx1112XXXbZ6aef/umnn9bV1Z1yyimiKEZ32LZtW2tr6759+/R6fW5ubmNj4/HHHx8dGJWXl6snzpw5c8SIEX6/f+zYsXV1dYcOHYrpXFZWVlZWJgjCxRdf/Mgjj3z44YeLFi3693//91GjRvXZLaP3kC4BAAAAADC4PPfcc9u2bQsEArt37/6Xf/mXG2644d13373gggtaW1uXLVv2zTffPPjgg3/84x8FQfjv//7vrKys6dOnKydGd7juuuvUEw8cOLB27dq6urrbbrtt8uTJkydPjun8q1/9aseOHXV1dRdffPFLL73U1NSk0+mys7O1fApIH/ZdAgAAAABgUJNl2e/3R081SuFEl8tltVp1urjfTd/e3p6dnW00GgVBcDqdOTk5BgNTXgaI/w9jSRXEjZRl2QAAAABJRU5ErkJggg==
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiID8+CjxvdHJzX2NvbmZpZyB2ZXJzaW9uPSIxLjAiIGluaXQ9IkZyYW1ld29yayI+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJGcm9udGVuZDo6TW9kdWxlIyMjQWRtaW5JbXBvcnRFeHBvcnQiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIFRyYW5zbGF0YWJsZT0iMSI+RnJvbnRlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGFnZW50IGludGVyZmFjZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5JbXBvcnRFeHBvcnQ8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWRtaW46Ok1vZHVsZVJlZ2lzdHJhdGlvbjwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxGcm9udGVuZE1vZHVsZVJlZz4KICAgICAgICAgICAgICAgIDxHcm91cD5hZG1pbjwvR3JvdXA+CiAgICAgICAgICAgICAgICA8RGVzY3JpcHRpb24gVHJhbnNsYXRhYmxlPSIxIj5JbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uPC9EZXNjcmlwdGlvbj4KICAgICAgICAgICAgICAgIDxUaXRsZT5JbXBvcnQvRXhwb3J0PC9UaXRsZT4KICAgICAgICAgICAgICAgIDxOYXZCYXJOYW1lPkFkbWluPC9OYXZCYXJOYW1lPgogICAgICAgICAgICAgICAgPE5hdkJhck1vZHVsZT4KICAgICAgICAgICAgICAgICAgICA8TW9kdWxlPktlcm5lbDo6T3V0cHV0OjpIVE1MOjpOYXZCYXJNb2R1bGVBZG1pbjwvTW9kdWxlPgogICAgICAgICAgICAgICAgICAgIDxOYW1lIFRyYW5zbGF0YWJsZT0iMSI+SW1wb3J0L0V4cG9ydDwvTmFtZT4KICAgICAgICAgICAgICAgICAgICA8QmxvY2s+U3lzdGVtPC9CbG9jaz4KICAgICAgICAgICAgICAgICAgICA8UHJpbz43MTA8L1ByaW8+CiAgICAgICAgICAgICAgICA8L05hdkJhck1vZHVsZT4KICAgICAgICAgICAgICAgIDxMb2FkZXI+CiAgICAgICAgICAgICAgICAgICAgPENTUz5JVFNNLkltcG9ydEV4cG9ydC5jc3M8L0NTUz4KICAgICAgICAgICAgICAgIDwvTG9hZGVyPgogICAgICAgICAgICA8L0Zyb250ZW5kTW9kdWxlUmVnPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IkltcG9ydEV4cG9ydDo6Rm9ybWF0QmFja2VuZFJlZ2lzdHJhdGlvbiMjI0NTViIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gVHJhbnNsYXRhYmxlPSIxIj5Gb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+SW1wb3J0RXhwb3J0PC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Rm9ybWF0QmFja2VuZDo6TW9kdWxlUmVnaXN0cmF0aW9uPC9TdWJHcm91cD4KICAgICAgICA8U2V0dGluZz4KICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik1vZHVsZSI+S2VybmVsOjpTeXN0ZW06OkltcG9ydEV4cG9ydDo6Rm9ybWF0QmFja2VuZDo6Q1NWPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJOYW1lIj5DU1Y8L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+Cjwvb3Ryc19jb25maWc+Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmNzX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdOb3bDoSDFoWFibG9uYSB6b2JyYXplbsOtJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdabmFrb3bDoSBzYWRhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ0R2b2p0ZcSNa2EgKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ1Nsb3VwZWMnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uIFNlcGFyYXRvcid9ID0gJ09kZMSbbG92YcSNIFNsb3VwY8WvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICdUZcSNa2EgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdTdMWZZWRuw61rICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bMOhdG9yIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXhwb3J0ZWQgcmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICcnOwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdJbXBvcnQvRXhwb3J0IFNwcsOhdmEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdaYWjDoWppdCBJbXBvcnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAnWmFow6FqaXQgRXhwb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAnfSA9ICdLcm9rJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnRWRpdGFjZSBvYmVjbsO9Y2ggaW5mb3JtYWPDrSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICdFZGl0YWNlIGluZm9ybWFjw60gbyBvYmpla3R1JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnRWRpdGFjZSBmb3Jtw6F0dSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gPSAnRWRpdGFjZSBtYXBvdsOhbsOtJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJ0VkaXRhY2UgdnlobGVkw6F2w6Fuw60nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdPbWV6aXQgRXhwb3J0IHZ5aGxlZMOhdsOhbsOtbSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICdJbmZvcm1hY2UgbyBJbXBvcnR1JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnWmRyb2pvdsO9IFNvdWJvcic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPayc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICdJbXBvcnQvRXhwb3J0JzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmRhX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdUaWxmw7hqIE1hcHBpbmctVGVtcGxhdGUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ1RlZ25zw6Z0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ0tvbG9uICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdLb2xvbm5lJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ1B1bmt0dW0gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdTZW1pa29sb24gKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYXRvciAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0V4cG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTa2lwcGVkJ30gPSAnJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAnSW1wb3J0L0VrcG9ydCBzdHlyaW5nJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAnU3RhcnQgaW1wb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ1N0YXJ0IGVrcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwJ30gPSAnVHJpbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJ1JldCBmw6ZsbGVzIGluZm9ybWF0aW9uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJ1JldCBvYmpla3QgaW5mb3JtYXRpb24nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICdSZXQgZm9ybWF0IGluZm9ybWF0aW9uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICdSZXQgbWFwcGluZyBpbmZvcm1hdGlvbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICdSZXQgc8O4Z2VpbmZvcm1hdGlvbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ0JlZ3LDpm5zIGVrcG9ydCBwci4gc8O4Z25pbmcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW1wb3J0IGluZm9ybWF0aW9uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnS2lsZGUgZmlsJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJyc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICdJbXBvcnQvRWtwb3J0JzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmRlX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdNYXBwaW5nLVRlbXBsYXRlIGhpbnp1ZsO8Z2VuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdaZWljaGVuc2F0eic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEb3BwZWxwdW5rdCAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnU3BhbHRlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdTcGFsdGVudHJlbm5lcic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUHVua3QgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdTZW1pY29sb24gKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYXRvciAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnTWl0IFNwYWx0ZW7DvGJlcnNjaHJpZnRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgc3VtbWFyeSBmb3InfSA9ICdJbXBvcnQtQmVyaWNodCBmw7xyJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydGVkIHJlY29yZHMnfSA9ICdJbXBvcnRpZXJ0ZSBEYXRlbnPDpHR6ZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFeHBvcnRlZCByZWNvcmRzJ30gPSAnRXhwb3J0aWVydGUgRGF0ZW5zw6R0emUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ0RhdGVuc8OkdHplJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICfDnGJlcnNwcnVuZ2VuJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAnSW1wb3J0L0V4cG9ydC1WZXJ3YWx0dW5nJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdFcnN0ZWxsZW4gZWluZXIgVm9ybGFnZSB6dW0gSW1wb3J0aWVyZW4gdW5kIEV4cG9ydGllcmVuIHZvbiBPYmpla3QtSW5mb3JtYXRpb25lbi4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAnSW1wb3J0IHN0YXJ0ZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAnRXhwb3J0IHN0YXJ0ZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCd9ID0gJ1NjaHJpdHQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICdBbGxnZW1laW5lIEluZm9ybWF0aW9uZW4gYmVhcmJlaXRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJ05hbWUgd2lyZCBiZW7DtnRpZ3QhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICdPYmpla3QgaXN0IGVyZm9yZGVybGljaCEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJ0Zvcm1hdCBpc3QgZXJmb3JkZXJsaWNoISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJ09iamVrdC1JbmZvcm1hdGlvbmVuIGJlYXJiZWl0ZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICdGb3JtYXQtSW5mb3JtYXRpb25lbiBiZWFyYmVpdGVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJ3dpcmQgYmVuw7Z0aWd0ISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICdNYXBwaW5nLUluZm9ybWF0aW9uZW4gYmVhcmJlaXRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnS2VpbmUgTWFwcGluZy1FbGVtZW50ZSBnZWZ1bmRlbi4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ01hcHBpbmctRWxlbWVudCBoaW56dWbDvGdlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJ1N1Y2gtSW5mb3JtYXRpb25lbiBiZWFyYmVpdGVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnRXhwb3J0IHBlciBTdWNoZSBlaW5zY2hyw6Rua2VuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0ltcG9ydC1JbmZvcm1hdGlvbmVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnUXVlbGwtRGF0ZWknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJ0VyZm9sZ3JlaWNoJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJ05pY2h0IGVyZm9sZ3JlaWNoJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ0RvcHBlbHRlIE5hbWVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnWnVsZXR6dCB2ZXJhcmJlaXRldGUgWmVpbGUgZGVyIEltcG9ydC1EYXRlaSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJ09rJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnRm9ybWF0LUJhY2tlbmQgTW9kdWwtUmVnaXN0cmF0aW9uIGRlcyBJbXBvcnQvRXhwb3J0IE1vZHVscy4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0ltcG90aWVyZW4gdW5kIEV4cG9ydGllcmVuIHZvbiBPYmpla3QtSW5mb3JtYXRpb25lbi4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ0ltcG9ydC9FeHBvcnQnOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmVzX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdBw7FhZGlyIHBsYW50aWxsYSBkZSBtYXBlbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAnSnVlZ28gZGUgY2FyYWN0ZXJlcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEb3MgcHVudG9zICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdDb2x1bW5hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdTZXBhcmFkb3IgZGUgQ29sdW1uYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUHVudG8gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdQdW50byB5IENvbWEgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYWRvciAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnSW5jbHVpciBDYWJlY2VyYSBkZSBsYSBDb2x1bW5hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ0ltcG9ydGFyIHJlc3VtZW4gcGFyYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnUmVnaXN0cm9zIGltcG9ydGFkb3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXhwb3J0ZWQgcmVjb3Jkcyd9ID0gJ1JlZ2lzdHJvcyBleHBvcnRhZG9zJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlY29yZHMnfSA9ICdSZWdpc3Ryb3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2tpcHBlZCd9ID0gJ1NhbHRhZG8nOwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdHZXN0acOzbiBkZSBJbXBvcnRhY2nDs24vRXhwb3J0YWNpw7NuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdDcmVhciB1bmEgcGxhbnRpbGxhIHBhcmEgaW1wb3J0YXIgeSBleHBvcnRhciBpbmZvcm1hY2lvbiBkZWwgb2JqZXRvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdJbmljaWFyIEltcG9ydGFjacOzbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdJbmljaWFyIEV4cG9ydGFjacOzbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwJ30gPSAnUGFzbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJ0VkaXRhciBpbmZvcm1hY2nDs24gY29tw7puJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnRWwgbm9tYnJlIGVzIHJlcXVlcmlkbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnwqFEZWJlIGVzcGVjaWZpY2FyIE9iamV0byEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJ8KhRGViZSBlc3BlY2lmaWNhciBGb3JtYXRvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJ0VkaXRhciBpbmZvcm1hY2nDs24gZGUgb2JqZXRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnRWRpdGFyIGluZm9ybWFjacOzbiBkZWwgZm9ybWF0byc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpcyByZXF1aXJlZCEnfSA9ICfCoWVzIHJlcXVlcmlkbyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gPSAnRWRpdGFyIGluZm9ybWFjacOzbiBkZSBtYXBlbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnTm8gc2UgZW5jb250cmFyb24gZWxlbWVudG9zIGRlIG1hcGVvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnQcOxYWRpciBNYXBlbyBkZSBFbGVtZW50b3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICdFZGl0YXIgaW5mb3JtYWNpw7NuIGRlIGLDunNxdWVkYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ1Jlc3RyaW5naXIgZXhwb3J0YWNpw7NuIHBvciBiw7pzcXVlZGEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW1wb3J0YXIgaW5mb3JtYWNpw7NuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnQXJjaGl2byBvcmlnZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJ8OJeGl0byc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICdGcmFjYXNhZG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnTm9tYnJlcyBkdXBsaWNhZG9zJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnw5psdGltYSBuw7ptZXJvIGRlIGzDrW5lYSBwcm9jZXNhZGEgZGVsIGFyY2hpdm8gaW1wb3J0YXInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPayc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ1JlZ2lzdHJvIGRlIG3Ds2R1bG8gZGUgZm9ybWF0byBiYWNrZW5kIHBhcmEgZWwgbcOzZHVsbyBpbXBvcnQvZXhwb3J0Lic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnSW1wb3J0YXIgeSBleHBvcnRhciBpbmZvcm1hY2nDs24gZGUgb2JqZXRvcy4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ0ltcG9ydGFyL0V4cG9ydGFyJzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmZhX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICfYp9i22KfZgdmHINqp2LHYr9mGINuM2qkg2YLYp9mE2Kgg2Ybar9in2LTYqic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAn2qnYr9io2YbYr9uMINin2LfZhNin2LnYp9iqJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ9iv2YjZhtmC2LfZhyAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAn2LPYqtmI2YYnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uIFNlcGFyYXRvcid9ID0gJ9is2K/Yp9qp2YbZhtiv2Ycg2LPYqtmI2YbigIzZh9inJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICfZhtmC2LfZhyAoLiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ9iz2YXbjCDaqdin2YTZhiAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGFidWxhdG9yIChUQUIpJ30gPSAn2KzYr9mI2YQg2LPYp9iyIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXhwb3J0ZWQgcmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICcnOwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICfZhdiv24zYsduM2Kog2YjYsdmI2K8v2LXYr9mI2LEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ9iz2KfYrtiqINmC2KfZhNio24wg2KjYsdin24wg2YjYsdmI2K8g2Ygg2LXYr9mI2LEg2KfYt9mE2KfYudin2Kog2KLYqNis2qnYqic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICfYtNix2YjYuSDYudmF2YTbjNin2Kog2YjYsdmI2K8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAn2LTYsdmI2Lkg2LnZhdmE24zYp9iqINi12K/ZiNixJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAnfSA9ICfar9in2YUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICfZiNuM2LHYp9uM2LQg2KfYt9mE2KfYudin2Kog2LnZhdmI2YXbjCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAn2KLYqNis2qnYqiDZhdmI2LHYryDZhtuM2KfYsiDYp9iz2KohJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICfZgtin2YTYqOKAjNio2YbYr9uMINmF2YjYsdivINmG24zYp9iyINin2LPYqiEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICfZiNuM2LHYp9uM2LQg2KfYt9mE2KfYudin2Kog2KLYqNis2qnYqtuMJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAn2YjbjNix2KfbjNi0INin2LfZhNin2LnYp9iqINmC2KfZhNio4oCM2KjZhtiv24wnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ9mI24zYsdin24zYtCDYp9i32YTYp9i52KfYqiDZhtqv2KfYtNiqJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICfZh9uM2oYg2LnZhti12LEg2Ybar9in2LTYqtuMINuM2KfZgdiqINmG2LTYry4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ9in2YHYstmI2K/ZhiDYudmG2LXYsSDZhtqv2KfYtNiqJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAn2YjbjNix2KfbjNi0INin2LfZhNin2LnYp9iqINis2LPYqtis2YgnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICfZhdit2K/ZiNiv2LPYp9iy24wg2LnZhdmE24zYp9iqINi12K/ZiNixINio2Ycg2KfYstin24wg2KzYs9iq2KzZiCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICfZiNix2YjYryDYp9i32YTYp9i52KfYqic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ9mB2KfbjNmEINmF2YbYqNi5JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJ9iq2KfbjNuM2K8nOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICfZgtin2YTYqOKAjNio2YbYr9uMINir2KjYqiDZhdin2pjZiNmEINio2LHYp9uMINmF2KfamNmI2YQg2YjYsdmI2K8v2LXYr9mI2LEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ9mI2LHZiNivINmIINi12K/ZiNixINin2LfZhNin2LnYp9iqINii2KjYrNqp2KonOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ9mI2LHZiNivL9i12K/ZiNixJzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmZyX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdBam91dGVyIHVuIHRlbXBsYXRlIGRlIG1hcHBhZ2UnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ0pldSBkZSBjYXJhY3TDqHJlcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEZXV4IHBvaW50cyAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnQ29sb25uZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnU8OpcGFyYXRldXIgZGUgY29sb25uZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUG9pbnQgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdQb2ludCB2aXJndWxlICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bGF0aW9uIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICdJbmNsdXJlIGxlcyBlbi10w6p0ZXMgZGUgY29sb25uZXMnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnUsOpc3Vtw6kgZGUgbFwnaW1wb3J0IHBvdXInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0ZWQgcmVjb3Jkcyd9ID0gJ0VucmVnaXN0cmVtZW50cyBpbXBvcnTDqXMnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXhwb3J0ZWQgcmVjb3Jkcyd9ID0gJ0VucmVnaXN0cmVtZW50cyBleHBvcnTDqXMnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ0VucmVnaXN0cmVtZW50cyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTa2lwcGVkJ30gPSAnUGFzc8OpKHMpJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAnR2VzdGlvbiBkZSBsXCdJbXBvcnQvRXhwb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdDcsOpZXIgdW4gbW9kw6hsZSBwb3VyIGltcG9ydGVyIGV0IGV4cG9ydGVyIGxlcyBpbmZvcm1hdGlvbnMgZFwnb2JqZXQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAnRMOpbWFycmVyIEltcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdEw6ltYXJyZXIgRXhwb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAnfSA9ICdFdGFwZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJ0VkaXRlciBsZXMgaW5mb3JtYXRpb25zIGNvbW11bmVzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnVW4gTm9tIGVzdCByZXF1aXMhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICdVbiBPYmpldCBlc3QgcmVxdWlzISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnVW4gRm9ybWF0IGVzdCByZXF1aXMhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnRWRpdGVyIGxlcyBpbmZvcm1hdGlvbnMgZGUgbFwnb2JqZXQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICdFZGl0ZXIgbGVzIGluZm9ybWF0aW9ucyBkZSBmb3JtYXQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAnZXN0IHJlcXVpcyAhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ0VkaXRlciBsZXMgaW5mb3JtYXRpb25zIGRlIG1hcHBhZ2UnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnRWRpdGVyIGxlcyBpbmZvcm1hdGlvbnMgZGUgcmVjaGVyY2hlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnUmVzdHJlaW5kcmUgbFwnZXhwb3J0IHBhciByZWNoZXJjaGUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW5mb3JtYXRpb25zIGRcJ2ltcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ0ZpY2hpZXIgU291cmNlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICdSw6l1c3NpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJ0VjaG91w6knOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnTm9tcyBlbiBkb3VibGUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICdOdW3DqXJvIGRlIGxhIGRlcm5pw6hyZSBsaWduZSB0cmFpdMOpZSBkYW5zIGxlIGZpY2hpZXIgZFwnaW1wb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnT2snOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0ltcG9ydGVyIGV0IGV4cG9ydGVyIGRlcyBpbmZvcm1hdGlvbnMgZFwnb2JqZXQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ0ltcG9ydGVyL0V4cG9ydGVyJzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6Oml0X0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdBZ2dpdW5naSBtYXBwYXR1cmEgbW9kZWxsbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAnQ2hhcnNldCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEdWUgcHVudGkgKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ0NvbG9ubmEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uIFNlcGFyYXRvcid9ID0gJ1NlcGFyYXRvcmUgZGkgY29sb25uYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUHVudG8gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdQdW50byBlIHZpcmdvbGEgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYXRvcmUgKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW5jbHVkZSBDb2x1bW4gSGVhZGVycyd9ID0gJ0luY2x1ZGkgbGUgQ29sb25uZSBkaSBJbnRlc3RhemlvbmUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnSW1wb3J0YSBpbCBzb21tYXJpbyBwZXInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0ZWQgcmVjb3Jkcyd9ID0gJ0ltcG9ydGEgdm9jaSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFeHBvcnRlZCByZWNvcmRzJ30gPSAnRXNwb3J0YSB2b2NpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlY29yZHMnfSA9ICdWb2NpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICdTYWx0YXRvJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAnR2VzdGlvbmUgSW1wb3J0YXppb25lL0VzcG9ydGF6aW9uZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnQ3JlYSB1biB0ZW1wbGF0ZSBwZXIgaW1wb3J0YXJlIGVkIGVzcG9ydGFyZSBsZSBpbmZvcm1hemlvbmkgZGVnbGkgb2dnZXR0aS4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAnSW5pemlhcmUgSW1wb3J0YXppb25lJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ0luaXppYXJlIEVzcG9ydGF6aW9uZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwJ30gPSAnUGFzc28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICdNb2RpZmljYSBpbmZvcm1hemlvbmkgY29tdW5pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnSWwgbm9tZSDDqCBvYmJsaWdhdG9yaW8hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICdMXCdvZ2dldHRvIMOoIG9iYmxpZ2F0b3JpbyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJ0lsIGZvcm1hdG8gw6ggb2JibGlnYXRvcmlvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJ01vZGlmaWNhIGluZm9ybWF6aW9uaSBvZ2dldHRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnTW9kaWZpY2EgZm9ybWF0byBpbmZvcm1hemlvbmUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAnw6ggb2JibGlnYXRvcmlvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICdNb2RpZmljYSBtYXBwYXR1cmEgaW5mb3JtYXppb25pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICdOZXNzdW4gZWxlbWVudG8gbWFwcGEgdHJvdmF0by4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ0FnZ2l1bmdpIHVuIGVsZW1lbnRvIGRpIG1hcHBhdHVyYS4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICdNb2RpZmljYSBpbmZvcm1hemlvbmkgZGkgcmljZXJjYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ1Jlc3RyaW5nZXJlIGVzcG9ydGF6aW9uZSBwZXIgcmljZXJjYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICdJbXBvcnRhcmUgaW5mb3JtYXppb25lJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnQXJjaGl2aW8gb3JpZ2luZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAnU3VjY2Vzc28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnRmFsbGl0byc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICdEdXBsaWNhIGkgbm9taSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJ051bWVybyBkZWxsXCd1bHRpbWEgcmlnYSBwcm9jZXNzYXRhIGRlbCBmaWxlIGRhIGltcG9ydGFyZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJ09rJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdJbXBvcnRhIGVkIGVzcG9ydGEgbGUgaW5mb3JtYXppb25pIHN1bGxcJ29nZ2V0dG8uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICdJbXBvcnRhcmUvRXNwb3J0YXJlJzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6Om1zX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdUYW1iYWggdGVtcGxhdCBwZXRhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdTZXQga2FyYWt0ZXInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sb24gKDopJ30gPSAnVGl0aWsgYmVydGluZGloICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdLb2x1bSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnS29sdW0gcGVtaXNhaCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnVGl0aWsgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdTZW1pa29sb24gKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYXRvciAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnVGVybWFzdWsgS29sdW0gS2VwYWxhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0V4cG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTa2lwcGVkJ30gPSAnJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAnUGVuZ3VydXNhbiBJbXBvcnQvRWtzcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnQ2lwdGEgdGVtcGxhdCB1bnR1ayBpbXBvcnQgZGFuIGVrc3BvcnQgaW5mb3JtYXNpIG9iamVrLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdNdWxhIGltcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdNdWxhIGVrc3BvcnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCd9ID0gJ0xhbmdrYWgnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICdBdWRpdCBpbmZvcm1hc2kgYmlhc2EnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ09iamVrIGFkYWxhaCBkaXBlcmx1a2FuISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnRm9ybWF0IGFkYWxhaCBkaXBlcmx1a2FuISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJ0F1ZGl0IGluZm9ybWFzaSBvYmplayc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJ0F1ZGl0IGluZm9ybWFzaSBmb3JtYXQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ0F1ZGl0IGluZm9ybWFzaSBwZXRhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICdUaWFkYSBlbGVtZW4gcGV0YSBkaXRlbXVpLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnVGFtYmFoIGVsZW1lbiBwZXRhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnQXVkaXQgaW5mb3JtYXNpIGNhcmlhbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ0RpbGFyYW5nIGVrc3BvcnQgcGVyIGNhcmlhbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICdJbmZvcm1hc2kgaW1wb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnU3VtYmVyIGZhaWwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnT2snOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICdGb3JtYXQgYmFja2VuZCBtb2R1bCBwZW5kYWZ0YXJhbiB1bnR1ayBtb2R1bCBpbXBvcnQvZWtzcG9ydC4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0luZm9ybWFzaSBvYmplayBpbXBvcnQgZGFuIGVrc3BvcnQuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICdJbXBvcnQvRWtzcG9ydCc7Cgp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6Om5iX05PX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdMZWdnIHRpbCBtYWwgZm9yIG1hcHBpbmcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ1RlZ25zZXR0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ0tvbG9uICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdLb2xvbm5lJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdLb2xvbm5lc2VwYXJhdG9yJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICdQdW5rdHVtICguKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTZW1pY29sb24gKDspJ30gPSAnU2VtaWtvbG9uICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bGF0b3IgKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW5jbHVkZSBDb2x1bW4gSGVhZGVycyd9ID0gJ0lua2x1ZGVyIGtvbG9ubmVvdmVyc2tyaWZ0ZXInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnT3Bwc3VtbWVyaW5nZXIgZm9yIGltcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnSW1wb3J0ZXJ0ZSByYWRlcic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFeHBvcnRlZCByZWNvcmRzJ30gPSAnRWtzcG9ydGVydGUgcmFkZXInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ1JhZGVyJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICdIb3BwZXQgb3Zlcic7CgogICAgIyBUZW1wbGF0ZTogQWRtaW5JbXBvcnRFeHBvcnQKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQgTWFuYWdlbWVudCd9ID0gJ0FkbWluaXN0cmFzam9uIGF2IEltcG9ydC9Fa3Nwb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdPcHByZXR0IGVuIG1hbCBmb3Igw6UgZWtzcG9ydGVyZSBvZyBpbXBvcnRlcmUgaW5mb3JtYXNqb24nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAnU3RhcnQgaW1wb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ1N0YXJ0IGVrc3BvcnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCd9ID0gJ1N0ZWcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICdFbmRyZSBmZWxsZXNpbmZvcm1hc2pvbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJ05hdm4gZXIgcMOla3JldmQhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICdPYmpla3QgZXIgcMOla3JldmQhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICdGb3JtYXQgZXIgcMOla3JldmQhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnRW5kcmUgb2JqZWt0ZXQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICdFbmRyZSBmb3JtYXRldCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpcyByZXF1aXJlZCEnfSA9ICdlciBww6VrcmV2ZCEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gPSAnRW5kcmUgbWFwcGluZyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnSW5nZW4gZWxlbWVudGVyIGZ1bm5ldC4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ0xlZ2cgdGlsIG1hcHBpbmctZWxlbWVudCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJ0VuZHJlIHPDuGsnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdCZWdyZW5zIGVrc3BvcnQgcGVyIHPDuGsnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW1wb3J0LWluZm9ybWFzam9uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnS2lsZGVmaWwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJ1ZlbGx5a2tldCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICdGZWlsZXQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnRHVwbGlrYXRlIG5hdm4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICdTaXN0ZSBwcm9zZXNzZXJ0ZSBsaW5qZW51bW1lciBhdiBpbXBvcnRmaWwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPayc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ0Jha3NpZGVtb2R1bC1yZWdpc3RyZXJpbmcgZm9yIGZvcm1hdGV0IHRpbCBpbXBvcnQvZWtzcG9ydC1tb2R1bGVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdJbmZvcm1hc2pvbiBmb3IgaW1wb3J0LSBvZyBla3Nwb3J0LW9iamVrdCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3J0L0Vrc3BvcnQnOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6Om5sX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdNYXBwaW5ndGVtcGxhdGUgdG9ldm9lZ2VuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdLYXJha3RlcnNldCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEdWJiZWxlIHB1bnQgKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ0tvbG9tJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdLb2xvbXNjaGVpZGluZ3N0ZWtlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUHVudCAoLiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ1B1bnRrb21tYSAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGFidWxhdG9yIChUQUIpJ30gPSAnVGFiJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydGVkIHJlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXhwb3J0ZWQgcmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICcnOwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdJbXBvcnQvRXhwb3J0IGJlaGVlcic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ0ltcG9ydCBzdGFydGVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ0V4cG9ydCBzdGFydGVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAnfSA9ICdTdGFwJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnQWxnZW1lbmUgaW5mb3JtYXRpZSBiZXdlcmtlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnT2JqZWN0IGlzIHZlcnBsaWNodC4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJ0Zvcm1hYXQgaXMgdmVycGxpY2h0Lic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJ09iamVjdC1pbmZvcm1hdGllIGJld2Vya2VuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnRm9ybWF0LWluZm9ybWF0aW9uZW4gYmV3ZXJrZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ01hcHBpbmctaW5mb3JtYXRpZSBiZXdlcmtlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnR2VlbiBlbGVtZW50ZW4gZ2V2b25kZW4uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICdab2VrLWluZm9ybWF0aWUgYmV3ZXJrZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdCZXBlcmsgZXhwb3J0IHRvdCB6b2Vrb3BkcmFjaHQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW1wb3J0LWluZm9ybWF0aWUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdCcm9uYmVzdGFuZCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPSyc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnSW1wb3J0IGVuIGV4cG9ydCBvYmplY3RpbmZvcm1hdGllJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICdJbXBvcnQvRXhwb3J0JzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnBsX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdEb2RhaiBzemFibG9uIG1hcG93YW5pYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAnS29kb3dhbmllJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ0R3dWtyb3BlayAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnS29sdW1uYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnU2VwYXJhdG9yIGtvbHVtbnknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ0tyb3BrYSAoLiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ8WacmVkbmlrICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bGF0b3IgKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW5jbHVkZSBDb2x1bW4gSGVhZGVycyd9ID0gJ1VtaWXFm8SHIG5hZ8WCw7N3a2kga29sdW1uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ1BvZHN1bW93YW5pZSBpbXBvcnR1IGRsYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnWmFpbXBvcnRvd2FuZSByZWtvcmR5JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0V4cG9ydGVkIHJlY29yZHMnfSA9ICdXeWVrc3BvcnRvd2FuZSByZWtvcmR5JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlY29yZHMnfSA9ICdSZWtvcmR5JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICdQb21pbmnEmXRlJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAnWmFyesSFZHphbmllIEltcG9ydGVtL0V4cG9ydGVtJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdVdHfDs3J6IHN6YWJsb24gZG8gaW1wb3J0dSBpIGVrc3BvcnR1IGRhbnljaCBvYmlla3TDs3cuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ1JvenBvY3puaWogaW1wb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ1JvenBvY3puaWogZWtzcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwJ30gPSAnS3Jvayc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJ0VkeXR1aiBpbmZvcm1hY2plIHdzcMOzbG5lJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICdPYmlla3QgamVzdCB3eW1hZ2FueSEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJ0Zvcm1hdCBqZXN0IHd5bWFnYW55ISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJ0VkeXR1aiBpbmZvcm1hY2plIG9iaWVrdHUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICdFZHl0dWogZm9ybWF0IG9iaWVrdHUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ0VkeXR1aiBpbmZvcm1hY2plIG1hcG93YW5pYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnTmllIHpuYWxlemlvbm8gZWxlbWVudMOzdyBtYXB5Lic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnRG9kYWogZWxlbWVudCBtYXBvd2FuaWEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICdFZHl0dWogaW5mb3JtYWNqZSB3eXN6dWtpd2FuaWEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdPZ3JhbmljeiBla3Nwb3J0IHByemV6IHd5c3p1a2FuaWUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW1wb3J0dWogaW5mb3JtYWNqZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ1BsaWsgxbpyw7NkxYJvd3knOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJ1Bvd29kemVuaWUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnTmllcG93b2R6ZW5pZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICdEdXBsaWthdHkgbmF6dyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJ05yIG9zdGFuaWVqIHByemV0d29yem9uZWogbGluaWkgcGxpa3UgaW1wb3J0b3dlZ28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPayc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ01vZHXFgiBmb3JtYXRvd2FuaWEgYmFja2VuZCBkbGEgbW9kdcWCdSBpbXBvcnQvZWtzcG9ydC4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0ltcG9ydHVqIGkgZWtzcG9ydHVqIGluZm9ybWFjamUgb2JpZWt0w7N3Lic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3J0L2Vrc3BvcnQnOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnB0X0JSX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdBZGljaW9uYXIgbW9kZWxvIGRlIG1hcGVhbWVudG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ0NvZGlmaWNhw6fDo28gZGUgQ2FyYWN0ZXJlcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEb2lzIFBvbnRvcyAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnQ29sdW5hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdTZXBhcmFkb3IgZGUgQ29sdW5hcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUG9udG8gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdQb250byBlIFbDrXJndWxhICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bGHDp8OjbyAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnSW5jbHVpciBDYWJlw6dhbGhvcyBkZSBDb2x1bmFzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ1Jlc3VtbyBkZSBpbXBvcnRhw6fDo28gcGFyYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnUmVnaXN0cm9zIGltcG9ydGFkb3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXhwb3J0ZWQgcmVjb3Jkcyd9ID0gJ1JlZ2lzdHJvcyBleHBvcnRhZG9zJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlY29yZHMnfSA9ICdSZWdpc3Ryb3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2tpcHBlZCd9ID0gJ0Rlc2NvbnNpZGVyYWRvcyc7CgogICAgIyBUZW1wbGF0ZTogQWRtaW5JbXBvcnRFeHBvcnQKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQgTWFuYWdlbWVudCd9ID0gJ0dlcmVuY2lhbWVudG8gZGUgSW1wb3J0YcOnw6NvL0V4cG9ydGHDp8Ojbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnQ3JpYXIgdW0gbW9kZWxvIHBhcmEgaW1wb3J0YXIgZSBleHBvcnRhciBpbmZvcm1hw6fDtWVzIGRlIG9iamV0by4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAnSW5pY2lhciBJbXBvcnRhw6fDo28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAnSW5pY2lhciBFeHBvcnRhw6fDo28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCd9ID0gJ1Bhc3NvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnRWRpdGFyIGluZm9ybWHDp8O1ZXMgY29tdW5zJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnTm9tZSDDqSBvYnJpZ2F0w7NyaW8hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICdPYmpldG8gw6kgb2JyaWdhdMOzcmlvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnTyBmb3JtYXRvIMOpIG9icmlnYXTDs3JpbyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICdFZGl0YXIgaW5mb3JtYcOnw7VlcyBkbyBvYmpldG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICdFZGl0YXIgaW5mb3JtYcOnw7VlcyBkbyBmb3JtYXRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJ8OpIG9icmlnYXTDs3JpbyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gPSAnRWRpdGFyIGluZm9ybWHDp8O1ZXMgZG8gbWFwZWFtZW50byc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnTsOjbyBow6EgZWxlbWVudG9zIG1hcGEgZW5jb250cmFkby4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ0FkaWNpb25hciBlbGVtZW50byBkZSBtYXBlYW1lbnRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnRWRpdGFyIGluZm9ybWHDp8O1ZXMgZGUgcGVzcXVpc2EnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdSZXN0cmluZ2lyIGV4cG9ydGHDp8OjbyBwb3IgcGVzcXVpc2EnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW5mb3JtYcOnw7VlcyBkZSBpbXBvcnRhw6fDo28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdBcnF1aXZvIGRlIE9yaWdlbSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAnU3VjZXNzbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICdGYWxob3UnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnTm9tZXMgZHVwbGljYWRvcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJ8OabHRpbW8gbsO6bWVybyBkZSBsaW5oYSBwcm9jZXNzYWRhIGRvIGFycXVpdm8gZGUgaW1wb3JhdGHDp8Ojbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJ09rJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnRm9ybWF0byBkZSByZWdpc3RybyBiYWNrZW5kIGRvIG3Ds2R1bG8gZGUgaW1wb3J0YcOnw6NvIC8gZXhwb3J0YcOnw6NvIG3Ds2R1bG8uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdJbXBvcnRhciBlIGV4cG9ydGFyIGluZm9ybWHDp8O1ZXMgZGUgb2JqZXRvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3J0YcOnw6NvL0V4cG9ydGHDp8Ojbyc7Cgp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnB0X0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdBZGljaW9uYXIgbW9kZWxvIGRlIG1hcGVhbWVudG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ0NvZGlmaWNhw6fDo28gZGUgQ2FyYWN0ZXJlcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEb2lzIFBvbnRvcyAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnQ29sdW5hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdTZXBhcmFkb3IgZGUgQ29sdW5hcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUG9udG8gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdQb250byBlIFbDrXJndWxhICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bGHDp8OjbyAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnSW5jbHVpciBDYWJlw6dhbGhvcyBkZSBDb2x1bmFzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ1Jlc3VtbyBkZSBpbXBvcnRhw6fDo28gcGFyYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnUmVnaXN0cm9zIGltcG9ydGFkb3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXhwb3J0ZWQgcmVjb3Jkcyd9ID0gJ1JlZ2lzdHJvcyBleHBvcnRhZG9zJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlY29yZHMnfSA9ICdSZWdpc3Ryb3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2tpcHBlZCd9ID0gJ0Rlc2NvbnNpZGVyYWRvcyc7CgogICAgIyBUZW1wbGF0ZTogQWRtaW5JbXBvcnRFeHBvcnQKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQgTWFuYWdlbWVudCd9ID0gJ0dlcmVuY2lhbWVudG8gZGUgSW1wb3J0YcOnw6NvL0V4cG9ydGHDp8Ojbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnQ3JpYXIgdW0gbW9kZWxvIHBhcmEgaW1wb3J0YXIgZSBleHBvcnRhciBpbmZvcm1hw6fDtWVzIGRlIG9iamV0by4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAnSW5pY2lhciBJbXBvcnRhw6fDo28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAnSW5pY2lhciBFeHBvcnRhw6fDo28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCd9ID0gJ1Bhc3NvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnRWRpdGFyIGluZm9ybWHDp8O1ZXMgY29tdW5zJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnTm9tZSDDqSBvYnJpZ2F0w7NyaW8hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICdPYmpldG8gw6kgb2JyaWdhdMOzcmlvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnTyBmb3JtYXRvIMOpIG9icmlnYXTDs3JpbyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICdFZGl0YXIgaW5mb3JtYcOnw7VlcyBkbyBvYmpldG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICdFZGl0YXIgaW5mb3JtYcOnw7VlcyBkbyBmb3JtYXRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICdFZGl0YXIgaW5mb3JtYcOnw7VlcyBkbyBtYXBlYW1lbnRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICdOw6NvIGjDoSBlbGVtZW50b3MgbWFwYSBlbmNvbnRyYWRvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnQWRpY2lvbmFyIGVsZW1lbnRvIGRlIG1hcGVhbWVudG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICdFZGl0YXIgaW5mb3JtYcOnw7VlcyBkZSBwZXNxdWlzYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ1Jlc3RyaW5naXIgZXhwb3J0YcOnw6NvIHBvciBwZXNxdWlzYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICdJbmZvcm1hw6fDtWVzIGRlIGltcG9ydGHDp8Ojbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ0FycXVpdm8gZGUgT3JpZ2VtJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICdTdWNlc3NvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJ0ZhbGhvdSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICdOb21lcyBkdXBsaWNhZG9zJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnw5psdGltbyBuw7ptZXJvIGRlIGxpbmhhIHByb2Nlc3NhZGEgZG8gYXJxdWl2byBkZSBpbXBvcmF0YcOnw6NvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnT2snOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICdGb3JtYXRvIGRlIHJlZ2lzdHJvIGJhY2tlbmQgZG8gbcOzZHVsbyBkZSBpbXBvcnRhw6fDo28gLyBleHBvcnRhw6fDo28gbcOzZHVsby4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0ltcG9ydGFyIGUgZXhwb3J0YXIgaW5mb3JtYcOnw7VlcyBkZSBvYmpldG8uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICdJbXBvcnRhw6fDo28vRXhwb3J0YcOnw6NvJzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnJ1X0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICfQlNC+0LHQsNCy0LjRgtGMINGI0LDQsdC70L7QvSDRgdC+0L/QvtGB0YLQsNCy0LvQtdC90LjRjyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAn0JrQvtC00LjRgNC+0LLQutCwJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ9CU0LLQvtC10YLQvtGH0LjQtSAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAn0KHRgtC+0LvQsdC10YYgJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICfQoNCw0LfQtNC10LvQuNGC0LXQu9GMJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICfQotC+0YfQutCwICguKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTZW1pY29sb24gKDspJ30gPSAn0KLQvtGH0LrQsCDRgSDQt9Cw0L/Rj9GC0L7QuSAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGFidWxhdG9yIChUQUIpJ30gPSAn0KLQsNCx0YPQu9GP0YbQuNGPIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICfQktC60LvRjtGH0LjRgtGMINC30LDQs9C+0LvQvtCy0LrQuCDRgdGC0L7Qu9Cx0YbQvtCyJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ9Ce0YLRh9C10YIg0L7QsSDQuNC80L/QvtGA0YLQtSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAn0JjQvNC/0L7RgNGC0LjRgNC+0LLQsNC90L4g0LfQsNC/0LjRgdC10LknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXhwb3J0ZWQgcmVjb3Jkcyd9ID0gJ9Ct0LrRgdC/0L7RgNGC0LjRgNC+0LLQsNC90L4g0LfQsNC/0LjRgdC10LknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ9CX0LDQv9C40YHQtdC5JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NraXBwZWQnfSA9ICfQn9GA0L7Qv9GD0YnQtdC90L4nOwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICfQo9C/0YDQsNCy0LvQtdC90LjQtSDQmNC80L/QvtGA0YLQvtC8L9Ct0LrRgdC/0L7RgNGC0L7QvCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAn0KHQvtC30LTQsNC50YLQtSDRiNCw0LHQu9C+0L0g0LTQu9GPINC40LzQv9C+0YDRgtCwINC4INGN0LrRgdC/0L7RgNGC0LAg0LjQvdGE0L7RgNC80LDRhtC40Lgg0L7QsdGK0LXQutGC0L7Qsic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICfQndCw0YfQsNGC0Ywg0LjQvNC/0L7RgNGCJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ9Cd0LDRh9Cw0YLRjCDRjdC60YHQv9C+0YDRgic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwJ30gPSAn0KjQsNCzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAn0KDQtdC00LDQutGC0LjRgNC+0LLQsNGC0Ywg0L7QsdGJ0YPRjiDQuNC90YTQvtGA0LzQsNGG0LjRjic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJ9Ci0YDQtdCx0YPQtdGC0YHRjyDQuNC80Y8hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICfQntCx0YrQtdC60YIg0L7QsdGP0LfQsNGC0LXQu9C10L0hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICfQpNC+0YDQvNCw0YIg0L7QsdGP0LfQsNGC0LXQu9C10L0hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAn0KDQtdC00LDQutGC0LjRgNC+0LLQsNGC0Ywg0LjQvdGE0L7RgNC80LDRhtC40Y4g0L7QsSDQvtCx0YrQtdC60YLQtSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJ9Cg0LXQtNCw0LrRgtC40YDQvtCy0LDRgtGMINGE0L7RgNC80LDRgiDQtNCw0L3QvdGL0YUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAn0L7QsdGP0LfQsNGC0LXQu9C10L0hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ9Cg0LXQtNCw0LrRgtC40YDQvtCy0LDRgtGMINC40L3RhNC+0YDQvNCw0YbQuNGOINGB0L7QvtGC0LLQtdGC0YHRgtCy0LjRjyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAn0J3QtdGCINGN0LvQtdC80LXQvdGC0L7QsiDRgdC+0L/QvtGB0YLQsNCy0LvQtdC90LjRjy4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ9CU0L7QsdCw0LLRjNGC0LUg0Y3Qu9C10LzQtdC90YIg0YHQvtC/0L7RgdGC0LDQstC70LXQvdC40Y8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICfQoNC10LTQsNC60YLQuNGA0L7QstCw0YLRjCDQv9C+0LjRgdC60L7QstGD0Y4g0LjQvdGE0L7RgNC80LDRhtC40Y4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICfQntCz0YDQsNC90LjRh9C40YLRjCDRjdC60YHQv9C+0YDRgiDQv9C+0LjRgdC60L7QvCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICfQmNC90YTQvtGA0LzQsNGG0LjRjyDQvtCxINC40LzQv9C+0YDRgtC1JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAn0JjRgdGF0L7QtNC90YvQuSDRhNCw0LnQuyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAn0KPRgdC/0LXRiNC90L4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAn0J3QtSDRg9C00LDQu9C+0YHRjCDQstGL0L/QvtC70L3QuNGC0YwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAn0JTRg9Cx0LvQuNGA0YPRjtGJ0LjQtSDQuNC80LXQvdCwJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAn0J3QvtC80LXRgCDQv9C+0YHQu9C10LTQvdC10Lkg0L7QsdGA0LDQsdC+0YLQsNC90L3QvtC5INGB0YLRgNC+0LrQuCDQuNC80L/QvtGA0YLQuNGA0YPQtdC80L7Qs9C+INGE0LDQudC70LAnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPayc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24g0LTQu9GPINC80L7QtNGD0LvRjyBpbXBvcnQvZXhwb3J0Lic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAn0JjQvNC/0L7RgNGCINC4INGN0LrRgdC/0L7RgNGCINC40L3RhNC+0YDQvNCw0YbQuNC4INC+0LEg0L7QsdGK0LXQutGC0LDRhSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAn0JjQvNC/0L7RgNGCL9Ct0LrRgdC/0L7RgNGCJzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnN3X0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdPbmdlemEga2llbGV6byBjaGEga3V3ZWthIHJhbWFuaSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAnU2V0aSB5YSBoZXJ1ZmknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sb24gKDopJ30gPSAnQ29sb24gKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ1NhZnV3aW1hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdLaXRlbmdhbmlzaGkgc2FmdSB3aW1hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICdOdWt0YSAoLiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ0FsYW1hIHlhIG51a3RhIG1rYXRvICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJvIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICdBbWJhdGlzaGEgdmljaHdhIHZ5YSBoYWJhcmkgdnlhIHNhZnV3aW1hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ0xldGEgbXVodGFzYXJpIGt3YSBhamlsaSB5YSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnS3VtYnVrdW1idSB6aWxpem9sZXR3YSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFeHBvcnRlZCByZWNvcmRzJ30gPSAnS3VtYnVrdW1idSB6aWxpem9oYW1pc2h3YSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnS3VtYnVrdW1idSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTa2lwcGVkJ30gPSAnUnVrd2EnOwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdVc2ltYW1pemkgd2Ega3VsZXRhL0hhbWlzaGEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ1RlbmdlbmV6YSBraW9sZXpvIGt1bGV0YSBuYSBrdWhhbWlzaGEgdGFhcmlmYSB6YSBraXBlbmdlZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdBbnphIGt1bGV0YSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdBbnphIGt1aGFtaXNoYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwJ30gPSAnSGF0dWEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICdIYXJpcmkgdGFhcmlmYSB6YSBrdWZhbmEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICdKaW5hIGxpbmFoaXRhamlrYSEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ0tpcGVuZ2VlIGtpbmFoaXRhamlrYSEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJ1VtYml6byB1bmFoaXRhamlrYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJ0hhcmlyaSB0YWFyaWZhIHphIGtpcGVuZ2VlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnSGFyaXJpIHRhYXJpZmEgemEgdW1iaXpvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJ0luYWhpdGFqaWthJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ0hhcmlyaSB0YXJpZmEgemEga3V3ZWthIHJhbWFuaSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnSGFrdW5hIGVsZW1lbnRpIHppbGl6b3BhdGlrYW5hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICdPbmdlemEgZWxlbWVudGkgemEga3V3ZWthIHJhbWFuaSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJ0hhcmlyaSB0YWFyaWZhIHphIGt1dGFmdXRhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnWnVpYSB1aGFtaXNoYWppIGt3YSBraWxhIHV0YWZ1dGFqaSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICdMZXRhIHRhYXJpZmEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdGYWlsaSBsYSBjaGFuem8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJ01hZmFuaWtpbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICdTaGluZHdhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ0ppbmEgbGltZWppcnVkaWE6JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnTmFtYmEgeWEgbXN0YXJpIHVuYW9rYXRpa2EgbWNoYWthdG8gd2EgbXdpc2hvIHdhIGZhaWxpIGxpbGlsb2xldHdhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnU2F3YSc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ1VtYml6YSB1c2FqaWxpIHdhIG1vZHVsaSB3YSBtYXppbmdpcmEgeWEgbnl1bWEga3dhIGFqaWxpIHlhIG1vZHVsaSB5YSBrdWxldGEva3VoYW1pc2hhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdMZXRhIG5hIGhhbWlzaGEgdGFhcmlmYSB6YSBraXBlbmdlZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnTGV0YS9IYW1pc2hhJzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnpoX0NOX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICflop7liqDmmKDlsITmqKHniYgnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ+Wtl+espumbhic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICflhpLlj7cgKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ+WIlyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAn5YiX5YiG6ZqU56ymJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICflj6Xlj7cgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICfliIblj7cgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ+WItuihqOmUriAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAn5YyF5ous5YiX5qCH6aKYJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ+WvvOWFpeaAu+e7kyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAn5a+85YWl6K6w5b2VJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0V4cG9ydGVkIHJlY29yZHMnfSA9ICflr7zlh7rorrDlvZUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ+iusOW9lSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTa2lwcGVkJ30gPSAn6Lez6L+H55qEJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAn5a+85YWlL+WvvOWHuueuoeeQhic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAn5Yib5bu65qih5p2/5a+85YWl5ZKM5a+85Ye65a+56LGh5L+h5oGv44CCJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ+W8gOWni+WvvOWFpSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICflvIDlp4vlr7zlh7onOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCd9ID0gJ+atpemqpCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJ+e8lui+keWFseeUqOS/oeaBryc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJ+WRveWQjeaYr+W/hemhu+eahO+8gSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAn5a+56LGh5piv5b+F6ZyA55qE77yBJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICfmoLzlvI/mmK/lv4XpnIDnmoTvvIEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICfnvJbovpHlr7nlg4/kv6Hmga8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICfnvJbovpHmoLzlvI/kv6Hmga8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAn5piv5b+F6aG755qE77yBJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ+e8lui+keaYoOWwhOS/oeaBryc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAn5rKh5pyJ5om+5Yiw5pig5bCE55qE5a2X5q61JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICfmt7vliqDmmKDlsITlrZfmrrUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICfnvJbovpHmkJzntKLkv6Hmga8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICfpmZDliLblr7zlh7rmr4/kuKrmkJzlr7snOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAn5a+85YWl5L+h5oGvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAn5rqQ5paH5Lu2JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICfmiJDlip8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAn5aSx6LSlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ+mHjeWkjeeahOWQjeensCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJ+WvvOWFpeaWh+S7tuacgOWQjuWkhOeQhueahOihjOaVsCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJ09LJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAn5qC85byP5ZCO5Y+w5qih5Z2X5a+55a+85YWlL+WvvOWHuuaooeWdl+azqOWGjCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAn5a+85YWl5ZKM5a+85Ye65a+56LGh5L+h5oGvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICflr7zlhaUv5a+85Ye6JzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnpoX1RXX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICflop7liqDmmKDlsITmqKHniYgnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ+Wtl+espumbhic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICflhpLomZ8gKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ+WIlyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAn5YiX5YiG6ZqU56ymJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICflj6XomZ8gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICfliIbomZ8gKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ+WItuihqOmNtSAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAn5YyF5ous5YiX5qiZ6aGMJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ+WwjuWFpee4vee1kCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAn5bCO5YWl57SA6YyEJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0V4cG9ydGVkIHJlY29yZHMnfSA9ICflsI7lh7rntIDpjIQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ+e0gOmMhCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTa2lwcGVkJ30gPSAn6Lez6YGO55qEJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAn5bCO5YWlL+WwjuWHuueuoeeQhic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAn5Ym15bu65qih5p2/5bCO5YWl5ZKM5bCO5Ye65bCN6LGh5L+h5oGv44CCJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ+mWi+Wni+WwjuWFpSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICfplovlp4vlsI7lh7onOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCd9ID0gJ+atpempnyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJ+e3qOi8r+WFseeUqOS/oeaBryc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAn5bCN6LGh5piv5b+F6ZyA55qE77yBJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICfmoLzlvI/mmK/lv4XpnIDnmoQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICfnt6jovK/lsI3osaHkv6Hmga8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICfnt6jovK/moLzlvI/kv6Hmga8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ+e3qOi8r+aYoOWwhOS/oeaBryc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAn5rKS5pyJ5om+5Yiw5pig5bCE55qE5a2X5q61JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICfmt7vliqDmmKDlsITlrZfmrrUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICfnt6jovK/mkJzntKLkv6Hmga8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICfpmZDliLblsI7lh7rmr4/lgIvmkJzlsIsnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAn5bCO5YWl5L+h5oGvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAn5rqQ5paH5Lu2JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICfmiJDlip8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAn5aSx5pWXJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ+mHjeikh+eahOWQjeeosSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJ+WwjuWFpeaWh+S7tuacgOW+jOiZleeQhueahOihjOaVuCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJ09rJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICflsI7lhaXlkozlsI7lh7rlsI3osaHkv6Hmga8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ+WwjuWFpS/lsI7lh7onOwoKfQoKMTsK
# --
# Copyright (C) 2001-2018 OTRS AG, https://otrs.com/
# --
# 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 https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Modules::AdminImportExport;

use strict;
use warnings;

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

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

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

    # check needed objects
    for my $Object (qw(ConfigObject ParamObject LogObject LayoutObject EncodeObject)) {
        if ( !$Self->{$Object} ) {
            $Self->{LayoutObject}->FatalError( Message => "Got no $Object!" );
        }
    }
    $Self->{ImportExportObject} = Kernel::System::ImportExport->new( %{$Self} );
    $Self->{ValidObject}        = Kernel::System::Valid->new( %{$Self} );

    return $Self;
}

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

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

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

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

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

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

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

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

        if ( !$SaveContinue ) {

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

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

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

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

        }

        # if save & continue
        my %ServerError;

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

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

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

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

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

        }

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

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

        # save to database
        my $Success;

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

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

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

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

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

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

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

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

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

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

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

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

        # save template starts here

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

        my %AttributeValues;

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

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

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

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

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

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

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

        }

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

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

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

    }

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

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

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

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

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

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

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

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

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

        # save starting here

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

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

        my $Error = 0;
        my %ServerError;

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

        for my $Header (@Headers) {

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

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

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

            $EmptyMap = 0;

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

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

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

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

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

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

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

                # output column counter
                $Self->{LayoutObject}->Block(
                    Name => 'TemplateEdit4MapNumberColumn',
                    Data => {
                        Counter => $AttributeRowCounter,
                    },
                );
            }

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

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

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

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

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

            $AttributeRowCounter++;
        }

        # output an empty list
        if ($EmptyMap) {

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

            $Counter++;
        }

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

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

                next MAPPINGID;
            }

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

                next MAPPINGID;
            }

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

                next MAPPINGID;
            }
        }

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

        $SourceFile{Content} ||= '';

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

        my $EmptyDatabase = 1;

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

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

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

            $EmptyDatabase = 0;

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

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

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

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

        # output an empty list
        if ($EmptyDatabase) {

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

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

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

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

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

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

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

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

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

    my $Class = ' Validate_Required ';

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

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

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

    if ( $Param{New} ) {

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

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

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

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

        $Class = ' Validate_Required ';

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

        my $Class = ' ';
        my $Value;

        my $DataTypeError;
        my $ErrorMessage;

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

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

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

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

        # error area

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

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

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

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

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

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

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

}

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

}

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRDaGVja2JveDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0TGF5b3V0Q2hlY2tib3ggLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKQWxsIGxheW91dCBmdW5jdGlvbnMgZm9yIGNoZWNrYm94IGVsZW1lbnRzIGluIGltcG9ydC9leHBvcnQuCgo9b3ZlciA0Cgo9Y3V0Cgo9aXRlbSBuZXcoKQoKY3JlYXRlIGFuIG9iamVjdAoKICAgICRCYWNrZW5kT2JqZWN0ID0gS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydExheW91dENoZWNrYm94LT5uZXcoCiAgICAgICAgJVBhcmFtLAogICAgKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgIyBjaGVjayBuZWVkZWQgb2JqZWN0cwogICAgZm9yIG15ICRPYmplY3QgKHF3KENvbmZpZ09iamVjdCBMb2dPYmplY3QgTWFpbk9iamVjdCBQYXJhbU9iamVjdCBMYXlvdXRPYmplY3QpKSB7CiAgICAgICAgJFNlbGYtPnskT2JqZWN0fSA9ICRQYXJhbXskT2JqZWN0fSB8fCBkaWUgIkdvdCBubyAkT2JqZWN0ISI7CiAgICB9CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aXRlbSBGb3JtSW5wdXRDcmVhdGUoKQoKY3JlYXRlIGEgaW5wdXQgc3RyaW5nCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkZvcm1JbnB1dENyZWF0ZSgKICAgICAgICBJdGVtICAgPT4gJEl0ZW1SZWYsCiAgICAgICAgUHJlZml4ID0+ICdQcmVmaXg6OicsICAjIChvcHRpb25hbCkKICAgICAgICBWYWx1ZSAgPT4gJ1ZhbHVlJywgICAgICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBGb3JtSW5wdXRDcmVhdGUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgaWYgKCAhJFBhcmFte0l0ZW19ICkgewogICAgICAgICRTZWxmLT57TG9nT2JqZWN0fS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAnTmVlZCBJdGVtIScsCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgJFBhcmFte1ByZWZpeH0gfHw9ICcnOwoKICAgIG15ICRDaGVja2VkID0gJFBhcmFte1ZhbHVlfSA/ICdjaGVja2VkPSJjaGVja2VkIicgOiAnJzsKCiAgICByZXR1cm4KICAgICAgICBxcXs8aW5wdXQgaWQ9IiRQYXJhbXtQcmVmaXh9JFBhcmFte0l0ZW19LT57S2V5fSIgdHlwZT0iY2hlY2tib3giIG5hbWU9IiRQYXJhbXtQcmVmaXh9JFBhcmFte0l0ZW19LT57S2V5fSIgJENoZWNrZWQgLz59Owp9Cgo9aXRlbSBGb3JtRGF0YUdldCgpCgpnZXQgZm9ybSBkYXRhCgogICAgbXkgJEZvcm1EYXRhID0gJEJhY2tlbmRPYmplY3QtPkZvcm1EYXRhR2V0KAogICAgICAgIEl0ZW0gICA9PiAkSXRlbVJlZiwKICAgICAgICBQcmVmaXggPT4gJ1ByZWZpeDo6JywgICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBGb3JtRGF0YUdldCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIEl0ZW0hJywKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAkUGFyYW17UHJlZml4fSB8fD0gJyc7CgogICAgIyBnZXQgZm9ybSBkYXRhCiAgICBteSAkRm9ybURhdGEgPSAkU2VsZi0+e1BhcmFtT2JqZWN0fS0+R2V0UGFyYW0oCiAgICAgICAgUGFyYW0gPT4gJFBhcmFte1ByZWZpeH0gLiAkUGFyYW17SXRlbX0tPntLZXl9LAogICAgKTsKCiAgICByZXR1cm4gJEZvcm1EYXRhOwp9CgoxOwoKPWJhY2sKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKEw8aHR0cHM6Ly9vdHJzLm9yZy8+KS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0Pi4KCj1jdXQK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRUVDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0TGF5b3V0VFQgLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKQWxsIGxheW91dCBmdW5jdGlvbnMgZm9yIGRpc3BsYXkgVFQgY29kZQoKPW92ZXIgNAoKPWN1dAoKPWl0ZW0gbmV3KCkKCmNyZWF0ZSBhbiBvYmplY3QKCiAgICAkQmFja2VuZE9iamVjdCA9IEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRUVC0+bmV3KAogICAgICAgICVQYXJhbSwKICAgICk7Cgo9Y3V0CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHt9OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgICMgY2hlY2sgbmVlZGVkIG9iamVjdHMKICAgIGZvciBteSAkT2JqZWN0IChxdyhDb25maWdPYmplY3QgTG9nT2JqZWN0IE1haW5PYmplY3QgUGFyYW1PYmplY3QgTGF5b3V0T2JqZWN0KSkgewogICAgICAgICRTZWxmLT57JE9iamVjdH0gPSAkUGFyYW17JE9iamVjdH0gfHwgZGllICJHb3Qgbm8gJE9iamVjdCEiOwogICAgfQoKICAgIHJldHVybiAkU2VsZjsKfQoKPWl0ZW0gRm9ybUlucHV0Q3JlYXRlKCkKCmNyZWF0ZSBhIGlucHV0IHN0cmluZwoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtSW5wdXRDcmVhdGUoCiAgICAgICAgSXRlbSA9PiAkSXRlbVJlZiwKICAgICk7Cgo9Y3V0CgpzdWIgRm9ybUlucHV0Q3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgSXRlbSEnCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgcmV0dXJuICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e0RhdGF9Owp9Cgo9aXRlbSBGb3JtRGF0YUdldCgpCgpnZXQgZm9ybSBkYXRhCgogICAgbXkgJEZvcm1EYXRhID0gJEJhY2tlbmRPYmplY3QtPkZvcm1EYXRhR2V0KCk7Cgo9Y3V0CgpzdWIgRm9ybURhdGFHZXQgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm47Cn0KCjE7Cgo9YmFjawoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIHNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoTDxodHRwczovL290cnMub3JnLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQ+LgoKPWN1dAo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRTZWxlY3Rpb247Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydExheW91dFNlbGVjdGlvbiAtIGxheW91dCBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIFNZTk9QU0lTCgpBbGwgbGF5b3V0IGZ1bmN0aW9ucyBmb3Igc2VsZWN0aW9uIGVsZW1lbnRzCgo9b3ZlciA0Cgo9Y3V0Cgo9aXRlbSBuZXcoKQoKY3JlYXRlIGFuIG9iamVjdAoKICAgICRCYWNrZW5kT2JqZWN0ID0gS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydExheW91dFNlbGVjdGlvbi0+bmV3KAogICAgICAgICVQYXJhbSwKICAgICk7Cgo9Y3V0CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHt9OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgICMgY2hlY2sgbmVlZGVkIG9iamVjdHMKICAgIGZvciBteSAkT2JqZWN0IChxdyhDb25maWdPYmplY3QgTG9nT2JqZWN0IE1haW5PYmplY3QgUGFyYW1PYmplY3QgTGF5b3V0T2JqZWN0KSkgewogICAgICAgICRTZWxmLT57JE9iamVjdH0gPSAkUGFyYW17JE9iamVjdH0gfHwgZGllICJHb3Qgbm8gJE9iamVjdCEiOwogICAgfQoKICAgIHJldHVybiAkU2VsZjsKfQoKPWl0ZW0gRm9ybUlucHV0Q3JlYXRlKCkKCmNyZWF0ZSBhIGlucHV0IHN0cmluZwoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtSW5wdXRDcmVhdGUoCiAgICAgICAgSXRlbSAgID0+ICRJdGVtUmVmLAogICAgICAgIFByZWZpeCA9PiAnUHJlZml4OjonLCAgIyAob3B0aW9uYWwpCiAgICAgICAgVmFsdWUgID0+ICdWYWx1ZScsICAgICAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgRm9ybUlucHV0Q3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgSXRlbSEnCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBzZXQgZGVmYXVsdCB2YWx1ZQogICAgJFBhcmFte1ByZWZpeH0gfHw9ICcnOwogICAgJFBhcmFte1ZhbHVlfSAgfHw9ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1ZhbHVlRGVmYXVsdH07CgogICAgaWYgKCAkUGFyYW17VmFsdWV9ICYmICRQYXJhbXtWYWx1ZX0gPX4gbXsgIyMjIyMgfXhtcyApIHsKICAgICAgICBteSBAVmFsdWVzID0gc3BsaXQgJyMjIyMjJywgJFBhcmFte1ZhbHVlfTsKICAgICAgICAkUGFyYW17VmFsdWV9ID0gXEBWYWx1ZXM7CiAgICB9CgogICAgIyBnZW5lcmF0ZSBvcHRpb24gc3RyaW5nCiAgICBteSAkU3RyaW5nID0gJFNlbGYtPntMYXlvdXRPYmplY3R9LT5CdWlsZFNlbGVjdGlvbigKICAgICAgICBJRCAgICAgICAgICAgPT4gJFBhcmFte1ByZWZpeH0gLiAkUGFyYW17SXRlbX0tPntLZXl9LAogICAgICAgIENsYXNzICAgICAgICA9PiAkUGFyYW17Q2xhc3N9LAogICAgICAgIE5hbWUgICAgICAgICA9PiAkUGFyYW17UHJlZml4fSAuICRQYXJhbXtJdGVtfS0+e0tleX0sCiAgICAgICAgRGF0YSAgICAgICAgID0+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e0RhdGF9IHx8IHt9LAogICAgICAgIFNlbGVjdGVkSUQgICA9PiAkUGFyYW17VmFsdWV9LAogICAgICAgIFRyYW5zbGF0aW9uICA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntUcmFuc2xhdGlvbn0sCiAgICAgICAgVHJlZVZpZXcgICAgID0+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1RyZWVWaWV3fSB8fCAwLAogICAgICAgIFBvc3NpYmxlTm9uZSA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntQb3NzaWJsZU5vbmV9LAogICAgICAgIE11bHRpcGxlICAgICA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntNdWx0aXBsZX0sCiAgICAgICAgU2l6ZSAgICAgICAgID0+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1NpemV9LAogICAgKTsKCiAgICByZXR1cm4gJFN0cmluZzsKfQoKPWl0ZW0gRm9ybURhdGFHZXQoKQoKZ2V0IGZvcm0gZGF0YQoKICAgIG15ICRGb3JtRGF0YSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtRGF0YUdldCgKICAgICAgICBJdGVtICAgPT4gJEl0ZW1SZWYsCiAgICAgICAgUHJlZml4ID0+ICdQcmVmaXg6OicsICAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgRm9ybURhdGFHZXQgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgaWYgKCAhJFBhcmFte0l0ZW19ICkgewogICAgICAgICRTZWxmLT57TG9nT2JqZWN0fS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAnTmVlZCBJdGVtIScKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAkUGFyYW17UHJlZml4fSB8fD0gJyc7CgogICAgIyBnZXQgZm9ybSBkYXRhCiAgICBteSBARm9ybURhdGFzID0gJFNlbGYtPntQYXJhbU9iamVjdH0tPkdldEFycmF5KAogICAgICAgIFBhcmFtID0+ICRQYXJhbXtQcmVmaXh9IC4gJFBhcmFte0l0ZW19LT57S2V5fSwKICAgICk7CgogICAgbXkgJEZvcm1EYXRhID0gam9pbiAnIyMjIyMnLCBARm9ybURhdGFzOwoKICAgIHJldHVybiAkRm9ybURhdGEgaWYgJEZvcm1EYXRhOwogICAgcmV0dXJuICRGb3JtRGF0YSBpZiAhJFBhcmFte0l0ZW19LT57SW5wdXR9LT57UmVxdWlyZWR9OwoKICAgICMgc2V0IGludmFsaWQgcGFyYW0KICAgICRQYXJhbXtJdGVtfS0+e0Zvcm19LT57SW52YWxpZH0gPSAxOwoKICAgIHJldHVybiAkRm9ybURhdGE7Cn0KCjE7Cgo9YmFjawoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIHNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoTDxodHRwczovL290cnMub3JnLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQ+LgoKPWN1dAo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRUZXh0OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnRMYXlvdXRUZXh0IC0gbGF5b3V0IGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgU1lOT1BTSVMKCkFsbCBsYXlvdXQgZnVuY3Rpb25zIGZvciB0ZXh0IGVsZW1lbnRzCgo9b3ZlciA0Cgo9Y3V0Cgo9aXRlbSBuZXcoKQoKY3JlYXRlIGFuIG9iamVjdAoKICAgICRCYWNrZW5kT2JqZWN0ID0gS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydExheW91dFRleHQtPm5ldygKICAgICAgICAlUGFyYW0sCiAgICApOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGNoZWNrIG5lZWRlZCBvYmplY3RzCiAgICBmb3IgbXkgJE9iamVjdCAocXcoQ29uZmlnT2JqZWN0IExvZ09iamVjdCBNYWluT2JqZWN0IFBhcmFtT2JqZWN0IExheW91dE9iamVjdCkpIHsKICAgICAgICAkU2VsZi0+eyRPYmplY3R9ID0gJFBhcmFteyRPYmplY3R9IHx8IGRpZSAiR290IG5vICRPYmplY3QhIjsKICAgIH0KCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1pdGVtIEZvcm1JbnB1dENyZWF0ZSgpCgpjcmVhdGUgYSBpbnB1dCBzdHJpbmcKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+Rm9ybUlucHV0Q3JlYXRlKAogICAgICAgIEl0ZW0gICA9PiAkSXRlbVJlZiwKICAgICAgICBQcmVmaXggPT4gJ1ByZWZpeDo6JywgICMgKG9wdGlvbmFsKQogICAgICAgIFZhbHVlICA9PiAnVmFsdWUnLCAgICAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1JbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIEl0ZW0hJwogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICRQYXJhbXtQcmVmaXh9IHx8PSAnJzsKCiAgICBteSAkVmFsdWUgPSAkUGFyYW17VmFsdWV9IHx8ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1ZhbHVlRGVmYXVsdH07CiAgICBteSAkU2l6ZSA9ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1NpemV9IHx8IDQwOwogICAgbXkgJFNpemVDbGFzczsKICAgIGlmICggJFNpemUgPCAxNSApIHsKICAgICAgICAkU2l6ZUNsYXNzID0gJ1cxMHBjJzsKICAgIH0KICAgIGVsc2lmICggJFNpemUgPCAzNSApIHsKICAgICAgICAkU2l6ZUNsYXNzID0gJ1czM3BjJzsKICAgIH0KICAgIGVsc2lmICggJFNpemUgPCA1MCApIHsKICAgICAgICAkU2l6ZUNsYXNzID0gJ1c1MHBjJzsKICAgIH0KICAgIGVsc2UgewogICAgICAgICRTaXplQ2xhc3MgPSAnVzc1cGMnOwogICAgfQoKICAgICMgcHJlcGFyZSBkYXRhCiAgICBteSAkSUQgPSAoICRQYXJhbXtQcmVmaXh9IHx8ICcnICkgLiAoICRQYXJhbXtJdGVtfS0+e0tleX0gKTsKICAgIG15ICROYW1lID0gKCAkUGFyYW17UHJlZml4fSB8fCAnJyApIC4gKCAkUGFyYW17TmFtZX0gfHwgJElEICk7CiAgICBteSAkQ2xhc3MgPSAoICRTaXplQ2xhc3MgfHwgJycgKSAuICggJFBhcmFte0NsYXNzfSB8fCAnJyApOwoKICAgIG15ICRTdHJpbmcgPSAiPGlucHV0IGlkPVwiJElEXCIgdHlwZT1cInRleHRcIiBuYW1lPVwiJE5hbWVcIiBjbGFzcz1cIiRDbGFzc1wiICI7CgogICAgaWYgKCRWYWx1ZSkgewoKICAgICAgICAjIHRyYW5zbGF0ZQogICAgICAgIGlmICggJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VHJhbnNsYXRpb259ICkgewogICAgICAgICAgICAkVmFsdWUgPSAkU2VsZi0+e0xheW91dE9iamVjdH0tPntMYW5ndWFnZU9iamVjdH0tPkdldCgkVmFsdWUpOwogICAgICAgIH0KCiAgICAgICAgIyB0cmFuc2Zvcm0gYXNjaWkgdG8gaHRtbAogICAgICAgICRWYWx1ZSA9ICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+QXNjaWkySHRtbCgKICAgICAgICAgICAgVGV4dCAgICAgICAgICAgPT4gJFZhbHVlLAogICAgICAgICAgICBIVE1MUmVzdWx0TW9kZSA9PiAxLAogICAgICAgICk7CgogICAgICAgICRTdHJpbmcgLj0gInZhbHVlPVwiJFZhbHVlXCIgIjsKICAgIH0KCiAgICAjIGFkZCBtYXhpbXVtIGxlbmd0aAogICAgaWYgKCAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntNYXhMZW5ndGh9ICkgewogICAgICAgICRTdHJpbmcgLj0gIm1heGxlbmd0aD1cIiRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e01heExlbmd0aH1cIiAiOwogICAgfQoKICAgICRTdHJpbmcgLj0gIi8+ICI7CgogICAgcmV0dXJuICRTdHJpbmc7Cn0KCj1pdGVtIEZvcm1EYXRhR2V0KCkKCmdldCBmb3JtIGRhdGEKCiAgICBteSAkRm9ybURhdGEgPSAkQmFja2VuZE9iamVjdC0+Rm9ybURhdGFHZXQoCiAgICAgICAgSXRlbSAgID0+ICRJdGVtUmVmLAogICAgICAgIFByZWZpeCA9PiAnUHJlZml4OjonLCAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1EYXRhR2V0IHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgSXRlbSEnCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgJFBhcmFte1ByZWZpeH0gfHw9ICcnOwoKICAgICMgZ2V0IGZvcm0gZGF0YQogICAgbXkgJEZvcm1EYXRhID0gJFNlbGYtPntQYXJhbU9iamVjdH0tPkdldFBhcmFtKAogICAgICAgIFBhcmFtID0+ICRQYXJhbXtQcmVmaXh9IC4gJFBhcmFte0l0ZW19LT57S2V5fSwKICAgICk7CgogICAgIyByZWdleCBjaGVjawogICAgaWYgKCAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntSZWdleH0gJiYgJEZvcm1EYXRhICF+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1JlZ2V4fSApIHsKCiAgICAgICAgJFBhcmFte0l0ZW19LT57Rm9ybX0tPntJbnZhbGlkfSA9IDE7CiAgICAgICAgcmV0dXJuICRGb3JtRGF0YTsKICAgIH0KCiAgICByZXR1cm4gJEZvcm1EYXRhIGlmICRGb3JtRGF0YTsKICAgIHJldHVybiAkRm9ybURhdGEgaWYgISRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1JlcXVpcmVkfTsKCiAgICAjIHNldCBpbnZhbGlkIHBhcmFtCiAgICAkUGFyYW17SXRlbX0tPntGb3JtfS0+e0ludmFsaWR9ID0gMTsKCiAgICByZXR1cm4gJEZvcm1EYXRhOwp9CgoxOwoKPWJhY2sKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKEw8aHR0cHM6Ly9vdHJzLm9yZy8+KS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0Pi4KCj1jdXQK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXRJbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgo9b3ZlcgoKPWl0ZW0gSW1wb3J0RXhwb3J0Rm9ybUlucHV0Q3JlYXRlKCkKCnJldHVybnMgYSBpbnB1dCBmaWVsZCBodG1sIHN0cmluZwoKICAgIG15ICRTdHJpbmcgPSAkTGF5b3V0T2JqZWN0LT5JbXBvcnRFeHBvcnRGb3JtSW5wdXRDcmVhdGUoCiAgICAgICAgSXRlbSAgPT4gJEl0ZW1SZWYsCiAgICAgICAgVmFsdWUgPT4gJ1ZhbHVlJywgICAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgSW1wb3J0RXhwb3J0Rm9ybUlucHV0Q3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkU2VsZi0+e0xvZ09iamVjdH0tPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgSXRlbSEnCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBsb2FkIGJhY2tlbmQKICAgIG15ICRCYWNrZW5kT2JqZWN0ID0gJFNlbGYtPl9JbXBvcnRFeHBvcnRMb2FkTGF5b3V0QmFja2VuZCgKICAgICAgICBUeXBlID0+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1R5cGV9LAogICAgKTsKCiAgICByZXR1cm4gJycgaWYgISRCYWNrZW5kT2JqZWN0OwoKICAgICMgbG9va3VwIGl0ZW0gdmFsdWUKICAgIG15ICRTdHJpbmcgPSAkQmFja2VuZE9iamVjdC0+Rm9ybUlucHV0Q3JlYXRlKCVQYXJhbSk7CgogICAgcmV0dXJuICRTdHJpbmc7Cn0KCj1pdGVtIEltcG9ydEV4cG9ydEZvcm1EYXRhR2V0KCkKCnJldHVybnMgdGhlIHZhbHVlcyBmcm9tIHRoZSBodG1sIGZvcm0gYXMgaGFzaCByZWZlcmVuY2UKCiAgICBteSAkRm9ybURhdGEgPSAkTGF5b3V0T2JqZWN0LT5JbXBvcnRFeHBvcnRGb3JtRGF0YUdldCgKICAgICAgICBJdGVtID0+ICRJdGVtUmVmLAogICAgKTsKCj1jdXQKCnN1YiBJbXBvcnRFeHBvcnRGb3JtRGF0YUdldCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIEl0ZW0hJwogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICMgbG9hZCBiYWNrZW5kCiAgICBteSAkQmFja2VuZE9iamVjdCA9ICRTZWxmLT5fSW1wb3J0RXhwb3J0TG9hZExheW91dEJhY2tlbmQoCiAgICAgICAgVHlwZSA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntUeXBlfSwKICAgICk7CgogICAgcmV0dXJuIGlmICEkQmFja2VuZE9iamVjdDsKCiAgICAjIGdldCBmb3JtIGRhdGEKICAgIG15ICRGb3JtRGF0YSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtRGF0YUdldCglUGFyYW0pOwoKICAgIHJldHVybiAkRm9ybURhdGE7Cn0KCj1pdGVtIF9JbXBvcnRFeHBvcnRMb2FkTGF5b3V0QmFja2VuZCgpCgp0byBsb2FkIGEgaW1wb3J0L2V4cG9ydCBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCiAgICBteSAkQmFja2VuZCA9ICRMYXlvdXRPYmplY3QtPl9JbXBvcnRFeHBvcnRMb2FkTGF5b3V0QmFja2VuZCgKICAgICAgICBUeXBlID0+ICdTZWxlY3Rpb24nLAogICAgKTsKCkFuIGluc3RhbmNlIG9mIHRoZSBsb2FkZWQgYmFja2VuZCBtb2R1bGUgaXMgcmV0dXJuZWQuCgo9Y3V0CgpzdWIgX0ltcG9ydEV4cG9ydExvYWRMYXlvdXRCYWNrZW5kIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgaWYgKCAhJFBhcmFte1R5cGV9ICkgewogICAgICAgICRTZWxmLT57TG9nT2JqZWN0fS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAnTmVlZCBUeXBlIScsCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgbXkgJEdlbmVyaWNNb2R1bGUgPSAiS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydExheW91dCRQYXJhbXtUeXBlfSI7CgogICAgIyBsb2FkIHRoZSBiYWNrZW5kIG1vZHVsZQogICAgaWYgKCAhJFNlbGYtPntNYWluT2JqZWN0fS0+UmVxdWlyZSgkR2VuZXJpY01vZHVsZSkgKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJDYW4ndCBsb2FkIGJhY2tlbmQgbW9kdWxlICRQYXJhbXtUeXBlfSEiCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBjcmVhdGUgbmV3IGluc3RhbmNlCiAgICBteSAkQmFja2VuZE9iamVjdCA9ICRHZW5lcmljTW9kdWxlLT5uZXcoCiAgICAgICAgJXskU2VsZn0sCiAgICAgICAgJVBhcmFtLAogICAgICAgIExheW91dE9iamVjdCA9PiAkU2VsZiwKICAgICk7CgogICAgaWYgKCAhJEJhY2tlbmRPYmplY3QgKSB7CiAgICAgICAgJFNlbGYtPntMb2dPYmplY3R9LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICJDYW4ndCBjcmVhdGUgYSBuZXcgaW5zdGFuY2Ugb2YgYmFja2VuZCBtb2R1bGUgJFBhcmFte1R5cGV9ISIsCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgcmV0dXJuICRCYWNrZW5kT2JqZWN0Owp9CgoxOwoKPWJhY2sK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgojIENvbW1vbgpbJSBUcmFuc2xhdGUoIkFkZCBtYXBwaW5nIHRlbXBsYXRlIikgfCBodG1sICVdClslIFRyYW5zbGF0ZSgiQ2hhcnNldCIpIHwgaHRtbCAlXQpbJSBUcmFuc2xhdGUoIkNvbG9uICg6KSIpIHwgaHRtbCAlXQpbJSBUcmFuc2xhdGUoIkNvbHVtbiIpIHwgaHRtbCAlXQpbJSBUcmFuc2xhdGUoIkNvbHVtbiBTZXBhcmF0b3IiKSB8IGh0bWwgJV0KWyUgVHJhbnNsYXRlKCJEb3QgKC4pIikgfCBodG1sICVdClslIFRyYW5zbGF0ZSgiU2VtaWNvbG9uICg7KSIpIHwgaHRtbCAlXQpbJSBUcmFuc2xhdGUoIlRhYnVsYXRvciAoVEFCKSIpIHwgaHRtbCAlXQpbJSBUcmFuc2xhdGUoIkluY2x1ZGUgQ29sdW1uIEhlYWRlcnMiKSB8IGh0bWwgJV0KWyUgVHJhbnNsYXRlKCJJbXBvcnQgc3VtbWFyeSBmb3IiKSB8IGh0bWwgJV0KWyUgVHJhbnNsYXRlKCJJbXBvcnRlZCByZWNvcmRzIikgfCBodG1sICVdClslIFRyYW5zbGF0ZSgiRXhwb3J0ZWQgcmVjb3JkcyIpIHwgaHRtbCAlXQpbJSBUcmFuc2xhdGUoIlJlY29yZHMiKSB8IGh0bWwgJV0KWyUgVHJhbnNsYXRlKCJTa2lwcGVkIikgfCBodG1sICVdCg==
# --
# Copyright (C) 2001-2018 OTRS AG, https://otrs.com/
# --
# 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 https://www.gnu.org/licenses/gpl-3.0.txt.
# --

[% RenderBlockStart("Overview") %]
<div class="MainBox ARIARoleMain LayoutFixedSidebar SidebarFirst">
    <h1>[% Translate("Import/Export Management") | html %]</h1>

    <div class="SidebarColumn">

[% RenderBlockStart("ActionList") %]
        <div class="WidgetSimple">
            <div class="Header">
                <h2>[% Translate("Actions") | html %]</h2>
            </div>
            <div class="Content">
                <ul class="ActionList">
[% RenderBlockStart("ActionOverview") %]
                    <li>
                        <a href="[% Env("Baselink") %]Action=[% Env("Action") %]" class="CallForAction"><span>[% Translate("Go to overview") | html %]</span></a>
                    </li>
[% RenderBlockEnd("ActionOverview") %]
[% RenderBlockStart("ActionAdd") %]
                    <li>
                        <a href="[% Env("Baselink") %]Action=[% Env("Action") %];Subaction=TemplateEdit1" class="CallForAction Plus"><span>[% Translate("Add template") | html %]</span></a>
                    </li>
[% RenderBlockEnd("ActionAdd") %]
                </ul>
            </div>
        </div>
[% RenderBlockEnd("ActionList") %]
        <div class="WidgetSimple">
            <div class="Header">
                <h2>[% Translate("Note") | html %]</h2>
            </div>
            <div class="Content">
                <p class="FieldExplanation">
                    [% Translate("Create a template to import and export object information.") | html %]
                </p>
            </div>
        </div>
    </div>

    <div class="ContentColumn">
        <div class="WidgetSimple">
[% RenderBlockStart("OverviewList") %]
            <div class="Header">
                <h2>[% Translate(Data.ObjectName) | html %]</h2>
            </div>
            <div class="Content">
                <table class="DataTable">
                    <thead>
                        <tr>
                            <th>[% Translate("Number") | html %]</th>
                            <th>[% Translate("Name") | html %]</th>
                            <th>[% Translate("Format") | html %]</th>
                            <th>[% Translate("valid") | html %]/[% Translate("invalid") | html %]</th>
                            <th class="Center">[% Translate("Delete") | html %]</th>
                            <th>[% Translate("Start Import") | html %]</th>
                            <th>[% Translate("Start Export") | html %]</th>
                        </tr>
                    </thead>
                    <tbody>
[% RenderBlockStart("NoDataFoundMsg") %]
                        <tr>
                            <td colspan="7">
                                [% Translate("No data found.") | html %]
                            </td>
                        </tr>
[% RenderBlockEnd("NoDataFoundMsg") %]
[% RenderBlockStart("OverviewListRow") %]
                        <tr>
                            <td>
                                <a class="AsBlock" href="[% Env("Baselink") %]Action=[% Env("Action") %];Subaction=TemplateEdit1;TemplateID=[% Data.TemplateID | uri %]">
                                    [% Data.Number | html %]
                                </a>
                            </td>
                            <td>[% Data.Name | html %]</td>
                            <td>[% Translate(Data.FormatName) | html %]</td>
                            <td>[% Translate(Data.Valid) | html %]</td>
                            <td class="Center">
                                <a href="[% Env("Baselink") %]Action=[% Env("Action") %];Subaction=TemplateDelete;TemplateID=[% Data.TemplateID | uri %]" title="[% Translate("Delete") | html %]">
                                    <i id="DeleteTemplateID[% Data.TemplateID | html %]" class="fa fa-trash-o">
                                        <span class="InvisibleText">[% Translate("Delete") | html %]</span>
                                    </i>
                                </a>
                            </td>
                            <td>
                                <a href="[% Env("Baselink") %]Action=[% Env("Action") %];Subaction=ImportInformation;TemplateID=[% Data.TemplateID | uri %]">
                                    [% Translate("Import") | html %]
                                </a>
                            </td>
                            <td>
                                <a href="[% Env("Baselink") %]Action=[% Env("Action") %];Subaction=Export;TemplateID=[% Data.TemplateID | uri %]">
                                    [% Translate("Export") | html %]
                                </a>
                            </td>
                        </tr>
[% RenderBlockEnd("OverviewListRow") %]
                    </tbody>
                </table>
            </div>
[% RenderBlockEnd("OverviewList") %]

[% RenderBlockStart("TemplateEdit1") %]
            <div class="Header">
                <h2>[% Translate("Step") | html %] 1 [% Translate("of") | html %] 5 - [% Translate("Edit common information") | html %]:</h2>
            </div>
            <div class="Content">
                <form action="[% Env("CGIHandle") %]" method="post" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="[% Env("Action") %]" />
                    <input type="hidden" name="Subaction" value="TemplateEdit1" />
                    <input type="hidden" name="TemplateID" value="[% Data.TemplateID %]" />

                    <fieldset class="TableLike">

                        <label for="Name">[% Translate("Name") | html %]:</label>
                        <div class="Field">

<!-- Validate_Required -->
                            <input id="Name" class="[% Data.NameClass %]" type="text" name="Name" value="[% Data.Name | html %]" size="50" maxlength="200" />
                            <div id="NameError" class="TooltipErrorMessage">
                                <p>[% Translate("Name is required!") | html %]</p>
                            </div>
                            <div id="NameServerError" class="TooltipErrorMessage">
                                <p>[% Translate("Name is required!") | html %]</p>
                            </div>

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

[% RenderBlockStart("NewObjectFormat") %]
                        <label for="Object">[% Translate("Object") | html %]:</label>
                        <div class="Field">

                            [% Data.ObjectOption %]
                            <div id="ObjectError" class="TooltipErrorMessage">
                                <p>[% Translate("Object is required!") | html %]</p>
                            </div>
                            <div id="ObjectServerError" class="TooltipErrorMessage">
                                <p>[% Translate("Object is required!") | html %]</p>
                            </div>

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

                        <label for="Format">[% Translate("Format") | html %]:</label>
                        <div class="Field">

                            [% Data.FormatOption %]
                            <div id="FormatError" class="TooltipErrorMessage">
                                <p>[% Translate("Format is required!") | html %]</p>
                            </div>
                            <div id="FormatServerError" class="TooltipErrorMessage">
                                <p>[% Translate("Format is required!") | html %]</p>
                            </div>

                        </div>
[% RenderBlockEnd("NewObjectFormat") %]
[% RenderBlockStart("EditObjectFormat") %]
                        <label>[% Translate("Object") | html %]:</label>
                        <div class="Field">
                            <span>[% Data.ObjectName | html %]</span>
                            <input type="hidden" name="Object" value="[% Data.Object | html %]" />
                        </div>
                        <div class="Clear"></div>

                        <label>[% Translate("Format") | html %]:</label>
                        <div class="Field">
                            <span>[% Data.FormatName | html %]</span>
                            <input type="hidden" name="Format" value="[% Data.Format | html %]" />
                        </div>
[% RenderBlockEnd("EditObjectFormat") %]

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

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

                        <label for="Comment">[% Translate("Comment") | html %]:</label>
                        <div class="Field">
                            <input id="Comment" type="text" name="Comment" value="[% Data.Comment | html %]" size="50" maxlength="200" />
                        </div>
                        <div class="Clear"></div>

                        <div class="Field SpacingTop">
                            <button class="Primary" type="submit" name="SubmitNext" value="SubmitNext">[% Translate("Next") | html %]</button>
                            [% Translate("or") | html %]
                            <a href="[% Env("Baselink") %]Action=[% Env("Action") %];Subaction='Overview'">[% Translate("Cancel") | html %] </a>
                        </div>
                    </fieldset>
                </form>
            </div>
[% RenderBlockEnd("TemplateEdit1") %]

[% RenderBlockStart("TemplateEdit2") %]
            <div class="Header">
                <h2>[% Translate("Step") | html %] 2 [% Translate("of") | html %] 5 - [% Translate("Edit object information") | html %]:</h2>
            </div>
            <div class="Content">
                <form action="[% Env("CGIHandle") %]" method="post" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="[% Env("Action") %]" />
                    <input type="hidden" name="Subaction" value="TemplateEdit2" />
                    <input type="hidden" name="TemplateID" value="[% Data.TemplateID | html %]" />

                    <fieldset class="TableLike">

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

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

[% RenderBlockStart("TemplateEdit2Element") %]
                        <label for="[% Data.ID %]">[% Translate(Data.Name) | html %]: </label>
                        <div class="Field">
                            [% Data.InputStrg %]
                            <div id="[% Data.ID %]Error" class="TooltipErrorMessage">
                                <p>[% Translate(Data.ErrorMessage) | html %]</p>
                            </div>
                            <div id="[% Data.ID %]ServerError" class="TooltipErrorMessage">
                                <p>[% Translate(Data.ErrorMessage) | html %]</p>
                            </div>
                        </div>
[% RenderBlockEnd("TemplateEdit2Element") %]
                        <div class="Field SpacingTop">
                            <button class="Back" type="button" name="Back" >[% Translate("Back") | html %]</button>
                            <button class="Primary" type="submit" name="SubmitNext" value="SubmitNext" >[% Translate("Next") | html %]</button>
                        </div>
                    </fieldset>
                </form>
            </div>
[% RenderBlockEnd("TemplateEdit2") %]

[% RenderBlockStart("TemplateEdit3") %]
            <div class="Header">
                <h2>[% Translate("Step") | html %] 3 [% Translate("of") | html %] 5 - [% Translate("Edit format information") | html %]:</h2>
            </div>
            <div class="Content">
                <form action="[% Env("CGIHandle") %]" method="post" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="[% Env("Action") %]" />
                    <input type="hidden" name="Subaction" value="TemplateEdit3" />
                    <input type="hidden" name="TemplateID" value="[% Data.TemplateID | html %]" />

                    <fieldset class="TableLike">

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

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

[% RenderBlockStart("TemplateEdit3Element") %]
                        <label for="[% Data.ID %]">[% Translate(Data.Name) | html %]: </label>
                        <div class="Field">
                            [% Data.InputStrg %]
[% RenderBlockStart("TemplateEdit3ElementRequired") %]
                            <div id="[% Data.ID %]Error" class="TooltipErrorMessage">
                                <p>[% Translate(Data.Name) | html %] [% Translate("is required!") | html %]</p>
                            </div>
                            <div id="[% Data.ID %]ServerError" class="TooltipErrorMessage">
                                <p>[% Translate(Data.Name) | html %] [% Translate("is required!") | html %]</p>
                            </div>
[% RenderBlockEnd("TemplateEdit3ElementRequired") %]
                        </div>
[% RenderBlockEnd("TemplateEdit3Element") %]
                        <div class="Field SpacingTop">
                            <button class="Back" type="button" name="Back" >[% Translate("Back") | html %]</button>
                            <button class="Primary" type="submit" name="SubmitNext" value="SubmitNext">[% Translate("Next") | html %]</button>
                        </div>
                    </fieldset>
                </form>
            </div>
[% RenderBlockEnd("TemplateEdit3") %]

[% RenderBlockStart("TemplateEdit4") %]
            <div class="Header">
                <h2>[% Translate("Step") | html %] 4 [% Translate("of") | html %] 5 - [% Translate("Edit mapping information") | html %]:</h2>
            </div>
            <div class="Content">
                <div class="MapHeaderRow SpacingTop">
                    <div class="Header">
                        <label>[% Translate("Name") | html %]:</label>
                        <div class="Field">
                            [% Data.Name | html %]
                        </div>
                    </div>

                    <div class="Header">
                        <label>[% Translate("Object") | html %]:</label>
                        <div class="Field">
                            [% Data.ObjectName | html %]
                        </div>
                    </div>

                    <div class="Header">
                        <label>[% Translate("Format") | html %]:</label>
                        <div class="Field">
                            [% Data.FormatName | html %]
                        </div>
                    </div>
                </div>
                <form action="[% Env("CGIHandle") %]" method="post" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="[% Env("Action") %]" />
                    <input type="hidden" name="Subaction" value="TemplateSave4" />
                    <input type="hidden" name="TemplateID" value="[% Data.TemplateID | html %]" />
                    <input type="hidden" name="MappingAdd" value="" />
                    <input type="hidden" name="SubmitNext" value="" />
                    <table class="DataTable SpacingTop">
                        <thead>
                            <tr>
[% RenderBlockStart("TemplateEdit4TableHeader") %]
                                <th class="Center">[% Translate(Data.Header) | html %]</th>
[% RenderBlockEnd("TemplateEdit4TableHeader") %]
                            </tr>
                        </thead>
                        <tbody>
[% RenderBlockStart("TemplateEdit4NoMapFound") %]
                            <tr>
                                <td colspan="[% Data.Columns %]">
                                    [% Translate("No map elements found.") | html %]
                                </td>
                            </tr>
[% RenderBlockEnd("TemplateEdit4NoMapFound") %]

[% RenderBlockStart("TemplateEdit4Row") %]
                            <tr>
[% RenderBlockStart("TemplateEdit4Column") %]
                                <td class="Center">
                                    [% Data.InputStrg %]
                                </td>
[% RenderBlockEnd("TemplateEdit4Column") %]
[% RenderBlockStart("TemplateEdit4MapNumberColumn") %]
                                <td class="Center">
                                    [% Data.Counter %]
                                </td>
[% RenderBlockEnd("TemplateEdit4MapNumberColumn") %]
                                <td class="Center">
[% RenderBlockStart("TemplateEdit4UpButton") %]
                                    <button class="ArrowUp"type="submit" name="MappingUp::[% Data.MappingID | html %]" value="[% Translate("Up") | html %]"> [% Translate("Up") | html %] </button>
[% RenderBlockEnd("TemplateEdit4UpButton") %]
[% RenderBlockStart("TemplateEdit4NoUpButton") %]
                                    <button class="ArrowUp" type="submit" disabled="disabled"> [% Translate("Up") | html %]</button>
[% RenderBlockEnd("TemplateEdit4NoUpButton") %]
                                </td>
                                <td class="Center">
[% RenderBlockStart("TemplateEdit4DownButton") %]
                                    <button class="ArrowDown" type="submit" name="MappingDown::[% Data.MappingID | html %]" value="[% Translate("Down") | html %]"> [% Translate("Down") | html %]</button>
[% RenderBlockEnd("TemplateEdit4DownButton") %]
[% RenderBlockStart("TemplateEdit4NoDownButton") %]
                                    <button class="ArrowDown" type="submit" disabled="disabled"> [% Translate("Down") | html %]</button>
[% RenderBlockEnd("TemplateEdit4NoDownButton") %]
                                </td>

                                <td class="Center">
                                    <a href="#" class="DeleteColumn" title="[% Translate("Delete") | html %]">
                                        <i class="fa fa-trash-o">
                                            <span class="InvisibleText">[% Translate("Delete") | html %]</span>
                                        </i>
                                    </a>
                                    <input type="hidden" name="MappingDelete::[% Data.MappingID | html %]" value="" />
                                </td>

                            </tr>
[% RenderBlockEnd("TemplateEdit4Row") %]
                        </tbody>
                    </table>
                    <div class="W100pc SpacingTopSmall Left">
                        <button class="CallForAction Plus" id="MappingAddButton" type="submit" name="MappingAddButton" value="[% Translate("Add") | html %]"><span> [% Translate("Add Mapping Element") | html %]</span> </button>
                    </div>

                    <div class="SpacingTop">
                        <button type="submit" name="SubmitBack" value="SubmitBack">[% Translate("Back") | html %]</button>
                        <button class="Primary" type="submit" name="SubmitNextButton" id="SubmitNextButton" value="1">[% Translate("Next") | html %]</button>
                    </div>
                </form>
            </div>

[% WRAPPER JSOnDocumentComplete %]
<script type="text/javascript">//<![CDATA[

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

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

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

    }).trigger('change');

    // set the hidden field to delete this column and submit the form
    $('.DeleteColumn').unbind('click').bind('click', function(Event) {
        $(this).closest('td').find('input[type="hidden"]').val(1);
        $(this).closest('form').submit();
        return true;
    });

    $('#MappingAddButton').bind('click', function (Event) {
        $('input[name=MappingAdd]').val('1');
        $('input[name=SubmitNext]').val('0');
    });

    $('#SubmitNextButton').bind('click', function (Event) {
        $('input[name=MappingAdd]').val('0');
        $('input[name=SubmitNext]').val('1');
    });

//]]></script>
[% END %]

[% RenderBlockEnd("TemplateEdit4") %]

[% RenderBlockStart("TemplateEdit5") %]
            <div class="Header">
                <h2>[% Translate("Step") | html %] 5 [% Translate("of") | html %] 5 - [% Translate("Edit search information") | html %]:</h2>
            </div>
            <div class="Content">
                <form action="[% Env("CGIHandle") %]" method="post" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="[% Env("Action") %]" />
                    <input type="hidden" name="Subaction" value="TemplateSave5" />
                    <input type="hidden" name="TemplateID" value="[% Data.TemplateID | html %]" />
                    <fieldset class="TableLike">
                        <label>[% Translate("Template Name") | html %]:</label>
                        <div class="Field">
                            [% Data.Name | html %]
                        </div>
                        <div class="Clear"></div>

                        <label for="RestrictExport">[% Translate("Restrict export per search") | html %]:</label>
                        <div class="Field SpacingBottom">
                            [% Data.RestrictExportStrg %]
                        </div>
                        <div class="Clear"></div>

[% RenderBlockStart("TemplateEdit5Element") %]
                        <label for="[% Data.ID %]">[% Translate(Data.Name) | html %]: </label>
                        <div class="Field">
                            [% Data.InputStrg %]
                        </div>
[% RenderBlockEnd("TemplateEdit5Element") %]

                        <div class="Field SpacingTop">
                            <button type="submit" name="SubmitBack" value="SubmitBack">[% Translate("Back") | html %]</button>
                            <button class="Primary" type="submit" name="SubmitNext" value="SubmitNext">[% Translate("Finish") | html %]</button>
                        </div>
                    </fieldset>
                </form>
            </div>
[% RenderBlockEnd("TemplateEdit5") %]

[% RenderBlockStart("ImportInformation") %]
            <div class="Header">
                <h2>[% Translate("Import information") | html %]:</h2>
            </div>
            <div class="Content">
                <form action="[% Env("CGIHandle") %]" method="post" enctype="multipart/form-data" class="Validate PreventMultipleSubmits">
                    <input type="hidden" name="Action" value="[% Env("Action") %]" />
                    <input type="hidden" name="Subaction" value="Import" />
                    <input type="hidden" name="TemplateID" value="[% Data.TemplateID | html %]" />

                    <fieldset class="TableLike">

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

                        <label for="SourceFile">[% Translate("Source File") | html %]:</label>
                        <div class="Field">
                            <input type="file" name="SourceFile" size="40" class="fixed" />
                        </div>
                        <div class="Clear"></div>

                        <div class="Field SpacingTop">
                            <button class="Primary" type="submit" value="[% Translate("Start Import") | html %]">[% Translate("Start Import") | html %]</button>
                        </div>
                    </fieldset>
                </form>
            </div>
[% RenderBlockEnd("ImportInformation") %]
        </div>
    </div>

[% WRAPPER JSOnDocumentComplete %]
    <script type="text/javascript">//<![CDATA[
        $('button.Back').bind('click', function () {
            location.href = "[% Env("Baselink") %][% Data.BackURL | html %]";
        });
    //]]></script>
[% END %]

[% WRAPPER JSOnDocumentComplete %]
<script type="text/javascript">//<![CDATA[

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

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

    });

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

    });

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

//]]></script>
[% END %]

</div>

[% RenderBlockEnd("Overview") %]

[% RenderBlockStart("ImportResult") %]
<div class="MainBox AriaRoleMain">
    <div class="W50pc SpacingTopLarge SpacingBottomLarge CenterBox">
        <div class="WidgetSimple">
            <div class="Header">
                <h2>[% Translate("Import summary for") | html %] [% Data.Object | html %]</h2>
            </div>
            <div class="Content">

                <form action="[% Env("CGIHandle") %]" method="post" enctype="multipart/form-data">
                    <input type="hidden" name="Action" value="[% Env("Action") %]"/>
                    <input type="hidden" name="Subaction" value="Overview"/>

                    <fieldset class="TableLike">
                        <label>[% Translate("Records") | html %]:</label>
                        <div class="Value">[% Data.Counter | html %]</div>
                        <div class="Clear"></div>

                        <label>[% Translate("Success") | html %]:</label>
                        <div class="Value">
                            [% Data.Success | html %]
[% RenderBlockStart("ImportResultReturnCode") %]
                            ([% Translate(Data.ReturnCodeName) | html %]: [% Data.ReturnCodeCount | html %])
[% RenderBlockEnd("ImportResultReturnCode") %]
                        </div>
                        <div class="Clear"></div>

                        <label>[% Translate("Failed") | html %]:</label>
                        <div class="Value">[% Data.Failed | html %]</div>
                        <div class="Clear"></div>

[% RenderBlockStart("ImportResultDuplicateNames") %]
                        <label>[% Translate("Duplicate names") | html %]:</label>
                        <div class="Value">[% Data.DuplicateNames | html %]</div>
                        <div class="Clear"></div>
[% RenderBlockEnd("ImportResultDuplicateNames") %]

[% RenderBlockStart("ImportResultLastLineNumber") %]
                        <label>[% Translate("Last processed line number of import file") | html %]:</label>
                        <div class="Value">[% Data.LastLineNumber | html %]</div>
                        <div class="Clear"></div>
[% RenderBlockEnd("ImportResultLastLineNumber") %]
                    </fieldset>

                    <p class="Center SpacingTopSmall">
                        <button type="submit" name="Ok" value="[% Translate("Ok") | html %]">[% Translate("Ok") | html %]</button>
                    </p>
                </form>
            </div>
        </div>
    </div>
</div>
[% RenderBlockEnd("ImportResult") %]

# --
# Copyright (C) 2001-2018 OTRS AG, https://otrs.com/
# --
# 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 https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::System::ImportExport;

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::CheckItem',
    'Kernel::System::DB',
    'Kernel::System::Log',
    'Kernel::System::Main',
);

=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::System::ObjectManager;
    local $Kernel::OM = Kernel::System::ObjectManager->new();
    my $ImportExportObject = $Kernel::OM->Get('Kernel::System::ImportExport');

=cut

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

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

    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} ) {
        $Kernel::OM->Get('Kernel::System::Log')->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
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL  => $SQL,
        Bind => \@BIND,
    );

    # fetch the result
    my @TemplateList;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->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
    $Kernel::OM->Get('Kernel::System::DB')->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 = $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

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

    # cleanup given params
    for my $Argument (qw(Object Format)) {
        $Kernel::OM->Get('Kernel::System::CheckItem')->StringClean(
            StringRef         => \$Param{$Argument},
            RemoveAllNewlines => 1,
            RemoveAllTabs     => 1,
            RemoveAllSpaces   => 1,
        );
    }
    for my $Argument (qw(Name Comment)) {
        $Kernel::OM->Get('Kernel::System::CheckItem')->StringClean(
            StringRef         => \$Param{$Argument},
            RemoveAllNewlines => 1,
            RemoveAllTabs     => 1,
        );
    }

    # find exiting template with same name
    $Kernel::OM->Get('Kernel::System::DB')->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 ( $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $NoAdd = 1;
    }

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

    # insert new template
    return if !$Kernel::OM->Get('Kernel::System::DB')->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
    $Kernel::OM->Get('Kernel::System::DB')->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 = $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

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

    # cleanup given params
    for my $Argument (qw(Name Comment)) {
        $Kernel::OM->Get('Kernel::System::CheckItem')->StringClean(
            StringRef         => \$Param{$Argument},
            RemoveAllNewlines => 1,
            RemoveAllTabs     => 1,
        );
    }

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

    # fetch the result
    my $Object;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $Object = $Row[0];
    }

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

    # find exiting template with same name
    $Kernel::OM->Get('Kernel::System::DB')->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 = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        if ( $Param{TemplateID} ne $Row[0] ) {
            $Update = 0;
        }
    }

    if ( !$Update ) {
        $Kernel::OM->Get('Kernel::System::Log')->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 $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    if ( !ref $Param{TemplateID} ) {
        $Param{TemplateID} = [ $Param{TemplateID} ];
    }
    elsif ( ref $Param{TemplateID} ne 'ARRAY' ) {
        $Kernel::OM->Get('Kernel::System::Log')->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 $Kernel::OM->Get('Kernel::System::DB')->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 = $Kernel::OM->Get('Kernel::Config')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->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} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Template with ID $Param{TemplateID} is incomplete!",
        );
        return;
    }

    # load backend
    my $Backend = $Kernel::OM->Get(
        'Kernel::System::ImportExport::ObjectBackend::' . $TemplateData->{Object}
    );

    return if !$Backend;

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

    return $Attributes;
}

=item ObjectDataGet()

get the object data from a template

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

=cut

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

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

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL  => 'SELECT data_key, data_value FROM imexport_object WHERE template_id = ?',
        Bind => [ \$Param{TemplateID} ],
    );

    # fetch the result
    my %ObjectData;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

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

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

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

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

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

        # insert one row
        $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    if ( !ref $Param{TemplateID} ) {
        $Param{TemplateID} = [ $Param{TemplateID} ];
    }
    elsif ( ref $Param{TemplateID} ne 'ARRAY' ) {
        $Kernel::OM->Get('Kernel::System::Log')->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 $Kernel::OM->Get('Kernel::System::DB')->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 = $Kernel::OM->Get('Kernel::Config')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->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} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Template with ID $Param{TemplateID} is incomplete!",
        );
        return;
    }

    # load backend
    my $Backend = $Kernel::OM->Get(
        'Kernel::System::ImportExport::FormatBackend::' . $TemplateData->{Format}
    );

    return if !$Backend;

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

    return $Attributes;
}

=item FormatDataGet()

get the format data from a template

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

=cut

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

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

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL  => 'SELECT data_key, data_value FROM imexport_format WHERE template_id = ?',
        Bind => [ \$Param{TemplateID} ],
    );

    # fetch the result
    my %FormatData;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

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

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

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

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

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

        # insert one row
        $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    if ( !ref $Param{TemplateID} ) {
        $Param{TemplateID} = [ $Param{TemplateID} ];
    }
    elsif ( ref $Param{TemplateID} ne 'ARRAY' ) {
        $Kernel::OM->Get('Kernel::System::Log')->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 $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

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

    # fetch the result
    my @MappingList;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    # find maximum position
    $Kernel::OM->Get('Kernel::System::DB')->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 = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {

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

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

    # find id of new mapping data row
    $Kernel::OM->Get('Kernel::System::DB')->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 = $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->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
        $Kernel::OM->Get('Kernel::System::DB')->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 $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->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
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL  => 'SELECT position FROM imexport_mapping WHERE id = ?',
        Bind => [ \$Param{MappingID} ],
    );

    # fetch the result
    my $Position;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $Position = $Row[0];
    }

    return 1 if !$Position;

    my $PositionUpper = $Position - 1;

    # update positions
    $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL  => 'UPDATE imexport_mapping SET position = ? WHERE template_id = ? AND position = ?',
        Bind => [ \$Position, \$Param{TemplateID}, \$PositionUpper ],
    );
    $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->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
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL  => 'SELECT position FROM imexport_mapping WHERE id = ?',
        Bind => [ \$Param{MappingID} ],
    );

    # fetch the result
    my $Position;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->FetchrowArray() ) {
        $Position = $Row[0];
    }

    my $PositionDown = $Position + 1;

    # update positions
    $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL  => 'UPDATE imexport_mapping SET position = ? WHERE template_id = ? AND position = ?',
        Bind => [ \$Position, \$Param{TemplateID}, \$PositionDown ],
    );
    $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->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} ) {
        $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->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} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Template with ID $Param{TemplateID} is incomplete!",
        );
        return;
    }

    # load backend
    my $Backend = $Kernel::OM->Get(
        'Kernel::System::ImportExport::ObjectBackend::' . $TemplateData->{Object}
    );

    return if !$Backend;

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

    return $Attributes;
}

=item MappingObjectDataDelete()

delete the existing object data of a mapping

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

    or

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

=cut

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

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

    if ( !ref $Param{MappingID} ) {
        $Param{MappingID} = [ $Param{MappingID} ];
    }
    elsif ( ref $Param{MappingID} ne 'ARRAY' ) {
        $Kernel::OM->Get('Kernel::System::Log')->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 $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

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

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

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

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

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

        # insert one mapping object row
        $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->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 = $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->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} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Template with ID $Param{TemplateID} is incomplete!",
        );
        return;
    }

    # load backend
    my $Backend = $Kernel::OM->Get(
        'Kernel::System::ImportExport::FormatBackend::' . $TemplateData->{Format}
    );

    return if !$Backend;

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

    return $Attributes;
}

=item MappingFormatDataDelete()

delete the existing format data of a mapping

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

    or

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

=cut

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

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

    if ( !ref $Param{MappingID} ) {
        $Param{MappingID} = [ $Param{MappingID} ];
    }
    elsif ( ref $Param{MappingID} ne 'ARRAY' ) {
        $Kernel::OM->Get('Kernel::System::Log')->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 $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

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

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

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

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

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

        # insert one mapping format row
        $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->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 = $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->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} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Template with ID $Param{TemplateID} is incomplete!",
        );
        return;
    }

    # load backend
    my $Backend = $Kernel::OM->Get(
        'Kernel::System::ImportExport::ObjectBackend::' . $TemplateData->{Object}
    );

    return if !$Backend;

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

    return $Attributes;
}

=item SearchDataGet()

get the search data from a template

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

=cut

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

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

    # ask database
    $Kernel::OM->Get('Kernel::System::DB')->Prepare(
        SQL  => 'SELECT data_key, data_value FROM imexport_search WHERE template_id = ?',
        Bind => [ \$Param{TemplateID} ],
    );

    # fetch the result
    my %SearchData;
    while ( my @Row = $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

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

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

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

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

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

        # insert one row
        $Kernel::OM->Get('Kernel::System::DB')->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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

    if ( !ref $Param{TemplateID} ) {
        $Param{TemplateID} = [ $Param{TemplateID} ];
    }
    elsif ( ref $Param{TemplateID} ne 'ARRAY' ) {
        $Kernel::OM->Get('Kernel::System::Log')->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 $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL  => "DELETE FROM imexport_search WHERE template_id IN ( $TemplateIDString )",
        Bind => \@BIND,
    );
}

=item Export()

export function

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

returns something like

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

=cut

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

    # check needed stuff
    for my $Argument (qw(TemplateID UserID)) {
        if ( !$Param{$Argument} ) {
            $Kernel::OM->Get('Kernel::System::Log')->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} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Template with ID $Param{TemplateID} is incomplete!",
        );
        return;
    }

    # load object backend
    my $ObjectBackend = $Kernel::OM->Get(
        'Kernel::System::ImportExport::ObjectBackend::' . $TemplateData->{Object}
    );

    return if !$ObjectBackend;

    # load format backend
    my $FormatBackend = $Kernel::OM->Get(
        'Kernel::System::ImportExport::FormatBackend::' . $TemplateData->{Format}
    );

    return if !$FormatBackend;

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

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

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

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

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

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

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

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

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

            push @ColumnNames, $ColumnName;
        }

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

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

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

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

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

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

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

    return \%Result;
}

=item Import()

import function

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

=cut

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

    # check needed stuff
    for my $Argument (qw(TemplateID UserID)) {
        if ( !$Param{$Argument} ) {
            $Kernel::OM->Get('Kernel::System::Log')->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} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "Template with ID $Param{TemplateID} is incomplete!",
        );
        return;
    }

    # load object backend
    my $ObjectBackend = $Kernel::OM->Get(
        'Kernel::System::ImportExport::ObjectBackend::' . $TemplateData->{Object}
    );

    return if !$ObjectBackend;

    # load format backend
    my $FormatBackend = $Kernel::OM->Get(
        'Kernel::System::ImportExport::FormatBackend::' . $TemplateData->{Format}
    );

    return if !$FormatBackend;

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

    return if !$ImportData;

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

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

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

        $Result{Counter}++;

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

        if ( !$ID ) {

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

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

    return \%Result;
}

1;

=back

=head1 TERMS AND CONDITIONS

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

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

=cut

# --
# Copyright (C) 2001-2018 OTRS AG, https://otrs.com/
# --
# 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 https://www.gnu.org/licenses/gpl-3.0.txt.
# --

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

use strict;
use warnings;

our @ObjectDependencies = (
    'Kernel::System::ImportExport',
    'Kernel::System::Log',
    'Kernel::System::Main',
);

=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::System::ObjectManager;
    local $Kernel::OM = Kernel::System::ObjectManager->new();
    my $ImportExportCSVBackendObject = $Kernel::OM->Get('Kernel::System::ImportExport::FormatBackend::CSV');

=cut

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

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

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

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

    return $Self;
}

=item FormatAttributesGet()

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

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

=cut

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

    # check needed stuff
    if ( !$Param{UserID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need UserID!',
        );
        return;
    }

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

    return $Attributes;
}

=item MappingFormatAttributesGet()

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

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

=cut

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

    # check needed object
    if ( !$Param{UserID} ) {
        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => 'Need UserID!',
        );
        return;
    }

    my $Attributes = [
        {
            Key   => 'Column',
            Name  => 'Column',
            Input => {
                Type     => 'TT',
                Data     => '',
                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} ) {
            $Kernel::OM->Get('Kernel::System::Log')->Log(
                Priority => 'error',
                Message  => "Need $Argument!",
            );
            return;
        }
    }

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

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

    # get format data
    my $FormatData = $Kernel::OM->Get('Kernel::System::ImportExport')->FormatDataGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # check format data
    if ( !$FormatData || ref $FormatData ne 'HASH' ) {
        $Kernel::OM->Get('Kernel::System::Log')->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 ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No valid charset found for the template id $Param{TemplateID}",
        );
        return;
    }

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

    # check the separator
    if ( !$Separator ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No valid separator found for the template id $Param{TemplateID}",
        );
        return;
    }

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

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

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

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

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

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

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

    # close the in memory file handle
    close $FH;

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

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

    return \@ImportData;
}

=item ExportDataSave()

export one row of the export data

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

=cut

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

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

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

    # get format data
    my $FormatData = $Kernel::OM->Get('Kernel::System::ImportExport')->FormatDataGet(
        TemplateID => $Param{TemplateID},
        UserID     => $Param{UserID},
    );

    # check format data
    if ( !$FormatData || ref $FormatData ne 'HASH' ) {
        $Kernel::OM->Get('Kernel::System::Log')->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 ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No valid charset found for the template id $Param{TemplateID}",
        );
        return;
    }

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

    # check the separator
    if ( !$Separator ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No valid separator found for the template id $Param{TemplateID}",
        );
        return;
    }

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

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

        $Kernel::OM->Get('Kernel::System::Log')->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 (L<https://otrs.org/>).

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

=cut

# --
# Copyright (C) 2001-2018 OTRS AG, https://otrs.com/
# --
# 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 https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

my $ConfigObject       = $Kernel::OM->Get('Kernel::Config');
my $UserObject         = $Kernel::OM->Get('Kernel::System::User');
my $ImportExportObject = $Kernel::OM->Get('Kernel::System::ImportExport');

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# create needed users
my @UserIDs;
{

    # disable email checks to create new user
    my $CheckEmailAddressesOrg =
        $ConfigObject->Get('CheckEmailAddresses') || 1;

    $ConfigObject->Set(
        Key   => 'CheckEmailAddresses',
        Value => 0,
    );

    for my $Counter ( 1 .. 2 ) {

        # create new users for the tests
        my $UserID = $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

    $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 = $ImportExportObject->TemplateList(
    UserID => 1,
);

# get original template list for later checks (all elements)
my $TemplateList1Object = $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 = $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 = $ImportExportObject->TemplateGet(
            TemplateID => $AddedTemplateIDs[-1],
            UserID     => $Item->{Add}->{UserID} || 1,
        );

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

    if ( $Item->{Update} ) {

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

        # update the template
        my $UpdateSucess = $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 = $ImportExportObject->TemplateGet(
            TemplateID => $AddedTemplateIDs[-1],
            UserID     => $Item->{Update}->{UserID} || 1,
        );

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

    # increment the counter
    $TestCount++;
}

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

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

$TestCount++;

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

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

$TestCount++;

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

# get template list with all elements
my $TemplateList2 = $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 = $ImportExportObject->TemplateList(
    Object => $ObjectName[0],
    UserID => 1,
);

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

# get template list with all elements
my $TemplateDelete1List2 = $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 = $ImportExportObject->TemplateDelete(
    TemplateID => $TemplateDeleteID,
    UserID     => 1,
);

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

# get template list with all elements
my $TemplateDelete1List3 = $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 = $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 =
    $ConfigObject->Get('ImportExport::ObjectBackendRegistration');

# set test list

$ConfigObject->Set(
    Key   => 'ImportExport::ObjectBackendRegistration',
    Value => $ObjectList1TestList,
);

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

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

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

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

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

    delete $ObjectList1TestList->{$Key};
}

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

# restore original object list

$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 =
    $ConfigObject->Get('ImportExport::FormatBackendRegistration');

# set test list

$ConfigObject->Set(
    Key   => 'ImportExport::FormatBackendRegistration',
    Value => $FormatList1TestList,
);

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

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

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

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

    next KEY if $FormatList1->{$Key} ne $FormatList1TestList->{$Key}->{Name};

    delete $FormatList1TestList->{$Key};
}

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

# restore original format list

$ConfigObject->Set(
    Key   => 'ImportExport::FormatBackendRegistration',
    Value => $FormatListOrg,
);

$TestCount++;

1;

# --
# Copyright (C) 2001-2018 OTRS AG, https://otrs.com/
# --
# 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 https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

use Data::Dumper;

my $ConfigObject        = $Kernel::OM->Get('Kernel::Config');
my $MainObject          = $Kernel::OM->Get('Kernel::System::Main');
my $ImportExportObject  = $Kernel::OM->Get('Kernel::System::ImportExport');
my $FormatBackendObject = $Kernel::OM->Get('Kernel::System::ImportExport::FormatBackend::CSV');

# ------------------------------------------------------------ #
# make preparations
# ------------------------------------------------------------ #

# get home directory
$Self->{Home} = $ConfigObject->Get('Home');

# add some test templates for later checks
my @TemplateIDs;
for ( 1 .. 30 ) {

    # add a test template for later checks
    my $TemplateID = $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 = $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 = $ImportExportObject->FormatAttributesGet(
    TemplateID => $TemplateIDs[0],
    UserID     => 1,
);

# check format attribute reference
$Self->True(
    $FormatAttributesGet1 && ref $FormatAttributesGet1 eq 'ARRAY',
    "Test $TestCount: FormatAttributesGet() - check array reference",
);

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

$Self->IsDeeply(
    $FormatAttributesGet1,
    $FormatAttributesGet1Reference,
    "Test $TestCount: FormatAttributesGet() - attributes of the row are identical",
);

$TestCount++;

# ------------------------------------------------------------ #
# FormatAttributesGet test 2 (check with non existing template)
# ------------------------------------------------------------ #

# get format attributes
my $FormatAttributesGet2 = $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 = $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     => 'TT',
            Data     => '',
            Required => 0,
        },
    },
];

$Self->IsDeeply(
    $MappingFormatAttributesGet1,
    $MappingFormatAttributesGet1Reference,
    "Test $TestCount: MappingFormatAttributesGet() - attributes of the row are identical",
);

$TestCount++;

# ------------------------------------------------------------ #
# MappingFormatAttributesGet test 2 (check with non existing template)
# ------------------------------------------------------------ #

# get mapping format attributes
my $MappingFormatAttributesGet2 = $ImportExportObject->MappingFormatAttributesGet(
    TemplateID => $TemplateIDs[-1] + 1,
    UserID     => 1,
);

# check false return
$Self->False(
    $MappingFormatAttributesGet2,
    "Test $TestCount: MappingFormatAttributesGet() - check false return",
);

$TestCount++;

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

my $ImportDataTests = [

    # ImportDataGet doesn't contains all data (check required attributes)
    {
        SourceImportData => {
            ImportDataGet => {
                UserID => 1,
            },
        },
    },

    # ImportDataGet doesn't contains all data (check required attributes)
    {
        SourceImportData => {
            ImportDataGet => {
                TemplateID => $TemplateIDs[1],
            },
        },
    },

    # no source content are given (empty array reference must be returned)
    {
        SourceImportData => {
            ImportDataGet => {
                TemplateID => $TemplateIDs[1],
                UserID     => 1,
            },
        },
        ReferenceImportData => [],
    },

    # source content must be a scalar reference (check return false)
    {
        SourceImportData => {
            ImportDataGet => {
                TemplateID    => $TemplateIDs[1],
                SourceContent => [],
                UserID        => 1,
            },
        },
    },

    # source content must be a scalar reference (check return false)
    {
        SourceImportData => {
            ImportDataGet => {
                TemplateID    => $TemplateIDs[1],
                SourceContent => {},
                UserID        => 1,
            },
        },
    },

    # source content must be a scalar reference (check return false)
    {
        SourceImportData => {
            ImportDataGet => {
                TemplateID    => $TemplateIDs[1],
                SourceContent => '',
                UserID        => 1,
            },
        },
    },

    # no existing template id is given (check return false)
    {
        SourceImportData => {
            ImportDataGet => {
                TemplateID    => $TemplateIDs[-1] + 1,
                SourceContent => \do {'Dummy'},
                UserID        => 1,
            },
        },
    },

    # no column Separator and charset are given (check return false)
    {
        SourceImportData => {
            ImportDataGet => {
                TemplateID    => $TemplateIDs[2],
                SourceContent => \do {'Dummy'},
                UserID        => 1,
            },
        },
    },

    # no column Separator is given (check return false)
    {
        SourceImportData => {
            FormatData => {
                Charset => 'UTF-8',
            },
            ImportDataGet => {
                TemplateID    => $TemplateIDs[2],
                SourceContent => \do {'Dummy'},
                UserID        => 1,
            },
        },
    },

    # no charset is given (check return false)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Dummy',
            },
            ImportDataGet => {
                TemplateID    => $TemplateIDs[2],
                SourceContent => \do {'Dummy'},
                UserID        => 1,
            },
        },
    },

    # invalid column Separator is given (check return false)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Dummy',
                Charset         => 'UTF-8',
            },
            ImportDataGet => {
                TemplateID    => $TemplateIDs[2],
                SourceContent => \do {'Dummy'},
                UserID        => 1,
            },
        },
    },

    # required values are given but source content is empty (empty array reference must be returned)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'UTF-8',
            },
            ImportDataGet => {
                TemplateID    => $TemplateIDs[3],
                SourceContent => \do {''},
                UserID        => 1,
            },
        },
        ReferenceImportData => [],
    },

    # source content is only a string with spaces (one cell array with the spaces must be returned)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'UTF-8',
            },
            ImportDataGet => {
                TemplateID    => $TemplateIDs[4],
                SourceContent => \do {'  '},
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            ['  '],
        ],
    },

    # all required values are given (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV001-MSExcel-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
            [ 'Row2-Col1', 'Row2-Col2', 'Row2-Col3' ],
            [ 'Row3-Col1', 'Row3-Col2', 'Row3-Col3' ],
        ],
    },

    # all required values are given, but Tabulator is used as Separator (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV001-MSExcel-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            ['Row1-Col1;Row1-Col2;Row1-Col3'],
            ['Row2-Col1;Row2-Col2;Row2-Col3'],
            ['Row3-Col1;Row3-Col2;Row3-Col3'],
        ],
    },

    # all required values are given (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV001-MSExcel-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
            [ 'Row2-Col1', 'Row2-Col2', 'Row2-Col3' ],
            [ 'Row3-Col1', 'Row3-Col2', 'Row3-Col3' ],
        ],
    },

    # all required values are given, but Semicolon is used as Separator (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV001-MSExcel-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            ["Row1-Col1\tRow1-Col2\tRow1-Col3"],
            ["Row2-Col1\tRow2-Col2\tRow2-Col3"],
            ["Row3-Col1\tRow3-Col2\tRow3-Col3"],
        ],
    },

    # all required values are given (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV001-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
            [ 'Row2-Col1', 'Row2-Col2', 'Row2-Col3' ],
            [ 'Row3-Col1', 'Row3-Col2', 'Row3-Col3' ],
        ],
    },

    # all required values are given (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV001-OpenOffice-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
            [ 'Row2-Col1', 'Row2-Col2', 'Row2-Col3' ],
            [ 'Row3-Col1', 'Row3-Col2', 'Row3-Col3' ],
        ],
    },

    # all required values are given (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV001-OpenOffice-Colon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
            [ 'Row2-Col1', 'Row2-Col2', 'Row2-Col3' ],
            [ 'Row3-Col1', 'Row3-Col2', 'Row3-Col3' ],
        ],
    },

    # all required values are given (newline checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV002-MSExcel-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[6],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ "\nTest 1 - 1", "Test 1 - 2",   "Test 1\n- 3",  'Test \n\t\r\s' ],
            [ "Test 2 \n- 1", "Te\nst 2 - 2", "Test 2 - 3\n", '' ],
        ],
    },

    # all required values are given (newline checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV002-MSExcel-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[6],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ "\nTest 1 - 1", 'Test 1 - 2',   "Test 1\n- 3",  'Test \n\t\r\s' ],
            [ "Test 2 \n- 1", "Te\nst 2 - 2", "Test 2 - 3\n", '' ],
        ],
    },

    # all required values are given (newline checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV002-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[6],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ "\nTest 1 - 1", "Test 1 - 2",   "Test 1\n- 3",  'Test \n\t\r\s' ],
            [ "Test 2 \n- 1", "Te\nst 2 - 2", "Test 2 - 3\n", '' ],
        ],
    },

    # all required values are given (newline checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV002-OpenOffice-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[6],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ "\nTest 1 - 1", "Test 1 - 2",   "Test 1\n- 3",  'Test \n\t\r\s' ],
            [ "Test 2 \n- 1", "Te\nst 2 - 2", "Test 2 - 3\n", '' ],
        ],
    },

    # all required values are given (newline checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV002-OpenOffice-Colon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[6],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ "\nTest 1 - 1", "Test 1 - 2",   "Test 1\n- 3",  'Test \n\t\r\s' ],
            [ "Test 2 \n- 1", "Te\nst 2 - 2", "Test 2 - 3\n", '' ],
        ],
    },

    # all required values are given (spaces checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV003-MSExcel-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[7],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ '  Test  ', '    ', 'Test  ' ],
            [ '    Test', '',     'Test' ],
            [ '',         '',     ' ' ],
        ],
    },

    # all required values are given (spaces checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV003-MSExcel-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[7],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ '  Test  ', '    ', 'Test  ' ],
            [ '    Test', '',     'Test' ],
            [ '',         '',     ' ' ],
        ],
    },

    # all required values are given (spaces checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV003-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[7],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ '  Test  ', '    ', 'Test  ' ],
            [ '    Test', '',     'Test' ],
            [ '',         '',     ' ' ],
        ],
    },

    # all required values are given (spaces checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV003-OpenOffice-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[7],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ '  Test  ', '    ', 'Test  ' ],
            [ '    Test', '',     'Test' ],
            [ '',         '',     ' ' ],
        ],
    },

    # all required values are given (spaces checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV003-OpenOffice-Colon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[7],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ '  Test  ', '    ', 'Test  ' ],
            [ '    Test', '',     'Test' ],
            [ '',         '',     ' ' ],
        ],
    },

    # all required values are given (special character checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV004-MSExcel-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[8],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Test;:_°^!"§$%&/()=?´`*+Test', '><@~\'}{[]\\' ],
            [ '"";;::..--__##',                  '' ],
        ],
    },

    # all required values are given (special character checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV004-MSExcel-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[8],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Test;:_°^!"§$%&/()=?´`*+Test', '><@~\'}{[]\\' ],
            [ '"";;::..--__##',                  '' ],
        ],
    },

    # all required values are given (special character checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV004-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[8],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Test;:_°^!"§$%&/()=?´`*+Test', '><@~\'}{[]\\' ],
            [ '"";;::..--__##',                  '' ],
        ],
    },

    # all required values are given (special character checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV004-OpenOffice-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[8],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Test;:_°^!"§$%&/()=?´`*+Test', '><@~\'}{[]\\' ],
            [ '"";;::..--__##',                  '' ],
        ],
    },

    # all required values are given (special character checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV004-OpenOffice-Colon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[8],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Test;:_°^!"§$%&/()=?´`*+Test', '><@~\'}{[]\\' ],
            [ '"";;::..--__##',                  '' ],
        ],
    },

    # all required values are given (ISO-8859 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV005-MSExcel-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[9],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'üöäß', 'ÜÖÄ' ],
            [ 'ßäöü', 'ÄÖÜ' ],
        ],
    },

    # all required values are given (ISO-8859 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV005-MSExcel-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[9],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'üöäß', 'ÜÖÄ' ],
            [ 'ßäöü', 'ÄÖÜ' ],
        ],
    },

    # all required values are given (ISO-8859 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV005-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[9],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'üöäß', 'ÜÖÄ' ],
            [ 'ßäöü', 'ÄÖÜ' ],
        ],
    },

    # all required values are given (ISO-8859 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV005-OpenOffice-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[9],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'üöäß', 'ÜÖÄ' ],
            [ 'ßäöü', 'ÄÖÜ' ],
        ],
    },

    # all required values are given (ISO-8859 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV005-OpenOffice-Colon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[9],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'üöäß', 'ÜÖÄ' ],
            [ 'ßäöü', 'ÄÖÜ' ],
        ],
    },

    # all required values are given (UTF-8 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'UTF-8',
            },
            SourceFile    => 'ImportExportFormatCSV006-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[10],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'ʩ ʬ ʮ',     ' ʡ ˤ Ό ' ],
            [ '  Η ϗ Ϡ  ', 'Ά Λ Ξ' ],
        ],
    },

    # all required values are given (UTF-8 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'UTF-8',
            },
            SourceFile    => 'ImportExportFormatCSV006-OpenOffice-Tabulator.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[10],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'ʩ ʬ ʮ',     ' ʡ ˤ Ό ' ],
            [ '  Η ϗ Ϡ  ', 'Ά Λ Ξ' ],
        ],
    },

    # all required values are given (UTF-8 checks)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'UTF-8',
            },
            SourceFile    => 'ImportExportFormatCSV006-OpenOffice-Colon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[10],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'ʩ ʬ ʮ',     ' ʡ ˤ Ό ' ],
            [ '  Η ϗ Ϡ  ', 'Ά Λ Ξ' ],
        ],
    },

    # all required values are given (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV007-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Row1-Col1', 'Row1-Col2',  'Row1-Col3' ],
            [ 'Row2-Col1', 'Row2-Col2',  'Row2-Col3' ],
            [ 'Row3-Col1', '0Row3-Col2', 'Row3-Col3' ],
        ],
    },

    # all required values are given (check the parsed content)
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            SourceFile    => 'ImportExportFormatCSV008-OpenOffice-Semicolon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[5],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Row1-Col1', 'Row1-Col2',  'Row1-Col3' ],
            [ 'Row2-Col1', '0Row2-Col2', 'Row2-Col3' ],
            [ 'Row3-Col1', 'Row3-Col2',  'Row3-Col3' ],
        ],
    },

];

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

TEST:
for my $Test ( @{$ImportDataTests} ) {

    # check SourceImportData attribute
    if ( !$Test->{SourceImportData} || ref $Test->{SourceImportData} ne 'HASH' ) {

        $Self->True(
            0,
            "Test $TestCount: No SourceImportData found for this test."
        );

        next TEST;
    }

    # set default ImportDataGet
    if ( !$Test->{SourceImportData}->{ImportDataGet} ) {
        $Test->{SourceImportData}->{ImportDataGet} = {};
    }

    # set source content
    if (
        $Test->{SourceImportData}->{SourceFile}
        && $Test->{SourceImportData}->{ImportDataGet}->{SourceContent}
        && $Test->{SourceImportData}->{ImportDataGet}->{SourceContent} eq 'SourceFile'
        )
    {

        my $SourceFile = $Test->{SourceImportData}->{SourceFile};

        # read source file
        my $SourceContent = $MainObject->FileRead(
            Location => $Self->{Home} . '/scripts/test/sample/ImportExport/' . $SourceFile,
            Result   => 'SCALAR',
            Mode     => 'binmode',
        );

        $Test->{SourceImportData}->{ImportDataGet}->{SourceContent} = $SourceContent;
    }

    # set the format data
    if (
        $Test->{SourceImportData}->{FormatData}
        && ref $Test->{SourceImportData}->{FormatData} eq 'HASH'
        && $Test->{SourceImportData}->{ImportDataGet}->{TemplateID}
        )
    {

        # save format data
        $ImportExportObject->FormatDataSave(
            TemplateID => $Test->{SourceImportData}->{ImportDataGet}->{TemplateID},
            FormatData => $Test->{SourceImportData}->{FormatData},
            UserID     => 1,
        );
    }

    # get import data
    my $ImportData = $FormatBackendObject->ImportDataGet(
        %{ $Test->{SourceImportData}->{ImportDataGet} },
    );

    if ( !$Test->{ReferenceImportData} ) {

        $Self->False(
            $ImportData,
            "Test $TestCount: ImportDataGet() - return false"
        );

        next TEST;
    }

    if ( ref $ImportData ne 'ARRAY' ) {

        # check array reference
        $Self->True(
            0,
            "Test $TestCount: ImportDataGet() - return value is an array reference",
        );

        next TEST;
    }

    # check number of rows
    $Self->Is(
        scalar @{$ImportData},
        scalar @{ $Test->{ReferenceImportData} },
        "Test $TestCount: ImportDataGet() - same number of rows",
    );

    # check content of import data
    my $CounterRow = 0;
    ROW:
    for my $ImportRow ( @{$ImportData} ) {

        # extract reference row
        my $ReferenceRow = $Test->{ReferenceImportData}->[$CounterRow];

        if ( ref $ImportRow ne 'ARRAY' || ref $ReferenceRow ne 'ARRAY' ) {

            # check array reference
            $Self->True(
                0,
                "Test $TestCount: ImportDataGet() - import row and reference row matched",
            );

            next TEST;
        }

        # check number of columns
        $Self->Is(
            scalar @{$ImportRow},
            scalar @{$ReferenceRow},
            "Test $TestCount: ImportDataGet() - same number of columns",
        );

        my $CounterColumn = 0;
        for my $Cell ( @{$ImportRow} ) {

            # set content if values are undef
            if ( !defined $Cell ) {
                $Cell = 'UNDEF-unittest';
            }
            if ( !defined $ReferenceRow->[$CounterColumn] ) {
                $ReferenceRow->[$CounterColumn] = 'UNDEF-unittest';
            }

            # check cell data
            $Self->Is(
                $Cell,
                $ReferenceRow->[$CounterColumn],
                "Test $TestCount: ImportDataGet() ",
            );

            $CounterColumn++;
        }

        $CounterRow++;
    }
}
continue {
    $TestCount++;
}

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

my $ExportDataTests = [

    # ExportDataSave doesn't contains all data (check required attributes)
    {
        SourceExportData => {
            ExportDataSave => {
                ExportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # ExportDataSave doesn't contains all data (check required attributes)
    {
        SourceExportData => {
            ExportDataSave => {
                TemplateID => $TemplateIDs[20],
                UserID     => 1,
            },
        },
    },

    # ExportDataSave doesn't contains all data (check required attributes)
    {
        SourceExportData => {
            ExportDataSave => {
                TemplateID    => $TemplateIDs[20],
                ExportDataRow => ['Dummy'],
            },
        },
    },

    # export data row must be an array reference (check return false)
    {
        SourceExportData => {
            ExportDataSave => {
                TemplateID    => $TemplateIDs[20],
                ExportDataRow => '',
                UserID        => 1,
            },
        },
    },

    # export data row must be an array reference (check return false)
    {
        SourceExportData => {
            ExportDataSave => {
                TemplateID    => $TemplateIDs[20],
                ExportDataRow => {},
                UserID        => 1,
            },
        },
    },

    # no existing template id is given (check return false)
    {
        SourceExportData => {
            ExportDataSave => {
                TemplateID    => $TemplateIDs[-1] + 1,
                ExportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # no column Separator and charset are given (check return false)
    {
        SourceExportData => {
            ExportDataSave => {
                TemplateID    => $TemplateIDs[21],
                ExportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # no column Separator is given (check return false)
    {
        SourceExportData => {
            FormatData => {
                Charset => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[21],
                ExportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # no charset is given (check return false)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dummy',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[21],
                ExportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # invalid column Separator is given (check return false)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dummy',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[21],
                ExportDataRow => ['Dummy'],
                UserID        => 1,
            },
        },
    },

    # export data are one cells with empty strings (one empty cell must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [''],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '""',
    },

    # export data are one cells with empty strings (one empty cell must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [''],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '""',
    },

    # export data are one cells with empty strings (one empty cell must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [''],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '""',
    },

    # export data are one cells with empty strings (one empty cell must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dot',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [''],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '""',
    },

    # export data are three cells with empty strings (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ '', '', '' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"";"";""',
    },

    # export data are three cells with empty strings (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ '', '', '' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => "\"\"\t\"\"\t\"\"",
    },

    # export data are three cells with empty strings (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ '', '', '' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"":"":""',
    },

    # export data are three cells with empty strings (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dot',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ '', '', '' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '""."".""',
    },

    # export data are three cells with empty and undef content (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ undef, '', undef ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => ';"";',
    },

    # export data are three cells with empty and undef content (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ undef, '', undef ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => "\t\"\"\t",
    },

    # export data are three cells with empty and undef content (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ undef, '', undef ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => ':"":',
    },

    # export data are three cells with empty and undef content (three empty cells must be returned)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dot',
                Charset         => 'UTF-8',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[22],
                ExportDataRow => [ undef, '', undef ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '."".',
    },

    # all required values are given (check the parsed content)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ExportDataRow => [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"Row1-Col1";"Row1-Col2";"Row1-Col3"',
    },

    # all required values are given (check the parsed content)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Comma',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ExportDataRow => [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"Row1-Col1","Row1-Col2","Row1-Col3"',
    },

    # all required values are given (check the parsed content)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ExportDataRow => [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => "\"Row1-Col1\"\t\"Row1-Col2\"\t\"Row1-Col3\"",
    },

    # all required values are given (check the parsed content)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ExportDataRow => [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"Row1-Col1":"Row1-Col2":"Row1-Col3"',
    },

    # all required values are given (check the parsed content)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dot',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[23],
                ExportDataRow => [ 'Row1-Col1', 'Row1-Col2', 'Row1-Col3' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"Row1-Col1"."Row1-Col2"."Row1-Col3"',
    },

    # all required values are given (newline checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ "\nTest 1", "Test \n 2", 'Test 3 \n\t\r\s', "Test 4\n" ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => qq{"\nTest 1";"Test \n 2";"Test 3 \\n\\t\\r\\s";"Test 4\n"},
    },

    # all required values are given (newline checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ "\nTest 1", "Test \n 2", 'Test 3 \n\t\r\s', "Test 4\n" ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent =>
            qq{"\nTest 1"\t"Test \n 2"\t"Test 3 \\n\\t\\r\\s"\t"Test 4\n"},
    },

    # all required values are given (newline checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ "\nTest 1", "Test \n 2", 'Test 3 \n\t\r\s', "Test 4\n" ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => qq{"\nTest 1":"Test \n 2":"Test 3 \\n\\t\\r\\s":"Test 4\n"},
    },

    # all required values are given (newline checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dot',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ "\nTest 1", "Test \n 2", 'Test 3 \n\t\r\s', "Test 4\n" ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => qq{"\nTest 1"."Test \n 2"."Test 3 \\n\\t\\r\\s"."Test 4\n"},
    },

    # all required values are given (spaces checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ '  Test  ', '    ', 'Test  ', '    Test', '', 'Test', '', ' ' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"  Test  ";"    ";"Test  ";"    Test";"";"Test";"";" "',
    },

    # all required values are given (spaces checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ '  Test  ', '    ', 'Test  ', '    Test', '', 'Test', '', ' ' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent =>
            "\"  Test  \"\t\"    \"\t\"Test  \"\t\"    Test\"\t\"\"\t\"Test\"\t\"\"\t\" \"",
    },

    # all required values are given (spaces checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ '  Test  ', '    ', 'Test  ', '    Test', '', 'Test', '', ' ' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"  Test  ":"    ":"Test  ":"    Test":"":"Test":"":" "',
    },

    # all required values are given (spaces checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dot',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[24],
                ExportDataRow => [ '  Test  ', '    ', 'Test  ', '    Test', '', 'Test', '', ' ' ],
                UserID        => 1,
            },
        },
        ReferenceDestinationContent => '"  Test  "."    "."Test  "."    Test".""."Test".""." "',
    },

    # all required values are given (special character checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Semicolon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ExportDataRow => [
                    'Test;:_^!"$%&/()=?`*+Test',
                    '><@~\'}{[]\\',
                    '',
                    '"";;::..--__##'
                ],
                UserID => 1,
            },
        },
        ReferenceDestinationContent =>
            '"Test;:_^!""$%&/()=?`*+Test";"><@~\'}{[]\";"";""""";;::..--__##"'
    },

    # all required values are given (special character checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Tabulator',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ExportDataRow => [
                    'Test;:_^!"$%&/()=?`*+Test',
                    '><@~\'}{[]\\',
                    '',
                    '"";;::..--__##'
                ],
                UserID => 1,
            },
        },
        ReferenceDestinationContent =>
            '"Test;:_^!""$%&/()=?`*+Test"'
            . "\t"
            . '"><@~\'}{[]\\"'
            . "\t"
            . '""'
            . "\t"
            . '""""";;::..--__##"',
    },

    # all required values are given (special character checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ExportDataRow => [
                    'Test;:_^!"$%&/()=?`*+Test',
                    '><@~\'}{[]\\',
                    '',
                    '"";;::..--__##'
                ],
                UserID => 1,
            },
        },
        ReferenceDestinationContent =>
            '"Test;:_^!""$%&/()=?`*+Test":"><@~\'}{[]\":"":""""";;::..--__##"',
    },

    # all required values are given (special character checks)
    {
        SourceExportData => {
            FormatData => {
                ColumnSeparator => 'Dot',
                Charset         => 'ISO-8859-1',
            },
            ExportDataSave => {
                TemplateID    => $TemplateIDs[25],
                ExportDataRow => [
                    'Test;:_^!"$%&/()=?`*+Test',
                    '><@~\'}{[]\\',
                    '',
                    '"";;::..--__##'
                ],
                UserID => 1,
            },
        },
        ReferenceDestinationContent =>
            '"Test;:_^!""$%&/()=?`*+Test"."><@~\'}{[]\\"."".""""";;::..--__##"',
    },

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

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

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

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

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

TEST:
for my $Test ( @{$ExportDataTests} ) {

    # check SourceExportData attribute
    if ( !$Test->{SourceExportData} || ref $Test->{SourceExportData} ne 'HASH' ) {

        $Self->True(
            0,
            "Test $TestCount: No SourceExportData found for this test."
        );

        next TEST;
    }

    # set default ExportDataSave
    if ( !$Test->{SourceExportData}->{ExportDataSave} ) {
        $Test->{SourceExportData}->{ExportDataSave} = {};
    }

    # set the format data
    if (
        $Test->{SourceExportData}->{FormatData}
        && ref $Test->{SourceExportData}->{FormatData} eq 'HASH'
        && $Test->{SourceExportData}->{ExportDataSave}->{TemplateID}
        )
    {

        # save format data
        $ImportExportObject->FormatDataSave(
            TemplateID => $Test->{SourceExportData}->{ExportDataSave}->{TemplateID},
            FormatData => $Test->{SourceExportData}->{FormatData},
            UserID     => 1,
        );
    }

    # get export data row
    my $ExportString = $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
$ImportExportObject->TemplateDelete(
    TemplateID => \@TemplateIDs,
    UserID     => 1,
);

1;

Um93MS1Db2wxO1JvdzEtQ29sMjtSb3cxLUNvbDMNClJvdzItQ29sMTtSb3cyLUNvbDI7Um93Mi1Db2wzDQpSb3czLUNvbDE7Um93My1Db2wyO1JvdzMtQ29sMw0K
Um93MS1Db2wxCVJvdzEtQ29sMglSb3cxLUNvbDMNClJvdzItQ29sMQlSb3cyLUNvbDIJUm93Mi1Db2wzDQpSb3czLUNvbDEJUm93My1Db2wyCVJvdzMtQ29sMw0K
IlJvdzEtQ29sMSI6IlJvdzEtQ29sMiI6IlJvdzEtQ29sMyINCiJSb3cyLUNvbDEiOiJSb3cyLUNvbDIiOiJSb3cyLUNvbDMiDQoiUm93My1Db2wxIjoiUm93My1Db2wyIjoiUm93My1Db2wzIg0K
IlJvdzEtQ29sMSI7IlJvdzEtQ29sMiI7IlJvdzEtQ29sMyINCiJSb3cyLUNvbDEiOyJSb3cyLUNvbDIiOyJSb3cyLUNvbDMiDQoiUm93My1Db2wxIjsiUm93My1Db2wyIjsiUm93My1Db2wzIg0K
IlJvdzEtQ29sMSIJIlJvdzEtQ29sMiIJIlJvdzEtQ29sMyINCiJSb3cyLUNvbDEiCSJSb3cyLUNvbDIiCSJSb3cyLUNvbDMiDQoiUm93My1Db2wxIgkiUm93My1Db2wyIgkiUm93My1Db2wzIg0K
IgpUZXN0IDEgLSAxIjtUZXN0IDEgLSAyOyJUZXN0IDEKLSAzIjtUZXN0IFxuXHRcclxzDQoiVGVzdCAyIAotIDEiOyJUZQpzdCAyIC0gMiI7IlRlc3QgMiAtIDMKIjsNCg==
IgpUZXN0IDEgLSAxIglUZXN0IDEgLSAyCSJUZXN0IDEKLSAzIglUZXN0IFxuXHRcclxzDQoiVGVzdCAyIAotIDEiCSJUZQpzdCAyIC0gMiIJIlRlc3QgMiAtIDMKIgkNCg==
IgpUZXN0IDEgLSAxIjoiVGVzdCAxIC0gMiI6IlRlc3QgMQotIDMiOiJUZXN0IFxuXHRcclxzIg0KIlRlc3QgMiAKLSAxIjoiVGUKc3QgMiAtIDIiOiJUZXN0IDIgLSAzCiI6DQo=
IgpUZXN0IDEgLSAxIjsiVGVzdCAxIC0gMiI7IlRlc3QgMQotIDMiOyJUZXN0IFxuXHRcclxzIg0KIlRlc3QgMiAKLSAxIjsiVGUKc3QgMiAtIDIiOyJUZXN0IDIgLSAzCiI7DQo=
IgpUZXN0IDEgLSAxIgkiVGVzdCAxIC0gMiIJIlRlc3QgMQotIDMiCSJUZXN0IFxuXHRcclxzIg0KIlRlc3QgMiAKLSAxIgkiVGUKc3QgMiAtIDIiCSJUZXN0IDIgLSAzCiIJDQo=
ICBUZXN0ICA7ICAgIDtUZXN0ICANCiAgICBUZXN0OztUZXN0DQo7OyANCg==
ICBUZXN0ICAJICAgIAlUZXN0ICANCiAgICBUZXN0CQlUZXN0DQoJCSANCg==
IiAgVGVzdCAgIjoiICAgICI6IlRlc3QgICINCiIgICAgVGVzdCI6OiJUZXN0Ig0KOjoiICINCg==
IiAgVGVzdCAgIjsiICAgICI7IlRlc3QgICINCiIgICAgVGVzdCI7OyJUZXN0Ig0KOzsiICINCg==
IiAgVGVzdCAgIgkiICAgICIJIlRlc3QgICINCiIgICAgVGVzdCIJCSJUZXN0Ig0KCQkiICINCg==
IlRlc3Q7Ol+wXiEiIqckJSYvKCk9P7RgKitUZXN0Ijs+PEB+J317W11cDQoiIiIiIjs7OjouLi0tX18jIyI7DQo=
IlRlc3Q7Ol+wXiEiIqckJSYvKCk9P7RgKitUZXN0Igk+PEB+J317W11cDQoiIiIiIjs7OjouLi0tX18jIyIJDQo=
IlRlc3Q7Ol+wXiEiIqckJSYvKCk9P7RgKitUZXN0IjoiPjxAfid9e1tdXCINCiIiIiIiOzs6Oi4uLS1fXyMjIjoNCg==
IlRlc3Q7Ol+wXiEiIqckJSYvKCk9P7RgKitUZXN0IjsiPjxAfid9e1tdXCINCiIiIiIiOzs6Oi4uLS1fXyMjIjsNCg==
IlRlc3Q7Ol+wXiEiIqckJSYvKCk9P7RgKitUZXN0IgkiPjxAfid9e1tdXCINCiIiIiIiOzs6Oi4uLS1fXyMjIgkNCg==
/Pbk3zvc1sQNCt/k9vw7xNbcDQo=
/Pbk3wnc1sQNCt/k9vwJxNbcDQo=
Ivz25N8iOiLc1sQiDQoi3+T2/CI6IsTW3CINCg==
Ivz25N8iOyLc1sQiDQoi3+T2/CI7IsTW3CINCg==
Ivz25N8iCSLc1sQiDQoi3+T2/CIJIsTW3CINCg==
IsqpIMqsIMquIjsiIMqhIMukIM6MICINCiIgIM6XIM+XIM+gICAiOyLOhiDOmyDOniINCg==
IsqpIMqsIMquIjoiIMqhIMukIM6MICINCiIgIM6XIM+XIM+gICAiOiLOhiDOmyDOniINCg==
IsqpIMqsIMquIgkiIMqhIMukIM6MICINCiIgIM6XIM+XIM+gICAiCSLOhiDOmyDOniINCg==
IlJvdzEtQ29sMSI7IlJvdzEtQ29sMiI7IlJvdzEtQ29sMyINCiJSb3cyLUNvbDEiOyJSb3cyLUNvbDIiOyJSb3cyLUNvbDMiDQoiUm93My1Db2wxIjsiMFJvdzMtQ29sMiI7IlJvdzMtQ29sMyINCg==
IlJvdzEtQ29sMSI7IlJvdzEtQ29sMiI7IlJvdzEtQ29sMyINCiJSb3cyLUNvbDEiOyIwUm93Mi1Db2wyIjsiUm93Mi1Db2wzIg0KIlJvdzMtQ29sMSI7IlJvdzMtQ29sMiI7IlJvdzMtQ29sMyINCg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIHZhcjo6cGFja2FnZXNldHVwOjpJbXBvcnRFeHBvcnQ7ICAgICMjIG5vIGNyaXRpYwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06OkRCJywKKTsKCj1oZWFkMSBOQU1FCgpJbXBvcnRFeHBvcnQucG0gLSBjb2RlIHRvIGV4Y2VjdXRlIGR1cmluZyBwYWNrYWdlIGluc3RhbGxhdGlvbgoKPWhlYWQxIFNZTk9QU0lTCgpBbGwgZnVuY3Rpb25zCgo9aGVhZDEgUFVCTElDIElOVEVSRkFDRQoKPW92ZXIgNAoKPWN1dAoKPWl0ZW0gbmV3KCkKCmNyZWF0ZSBhbiBvYmplY3QKCiAgICB1c2UgS2VybmVsOjpTeXN0ZW06Ok9iamVjdE1hbmFnZXI7CiAgICBsb2NhbCAkS2VybmVsOjpPTSA9IEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyLT5uZXcoKTsKICAgIG15ICRDb2RlT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgndmFyOjpwYWNrYWdlc2V0dXA6OkltcG9ydEV4cG9ydCcpOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGFsd2F5cyBkaXNjYXJkIHRoZSBjb25maWcgb2JqZWN0IGJlZm9yZSBwYWNrYWdlIGNvZGUgaXMgZXhlY3V0ZWQsCiAgICAjIHRvIG1ha2Ugc3VyZSB0aGF0IHRoZSBjb25maWcgb2JqZWN0IHdpbGwgYmUgY3JlYXRlZCBuZXdseSwgc28gdGhhdCBpdAogICAgIyB3aWxsIHVzZSB0aGUgcmVjZW50bHkgd3JpdHRlbiBuZXcgY29uZmlnIGZyb20gdGhlIHBhY2thZ2UKICAgICRLZXJuZWw6Ok9NLT5PYmplY3RzRGlzY2FyZCgKICAgICAgICBPYmplY3RzID0+IFsnS2VybmVsOjpDb25maWcnXSwKICAgICk7CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aXRlbSBDb2RlSW5zdGFsbCgpCgpydW4gdGhlIGNvZGUgaW5zdGFsbCBwYXJ0CgogICAgbXkgJFJlc3VsdCA9ICRDb2RlT2JqZWN0LT5Db2RlSW5zdGFsbCgpOwoKPWN1dAoKc3ViIENvZGVJbnN0YWxsIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIDE7Cn0KCj1pdGVtIENvZGVSZWluc3RhbGwoKQoKcnVuIHRoZSBjb2RlIHJlaW5zdGFsbCBwYXJ0CgogICAgbXkgJFJlc3VsdCA9ICRDb2RlT2JqZWN0LT5Db2RlUmVpbnN0YWxsKCk7Cgo9Y3V0CgpzdWIgQ29kZVJlaW5zdGFsbCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiAxOwp9Cgo9aXRlbSBDb2RlVXBncmFkZSgpCgpydW4gdGhlIGNvZGUgdXBncmFkZSBwYXJ0CgogICAgbXkgJFJlc3VsdCA9ICRDb2RlT2JqZWN0LT5Db2RlVXBncmFkZSgpOwoKPWN1dAoKc3ViIENvZGVVcGdyYWRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIDE7Cn0KCj1pdGVtIENvZGVVcGdyYWRlRnJvbUJlZm9yZV8yXzBfMygpCgpUaGlzIGZ1bmN0aW9uIGlzIG9ubHkgZXhlY3V0ZWQgaWYgdGhlIGluc3RhbGxlZCBtb2R1bGUgdmVyc2lvbiBpcyBzbWFsbGVyIHRoYW4gMi4wLjMuCgogICAgbXkgJFJlc3VsdCA9ICRDb2RlT2JqZWN0LT5Db2RlVXBncmFkZUZyb21CZWZvcmVfMl8wXzMoKTsKCj1jdXQKCnN1YiBDb2RlVXBncmFkZUZyb21CZWZvcmVfMl8wXzMgeyAgICAjIyBubyBjcml0aWMKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBmaXggYSB0eXBvIGluIHRoZSBkYXRhYmFzZQogICAgJFNlbGYtPl9GaXhEYXRhYmFzZVR5cG8oKTsKCiAgICByZXR1cm4gMTsKfQoKPWl0ZW0gQ29kZVVuaW5zdGFsbCgpCgpydW4gdGhlIGNvZGUgdW5pbnN0YWxsIHBhcnQKCiAgICBteSAkUmVzdWx0ID0gJENvZGVPYmplY3QtPkNvZGVVbmluc3RhbGwoKTsKCj1jdXQKCnN1YiBDb2RlVW5pbnN0YWxsIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIDE7Cn0KCj1iZWdpbiBJbnRlcm5hbDoKCj1pdGVtIF9GaXhEYXRhYmFzZVR5cG8oKQoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+X0ZpeERhdGFiYXNlVHlwbygpOwoKPWN1dAoKc3ViIF9GaXhEYXRhYmFzZVR5cG8gewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGZpeCB0aGUgQ29sdW1uU2VwZXJhdG9yIHR5cG8gKGNvcnJlY3QgaXMgQ29sdW1uU2VwYXJhdG9yKQogICAgcmV0dXJuIGlmICEkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6REInKS0+RG8oCiAgICAgICAgU1FMID0+ICJVUERBVEUgaW1leHBvcnRfZm9ybWF0ICIKICAgICAgICAgICAgLiAiU0VUIGRhdGFfa2V5ID0gJ0NvbHVtblNlcGFyYXRvcicgIgogICAgICAgICAgICAuICJXSEVSRSBkYXRhX2tleSA9ICdDb2x1bW5TZXBlcmF0b3InIiwKICAgICk7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cgo9ZW5kIEludGVybmFsOgoKPWJhY2sKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBTb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKGh0dHA6Ly9vdHJzLm9yZy8pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQ+LgoKPWN1dAo=
LyoqCkNvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQuCiovCgovKioKICogQHBhY2thZ2UgICAgIFNraW4gIkRlZmF1bHQiCiAqIEBzZWN0aW9uICAgICBJbXBvcnQgRXhwb3J0IFNjcmVlbgogKi8KCkBtZWRpYSBzY3JlZW4scHJvamVjdGlvbix0dixoYW5kaGVsZCB7CgovKioKICogQHN1YnNlY3Rpb24KICovCgoKLk1hcEhlYWRlclJvdyBsYWJlbCB7CiAgICBjb2xvcjogIzkyOTI5MjsKfQoKLk1hcEhlYWRlclJvdyAuSGVhZGVyLAouTWFwSGVhZGVyUm93IC5GaWVsZCB7CiAgICBkaXNwbGF5OiBpbmxpbmU7CiAgICBtYXJnaW4tcmlnaHQ6IDE1cHg7CiAgICBwYWRkaW5nLWxlZnQ6IDJweDsKfQoKYnV0dG9uLkFycm93VXAsCmJ1dHRvbi5BcnJvd0Rvd24gewogICAgaGVpZ2h0OiAxNnB4OwogICAgd2lkdGg6IDE2cHg7CiAgICBwYWRkaW5nOiAxcHg7CiAgICBtYXJnaW4tdG9wOiAwOwogICAgbWFyZ2luLWJvdHRvbTogMXB4OwogICAgYm9yZGVyLXN0eWxlOiBub25lOwogICAgdGV4dC1pbmRlbnQ6IC05OTk5cHg7CiAgICBkaXNwbGF5OiBpbmxpbmUtYmxvY2s7CiAgICB2ZXJ0aWNhbC1hbGlnbjogbWlkZGxlOwogICAgY3Vyc29yOiBwb2ludGVyOwp9CgpidXR0b24uQXJyb3dVcDphY3RpdmUsCmJ1dHRvbi5BcnJvd0Rvd246YWN0aXZlIHsKICAgIG1hcmdpbi10b3A6IDFweDsKICAgIG1hcmdpbi1ib3R0b206IDA7Cn0KCmJ1dHRvbi5BcnJvd1VwW2Rpc2FibGVkPSJkaXNhYmxlZCJdOmFjdGl2ZSwKYnV0dG9uLkFycm93RG93bltkaXNhYmxlZD0iZGlzYWJsZWQiXTphY3RpdmUgewogICAgbWFyZ2luLXRvcDogMHB4OwogICAgbWFyZ2luLWJvdHRvbTogMXB4Owp9CgpidXR0b24uQXJyb3dVcCB7CiAgICBiYWNrZ3JvdW5kOiB1cmwoLi4vaW1nL2ljb25zL2ltcG9ydGV4cG9ydF9hcnJvd191cC5wbmcpOwp9CgpidXR0b24uQXJyb3dVcFtkaXNhYmxlZD0iZGlzYWJsZWQiXSB7CiAgICBiYWNrZ3JvdW5kOiB1cmwoLi4vaW1nL2ljb25zL2ltcG9ydGV4cG9ydF9hcnJvd191cF9kaXNhYmxlZC5wbmcpOwogICAgY3Vyc29yOiBkZWZhdWx0Owp9CgpidXR0b24uQXJyb3dEb3duIHsKICAgIGJhY2tncm91bmQ6IHVybCguLi9pbWcvaWNvbnMvaW1wb3J0ZXhwb3J0X2Fycm93X2Rvd24ucG5nKTsKfQoKYnV0dG9uLkFycm93RG93bltkaXNhYmxlZD0iZGlzYWJsZWQiXXsKICAgIGJhY2tncm91bmQ6IHVybCguLi9pbWcvaWNvbnMvaW1wb3J0ZXhwb3J0X2Fycm93X2Rvd25fZGlzYWJsZWQucG5nKTsKICAgIGN1cnNvcjogZGVmYXVsdDsKfQoKfSAvKiBlbmQgQG1lZGlhICovCg==
iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAZlJREFUeNqkUztLA0EQnr3bvEhhI9FC09gJInJiZbAQREHUWhTtbK8ULEWwFVLZaJE/YKWFgn2SQgxBgjYSIoYkEsxxuec6c/FxCXmAGZjbmdmZ7+a1TAgBw5AEQxL3K7OnjxAMhyEQCkEgGARJllfJ7jrOjWWaYBkGmM0mPBzOdAdoIyF2MFD9lkfxmxqYgT/Ysix1fzOukHp59apCq1epgT0QFGya6u56XMnnG0C8vTahkI3u+gJ4wYah7m1NKcWiBrregFqtArlcCTaWxhW66wRpKwEdEjTW5EUmS/riXEzhXMB9upSFdMnzYYwl/KXwjgwOsHEgXBds285oQgYHAQ1dB875PJMkmkz/Jjq2DbqmeXJFDoHDwAMwUI9Eo/0BtHod9EbjVy/zKDgugIkARHSy52uUFroDFM9WgI9NQ2T5qJUBj2AJfwD63QnY73mUznuPkRy022MwcfNsJnlMMtlawYMXadItP8Wst1yhmmz9SVQLBbTRUn0iv1CrfpyZ/zXiiOgYQY72WHDq5QfGuF0B/kNfAgwAlIbWNoRkTzIAAAAASUVORK5CYII=
iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAQAAAC1+jfqAAAAAXNSR0IArs4c6QAAAAJiS0dEAP+Hj8y/AAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH2goUEigB30KkogAAAMpJREFUKM+lkT0OAWEQhp/9ESJKF5FPK1oSCQfYXvvVDuAcCola5w5sKdkL6EQhYv+NYpddSyi8xUwxT2bemTGE7zJ/1LGzNKNBnRrWANJNTEjAvAzkclINtFm+dQAQJ9ZjBWstFIhZlCM9Uh4eQxVpcSqAOKGeqAM+J/b0VfhE8hFRT1i40FEWWxfA6GVjckCmKTeSXcoNH7trYlVNJlwBQQiAZhU4c8k6IfiAT+sVOD52AQJ4xsJk6VrRp1PHWXJXgOGWAePvb94BXKRFykgCmkUAAAAASUVORK5CYII=
iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAZBJREFUeNrEU79Lw0AUfpemhZI4CIpCJ92cHE4HJ0HEQcFRQdShIJ0E/5SCOBSkgxZE7CS4iIt06NIMnVxUcKhWCJb+iPl1l/gu1ppKKkIHH1zu3r33ffnu3Tvi+z4MYxIMaXLYWcy/9taEkBxOtOtqqDTzFbtNT0QTeIyFXbq5kgoIzq9rf1PgOk5fsNVyQZTo5/5AgqauQ1JVe75puuB5AI5lfe91OvidiiYwms0gOakogd9ovYPjuuCY5ifYMMC17cEKhDEEGO02yLIMNYsB9wjYSMCwPh7nvx8BK51DzZSjbo5E9biCBBAQoFW6ORpOmegi2nYJEyhNr9EHPOo9qvXAB3V1iY4mAF7OrjS83tLARkJwwbWsbPmoqCkJCXQmwRuPQVyW4Clf1ERM5IQxJNzK6vwWSmcQm1nfxv2Dsb0dyjHcOD4Vf87yu8uCVb2AMKaPAJPEJK5gRF7Y3yCTs7uBsnr1hJUPi7gUd9j2Q6AognEcKRwxMr08FxA83lS6KaLXnxHjRRL8y2v8EGAAqEvNnH+aSSAAAAAASUVORK5CYII=
iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAQAAAC1+jfqAAAAAXNSR0IArs4c6QAAAAJiS0dEAP+Hj8y/AAAACXBIWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH2goUEig6bklNhgAAAMlJREFUKM+lkT0KAjEQhb9IFEE9TToP4BU8gq2exG23FAKCYCmIra25iYL4w/6asVhddlW0cJo3yXy8CS9K+F6NH3N0IVMAVIgBnIwAJlXgVogZGNh8ckgfxzNS9jXgQAeACE8MwLUOnIjpACcyYuBa+uinVc4ZzQVPRI5/XSGhN54cQYhhB+IYVYBkK2ZojkR4+qbNyqltLSixSTBzLRIyNEuXBGJfkhSbBtb16LJwaTmuPBLEZszHoIKmfcvhieR70Ovqnfr7N++lLlCSshf+TwAAAABJRU5ErkJggg==