ImportExport
8.0.8
OFORK
https://o-fork.de/
GNU AFFERO GENERAL PUBLIC LICENSE Version 3, November 2007
New release.
New release.
Added new functions.
Added new functions.
Added new functions.
Added new functions.
Added new functions.
Init release.
The ImportExport package.
Das ImportExport Paket.
8.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>
# create the package name
my $CodeModule = 'var::packagesetup::' . $Param{Structure}->{Name}->{Content};
$Kernel::OM->Get($CodeModule)->CodeInstall();
2019-09-02 08:43:42
localhost
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uWpmXEhISEhIiJMbBYCeJmevuHmASVr13N/ff/To0X//+99d3R3PExsbK62h3u2USiUz4wCPQHQJAICfsKo3AHQjNw8qybq2xjksZWZm9kS1L774Yk9UC6Dbse4SAAA/kW6EOnyKEADA5SIiIhQ2IiIiXN0vALgZkbuEm8sLL7wg/xcAbJG7BACe4tKlS67uAgDgR54UXdLr9YcOHTp16pRWq21ubu7bt+/AgQNHjhw5YcIEPz8/eTfHAwcvvPCC1c6+vr79+/e/5ZZb7r777sDAQKv9TSbT4cOHi4qKqqqqWlpa+vTpM3DgwKFDh44ZM2bQoEEONlpeXv7tt9+eP3++oaHB399/6NChEydOHDJkiCO97cJbAIBOIboEAABuTr34l3j7g+qVQ3Y+j4kunT9//oMPPqivr5dLGhoaGhoazp8/f+DAgUWLFjkSoOmQwWDQarVarfbYsWPLli2zfGCQXq//3//937KyMrmkqamprKysrKwsPz/fwX+O//73v7/66it5/nl9fX1RUVFRURH/mgHATZhMJkF0CQAAwNP0+uiYmw/NM6JLZWVl77zzjtFojIyMvPvuu4cMGeLv79/U1HThwoUDBw5oNJp33nnn17/+dXR0tGjrjHf4l5DfamhoKCsr+/zzz2tqavbu3fvAAw/I+/z73/8uKysLCAi45557br311sDAQIPBcPny5XPnzhUVFTkyioKCgn//+99eXl6TJ08eN25caGhoc3NzaWnpt99+6/ipcPN/TwDg6chdAgAA6JW4m+5RHhBdMhqNH330kdFovPPOO+fMmSOvtBoYGDh69OhRo0b961//OnLkyIcffrh8+XJvb+8baSsgIGDUqFH9+vV76623SktLLd86fvy4EGLBggUjRoyQSry9vaOjo6Ojo6dOndphzQ0NDXv27BFCzJ49e9y4cXJzt99+++23334jfQYAdCOiSwAAdBej0XiDN2gAPIUHRJe+//57nU4XFhY2e/Zs24f4eHl5zZ49W6PRSNPZ7rzzzhtvUVpEqaWlxbKwrq5OCBETE9O1Oo8eParX62NiYuTQUk+QYrF//OMfDx8+XFhYePnyZYPB8Mc//lEIodFojhw58sMPP9TU1CgUipCQkJEjR06ZMsXf39+2hhdeeOHIkSMFBQVardbLyysmJiYxMVE6LUePHpXKzWZzTExMUlKS7ZpTNTU1arX67NmztbW1Pj4+KpVq0qRJo0aNknf47//+b51O99hjj6lUKqkkLy/v6NGjQog77rhj/vz5UuHFixezs7NDQ0PT09PlY8vKytRqdVlZWVNTk7+//+DBgydPniylrTlyHmy1tLR8/PHHJSUlXl5eKSkpEydO7ORZB9CrEF0CcHNSKBTy6g29Rl5eXmpqamBg4B133LFv3z5Xd8eNSF9z3fsXf/7559euXSu9NpvN5eXldXV1qampx44dk/fZv3//m2++2dDQ8M9//tPX13fr1q3SlX975VbVwvlu/PZQvjUrLCw8fPjw5cuXFQpFVFTUlClThg8fbtXcjdzryXlJlglKnU1W+uSTTw4dOhQcHPzoo49aLsRcX1//5ptv1tbWTpgw4f777/fEoTmBB0SXSkpKhBAJCQnthb29vb0nTZr0ySeflJSUdEt06eLFi0KI4OBgy8KgoKCrV6+eOXNm9OjRXajz7NmzQoixY8feePc69Omnnx46dMiqcNOmTZably9fvnz5cnFx8W9/+9t+/fpZ7bxr1y7L+XqnT58+f/78b3/724KCgu+++04uP3v2bHl5+WOPPaZUKi0Lt27dqtfrpc3W1tZz586dO3fu5z//+b333isVDh8+vLCw8Ny5c3J06dy5c1YvhBBS+tiwYcPkkoKCgp07d1ouXFVcXHzixIn7779//PjxjpwHKzU1Ne+9915VVVW/fv0efPBBy7YA3JxsrwacwGw2V1ZWent7h4WFEdgC0KNUKpV0rWulU4GGxsZG2wvInhYUFCT93Ou41NRUyziFc3Shn85nNpu79+vm9OnTNTU18uaxY8dKS0vnzp1rtYTIo48+evDgQWlxW7VanZycLK2r2165EKKmpqa0tNQl386Q3cjtoWTnzp0FBQXyZmlp6blz52bOnGn503633+t1QUpKSlVV1YULF7Zt2/arX/1KCkEYjcatW7fW1tYOHjw4JSXFQ4fmBB4QXaqoqBDXhxhsSe9Ke96IxsbG8+fPf/7550IIqwlrY8eOzc/P//jjj8+ePTty5MioqKiAgADHa66qqhJCxMTEHD9+XK1WV1VVeXt7h4eHjxs37s477+zeD/fDhw9Pnjw5Li5OqVTK2V5Dhw4dP3784MGDAwMD9Xp9RUXF3r17NRpNfn7+9OnTrWooKCi4++674+LigoODL126tH37dq1W++6779bW1tqW5+fnz507VzqwtrZ227Zter3+9ttvnzx58oABA5qamo4fP75v3779+/ePGDFCWnx92LBhUnRp8uTJQojLly/X1tYGBgaazeba2trLly8PHDhQXIs0yX/6ysrKXbt2mc3mcePGTZkypX///jqdLj8//8iRIzt37oyJiQkPD+/wPFj64Ycftm3b1tjYGB4e/stf/tJyEXcAcCaTySRdRlt9jgFAt6usrLQq2blz56xZs4RFgEm6NH311VdXr179+eefnzhx4r/+67/eeeed+fPna7XamTNnHjp0KDw8/MEHH3zllVekMFN4eHhra2t1dbUQQqlU+vr6Xrp0Sa7qiy+++OUvfzl48OCPPvpIuho0GAz/7//9v7y8PLPZPG/evLVr10rPgJb2r6mpefrpp7dt2/a73/1u/fr1Up319fXyNXOHsbAhQ4ZcuHBBCLFgwQLLQzrVrp3zIIRYvnz57t27y8vLJ0+evGrVqnvuuafDfsbHxwshLO/G22R1uNVmm+22d96am5ufffbZ999/38vLa926dX/+859PnTolN7R37960tLSqqirLGFyn6pdt2rQpOTlZ3hw7dmybP6u3tLTIl9yTJ09uaGiwXy6ESExMzM7Ofvnll+2fNPSoLt8eWtYwfvz4u+++OyQk5OrVq19//XVhYeFnn302dOjQsLAw0R33evIT4W8kqcfb2zs1NfXNN9+8cOHCzp07Z8+eLYT49NNPy8rKQkJCUlNTbVNePGVoTtDGLbe7aWxsFDaZRFakd6U9u+CFa1555ZWtW7fqdLrbb7/95z//ueU+06ZNu+OOO1pbWw8fPvyPf/xj/fr1r776qhRscqSJpqYmIURRUdEHH3yg0WgMBkNzc/OFCxe2b9/+wQcfOP5j0Qs2bPeZOHFicnLywIEDLUMqS5cuvf3220NCQry9vf39/UeMGLFo0SIhhOUXjOzuu+9OSkoaMGCAr69vdHT0zJkzhRA6na7Ncstso2+++aalpeWOO+544IEHVCqVn59fSEjI5MmT77vvPmHxVSoFjM6fP280GoVFFEkqlzaNRuP58+eFRXTp22+/NZlMsbGxc+fOHTBggLe394ABA+bNm3fLLbcYjcZvvvnGkfMgO3To0DvvvNPY2Dhy5Eir5wMCgJNJH4ZtflgBQE+bOXOm1bWotGk0Gjdv3jxr1qyioqLN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i2NjY1tbW99///1jx47dyMAB4EYwMw5Ab2WV/oPeKjY21moJnu6iVCqZGdcLWN3E7dixo6SkxPImrlvu9aR0of/7v/9zPBJtq7KyUrpJl54EFxMTk5SUZDab//nPf1o9ddHjhtbTPCB3SQgRExOTlpa2bdu2iooKKZpjKSAg4MEHH4yJienGFm+77bbQ0NArV66UlJRIEUqtVqvVaqWl9Wy7N23aNPsVKhSKhQsXvvXWW/X19Xl5eZZv3XHHHdLy3j1n2rRp//jHP86cOSM9000SHx//3XffdW9DQUFBDz300NatW0tKSqQpb3YMHz5citdGRETID3cMCAgIDw+XflWzXNJbCDFo0KCUlJSdO3cePnzY8pcfhUIxc+ZMKYLWKT4+PqmpqR988MHJkyc//vjjlpYWz12fH0C3MBgMNTU10pNindku0SUAvRXrMd88MjMze6LaF198sSeqhZNNnDixoKDA6iZuxowZ8lzdbrnXGz16dH5+/mefffbZZ59JJVbLeLe3qrdcrtfrP/zww9bW1vHjx//sZz+TChMSEi5cuFBcXLxt27bf/OY3VlOC3GRo7sAzoktCiCFDhqSnpx86dKikpESr1TY3N/ft2zcsLCw2NnbChAndnh7m5eV111137dy588CBA1J06fHHHz958uT58+erqqqkZ5/169cvIiJizJgxd9xxhyO3BEql8j//8z/z8/NLSkpqa2t9fX0HDRo0fvz47prQZ8ett9760EMP7d+//+LFi15eXmFhYePHj4+Li+v26JIQIjo6+ve//730l7p8+bL0pIwBAwbceuutVhleMTExUoKfVRRp+PDhly5d8vHxsUodFEJMnDgxIiLi4MGDFy5caGpq8vf3Hzx48OTJk7scW/T29l60aNFHH310/PjxTz75pLm52fZRkQBuHgaD4erVq9KnljPbJboEAF0QEREhLVxiSf6dEoD7mDlzZnh4eGFh4eXLlxUKRVRU1JQpU0aMGGG5z43f6/3iF78wm83FxcVXr15tb9Vm+z799NPLly+Hh4fPmDHDsnzu3LmXLl2qrKz89NNP586d64lDc4IbeiIpAAC9iV6vr62t9fHx6aH0/vZcvHixvr4+PDxcXgAS6Ky4uLijR4+67RUnXCI/Pz89Pb2wsNDVHQF6SkJCQmZmZmJioqs7gnZJKTZumGhz43rx0LrGY3KX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fX19FotlWKdcvHjRuzEAgLdEqR0AAHgoKipKr9c3NTVdffXV9vFHO3fufOWVV8aMGSOE2LZt23XXXffkk08qu5555hn7N34DnutosLzAiy+++Nprr+Xk5Aghnn322W9/+9tXfiFGo3HHjh0lJSV9fX29vb3x8fF5eXm//vWv7Qd88MEHLk4fM2bM1Vdf7XHrp0+fvpKkGAYky7LJZBJCxMXFqR2L91VVVU2ePFntKADAE59//rnSgzn+R698JfDUU09t2rTpzTff7O3tXbRo0e7duxcuXKgc0NzcXFRU9Mknn2RkZNxxxx2//e1vExIShBC5ubn19fU5OTl1dXVCiNGjR587d+6qq646f/68UueJEyfuuOOO+vr6BQsW/P73v09MTFQq7O7uvvfeeysrK00mU15e3r59+5KTkx2DaWtrW7t27YEDB+6+++5nnnlGCHHNNdfU1taKr47pdv01xpDXJYRYsWLFn/70p9TU1C1btixatMjxxAGDd31dA95e+1l79+594oknTp48KYTYt2+fEkZPT8+aNWsOHDgwZsyYP/7xjxMnThzyugAEJsYuAQhWr7zyytGjR6dMmTJ+/Hj7MKJz585NnDhRkiRJkkaOHHnhwgX78Y4pmAHPdZSVlTVgzqW+vn7s2LHK9jXXXOO4S6/XDzjcaUi9vb3t7e1Wq9X+vMhymAhk3/rWt4qKitSOAgA8MWnSpP6ZC6XEarXu3r37e9/73n/913/t3LmzuLjYfsC6deu2bdtmNptra2tnz569bt06pbyuru7MmTM/+MEPDAaDzWYrKSmpq6s7f/68vc5jx44dPXq0o6Nj6tSpDz/8sL3CjRs3bty48fz5842NjevWrdu4caNTMN///vdvuukmvV6vpJaEEGfOnFF2yQ5cX+yQ1yWEeO655xoaGl588cUf/ehHTicOGLzr6xrw9trPevPNNw8ePGgymfbt22cPY/369TqdrqGhoaKi4siRI4LUEhC8ZAAIbHFxcUajUdlWBoQ77rXZbG+88cbIkSOVjxMmTPjyyy/7VzJgd+d0rqM77rhjz549/U//2te+duLECWW7qqrKsdrdu3cvWbLE7cty1traunHjxoSEhJiYmNTU1J6eHo+rgupsNltNTU1NTY3ValU7FoSLBx98MD09Xe0oAP/Jysqqqanx7Nz+TwX2EiGEzWZzOsbpt02d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vf/++8OHD3/33XeVU1u3bi0tLd29e3d7e/vu3btLS0u3b9+uXjVq1Kjdu3d7PJ64u3esp6NgMDhlypSXXnpJefniiy9OnTo1GAz2dklIDysMAQAAAAAYEtRthiRJ+vjjj6+++mqlvbGxsaCg4JNPPhk7dqzactlll507d07pvH379ltuuUU5tW3btp/97GcffPCBEOLaa69dsmTJbbfdppwqLy9/9tln//jHPypX7d27d+rUqR3vHtvSsUi1T0NDw7XXXvurX/1KCDF//vw//elPBQUF6oW9VBLSw/sIAAAAAMCQEJsuRSIRnU4Xe6pji9q5paUlOztbaW9paRk5cqSybs7pdJ48eTInJ0c9dfHFFzc3NytX+Xy+jIyMjgMmaYnzySefzJ8/32AwvP7662ry1aslIT3suwQAAAAAwJATGyQlajlPsTlOev785z87HI5AIHD27Nl+UhI6RboEAAAAAACS2bdvn3r84YcffvWrX1WOL7vssr1798ae+spXvpJoEIPBEIlEUr/pp59++uMf//jtt99+6623vvvd7548eVLzkpAI6RIAAAAAAEjmkUce+fDDD91u9wcffPC9733vhz/8odL++OOPP/roox988IF6aunSpYkGGT58+HvvvReNRlO5o8fjmTdv3s9+9rOLL754/Pjx/+///b+5c+f6/X4NS0ISrDAEAAAAAGBIiN1KqcstkGI7v/zyy2vXrj1z5kxZWdmTTz65aNEitdu6deuUUyNGjFi6dOl3vvOdRANu3br18ccfP3v2bCQSUUfuWKRy6r777tPr9a+88orafvfdd1sslk2bNvVqSUgP6RIAAAAAAEioH2593Q9LGuJYGQcAAAAAAID0kS4BAAAAAAAgfcwlAwAAAAAAQPqYuwQAAAAAAID0kS4BAAAAAAAgfaRLAAAAAAAASB/pEgAAAAAAANJHugQAAAAAwJBz5syZyspK5djtdp8+fVo5DgQCTU1NQgiPx1NVVVVVVVVXVxd7YWznpqamo0ePJhm50w4YfPjOOAAAAAAAhpbt27f//e9/b25uvvDCC+fOnfv973//iiuuMJvNt95663/+5382NjZu27bt448//p//+R+fz3fw4MGdO3cqF9bW1qqdr7/++pUrV06aNOnkyZPr1q3rOPLkyZM7dsCgRLoEAAAAAMBQ1N7e/uCDD06aNGnMmDH/9m//NmfOnK1btwohbrvttm3btil9fvnLXxoMhrvuukt5+eKLL6qdFy9efOLEiQULFjz00EO/+tWvOo589913J+qAQYaVcQAAAAAADDmyLD/xxBNLlixpbGzMz89P1K28vHz27NmnTp36+OOPfT5fbOexY8ceOHDg0UcfnTVrltohduTYDn30VNAI6RIAAAAAAEPO448/fscdd1x55ZUlJSU1NTVCCEmS4vqcOHGitLQ0IyOjpqbm6NGjfr8/tvOmTZu+//3vv/7661u3blU7xI4c26HvHxB9yaB1AQAAAAAAoE/94he/2LNnTyAQOHTo0De/+c0HHnjgD3/4w80339zc3LxixYq///3vzzzzzLJly1555ZVvfetbQogpU6ZMmTJFCDF37ly185gxY9asWTNq1KgLLrhA7RA78owZM9QO2j4vehv7LgEAAAAAMKTJsuz3+y0WS3c7h8Nhn89nt9sTde6yAwaH/w/RBIPnLF0HVwAAAABJRU5ErkJggg==
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiID8+CjxvZm9ya19jb25maWcgdmVyc2lvbj0iMi4wIiBpbml0PSJGcmFtZXdvcmsiPgogICAgPFNldHRpbmcgTmFtZT0iRnJvbnRlbmQ6Ok1vZHVsZSMjI0FkbWluSW1wb3J0RXhwb3J0IiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBUcmFuc2xhdGFibGU9IjEiPkZyb250ZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBhZ2VudCBpbnRlcmZhY2UuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8TmF2aWdhdGlvbj5Gcm9udGVuZDo6QWRtaW46Ok1vZHVsZVJlZ2lzdHJhdGlvbjwvTmF2aWdhdGlvbj4KICAgICAgICA8VmFsdWU+CiAgICAgICAgICAgIDxJdGVtIFZhbHVlVHlwZT0iRnJvbnRlbmRSZWdpc3RyYXRpb24iPgogICAgICAgICAgICAgICAgPEhhc2g+CiAgICAgICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJHcm91cFJvIj4KICAgICAgICAgICAgICAgICAgICAgICAgPEFycmF5PgogICAgICAgICAgICAgICAgICAgICAgICA8L0FycmF5PgogICAgICAgICAgICAgICAgICAgIDwvSXRlbT4KICAgICAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ikdyb3VwIj4KICAgICAgICAgICAgICAgICAgICAgICAgPEFycmF5PgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPEl0ZW0+YWRtaW48L0l0ZW0+CiAgICAgICAgICAgICAgICAgICAgICAgIDwvQXJyYXk+CiAgICAgICAgICAgICAgICAgICAgPC9JdGVtPgogICAgICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iRGVzY3JpcHRpb24iIFRyYW5zbGF0YWJsZT0iMSI+SW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLjwvSXRlbT4KICAgICAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IlRpdGxlIiBUcmFuc2xhdGFibGU9IjEiPkltcG9ydC9FeHBvcnQ8L0l0ZW0+CiAgICAgICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJOYXZCYXJOYW1lIj5BZG1pbjwvSXRlbT4KICAgICAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICAgICAgPC9JdGVtPgogICAgICAgIDwvVmFsdWU+CiAgICA8L1NldHRpbmc+CiAgICA8U2V0dGluZyBOYW1lPSJMb2FkZXI6Ok1vZHVsZTo6QWRtaW5JbXBvcnRFeHBvcnQjIyMwMDEtSW1wb3J0RXhwb3J0IiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBUcmFuc2xhdGFibGU9IjEiPkxvYWRlciBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgYWdlbnQgaW50ZXJmYWNlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPE5hdmlnYXRpb24+RnJvbnRlbmQ6OkFkbWluOjpNb2R1bGVSZWdpc3RyYXRpb246OkxvYWRlcjwvTmF2aWdhdGlvbj4KICAgICAgICA8VmFsdWU+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJDU1MiPgogICAgICAgICAgICAgICAgICAgIDxBcnJheT4KICAgICAgICAgICAgICAgICAgICAgICAgPEl0ZW0+SVRTTS5JbXBvcnRFeHBvcnQuY3NzPC9JdGVtPgogICAgICAgICAgICAgICAgICAgIDwvQXJyYXk+CiAgICAgICAgICAgICAgICA8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IkphdmFTY3JpcHQiPgogICAgICAgICAgICAgICAgICAgIDxBcnJheT4KICAgICAgICAgICAgICAgICAgICAgICAgPEl0ZW0+SVRTTS5BZG1pbi5JbXBvcnRFeHBvcnQuanM8L0l0ZW0+CiAgICAgICAgICAgICAgICAgICAgPC9BcnJheT4KICAgICAgICAgICAgICAgIDwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvVmFsdWU+CiAgICA8L1NldHRpbmc+CiAgICA8U2V0dGluZyBOYW1lPSJGcm9udGVuZDo6TmF2aWdhdGlvbiMjI0FkbWluSW1wb3J0RXhwb3J0IyMjMDAxLUltcG9ydEV4cG9ydCIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIwIj4KICAgICAgICA8RGVzY3JpcHRpb24gVHJhbnNsYXRhYmxlPSIxIj5NYWluIG1lbnUgaXRlbSByZWdpc3RyYXRpb24uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8TmF2aWdhdGlvbj5Gcm9udGVuZDo6QWRtaW46Ok1vZHVsZVJlZ2lzdHJhdGlvbjo6TWFpbk1lbnU8L05hdmlnYXRpb24+CiAgICAgICAgPFZhbHVlPgogICAgICAgICAgICA8QXJyYXk+CiAgICAgICAgICAgICAgICA8RGVmYXVsdEl0ZW0gVmFsdWVUeXBlPSJGcm9udGVuZE5hdmlnYXRpb24iPgogICAgICAgICAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICAgICAgICAgIDwvRGVmYXVsdEl0ZW0+CiAgICAgICAgICAgIDwvQXJyYXk+CiAgICAgICAgPC9WYWx1ZT4KICAgIDwvU2V0dGluZz4KICAgIDxTZXR0aW5nIE5hbWU9IkZyb250ZW5kOjpOYXZpZ2F0aW9uTW9kdWxlIyMjQWRtaW5JbXBvcnRFeHBvcnQiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIFRyYW5zbGF0YWJsZT0iMSI+QWRtaW4gYXJlYSBuYXZpZ2F0aW9uIGZvciB0aGUgYWdlbnQgaW50ZXJmYWNlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPE5hdmlnYXRpb24+RnJvbnRlbmQ6OkFkbWluOjpNb2R1bGVSZWdpc3RyYXRpb246OkFkbWluT3ZlcnZpZXc8L05hdmlnYXRpb24+CiAgICAgICAgPFZhbHVlPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iR3JvdXAiPgogICAgICAgICAgICAgICAgICAgIDxBcnJheT4KICAgICAgICAgICAgICAgICAgICAgICAgPEl0ZW0+YWRtaW48L0l0ZW0+CiAgICAgICAgICAgICAgICAgICAgPC9BcnJheT4KICAgICAgICAgICAgICAgIDwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iR3JvdXBSbyI+CiAgICAgICAgICAgICAgICAgICAgPEFycmF5PgogICAgICAgICAgICAgICAgICAgIDwvQXJyYXk+CiAgICAgICAgICAgICAgICA8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik1vZHVsZSI+S2VybmVsOjpPdXRwdXQ6OkhUTUw6Ok5hdkJhcjo6TW9kdWxlQWRtaW48L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik5hbWUiIFRyYW5zbGF0YWJsZT0iMSI+SW1wb3J0L0V4cG9ydDwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iQmxvY2siPkFkbWluaXN0cmF0aW9uPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJEZXNjcmlwdGlvbiI+TWFuYWdlIGltcG9ydCBhbmQgZXhwb3J0IG9mIG9iamVjdHMuPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJJY29uQmlnIj5mYS1hcnJvd3MtaDwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iSWNvblNtYWxsIj48L0l0ZW0+CiAgICAgICAgICAgIDwvSGFzaD4KICAgICAgICA8L1ZhbHVlPgogICAgPC9TZXR0aW5nPgogICAgPFNldHRpbmcgTmFtZT0iSW1wb3J0RXhwb3J0OjpGb3JtYXRCYWNrZW5kUmVnaXN0cmF0aW9uIyMjQ1NWIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBUcmFuc2xhdGFibGU9IjEiPkZvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxOYXZpZ2F0aW9uPkNvcmU6OkltcG9ydEV4cG9ydDwvTmF2aWdhdGlvbj4KICAgICAgICA8VmFsdWU+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJNb2R1bGUiPktlcm5lbDo6U3lzdGVtOjpJbXBvcnRFeHBvcnQ6OkZvcm1hdEJhY2tlbmQ6OkNTVjwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iTmFtZSI+Q1NWPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9WYWx1ZT4KICAgIDwvU2V0dGluZz4KPC9vZm9ya19jb25maWc+Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9kZV9JbXBvcnRFeHBvcnQucG0KIyBNb2RpZmllZCB2ZXJzaW9uIG9mIHRoZSB3b3JrOgojIENvcHlyaWdodCAoQykgMjAxMC0yMDE4IE9GT1JLLCBodHRwczovL28tZm9yay5kZQojIGJhc2VkIG9uIHRoZSBvcmlnaW5hbCB3b3JrIG9mOgojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLmNvbS8KIyAtLQojICRJZDogZGVfSW1wb3J0RXhwb3J0LnBtLHYgMS4xLjEuMSAyMDE4LzEwLzAyIDE1OjE2OjA0IHVkIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmRlX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdJbXBvcnQvRXhwb3J0LVZlcndhbHR1bmcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0Vyc3RlbGxlbiBlaW5lciBWb3JsYWdlIHp1bSBJbXBvcnRpZXJlbiB1bmQgRXhwb3J0aWVyZW4gdm9uIE9iamVrdC1JbmZvcm1hdGlvbmVuLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdJbXBvcnQgc3RhcnRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdFeHBvcnQgc3RhcnRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDEgb2YgNSAtIEVkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnU2Nocml0dCAxIHZvbiA1IC0gQWxsZ2VtZWluZSBJbmZvcm1hdGlvbmVuIGJlYXJiZWl0ZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICdOYW1lIHdpcmQgYmVuw7Z0aWd0ISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnT2JqZWt0IGlzdCBlcmZvcmRlcmxpY2ghJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICdGb3JtYXQgaXN0IGVyZm9yZGVybGljaCEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAyIG9mIDUgLSBFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJ1NjaHJpdHQgMiB2b24gNSAtIE9iamVrdC1JbmZvcm1hdGlvbmVuIGJlYXJiZWl0ZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAzIG9mIDUgLSBFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJ1NjaHJpdHQgMyB2b24gNSAtIEZvcm1hdGluZm9ybWF0aW9uZW4gYmVhcmJlaXRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpcyByZXF1aXJlZCEnfSA9ICd3aXJkIGJlbsO2dGlndCEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA0IG9mIDUgLSBFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICdTY2hyaXR0IDQgdm9uIDUgLSBNYXBwaW5nLUluZm9ybWF0aW9uZW4gYmVhcmJlaXRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnS2VpbmUgTWFwcGluZy1FbGVtZW50ZSBnZWZ1bmRlbi4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ01hcHBpbmctRWxlbWVudCBoaW56dWbDvGdlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDUgb2YgNSAtIEVkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnU2Nocml0dCA1IHZvbiA1IC0gU3VjaC1JbmZvcm1hdGlvbmVuIGJlYXJiZWl0ZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdFeHBvcnQgcGVyIFN1Y2hlIGVpbnNjaHLDpG5rZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW1wb3J0LUluZm9ybWF0aW9uZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdRdWVsbC1EYXRlaSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgc3VtbWFyeSBmb3IgJXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ0RhdGVuc8OkdHplJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICdFcmZvbGdyZWljaCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICdEb3BwZWx0ZSBOYW1lbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJ1p1bGV0enQgdmVyYXJiZWl0ZXRlIFplaWxlIGRlciBJbXBvcnQtRGF0ZWknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPayc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJ0tlaW4gQmFja2VuZCBmw7xyIGRhcyBPYmpla3QgdmVyZsO8Z2JhciEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gZm9ybWF0IGJhY2tlbmQgZm91bmQhJ30gPSAnS2VpbiBCYWNrZW5kIGbDvHIgZGFzIEZvcm1hdCB2ZXJmw7xnYmFyISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBub3QgZm91bmQhJ30gPSAnVm9ybGFnZSBuaWNodCB2ZXJmw7xnYmFyISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICdEZWkgVm9ybGFnZSBrb25udGUgbmljaHQgZXJzdGVsbHQgb2RlciBha3R1YWxpc2llcnQgd2VyZGVuISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOZWVkZWQgVGVtcGxhdGVJRCEnfSA9ICdEaWUgSUQgZGVyIFZvcmxhZ2Ugd2lyZCBiZW7DtnRpZ3QhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICdGZWhsZXIgYXVmZ2V0cmV0ZW4uIEltcG9ydGllcmVuIHVubcO2Z2xpY2ghIEbDvHIgRGV0YWlscyBzaWVoZSBTeXNsb2cuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBFeHBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICdGZWhsZXIgYXVmZ2V0cmV0ZW4uIEV4cG9ydGllcmVuIHVubcO2Z2xpY2ghIEbDvHIgRGV0YWlscyBzaWVoZSBTeXNsb2cuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RlbXBsYXRlIExpc3QnfSA9ICdWb3JsYWdlbmxpc3RlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlcid9ID0gJ1phaGwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnbnVtYmVyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICdaYWhsIGdyw7bDn2VyIGFscyBOdWxsJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXInfSA9ICdHYW56emFobCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICdHYW56emFobCBncsO2w59lciBhbHMgTnVsbCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICdFbGVtZW50IGVyZm9yZGVybGljaCwgYml0dGUgZsO8Z2VuIFNpZSBEYXRlbiBlaW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW52YWxpZCBkYXRhLCBwbGVhc2UgaW5zZXJ0IGEgdmFsaWQgJXMnfSA9ICdVbmfDvGx0aWdlIERhdGVuLCBiaXR0ZSBnZWJlbiBTaWUgZWluIGfDvGx0aWdlcyAlcyBlaW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IG5vdCBmb3VuZCEnfSA9ICdGb3JtYXQgbmljaHQgZ2VmdW5kZW4hJzsKCiAgICAjIFBlcmwgTW9kdWxlOiBLZXJuZWwvU3lzdGVtL0ltcG9ydEV4cG9ydC9Gb3JtYXRCYWNrZW5kL0NTVi5wbQogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uIFNlcGFyYXRvcid9ID0gJ1NwYWx0ZW50cmVubmVyJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYXRvciAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTZW1pY29sb24gKDspJ30gPSAnU2VtaWNvbG9uICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEb3BwZWxwdW5rdCAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ1B1bmt0ICguKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb21tYSAoLCknfSA9ICdLb21tYSAoLCknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ1plaWNoZW5zYXR6JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICdNaXQgU3BhbHRlbsO8YmVyc2NocmlmdGVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ1NwYWx0ZSc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ0Zvcm1hdC1CYWNrZW5kIE1vZHVsLVJlZ2lzdHJhdGlvbiBkZXMgSW1wb3J0L0V4cG9ydCBNb2R1bHMuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdJbXBvcnRpZXJlbiB1bmQgRXhwb3J0aWVyZW4gdm9uIE9iamVrdC1JbmZvcm1hdGlvbmVuLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3J0L0V4cG9ydCc7CgoKICAgIHB1c2ggQHsgJFNlbGYtPntKYXZhU2NyaXB0U3RyaW5nc30gLy8gW10gfSwgKAogICAgKTsKCn0KCjE7Cg==
# --
# Kernel/Modules/AdminImportExport.pm
# Modified version of the work:
# Copyright (C) 2010-2018 OFORK, https://o-fork.de
# based on the original work of:
# Copyright (C) 2001-2018 OTRS AG, http://otrs.com/
# --
# $Id: AdminImportExport.pm,v 1.1.1.1 2018/10/02 15:16:04 ud Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::Modules::AdminImportExport;

use strict;
use warnings;

use Kernel::Language qw(Translatable);

our $ObjectManagerDisabled = 1;

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

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

    return $Self;
}

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

    # get needed objects
    my $ImportExportObject = $Kernel::OM->Get('Kernel::System::ImportExport');
    my $ParamObject        = $Kernel::OM->Get('Kernel::System::Web::Request');
    my $LayoutObject       = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

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

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

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

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

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

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

        my $SaveContinue = $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 = $ImportExportObject->TemplateGet(
                TemplateID => $TemplateID,
                UserID     => $Self->{UserID},
            );

            if ( !$TemplateData->{TemplateID} ) {
                $LayoutObject->FatalError(
                    Message => Translatable('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} = $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} = $ImportExportObject->TemplateAdd(
                %{$TemplateData},
                UserID => $Self->{UserID},
            );
            $Success = $TemplateData->{TemplateID};
        }
        else {
            $Success = $ImportExportObject->TemplateUpdate(
                UserID => $Self->{UserID},
                %{$TemplateData},
            );
        }

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

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

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

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

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

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

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

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

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

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

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

        # save template starts here

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

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

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

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

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

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

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

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

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

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

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

        # save starting here

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

        # get format data
        my $FormatData = $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} } = $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
        $ImportExportObject->FormatDataSave(
            TemplateID => $TemplateID,
            FormatData => \%AttributeValues,
            UserID     => $Self->{UserID},
        );

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

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

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

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

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

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

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

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

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

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

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

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

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

        $LayoutObject->AddJSData(
            Key   => 'TemplateEdit4',
            Value => 1,
        );

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

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

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

        # get format attributes
        my $MappingFormatAttributes = $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
            $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
            $LayoutObject->Block(
                Name => 'TemplateEdit4Row',
                Data => {
                    MappingID => $MappingID,
                },
            );

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

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

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

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

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

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

                # output column counter
                $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
            $LayoutObject->Block(
                Name => $UpBlock,
                Data => { MappingID => $MappingID },
            );

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

            $AttributeRowCounter++;
        }

        # output an empty list
        if ($EmptyMap) {

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

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

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

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

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

        # get template id
        my $TemplateID = $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 !$ParamObject->GetParam( Param => $SubmitKey );

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

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

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

        # get format attributes
        my $MappingFormatAttributes = $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} } = $LayoutObject->ImportExportFormDataGet(
                    Item   => $Item,
                    Prefix => 'Object::' . $Counter . '::',
                );
            }

            # save the mapping object data
            $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} } = $LayoutObject->ImportExportFormDataGet(
                    Item   => $Item,
                    Prefix => 'Format::' . $Counter . '::',
                );
            }

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

            $Counter++;
        }

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

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

                next MAPPINGID;
            }

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

                next MAPPINGID;
            }

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

                next MAPPINGID;
            }
        }

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

        # get template id
        my $TemplateID = $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 !$ParamObject->GetParam( Param => $SubmitKey );

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

        $SourceFile{Content} ||= '';

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

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

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

        # output import results
        $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 {
                $LayoutObject->Block(
                    Name => 'ImportResultReturnCode',
                    Data => {
                        ReturnCodeName  => $RetCode,
                        ReturnCodeCount => $Result->{RetCode}->{$RetCode},
                    },
                );
            }
        }

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

        # get valid list
        my %ValidList = $Kernel::OM->Get('Kernel::System::Valid')->ValidList();

        my $EmptyDatabase = 1;

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

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

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

            $EmptyDatabase = 0;

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

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

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

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

        # output an empty list
        if ($EmptyDatabase) {

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

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

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

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

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

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

    # get layout object
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

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

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

    # generate ValidOptionStrg
    my %ValidList        = $Kernel::OM->Get('Kernel::System::Valid')->ValidList();
    my %ValidListReverse = reverse %ValidList;
    my $ValidOptionStrg  = $LayoutObject->BuildSelection(
        Name       => 'ValidID',
        Data       => \%ValidList,
        SelectedID => $Param{ValidID} || $ValidListReverse{valid},
        Class      => 'Modernize',
    );

    my $Class = ' Validate_Required ';

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

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

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

    if ( $Param{New} ) {

        # get ImportExport object
        my $ImportExportObject = $Kernel::OM->Get('Kernel::System::ImportExport');

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

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

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

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

        $Class = ' Validate_Required ';

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

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

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

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

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

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

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

    $Output .= $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} };
    }

    # get layout object
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

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

    # get ImportExport object
    my $ImportExportObject = $Kernel::OM->Get('Kernel::System::ImportExport');

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

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

    $LayoutObject->AddJSData(
        Key   => 'BackURL',
        Value => "Action=$Self->{Action};Subaction=TemplateEdit1;TemplateID=$TemplateID",
    );

    $LayoutObject->AddJSData(
        Key   => 'BaseLink',
        Value => $LayoutObject->{Baselink},
    );

    $LayoutObject->AddJSData(
        Key   => 'TemplateOverview',
        Value => 1,
    );

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

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

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

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

    # get object data
    my $ObjectData = $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} = $LayoutObject->{LanguageObject}->Translate('number');
    $PredefinedErrorMessages{NumberBiggerThanZero}
        = $LayoutObject->{LanguageObject}->Translate('number bigger than zero');
    $PredefinedErrorMessages{Integer} = $LayoutObject->{LanguageObject}->Translate('integer');
    $PredefinedErrorMessages{IntegerBiggerThanZero}
        = $LayoutObject->{LanguageObject}->Translate('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 = $LayoutObject->{LanguageObject}->Translate('Element required, please insert data');
        }

        if ( $Item->{Input}->{DataType} ) {
            $Class .= " $JSClass{ $Item->{Input}->{DataType} }";
            $ErrorMessage = $LayoutObject->{LanguageObject}->Translate(
                'Invalid data, please insert a valid %s',
                $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 = $LayoutObject->ImportExportFormInputCreate(
            Item  => $Item,
            Class => $Class . ' Modernize',
            Value => $Value,
        );

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

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

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

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

    $Output .= $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};
    }

    # get layout object
    my $LayoutObject = $Kernel::OM->Get('Kernel::Output::HTML::Layout');

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

    # get ImportExport object
    my $ImportExportObject = $Kernel::OM->Get('Kernel::System::ImportExport');

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

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

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

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

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

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

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

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

    if ( !$FormatData ) {
        $LayoutObject->FatalError(
            Message => Translatable('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 = $LayoutObject->ImportExportFormInputCreate(
            Item  => $Item,
            Class => $Class . ' Modernize',
            Value => $FormatData->{ $Item->{Key} },
        );

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

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

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

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

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

}

1;

IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnQvTGF5b3V0Q2hlY2tib3gucG0KIyBNb2RpZmllZCB2ZXJzaW9uIG9mIHRoZSB3b3JrOgojIENvcHlyaWdodCAoQykgMjAxMC0yMDE4IE9GT1JLLCBodHRwczovL28tZm9yay5kZQojIGJhc2VkIG9uIHRoZSBvcmlnaW5hbCB3b3JrIG9mOgojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLmNvbS8KIyAtLQojICRJZDogTGF5b3V0Q2hlY2tib3gucG0sdiAxLjEuMS4xIDIwMTgvMTAvMDIgMTU6MTY6MDQgdWQgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydDo6TGF5b3V0Q2hlY2tib3g7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6TG9nJywKICAgICdLZXJuZWw6OlN5c3RlbTo6V2ViOjpSZXF1ZXN0JywKKTsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0OjpMYXlvdXRDaGVja2JveCAtIGxheW91dCBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIERFU0NSSVBUSU9OCgpBbGwgbGF5b3V0IGZ1bmN0aW9ucyBmb3IgY2hlY2tib3ggZWxlbWVudHMgaW4gaW1wb3J0L2V4cG9ydC4KCj1jdXQKCj1oZWFkMiBuZXcoKQoKQ3JlYXRlIGFuIG9iamVjdAoKICAgICRCYWNrZW5kT2JqZWN0ID0gS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydDo6TGF5b3V0Q2hlY2tib3gtPm5ldygKICAgICAgICAlUGFyYW0sCiAgICApOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1oZWFkMiBGb3JtSW5wdXRDcmVhdGUoKQoKQ3JlYXRlIGEgaW5wdXQgc3RyaW5nCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkZvcm1JbnB1dENyZWF0ZSgKICAgICAgICBJdGVtICAgPT4gJEl0ZW1SZWYsCiAgICAgICAgUHJlZml4ID0+ICdQcmVmaXg6OicsICAjIChvcHRpb25hbCkKICAgICAgICBWYWx1ZSAgPT4gJ1ZhbHVlJywgICAgICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBGb3JtSW5wdXRDcmVhdGUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgaWYgKCAhJFBhcmFte0l0ZW19ICkgewogICAgICAgICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAnTmVlZCBJdGVtIScsCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgJFBhcmFte1ByZWZpeH0gfHw9ICcnOwoKICAgIG15ICRDaGVja2VkID0gJFBhcmFte1ZhbHVlfSA/ICdjaGVja2VkPSJjaGVja2VkIicgOiAnJzsKCiAgICByZXR1cm4KICAgICAgICBxcXs8aW5wdXQgaWQ9IiRQYXJhbXtQcmVmaXh9JFBhcmFte0l0ZW19LT57S2V5fSIgdHlwZT0iY2hlY2tib3giIG5hbWU9IiRQYXJhbXtQcmVmaXh9JFBhcmFte0l0ZW19LT57S2V5fSIgJENoZWNrZWQgLz59Owp9Cgo9aGVhZDIgRm9ybURhdGFHZXQoKQoKR2V0IGZvcm0gZGF0YQoKICAgIG15ICRGb3JtRGF0YSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtRGF0YUdldCgKICAgICAgICBJdGVtICAgPT4gJEl0ZW1SZWYsCiAgICAgICAgUHJlZml4ID0+ICdQcmVmaXg6OicsICAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgRm9ybURhdGFHZXQgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgaWYgKCAhJFBhcmFte0l0ZW19ICkgewogICAgICAgICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAnTmVlZCBJdGVtIScsCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgJFBhcmFte1ByZWZpeH0gfHw9ICcnOwoKICAgICMgZ2V0IGZvcm0gZGF0YQogICAgbXkgJEZvcm1EYXRhID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OldlYjo6UmVxdWVzdCcpLT5HZXRQYXJhbSgKICAgICAgICBQYXJhbSA9PiAkUGFyYW17UHJlZml4fSAuICRQYXJhbXtJdGVtfS0+e0tleX0sCiAgICApOwoKICAgIHJldHVybiAkRm9ybURhdGE7Cn0KCjE7Cgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKCkxheW91dENoZWNrYm94ClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIEw8aHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0Pi4KCj1jdXQK
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnQvTGF5b3V0VFQucG0KIyBNb2RpZmllZCB2ZXJzaW9uIG9mIHRoZSB3b3JrOgojIENvcHlyaWdodCAoQykgMjAxMC0yMDE4IE9GT1JLLCBodHRwczovL28tZm9yay5kZQojIGJhc2VkIG9uIHRoZSBvcmlnaW5hbCB3b3JrIG9mOgojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLmNvbS8KIyAtLQojICRJZDogTGF5b3V0VFQucG0sdiAxLjEuMS4xIDIwMTgvMTAvMDIgMTU6MTY6MDQgdWQgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydDo6TGF5b3V0VFQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6TG9nJywKKTsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0OjpMYXlvdXRUVCAtIGxheW91dCBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIERFU0NSSVBUSU9OCgpBbGwgbGF5b3V0IGZ1bmN0aW9ucyBmb3IgZGlzcGxheSBDPFRUPiBjb2RlCgo9Y3V0Cgo9aGVhZDIgbmV3KCkKCkNyZWF0ZSBhbiBvYmplY3QKCiAgICAkQmFja2VuZE9iamVjdCA9IEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnQ6OkxheW91dFRULT5uZXcoCiAgICAgICAgJVBhcmFtLAogICAgKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aGVhZDIgRm9ybUlucHV0Q3JlYXRlKCkKCkNyZWF0ZSBhIGlucHV0IHN0cmluZwoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtSW5wdXRDcmVhdGUoCiAgICAgICAgSXRlbSA9PiAkSXRlbVJlZiwKICAgICk7Cgo9Y3V0CgpzdWIgRm9ybUlucHV0Q3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgSXRlbSEnCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgcmV0dXJuICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e0RhdGF9Owp9Cgo9aGVhZDIgRm9ybURhdGFHZXQoKQoKR2V0IGZvcm0gZGF0YQoKICAgIG15ICRGb3JtRGF0YSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtRGF0YUdldCgpOwoKPWN1dAoKc3ViIEZvcm1EYXRhR2V0IHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuOwp9CgoxOwoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIHNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9GT1JLIHByb2plY3QgKEw8aHR0cHM6Ly9vLWZvcmsuZGUvPikuCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dD4uCgo9Y3V0Cg==
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnQvTGF5b3V0U2VsZWN0aW9uLnBtCiMgTW9kaWZpZWQgdmVyc2lvbiBvZiB0aGUgd29yazoKIyBDb3B5cmlnaHQgKEMpIDIwMTAtMjAxOCBPRk9SSywgaHR0cHM6Ly9vLWZvcmsuZGUKIyBiYXNlZCBvbiB0aGUgb3JpZ2luYWwgd29yayBvZjoKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxOCBPVFJTIEFHLCBodHRwOi8vb3Rycy5jb20vCiMgLS0KIyAkSWQ6IExheW91dFNlbGVjdGlvbi5wbSx2IDEuMS4xLjEgMjAxOC8xMC8wMiAxNToxNjowNCB1ZCBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0OjpMYXlvdXRTZWxlY3Rpb247Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6TG9nJywKICAgICdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0JywKICAgICdLZXJuZWw6OlN5c3RlbTo6V2ViOjpSZXF1ZXN0JywKKTsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0OjpMYXlvdXRTZWxlY3Rpb24gLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBERVNDUklQVElPTgoKQWxsIGxheW91dCBmdW5jdGlvbnMgZm9yIHNlbGVjdGlvbiBlbGVtZW50cwoKPWN1dAoKPWhlYWQyIG5ldygpCgpDcmVhdGUgYW4gb2JqZWN0CgogICAgJEJhY2tlbmRPYmplY3QgPSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0OjpMYXlvdXRTZWxlY3Rpb24tPm5ldygKICAgICAgICAlUGFyYW0sCiAgICApOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1oZWFkMiBGb3JtSW5wdXRDcmVhdGUoKQoKQ3JlYXRlIGEgaW5wdXQgc3RyaW5nCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkZvcm1JbnB1dENyZWF0ZSgKICAgICAgICBJdGVtICAgPT4gJEl0ZW1SZWYsCiAgICAgICAgUHJlZml4ID0+ICdQcmVmaXg6OicsICAjIChvcHRpb25hbCkKICAgICAgICBWYWx1ZSAgPT4gJ1ZhbHVlJywgICAgICMgKG9wdGlvbmFsKQogICAgICAgIENsYXNzICA9PiAnTW9kZXJuaXplJyAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1JbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpLT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIEl0ZW0hJwogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICMgc2V0IGRlZmF1bHQgdmFsdWUKICAgICRQYXJhbXtQcmVmaXh9IHx8PSAnJzsKICAgICRQYXJhbXtWYWx1ZX0gIHx8PSAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntWYWx1ZURlZmF1bHR9OwoKICAgIGlmICggJFBhcmFte1ZhbHVlfSAmJiAkUGFyYW17VmFsdWV9ID1+IG17ICMjIyMjIH14bXMgKSB7CiAgICAgICAgbXkgQFZhbHVlcyA9IHNwbGl0ICcjIyMjIycsICRQYXJhbXtWYWx1ZX07CiAgICAgICAgJFBhcmFte1ZhbHVlfSA9IFxAVmFsdWVzOwogICAgfQoKICAgICMgZ2VuZXJhdGUgb3B0aW9uIHN0cmluZwogICAgbXkgJFN0cmluZyA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnKS0+QnVpbGRTZWxlY3Rpb24oCiAgICAgICAgSUQgICAgICAgICAgID0+ICRQYXJhbXtQcmVmaXh9IC4gJFBhcmFte0l0ZW19LT57S2V5fSwKICAgICAgICBDbGFzcyAgICAgICAgPT4gJFBhcmFte0NsYXNzfSwKICAgICAgICBOYW1lICAgICAgICAgPT4gJFBhcmFte1ByZWZpeH0gLiAkUGFyYW17SXRlbX0tPntLZXl9LAogICAgICAgIERhdGEgICAgICAgICA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntEYXRhfSB8fCB7fSwKICAgICAgICBTZWxlY3RlZElEICAgPT4gJFBhcmFte1ZhbHVlfSwKICAgICAgICBUcmFuc2xhdGlvbiAgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VHJhbnNsYXRpb259LAogICAgICAgIFRyZWVWaWV3ICAgICA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntUcmVlVmlld30gfHwgMCwKICAgICAgICBQb3NzaWJsZU5vbmUgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57UG9zc2libGVOb25lfSwKICAgICAgICBNdWx0aXBsZSAgICAgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57TXVsdGlwbGV9LAogICAgICAgIFNpemUgICAgICAgICA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntTaXplfSwKICAgICk7CgogICAgcmV0dXJuICRTdHJpbmc7Cn0KCj1oZWFkMiBGb3JtRGF0YUdldCgpCgpHZXQgZm9ybSBkYXRhCgogICAgbXkgJEZvcm1EYXRhID0gJEJhY2tlbmRPYmplY3QtPkZvcm1EYXRhR2V0KAogICAgICAgIEl0ZW0gICA9PiAkSXRlbVJlZiwKICAgICAgICBQcmVmaXggPT4gJ1ByZWZpeDo6JywgICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBGb3JtRGF0YUdldCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpLT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIEl0ZW0hJwogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICRQYXJhbXtQcmVmaXh9IHx8PSAnJzsKCiAgICAjIGdldCBmb3JtIGRhdGEKICAgIG15IEBGb3JtRGF0YXMgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6V2ViOjpSZXF1ZXN0JyktPkdldEFycmF5KAogICAgICAgIFBhcmFtID0+ICRQYXJhbXtQcmVmaXh9IC4gJFBhcmFte0l0ZW19LT57S2V5fSwKICAgICk7CgogICAgbXkgJEZvcm1EYXRhID0gam9pbiAnIyMjIyMnLCBARm9ybURhdGFzOwoKICAgIHJldHVybiAkRm9ybURhdGEgaWYgJEZvcm1EYXRhOwogICAgcmV0dXJuICRGb3JtRGF0YSBpZiAhJFBhcmFte0l0ZW19LT57SW5wdXR9LT57UmVxdWlyZWR9OwoKICAgICMgc2V0IGludmFsaWQgcGFyYW0KICAgICRQYXJhbXtJdGVtfS0+e0Zvcm19LT57SW52YWxpZH0gPSAxOwoKICAgIHJldHVybiAkRm9ybURhdGE7Cn0KCjE7Cgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKClRoaXMgc29mdHdhcmUgaXMgcGFydCBvZiB0aGUgT0ZPUksgcHJvamVjdCAoTDxodHRwczovL28tZm9yay5kZS8+KS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIEw8aHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0Pi4KCj1jdXQK
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9JbXBvcnRFeHBvcnQvTGF5b3V0VGV4dC5wbQojIE1vZGlmaWVkIHZlcnNpb24gb2YgdGhlIHdvcms6CiMgQ29weXJpZ2h0IChDKSAyMDEwLTIwMTggT0ZPUkssIGh0dHBzOi8vby1mb3JrLmRlCiMgYmFzZWQgb24gdGhlIG9yaWdpbmFsIHdvcmsgb2Y6CiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTggT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgJElkOiBMYXlvdXRUZXh0LnBtLHYgMS4xLjEuMSAyMDE4LzEwLzAyIDE1OjE2OjA0IHVkIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnQ6OkxheW91dFRleHQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6TG9nJywKICAgICdLZXJuZWw6OlN5c3RlbTo6V2ViOjpSZXF1ZXN0JywKICAgICdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0JywKKTsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0OjpMYXlvdXRUZXh0IC0gbGF5b3V0IGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgREVTQ1JJUFRJT04KCkFsbCBsYXlvdXQgZnVuY3Rpb25zIGZvciB0ZXh0IGVsZW1lbnRzCgo9Y3V0Cgo9aGVhZDIgbmV3KCkKCkNyZWF0ZSBhbiBvYmplY3QKCiAgICAkQmFja2VuZE9iamVjdCA9IEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnQ6OkxheW91dFRleHQtPm5ldygKICAgICAgICAlUGFyYW0sCiAgICApOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1oZWFkMiBGb3JtSW5wdXRDcmVhdGUoKQoKQ3JlYXRlIGEgaW5wdXQgc3RyaW5nCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkZvcm1JbnB1dENyZWF0ZSgKICAgICAgICBJdGVtICAgICA9PiAkSXRlbVJlZiwKICAgICAgICBQcmVmaXggICA9PiAnUHJlZml4OjonLCAgIyAob3B0aW9uYWwpCiAgICAgICAgVmFsdWUgICAgPT4gJ1ZhbHVlJywgICAgICMgKG9wdGlvbmFsKQogICAgICAgIFJlYWRvbmx5ID0+IDEsICAgICAgICAgICAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgRm9ybUlucHV0Q3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgSXRlbSEnLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICRQYXJhbXtQcmVmaXh9IHx8PSAnJzsKCiAgICBteSAkVmFsdWUgPSAkUGFyYW17VmFsdWV9IHx8ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1ZhbHVlRGVmYXVsdH07CiAgICBteSAkU2l6ZSA9ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1NpemV9IHx8IDQwOwogICAgbXkgJFNpemVDbGFzczsKICAgIGlmICggJFNpemUgPCAxNSApIHsKICAgICAgICAkU2l6ZUNsYXNzID0gJ1cxMHBjJzsKICAgIH0KICAgIGVsc2lmICggJFNpemUgPCAzNSApIHsKICAgICAgICAkU2l6ZUNsYXNzID0gJ1czM3BjJzsKICAgIH0KICAgIGVsc2lmICggJFNpemUgPCA1MCApIHsKICAgICAgICAkU2l6ZUNsYXNzID0gJ1c1MHBjJzsKICAgIH0KICAgIGVsc2UgewogICAgICAgICRTaXplQ2xhc3MgPSAnVzc1cGMnOwogICAgfQoKICAgICMgcHJlcGFyZSBkYXRhCiAgICBteSAkSUQgPSAoICRQYXJhbXtQcmVmaXh9IHx8ICcnICkgLiAoICRQYXJhbXtJdGVtfS0+e0tleX0gKTsKICAgIG15ICROYW1lID0gKCAkUGFyYW17UHJlZml4fSB8fCAnJyApIC4gKCAkUGFyYW17TmFtZX0gfHwgJElEICk7CiAgICBteSAkQ2xhc3MgPSAoICRTaXplQ2xhc3MgfHwgJycgKSAuICggJFBhcmFte0NsYXNzfSB8fCAnJyApOwoKICAgIG15ICRTdHJpbmcgPSAiPGlucHV0IGlkPVwiJElEXCIgdHlwZT1cInRleHRcIiBuYW1lPVwiJE5hbWVcIiBjbGFzcz1cIiRDbGFzc1wiICI7CgogICAgaWYgKCRWYWx1ZSkgewoKICAgICAgICAjIGdldCBsYXlvdXQgb2JqZWN0CiAgICAgICAgbXkgJExheW91dE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnKTsKCiAgICAgICAgIyB0cmFuc2xhdGUKICAgICAgICBpZiAoICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1RyYW5zbGF0aW9ufSApIHsKICAgICAgICAgICAgJFZhbHVlID0gJExheW91dE9iamVjdC0+e0xhbmd1YWdlT2JqZWN0fS0+VHJhbnNsYXRlKCRWYWx1ZSk7CiAgICAgICAgfQoKICAgICAgICAjIHRyYW5zZm9ybSBhc2NpaSB0byBodG1sCiAgICAgICAgJFZhbHVlID0gJExheW91dE9iamVjdC0+QXNjaWkySHRtbCgKICAgICAgICAgICAgVGV4dCAgICAgICAgICAgPT4gJFZhbHVlLAogICAgICAgICAgICBIVE1MUmVzdWx0TW9kZSA9PiAxLAogICAgICAgICk7CgogICAgICAgICRTdHJpbmcgLj0gInZhbHVlPVwiJFZhbHVlXCIgIjsKICAgIH0KCiAgICAjIGFkZCBtYXhpbXVtIGxlbmd0aAogICAgaWYgKCAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntNYXhMZW5ndGh9ICkgewogICAgICAgICRTdHJpbmcgLj0gIm1heGxlbmd0aD1cIiRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e01heExlbmd0aH1cIiAiOwogICAgfQoKICAgICMgYWRkIHJlYWRvbmx5CiAgICBpZiAoICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1JlYWRvbmx5fSApIHsKICAgICAgICAkU3RyaW5nIC49ICdyZWFkb25seT0icmVhZG9ubHkiICc7CiAgICB9CgogICAgJFN0cmluZyAuPSAiLz4gIjsKCiAgICByZXR1cm4gJFN0cmluZzsKfQoKPWhlYWQyIEZvcm1EYXRhR2V0KCkKCkdldCBmb3JtIGRhdGEKCiAgICBteSAkRm9ybURhdGEgPSAkQmFja2VuZE9iamVjdC0+Rm9ybURhdGFHZXQoCiAgICAgICAgSXRlbSAgID0+ICRJdGVtUmVmLAogICAgICAgIFByZWZpeCA9PiAnUHJlZml4OjonLCAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1EYXRhR2V0IHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgSXRlbSEnCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgJFBhcmFte1ByZWZpeH0gfHw9ICcnOwoKICAgICMgZ2V0IGZvcm0gZGF0YQogICAgbXkgJEZvcm1EYXRhID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OldlYjo6UmVxdWVzdCcpLT5HZXRQYXJhbSgKICAgICAgICBQYXJhbSA9PiAkUGFyYW17UHJlZml4fSAuICRQYXJhbXtJdGVtfS0+e0tleX0sCiAgICApOwoKICAgICMgcmVnZXggY2hlY2sKICAgIGlmICggJFBhcmFte0l0ZW19LT57SW5wdXR9LT57UmVnZXh9ICYmICRGb3JtRGF0YSAhfiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntSZWdleH0gKSB7CgogICAgICAgICRQYXJhbXtJdGVtfS0+e0Zvcm19LT57SW52YWxpZH0gPSAxOwogICAgICAgIHJldHVybiAkRm9ybURhdGE7CiAgICB9CgogICAgcmV0dXJuICRGb3JtRGF0YSBpZiAkRm9ybURhdGE7CiAgICByZXR1cm4gJEZvcm1EYXRhIGlmICEkUGFyYW17SXRlbX0tPntJbnB1dH0tPntSZXF1aXJlZH07CgogICAgIyBzZXQgaW52YWxpZCBwYXJhbQogICAgJFBhcmFte0l0ZW19LT57Rm9ybX0tPntJbnZhbGlkfSA9IDE7CgogICAgcmV0dXJuICRGb3JtRGF0YTsKfQoKMTsKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPRk9SSyBwcm9qZWN0IChMPGh0dHBzOi8vby1mb3JrLmRlLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQ+LgoKPWN1dAo=
IyAtLQojIEtlcm5lbC9PdXRwdXQvSFRNTC9MYXlvdXQvSW1wb3J0RXhwb3J0LnBtCiMgTW9kaWZpZWQgdmVyc2lvbiBvZiB0aGUgd29yazoKIyBDb3B5cmlnaHQgKEMpIDIwMTAtMjAxOCBPRk9SSywgaHR0cHM6Ly9vLWZvcmsuZGUKIyBiYXNlZCBvbiB0aGUgb3JpZ2luYWwgd29yayBvZjoKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAxOCBPVFJTIEFHLCBodHRwOi8vb3Rycy5jb20vCiMgLS0KIyAkSWQ6IEltcG9ydEV4cG9ydC5wbSx2IDEuMS4xLjEgMjAxOC8xMC8wMiAxNToxNjowNCB1ZCBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChBR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0OjpJbXBvcnRFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgJE9iamVjdE1hbmFnZXJEaXNhYmxlZCA9IDE7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkxheW91dDo6SW1wb3J0RXhwb3J0Cgo9Y3V0Cgo9aGVhZDIgSW1wb3J0RXhwb3J0Rm9ybUlucHV0Q3JlYXRlKCkKClJldHVybnMgYSBpbnB1dCBmaWVsZCBodG1sIHN0cmluZwoKICAgIG15ICRTdHJpbmcgPSAkTGF5b3V0T2JqZWN0LT5JbXBvcnRFeHBvcnRGb3JtSW5wdXRDcmVhdGUoCiAgICAgICAgSXRlbSAgPT4gJEl0ZW1SZWYsCiAgICAgICAgVmFsdWUgPT4gJ1ZhbHVlJywgICAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgSW1wb3J0RXhwb3J0Rm9ybUlucHV0Q3JlYXRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgSXRlbSEnCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBsb2FkIGJhY2tlbmQKICAgIG15ICRCYWNrZW5kT2JqZWN0ID0gJFNlbGYtPl9JbXBvcnRFeHBvcnRMb2FkTGF5b3V0QmFja2VuZCgKICAgICAgICBUeXBlID0+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1R5cGV9LAogICAgKTsKCiAgICByZXR1cm4gJycgaWYgISRCYWNrZW5kT2JqZWN0OwoKICAgICMgbG9va3VwIGl0ZW0gdmFsdWUKICAgIG15ICRTdHJpbmcgPSAkQmFja2VuZE9iamVjdC0+Rm9ybUlucHV0Q3JlYXRlKCVQYXJhbSk7CgogICAgcmV0dXJuICRTdHJpbmc7Cn0KCj1oZWFkMiBJbXBvcnRFeHBvcnRGb3JtRGF0YUdldCgpCgpSZXR1cm5zIHRoZSB2YWx1ZXMgZnJvbSB0aGUgaHRtbCBmb3JtIGFzIGhhc2ggcmVmZXJlbmNlCgogICAgbXkgJEZvcm1EYXRhID0gJExheW91dE9iamVjdC0+SW1wb3J0RXhwb3J0Rm9ybURhdGFHZXQoCiAgICAgICAgSXRlbSA9PiAkSXRlbVJlZiwKICAgICk7Cgo9Y3V0CgpzdWIgSW1wb3J0RXhwb3J0Rm9ybURhdGFHZXQgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgaWYgKCAhJFBhcmFte0l0ZW19ICkgewogICAgICAgICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAnTmVlZCBJdGVtIScKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAjIGxvYWQgYmFja2VuZAogICAgbXkgJEJhY2tlbmRPYmplY3QgPSAkU2VsZi0+X0ltcG9ydEV4cG9ydExvYWRMYXlvdXRCYWNrZW5kKAogICAgICAgIFR5cGUgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VHlwZX0sCiAgICApOwoKICAgIHJldHVybiBpZiAhJEJhY2tlbmRPYmplY3Q7CgogICAgIyBnZXQgZm9ybSBkYXRhCiAgICBteSAkRm9ybURhdGEgPSAkQmFja2VuZE9iamVjdC0+Rm9ybURhdGFHZXQoJVBhcmFtKTsKCiAgICByZXR1cm4gJEZvcm1EYXRhOwp9Cgo9aGVhZDIgX0ltcG9ydEV4cG9ydExvYWRMYXlvdXRCYWNrZW5kKCkKClRvIGxvYWQgYSBpbXBvcnQvZXhwb3J0IGxheW91dCBiYWNrZW5kIG1vZHVsZQoKICAgIG15ICRCYWNrZW5kID0gJExheW91dE9iamVjdC0+X0ltcG9ydEV4cG9ydExvYWRMYXlvdXRCYWNrZW5kKAogICAgICAgIFR5cGUgPT4gJ1NlbGVjdGlvbicsCiAgICApOwoKQW4gaW5zdGFuY2Ugb2YgdGhlIGxvYWRlZCBiYWNrZW5kIG1vZHVsZSBpcyByZXR1cm5lZC4KCj1jdXQKCnN1YiBfSW1wb3J0RXhwb3J0TG9hZExheW91dEJhY2tlbmQgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGdldCBsb2cgb2JqZWN0CiAgICBteSAkTG9nT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpOwoKICAgIGlmICggISRQYXJhbXtUeXBlfSApIHsKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIFR5cGUhJywKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICBteSAkR2VuZXJpY01vZHVsZSA9ICJLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0OjpMYXlvdXQkUGFyYW17VHlwZX0iOwoKICAgICMgbG9hZCB0aGUgYmFja2VuZCBtb2R1bGUKICAgIGlmICggISRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpNYWluJyktPlJlcXVpcmUoJEdlbmVyaWNNb2R1bGUpICkgewogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gIkNhbid0IGxvYWQgYmFja2VuZCBtb2R1bGUgJFBhcmFte1R5cGV9ISIsCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBjcmVhdGUgbmV3IGluc3RhbmNlCiAgICBteSAkQmFja2VuZE9iamVjdCA9ICRHZW5lcmljTW9kdWxlLT5uZXcoCiAgICAgICAgJXskU2VsZn0sCiAgICAgICAgJVBhcmFtLAogICAgICAgIExheW91dE9iamVjdCA9PiAkU2VsZiwKICAgICk7CgogICAgaWYgKCAhJEJhY2tlbmRPYmplY3QgKSB7CiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiQ2FuJ3QgY3JlYXRlIGEgbmV3IGluc3RhbmNlIG9mIGJhY2tlbmQgbW9kdWxlICRQYXJhbXtUeXBlfSEiLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIHJldHVybiAkQmFja2VuZE9iamVjdDsKfQoKMTsKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPRk9SSyBwcm9qZWN0IChMPGh0dHBzOi8vby1mb3JrLmRlLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQ+LgoKPWN1dAo=
# --
# Kernel/Output/HTML/Templates/Standard/AdminImportExport.tt
# Modified version of the work:
# Copyright (C) 2010-2019 OFORK, https://o-fork.de/
# based on the original work of:
# Copyright (C) 2010-2019 OFORK, https://o-fork.de/
# --
# $Id: AdminImportExport.tt,v 1.3 2019/06/02 08:55:40 ud Exp $
# ---
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

<div class="MainBox ARIARoleMain LayoutFixedSidebar SidebarFirst">
    <h1 class="InvisibleText">[% Translate("Request Field Management") | html %]</h1>

    [% BreadcrumbPath = [
            {
                Name => Translate('Import/Export Management'),
                Link => Env("Action"),
            },
        ]
    %]

    [% SWITCH Data.Action %]
        [% CASE 'Add' %]
            [% BreadcrumbPath.push({ Name => Translate("Add Group"),}) %]
        [% CASE 'Change' %]
            [% USE EditTitle = String(Translate("Edit Group")) %]
            [% BreadcrumbPath.push({ Name => EditTitle.append( ': ', Data.Name ) }) %]
    [% END %]

    [% INCLUDE "Breadcrumb.tt" Path = BreadcrumbPath %]

[% 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 Fullsize Center"><span><i class="fa fa-caret-left"></i>[% Translate("Go to overview") | html %]</span></a>
                    </li>
[% RenderBlockEnd("ActionOverview") %]
[% RenderBlockStart("ActionAdd") %]
                    <li>
                        <a href="[% Env("Baselink") %]Action=[% Env("Action") %];Subaction=TemplateEdit1" class="CallForAction Fullsize Center"><span><i class="fa fa-plus-square"></i>[% 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("Validity") | 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 [% IF Data.ValidID != 1%]class="Invalid"[% END %]>
                            <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 1 of 5 - 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 %]" />
                    <input type="hidden" name="SubmitNext" value="1" />

                    <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 CallForAction" type="submit"><span>[% Translate("Next") | html %]</span></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 2 of 5 - 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 %]" />
                    <input type="hidden" name="SubmitNext" value="1" />

                    <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>
                        <div class="Clear"></div>                        
[% RenderBlockEnd("TemplateEdit2Element") %]
                        <div class="Field SpacingTop">
                            <button class="Back CallForAction" type="button" name="Back" ><span>[% Translate("Back") | html %]</span></button>
                            <button class="Primary CallForAction" type="submit"><span>[% Translate("Next") | html %]</span></button>
                        </div>
                    </fieldset>
                </form>
            </div>
[% RenderBlockEnd("TemplateEdit2") %]

[% RenderBlockStart("TemplateEdit3") %]
            <div class="Header">
                <h2>[% Translate("Step 3 of 5 - 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 %]" />
                    <input type="hidden" name="SubmitNext" value="1" />

                    <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>
                        <div class="Clear"></div>                    
                            
[% RenderBlockEnd("TemplateEdit3ElementRequired") %]
                        </div>
[% RenderBlockEnd("TemplateEdit3Element") %]
                        <div class="Field SpacingTop">
                            <button class="Back CallForAction" type="button" name="Back" ><span>[% Translate("Back") | html %]</span></button>
                            <button class="Primary CallForAction" type="submit"><span>[% Translate("Next") | html %]</span></button>
                        </div>
                    </fieldset>
                </form>
            </div>
[% RenderBlockEnd("TemplateEdit3") %]

[% RenderBlockStart("TemplateEdit4") %]
            <div class="Header">
                <h2>[% Translate("Step 4 of 5 - 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><i class="fa fa-plus-square"></i>[% Translate("Add Mapping Element") | html %]</span> </button>
                    </div>

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

[% RenderBlockEnd("TemplateEdit4") %]

[% RenderBlockStart("TemplateEdit5") %]
            <div class="Header">
                <h2>[% Translate("Step 5 of 5 - 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 %]" />
                    <input type="hidden" name="SubmitNext" value="1" />
                    <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>
                        <div class="Clear"></div>                        
[% RenderBlockEnd("TemplateEdit5Element") %]

                        <div class="Field SpacingTop">
                            <button class="Back CallForAction" type="submit" name="SubmitBack" value="SubmitBack"><span>[% Translate("Back") | html %]</span></button>
                            <button class="Primary CallForAction" type="submit"><span>[% Translate("Finish") | html %]</span></button>
                        </div>
                    </fieldset>
                </form>
            </div>
            <div class="Clear"></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 CallForAction" type="submit" value="[% Translate("Start Import") | html %]"><span>[% Translate("Start Import") | html %]</span></button>
                        </div>
                    </fieldset>
                </form>
            </div>
[% RenderBlockEnd("ImportInformation") %]
        </div>
    </div>

</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 %s", 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 class="Primary CallForAction" type="submit" name="Ok" value="[% Translate("Ok") | html %]"><span>[% Translate("Ok") | html %]</span></button>
                    </p>
                </form>
            </div>
        </div>
    </div>
</div>
[% RenderBlockEnd("ImportResult") %]

# --
# Kernel/System/ImportExport.pm
# Modified version of the work:
# Copyright (C) 2010-2018 OFORK, https://o-fork.de
# based on the original work of:
# Copyright (C) 2001-2018 OTRS AG, http://otrs.com/
# --
# $Id: ImportExport.pm,v 1.1.1.1 2018/10/02 15:16:04 ud Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::System::ImportExport;

use strict;
use warnings;

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

=head1 NAME

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

=head1 DESCRIPTION

All import and export functions.

=head1 PUBLIC INTERFACE

=cut

=head2 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;
}

=head2 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';

    # get DB object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

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

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

    return \@TemplateList;
}

=head2 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} };

    # get DB object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

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

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

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

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

    return \%TemplateData;
}

=head2 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 ) = @_;

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

    # get CheckItem object
    my $CheckItemObject = $Kernel::OM->Get('Kernel::System::CheckItem');

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

    # get DB object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

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

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

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

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

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

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

    return $TemplateID;
}

=head2 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 ) = @_;

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    # check needed stuff
    for my $Argument (qw(TemplateID Name ValidID UserID)) {
        if ( !$Param{$Argument} ) {
            $LogObject->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 DB object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

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

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

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

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

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

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

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

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

=head2 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 ) = @_;

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

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

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

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

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

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

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

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

    # delete templates
    return $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL  => "DELETE FROM imexport_template WHERE id IN ( $TemplateIDString )",
        Bind => \@BIND,
    );
}

=head2 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;
}

=head2 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 ) = @_;

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

    # check template data
    if ( !$TemplateData || !$TemplateData->{Object} ) {
        $LogObject->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;
}

=head2 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;
        }
    }

    # get DB object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

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

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

    return \%ObjectData;
}

=head2 ObjectDataSave()

Save the object data of a template

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

=cut

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

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

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

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

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

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

        # insert one row
        $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;
}

=head2 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 ) = @_;

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

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

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

    # delete templates
    return $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL  => "DELETE FROM imexport_object WHERE template_id IN ( $TemplateIDString )",
        Bind => \@BIND,
    );
}

=head2 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;
}

=head2 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 ) = @_;

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

    # check template data
    if ( !$TemplateData || !$TemplateData->{Format} ) {
        $LogObject->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;
}

=head2 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;
        }
    }

    # get DB object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

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

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

    return \%FormatData;
}

=head2 FormatDataSave()

Save the format data of a template

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

=cut

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

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

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

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

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

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

        # insert one row
        $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;
}

=head2 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 ) = @_;

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

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

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

    # delete templates
    return $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL  => "DELETE FROM imexport_format WHERE template_id IN ( $TemplateIDString )",
        Bind => \@BIND,
    );
}

=head2 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;
        }
    }

    # get DB object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

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

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

    return \@MappingList;
}

=head2 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;
        }
    }

    # get DB object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

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

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

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

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

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

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

    return $MappingID;
}

=head2 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;
        }
    }

    # get DB object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

    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
        $DBObject->Do(
            SQL  => 'DELETE FROM imexport_mapping WHERE id = ?',
            Bind => [ \$Param{MappingID} ],
        );

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

        return 1;
    }
    else {

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

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

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

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

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

=head2 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];

    # get DB object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

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

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

    return 1 if !$Position;

    my $PositionUpper = $Position - 1;

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

    return 1;
}

=head2 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];

    # get DB object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

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

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

    my $PositionDown = $Position + 1;

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

    return 1;
}

=head2 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;
}

=head2 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 ) = @_;

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

    # check template data
    if ( !$TemplateData || !$TemplateData->{Object} ) {
        $LogObject->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;
}

=head2 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 ) = @_;

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

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

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

    # delete mapping object data
    return $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL  => "DELETE FROM imexport_mapping_object WHERE mapping_id IN ( $MappingIDString )",
        Bind => \@BIND,
    );
}

=head2 MappingObjectDataSave()

Save the object data of a mapping

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

=cut

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

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

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

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

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

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

        # insert one mapping object row
        $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;
}

=head2 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;
        }
    }

    # get DB object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

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

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

    return \%MappingObjectData;
}

=head2 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 ) = @_;

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

    # check template data
    if ( !$TemplateData || !$TemplateData->{Format} ) {
        $LogObject->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;
}

=head2 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 ) = @_;

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

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

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

    # delete mapping format data
    return $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL  => "DELETE FROM imexport_mapping_format WHERE mapping_id IN ( $MappingIDString )",
        Bind => \@BIND,
    );
}

=head2 MappingFormatDataSave()

Save the format data of a mapping

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

=cut

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

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

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

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

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

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

        # insert one mapping format row
        $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;
}

=head2 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;
        }
    }

    # get DB object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

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

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

    return \%MappingFormatData;
}

=head2 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 ) = @_;

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

    # check template data
    if ( !$TemplateData || !$TemplateData->{Object} ) {
        $LogObject->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;
}

=head2 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;
        }
    }

    # get DB object
    my $DBObject = $Kernel::OM->Get('Kernel::System::DB');

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

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

    return \%SearchData;
}

=head2 SearchDataSave()

Save the search data of a template

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

=cut

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

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

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

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

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

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

        # insert one row
        $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;
}

=head2 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 ) = @_;

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

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

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

    # delete templates
    return $Kernel::OM->Get('Kernel::System::DB')->Do(
        SQL  => "DELETE FROM imexport_search WHERE template_id IN ( $TemplateIDString )",
        Bind => \@BIND,
    );
}

=head2 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 ) = @_;

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

    # check template data
    if ( !$TemplateData || !$TemplateData->{Object} || !$TemplateData->{Format} ) {
        $LogObject->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
    $LogObject->Log(
        Priority => 'notice',
        Message  => "Export of $Result{Failed} records ($TemplateData->{Object}): failed!",
    );
    $LogObject->Log(
        Priority => 'notice',
        Message  => "Export of $Result{Success} records ($TemplateData->{Object}): successful!",
    );

    return \%Result;
}

=head2 Import()

Import function

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

=cut

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

    my $CacheObject = $Kernel::OM->Get('Kernel::System::Cache');

    # Disable the cache for faster import.
    $CacheObject->Configure(
        CacheInMemory  => 0,
        CacheInBackend => 0,
    );

    # get log object
    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

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

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

    # check template data
    if ( !$TemplateData || !$TemplateData->{Object} || !$TemplateData->{Format} ) {
        $LogObject->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
    $LogObject->Log(
        Priority => 'notice',
        Message =>
            "Import of $Result{Counter} $Result{Object} records: "
            . "$Result{Failed} failed, $Result{Success} succeeded",
    );
    for my $RetCode ( sort keys %{ $Result{RetCode} } ) {
        my $Count = $Result{RetCode}->{$RetCode} || 0;
        $LogObject->Log(
            Priority => 'notice',
            Message =>
                "Import of $Result{Counter} $Result{Object} records: $Count $RetCode",
        );
    }
    if ( $Result{Failed} ) {
        $LogObject->Log(
            Priority => 'notice',
            Message  => "Last processed line number of import file: $Result{Counter}",
        );
    }

    # Cleanup the cache after import.
    $CacheObject->CleanUp();

    return \%Result;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OFORK project (L<https://o-fork.de/>).

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

=cut

# --
# Kernel/System/ImportExport/FormatBackend/CSV.pm
# Modified version of the work:
# Copyright (C) 2010-2018 OFORK, https://o-fork.de
# based on the original work of:
# Copyright (C) 2001-2018 OTRS AG, http://otrs.com/
# --
# $Id: CSV.pm,v 1.1.1.1 2018/10/02 15:16:04 ud Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

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

use strict;
use warnings;

use Kernel::Language qw(Translatable);

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

=head1 NAME

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

=head1 DESCRIPTION

All functions to import and export a csv format

=cut

=head2 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;
}

=head2 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  => Translatable('Column Separator'),
            Input => {
                Type => 'Selection',
                Data => {
                    Tabulator => Translatable('Tabulator (TAB)'),
                    Semicolon => Translatable('Semicolon (;)'),
                    Colon     => Translatable('Colon (:)'),
                    Dot       => Translatable('Dot (.)'),
                    Comma     => Translatable('Comma (,)'),
                },
                Required     => 1,
                Translation  => 1,
                PossibleNone => 1,
            },
        },
        {
            Key   => 'Charset',
            Name  => Translatable('Charset'),
            Input => {
                Type         => 'Text',
                ValueDefault => 'UTF-8',
                Required     => 1,
                Translation  => 0,
                Size         => 20,
                MaxLength    => 20,
                Readonly     => 1,
            },
        },
        {
            Key   => 'IncludeColumnHeaders',
            Name  => Translatable('Include Column Headers'),
            Input => {
                Type => 'Selection',
                Data => {
                    0 => Translatable('No'),
                    1 => Translatable('Yes'),
                },
                Translation  => 1,
                PossibleNone => 0,
            },
        },
    ];

    return $Attributes;
}

=head2 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  => Translatable('Column'),
            Input => {
                Type     => 'TT',
                Data     => '',
                Required => 0,
            },
        },
    ];

    return $Attributes;
}

=head2 ImportDataGet()

Get import data as C<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} ||= '';

    # check the charset
    if ( $Charset ne 'UTF-8' ) {

        $Kernel::OM->Get('Kernel::System::Log')->Log(
            Priority => 'error',
            Message  => "No valid charset found for the template id $Param{TemplateID}. Charset must be UTF-8!",
        );
        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;

    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;

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

    return \@ImportData;
}

=head2 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} ||= '';

    # check the charset
    if ( $Charset ne 'UTF-8' ) {

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

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

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

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

    return $String;
}

1;

=head1 TERMS AND CONDITIONS

This software is part of the OFORK project (L<https://o-fork.de/>).

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

=cut

IyAtLQojIEtlcm5lbC9TeXN0ZW0vQ29uc29sZS9Db21tYW5kL0FkbWluL0lUU00vSW1wb3J0RXhwb3J0L0ltcG9ydC5wbQojIE1vZGlmaWVkIHZlcnNpb24gb2YgdGhlIHdvcms6CiMgQ29weXJpZ2h0IChDKSAyMDEwLTIwMTggT0ZPUkssIGh0dHBzOi8vby1mb3JrLmRlCiMgYmFzZWQgb24gdGhlIG9yaWdpbmFsIHdvcmsgb2Y6CiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTggT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgJElkOiBJbXBvcnQucG0sdiAxLjEuMS4xIDIwMTgvMTAvMDIgMTU6MTY6MDQgdWQgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OkNvbnNvbGU6OkNvbW1hbmQ6OkFkbWluOjpJVFNNOjpJbXBvcnRFeHBvcnQ6OkltcG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSBwYXJlbnQgcXcoS2VybmVsOjpTeXN0ZW06OkNvbnNvbGU6OkJhc2VDb21tYW5kKTsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpNYWluJywKICAgICdLZXJuZWw6OlN5c3RlbTo6SW1wb3J0RXhwb3J0JywKKTsKCnN1YiBDb25maWd1cmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAkU2VsZi0+RGVzY3JpcHRpb24oJ1RoZSB0b29sIGZvciBpbXBvcnRpbmcgY29uZmlnIGl0ZW1zJyk7CiAgICAkU2VsZi0+QWRkT3B0aW9uKAogICAgICAgIE5hbWUgICAgICAgID0+ICd0ZW1wbGF0ZS1udW1iZXInLAogICAgICAgIERlc2NyaXB0aW9uID0+ICJTcGVjaWZ5IGEgdGVtcGxhdGUgbnVtYmVyIHRvIGJlIGltcG9lcnRlZC4iLAogICAgICAgIFJlcXVpcmVkICAgID0+IDEsCiAgICAgICAgSGFzVmFsdWUgICAgPT4gMSwKICAgICAgICBWYWx1ZVJlZ2V4ICA9PiBxci9cZCsvc214LAogICAgKTsKICAgICRTZWxmLT5BZGRBcmd1bWVudCgKICAgICAgICBOYW1lICAgICAgICA9PiAnc291cmNlJywKICAgICAgICBEZXNjcmlwdGlvbiA9PiAiU3BlY2lmeSB0aGUgcGF0aCB0byB0aGUgZmlsZSB3aGljaCBjb250YWluaW5nIHRoZSBjb25maWcgaXRlbSBkYXRhIGZvciBpbXBvcnRpbmcuIiwKICAgICAgICBSZXF1aXJlZCAgICA9PiAxLAogICAgICAgIFZhbHVlUmVnZXggID0+IHFyLy4qL3NteCwKICAgICk7CgogICAgcmV0dXJuOwp9CgpzdWIgUHJlUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJFNvdXJjZVBhdGggPSAkU2VsZi0+R2V0QXJndW1lbnQoJ3NvdXJjZScpOwogICAgaWYgKCAkU291cmNlUGF0aCAmJiAhLXIgJFNvdXJjZVBhdGggKSB7CiAgICAgICAgZGllICJGaWxlICRTb3VyY2VQYXRoIGRvZXMgbm90IGV4aXN0LCBjYW4gbm90IGJlIHJlYWQuXG4iOwogICAgfQoKICAgIHJldHVybjsKfQoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRUZW1wbGF0ZUlEID0gJFNlbGYtPkdldE9wdGlvbigndGVtcGxhdGUtbnVtYmVyJyk7CgogICAgIyBnZXQgdGVtcGxhdGUgZGF0YQogICAgbXkgJFRlbXBsYXRlRGF0YSA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJbXBvcnRFeHBvcnQnKS0+VGVtcGxhdGVHZXQoCiAgICAgICAgVGVtcGxhdGVJRCA9PiAkVGVtcGxhdGVJRCwKICAgICAgICBVc2VySUQgICAgID0+IDEsCiAgICApOwoKICAgIGlmICggISRUZW1wbGF0ZURhdGEtPntUZW1wbGF0ZUlEfSApIHsKICAgICAgICAkU2VsZi0+UHJpbnRFcnJvcigiVGVtcGxhdGUgJFRlbXBsYXRlSUQgbm90IGZvdW5kIS5cbiIpOwogICAgICAgICRTZWxmLT5QcmludEVycm9yKCJFeHBvcnQgYWJvcnRlZC4uXG4iKTsKICAgICAgICByZXR1cm4gJFNlbGYtPkV4aXRDb2RlRXJyb3IoKTsKICAgIH0KCiAgICAkU2VsZi0+UHJpbnQoIjx5ZWxsb3c+SW1wb3J0aW5nIGNvbmZpZyBpdGVtcy4uLjwveWVsbG93PlxuIik7CiAgICAkU2VsZi0+UHJpbnQoICI8eWVsbG93PiIgLiAoICc9JyB4IDY5ICkgLiAiPC95ZWxsb3c+XG4iICk7CgogICAgbXkgJFNvdXJjZUNvbnRlbnQ7CiAgICBteSAkU291cmNlRmlsZSA9ICRTZWxmLT5HZXRBcmd1bWVudCgnc291cmNlJyk7CgogICAgaWYgKCRTb3VyY2VGaWxlKSB7CgogICAgICAgICRTZWxmLT5QcmludCgiPHllbGxvdz5SZWFkIEZpbGUgJFNvdXJjZUZpbGUuPC95ZWxsb3c+XG4iKTsKCiAgICAgICAgIyByZWFkIHNvdXJjZSBmaWxlCiAgICAgICAgJFNvdXJjZUNvbnRlbnQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TWFpbicpLT5GaWxlUmVhZCgKICAgICAgICAgICAgTG9jYXRpb24gPT4gJFNvdXJjZUZpbGUsCiAgICAgICAgICAgIFJlc3VsdCAgID0+ICdTQ0FMQVInLAogICAgICAgICAgICBNb2RlICAgICA9PiAnYmlubW9kZScsCiAgICAgICAgKTsKCiAgICAgICAgaWYgKCAhJFNvdXJjZUNvbnRlbnQgKSB7CiAgICAgICAgICAgICRTZWxmLT5QcmludEVycm9yKCJDYW4ndCByZWFkIGZpbGUgJFNvdXJjZUZpbGUuXG5JbXBvcnQgYWJvcnRlZC5cbiIpIGlmICEkU291cmNlQ29udGVudDsKICAgICAgICAgICAgcmV0dXJuICRTZWxmLT5FeGl0Q29kZUVycm9yKCk7CiAgICAgICAgfQoKICAgIH0KCiAgICAjIGltcG9ydCBkYXRhCiAgICBteSAkUmVzdWx0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkltcG9ydEV4cG9ydCcpLT5JbXBvcnQoCiAgICAgICAgVGVtcGxhdGVJRCAgICA9PiAkVGVtcGxhdGVJRCwKICAgICAgICBTb3VyY2VDb250ZW50ID0+ICRTb3VyY2VDb250ZW50LAogICAgICAgIFVzZXJJRCAgICAgICAgPT4gMSwKICAgICk7CgogICAgaWYgKCAhJFJlc3VsdCApIHsKICAgICAgICAkU2VsZi0+UHJpbnRFcnJvcigiXG5FcnJvciBvY2N1cnJlZC4gSW1wb3J0IGltcG9zc2libGUhIFNlZSB0aGUgT0ZPUksgbG9nIGZvciBkZXRhaWxzLlxuIik7CiAgICAgICAgcmV0dXJuICRTZWxmLT5FeGl0Q29kZUVycm9yKCk7CiAgICB9CgogICAgIyBwcmludCByZXN1bHQKICAgICRTZWxmLT5QcmludCgiXG48Z3JlZW4+SW1wb3J0IG9mICRSZXN1bHQtPntDb3VudGVyfSAkUmVzdWx0LT57T2JqZWN0fSByZWNvcmRzOjwvZ3JlZW4+XG4iKTsKICAgICRTZWxmLT5QcmludCggIjxncmVlbj4iIC4gKCAnLScgeCA2OSApIC4gIjwvZ3JlZW4+XG4iICk7CiAgICAkU2VsZi0+UHJpbnQoIjxncmVlbj5TdWNjZXNzOiAkUmVzdWx0LT57U3VjY2Vzc30gc3VjY2VlZGVkPC9ncmVlbj5cbiIpOwogICAgaWYgKCAkUmVzdWx0LT57RmFpbGVkfSApIHsKICAgICAgICAkU2VsZi0+UHJpbnRFcnJvcigiJFJlc3VsdC0+e0ZhaWxlZH0gZmFpbGVkLlxuIik7CiAgICB9CiAgICBlbHNlIHsKICAgICAgICAkU2VsZi0+UHJpbnQoIjxncmVlbj5FcnJvcjogJFJlc3VsdC0+e0ZhaWxlZH0gZmFpbGVkLjwvZ3JlZW4+XG4iKTsKICAgIH0KCiAgICBmb3IgbXkgJFJldENvZGUgKCBzb3J0IGtleXMgJXsgJFJlc3VsdC0+e1JldENvZGV9IH0gKSB7CiAgICAgICAgbXkgJENvdW50ID0gJFJlc3VsdC0+e1JldENvZGV9LT57JFJldENvZGV9IHx8IDA7CiAgICAgICAgJFNlbGYtPlByaW50KCI8Z3JlZW4+SW1wb3J0IG9mICRSZXN1bHQtPntDb3VudGVyfSAkUmVzdWx0LT57T2JqZWN0fSByZWNvcmRzOiAkQ291bnQgJFJldENvZGU8L2dyZWVuPlxuIik7CiAgICB9CiAgICBpZiAoICRSZXN1bHQtPntGYWlsZWR9ICkgewogICAgICAgICRTZWxmLT5QcmludCgiPGdyZWVuPkxhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlOiAkUmVzdWx0LT57Q291bnRlcn08L2dyZWVuPlxuIik7CiAgICB9CgogICAgJFNlbGYtPlByaW50KCI8Z3JlZW4+SW1wb3J0IGNvbXBsZXRlLjwvZ3JlZW4+XG4iKTsKICAgICRTZWxmLT5QcmludCggIjxncmVlbj4iIC4gKCAnLScgeCA2OSApIC4gIjwvZ3JlZW4+XG4iICk7CiAgICAkU2VsZi0+UHJpbnQoIjxncmVlbj5Eb25lLjwvZ3JlZW4+XG4iKTsKCiAgICByZXR1cm4gJFNlbGYtPkV4aXRDb2RlT2soKTsKfQoKMTsKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPRk9SSyBwcm9qZWN0IChMPGh0dHBzOi8vby1mb3JrLmRlLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQ+LgoKPWN1dAo=
IyAtLQojIEtlcm5lbC9TeXN0ZW0vQ29uc29sZS9Db21tYW5kL0FkbWluL0lUU00vSW1wb3J0RXhwb3J0L0V4cG9ydC5wbQojIE1vZGlmaWVkIHZlcnNpb24gb2YgdGhlIHdvcms6CiMgQ29weXJpZ2h0IChDKSAyMDEwLTIwMTggT0ZPUkssIGh0dHBzOi8vby1mb3JrLmRlCiMgYmFzZWQgb24gdGhlIG9yaWdpbmFsIHdvcmsgb2Y6CiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTggT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIC0tCiMgJElkOiBFeHBvcnQucG0sdiAxLjEuMS4xIDIwMTgvMTAvMDIgMTU6MTY6MDQgdWQgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpTeXN0ZW06OkNvbnNvbGU6OkNvbW1hbmQ6OkFkbWluOjpJVFNNOjpJbXBvcnRFeHBvcnQ6OkV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSBwYXJlbnQgcXcoS2VybmVsOjpTeXN0ZW06OkNvbnNvbGU6OkJhc2VDb21tYW5kKTsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpNYWluJywKICAgICdLZXJuZWw6OlN5c3RlbTo6SW1wb3J0RXhwb3J0JywKKTsKCnN1YiBDb25maWd1cmUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAkU2VsZi0+RGVzY3JpcHRpb24oJ1RoZSB0b29sIGZvciBleHBvcnRpbmcgY29uZmlnIGl0ZW1zJyk7CiAgICAkU2VsZi0+QWRkT3B0aW9uKAogICAgICAgIE5hbWUgICAgICAgID0+ICd0ZW1wbGF0ZS1udW1iZXInLAogICAgICAgIERlc2NyaXB0aW9uID0+ICJTcGVjaWZ5IGEgdGVtcGxhdGUgbnVtYmVyIHRvIGJlIGV4cG9ydGVkLiIsCiAgICAgICAgUmVxdWlyZWQgICAgPT4gMSwKICAgICAgICBIYXNWYWx1ZSAgICA9PiAxLAogICAgICAgIFZhbHVlUmVnZXggID0+IHFyL1xkKy9zbXgsCiAgICApOwogICAgJFNlbGYtPkFkZEFyZ3VtZW50KAogICAgICAgIE5hbWUgICAgICAgID0+ICdkZXN0aW5hdGlvbicsCiAgICAgICAgRGVzY3JpcHRpb24gPT4gIlNwZWNpZnkgdGhlIHBhdGggdG8gYSBmaWxlIHdoZXJlIGNvbmZpZyBpdGVtIGRhdGEgc2hvdWxkIGJlIGV4cG9ydGVkLiIsCiAgICAgICAgUmVxdWlyZWQgICAgPT4gMSwKICAgICAgICBWYWx1ZVJlZ2V4ICA9PiBxci8uKi9zbXgsCiAgICApOwoKICAgIHJldHVybjsKfQoKc3ViIFJ1biB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIG15ICRUZW1wbGF0ZUlEID0gJFNlbGYtPkdldE9wdGlvbigndGVtcGxhdGUtbnVtYmVyJyk7CgogICAgIyBnZXQgdGVtcGxhdGUgZGF0YQogICAgbXkgJFRlbXBsYXRlRGF0YSA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJbXBvcnRFeHBvcnQnKS0+VGVtcGxhdGVHZXQoCiAgICAgICAgVGVtcGxhdGVJRCA9PiAkVGVtcGxhdGVJRCwKICAgICAgICBVc2VySUQgICAgID0+IDEsCiAgICApOwoKICAgIGlmICggISRUZW1wbGF0ZURhdGEtPntUZW1wbGF0ZUlEfSApIHsKICAgICAgICAkU2VsZi0+UHJpbnRFcnJvcigiVGVtcGxhdGUgJFRlbXBsYXRlSUQgbm90IGZvdW5kIS5cbiIpOwogICAgICAgICRTZWxmLT5QcmludEVycm9yKCJFeHBvcnQgYWJvcnRlZC4uXG4iKTsKICAgICAgICByZXR1cm4gJFNlbGYtPkV4aXRDb2RlRXJyb3IoKTsKICAgIH0KCiAgICAkU2VsZi0+UHJpbnQoIjx5ZWxsb3c+RXhwb3J0aW5nIGNvbmZpZyBpdGVtcy4uLjwveWVsbG93PlxuIik7CiAgICAkU2VsZi0+UHJpbnQoICI8eWVsbG93PiIgLiAoICc9JyB4IDY5ICkgLiAiPC95ZWxsb3c+XG4iICk7CgogICAgIyBleHBvcnQgZGF0YQogICAgbXkgJFJlc3VsdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJbXBvcnRFeHBvcnQnKS0+RXhwb3J0KAogICAgICAgIFRlbXBsYXRlSUQgPT4gJFRlbXBsYXRlSUQsCiAgICAgICAgVXNlcklEICAgICA9PiAxLAogICAgKTsKCiAgICBpZiAoICEkUmVzdWx0ICkgewogICAgICAgICRTZWxmLT5QcmludEVycm9yKCJFcnJvciBvY2N1cnJlZC4gRXhwb3J0IGltcG9zc2libGUhIFNlZSBTeXNsb2cgZm9yIGRldGFpbHMuXG4iKTsKICAgICAgICByZXR1cm4gJFNlbGYtPkV4aXRDb2RlRXJyb3IoKTsKICAgIH0KCiAgICAkU2VsZi0+UHJpbnQoICI8Z3JlZW4+IiAuICggJy0nIHggNjkgKSAuICI8L2dyZWVuPlxuIiApOwogICAgJFNlbGYtPlByaW50KCI8Z3JlZW4+U3VjY2VzczogJFJlc3VsdC0+e1N1Y2Nlc3N9IHN1Y2NlZWRlZDwvZ3JlZW4+XG4iKTsKICAgIGlmICggJFJlc3VsdC0+e0ZhaWxlZH0gKSB7CiAgICAgICAgJFNlbGYtPlByaW50RXJyb3IoIiRSZXN1bHQtPntGYWlsZWR9IGZhaWxlZC5cbiIpOwogICAgfQogICAgZWxzZSB7CiAgICAgICAgJFNlbGYtPlByaW50KCI8Z3JlZW4+RXJyb3I6ICRSZXN1bHQtPntGYWlsZWR9IGZhaWxlZC48L2dyZWVuPlxuIik7CiAgICB9CgogICAgbXkgJERlc3RpbmF0aW9uRmlsZSA9ICRTZWxmLT5HZXRBcmd1bWVudCgnZGVzdGluYXRpb24nKTsKCiAgICBpZiAoJERlc3RpbmF0aW9uRmlsZSkgewoKICAgICAgICBteSAkRmlsZUNvbnRlbnQgPSBqb2luICJcbiIsIEB7ICRSZXN1bHQtPntEZXN0aW5hdGlvbkNvbnRlbnR9IH07CgogICAgICAgICMgc2F2ZSBkZXN0aW5hdGlvbiBjb250ZW50IHRvIGZpbGUKICAgICAgICBteSAkU3VjY2VzcyA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpNYWluJyktPkZpbGVXcml0ZSgKICAgICAgICAgICAgTG9jYXRpb24gPT4gJERlc3RpbmF0aW9uRmlsZSwKICAgICAgICAgICAgQ29udGVudCAgPT4gXCRGaWxlQ29udGVudCwKICAgICAgICApOwoKICAgICAgICBpZiAoICEkU3VjY2VzcyApIHsKICAgICAgICAgICAgJFNlbGYtPlByaW50RXJyb3IoIkNhbid0IHdyaXRlIGZpbGUgJERlc3RpbmF0aW9uRmlsZS5cbkV4cG9ydCBhYm9ydGVkLlxuIik7CiAgICAgICAgICAgIHJldHVybiAkU2VsZi0+RXhpdENvZGVFcnJvcigpOwogICAgICAgIH0KCiAgICAgICAgJFNlbGYtPlByaW50KCI8Z3JlZW4+RmlsZSAkRGVzdGluYXRpb25GaWxlIHNhdmVkLjwvZ3JlZW4+XG4iKTsKCiAgICB9CgogICAgJFNlbGYtPlByaW50KCI8Z3JlZW4+RXhwb3J0IGNvbXBsZXRlLjwvZ3JlZW4+XG4iKTsKICAgICRTZWxmLT5QcmludCggIjxncmVlbj4iIC4gKCAnLScgeCA2OSApIC4gIjwvZ3JlZW4+XG4iICk7CiAgICAkU2VsZi0+UHJpbnQoIjxncmVlbj5Eb25lLjwvZ3JlZW4+XG4iKTsKCiAgICByZXR1cm4gJFNlbGYtPkV4aXRDb2RlT2soKTsKfQoKMTsKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPRk9SSyBwcm9qZWN0IChMPGh0dHBzOi8vby1mb3JrLmRlLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQ+LgoKPWN1dAo=
# --
# Copyright (C) 2001-2018 OTRS AG, http://otrs.com/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

# get needed objects
my $ConfigObject       = $Kernel::OM->Get('Kernel::Config');
my $UserObject         = $Kernel::OM->Get('Kernel::System::User');
my $ImportExportObject = $Kernel::OM->Get('Kernel::System::ImportExport');
my $Helper             = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

# ------------------------------------------------------------ #
# 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 . $Helper->GetRandomID(),
            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' . $Helper->GetRandomID();
}

# create needed random object names
my @ObjectName;
push @ObjectName, 'UnitTest' . $Helper->GetRandomID();

# 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 successfully
    {
        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 successfully
    {
        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 successfully (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 successfully (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 variable
        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 element more
$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 successful
$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 unit test 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,
);

1;

# --
# Copyright (C) 2001-2018 OTRS AG, http://otrs.com/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

use Data::Dumper;

# get needed objects
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');

# get helper object
$Kernel::OM->ObjectParamAdd(
    'Kernel::System::UnitTest::Helper' => {
        RestoreDatabase => 1,
    },
);
my $Helper = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');

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

# get home directory
$Self->{Home} = $Kernel::OM->Get('Kernel::Config')->Get('Home');

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

    my $RandomID = $Helper->GetRandomID();

    # add a test template for later checks
    my $TemplateID = $ImportExportObject->TemplateAdd(
        Object  => 'UnitTest' . $RandomID,
        Format  => 'CSV',
        Name    => 'UnitTest' . $RandomID,
        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,
            Readonly     => 1,
        },
    },
    {
        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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            SourceFile    => 'ImportExportFormatCSV004-OpenOffice-Colon.csv',
            ImportDataGet => {
                TemplateID    => $TemplateIDs[8],
                SourceContent => 'SourceFile',
                UserID        => 1,
            },
        },
        ReferenceImportData => [
            [ 'Test;:_°^!"§$%&/()=?´`*+Test', '><@~\'}{[]\\' ],
            [ '"";;::..--__##',                  '' ],
        ],
    },

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

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

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

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

    # all required values are given
    {
        SourceImportData => {
            FormatData => {
                ColumnSeparator => 'Colon',
                Charset         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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;
        }

        # # print the file name
        # $Self->True(
        #     1,
        #     $Test->{SourceImportData}->{SourceFile},
        # );

        # 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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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         => 'UTF-8',
            },
            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++;
}

# cleanup is done by RestoreDatabase.

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLmNvbS8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCiMjIG5vIGNyaXRpYyAoTW9kdWxlczo6UmVxdWlyZUV4cGxpY2l0UGFja2FnZSkKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwp1c2UgdXRmODsKCnVzZSB2YXJzIChxdygkU2VsZikpOwp1c2UgRmlsZTo6UGF0aCBxdyhybXRyZWUpOwoKIyBnZXQgbmVlZGVkIG9iamVjdHMKbXkgJENvbW1hbmRPYmplY3QgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpDb25zb2xlOjpDb21tYW5kOjpBZG1pbjo6SVRTTTo6SW1wb3J0RXhwb3J0OjpJbXBvcnQnKTsKbXkgJEltcG9ydEV4cG9ydE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJbXBvcnRFeHBvcnQnKTsKCiMgZ2V0IGhlbHBlciBvYmplY3QKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCiMgdGVzdCBjb21tYW5kIHdpdGhvdXQgLS10ZW1wbGF0ZS1udW1iZXIgb3B0aW9uCm15ICRFeGl0Q29kZSA9ICRDb21tYW5kT2JqZWN0LT5FeGVjdXRlKCk7CgokU2VsZi0+SXMoCiAgICAkRXhpdENvZGUsCiAgICAxLAogICAgIk5vIC0tdGVtcGxhdGUtbnVtYmVyICAtIGV4aXQgY29kZSIsCik7CgojIGFkZCB0ZXN0IHRlbXBsYXRlCm15ICRUZW1wbGF0ZUlEID0gJEltcG9ydEV4cG9ydE9iamVjdC0+VGVtcGxhdGVBZGQoCiAgICBPYmplY3QgID0+ICdJVFNNQ29uZmlnSXRlbScsCiAgICBGb3JtYXQgID0+ICdDU1YnLAogICAgTmFtZSAgICA9PiAnVGVtcGxhdGUnIC4gJEhlbHBlci0+R2V0UmFuZG9tSUQoKSwKICAgIFZhbGlkSUQgPT4gMSwKICAgIENvbW1lbnQgPT4gJ0NvbW1lbnQnLAogICAgVXNlcklEICA9PiAxLAopOwoKJFNlbGYtPlRydWUoCiAgICAkVGVtcGxhdGVJRCwKICAgICJJbXBvcnQvRXhwb3J0IHRlbXBsYXRlIGlzIGNyZWF0ZWQgLSAkVGVtcGxhdGVJRCIsCik7CgojIGdldCAnSGFyZHdhcmUnIGNhdGFsb2cgY2xhc3MgSUQKbXkgJENvbmZpZ0l0ZW1EYXRhUmVmID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkdlbmVyYWxDYXRhbG9nJyktPkl0ZW1HZXQoCiAgICBDbGFzcyA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgPT4gJ0hhcmR3YXJlJywKKTsKbXkgJEhhcmR3YXJlQ29uZmlnSXRlbUlEID0gJENvbmZpZ0l0ZW1EYXRhUmVmLT57SXRlbUlEfTsKCiMgZ2V0IG9iamVjdCBkYXRhIGZvciB0ZXN0IHRlbXBsYXRlCm15ICVUZW1wbGF0ZVJlZiA9ICgKICAgICdDbGFzc0lEJyAgPT4gJEhhcmR3YXJlQ29uZmlnSXRlbUlELAogICAgJ0NvdW50TWF4JyA9PiAxMCwKKTsKbXkgJFN1Y2Nlc3MgPSAkSW1wb3J0RXhwb3J0T2JqZWN0LT5PYmplY3REYXRhU2F2ZSgKICAgIFRlbXBsYXRlSUQgPT4gJFRlbXBsYXRlSUQsCiAgICBPYmplY3REYXRhID0+IFwlVGVtcGxhdGVSZWYsCiAgICBVc2VySUQgICAgID0+IDEsCik7CgokU2VsZi0+VHJ1ZSgKICAgICRTdWNjZXNzLAogICAgIk9iamVjdERhdGEgZm9yIHRlc3QgdGVtcGxhdGUgaXMgYWRkZWQiLAopOwoKIyBhZGQgdGhlIGZvcm1hdCBkYXRhIG9mIHRoZSB0ZXN0IHRlbXBsYXRlCm15ICVGb3JtYXREYXRhID0gKAogICAgQ2hhcnNldCAgICAgICAgICAgICAgPT4gJ1VURi04JywKICAgIENvbHVtblNlcGFyYXRvciAgICAgID0+ICdDb21tYScsCiAgICBJbmNsdWRlQ29sdW1uSGVhZGVycyA9PiAxLAopOwokU3VjY2VzcyA9ICRJbXBvcnRFeHBvcnRPYmplY3QtPkZvcm1hdERhdGFTYXZlKAogICAgVGVtcGxhdGVJRCA9PiAkVGVtcGxhdGVJRCwKICAgIEZvcm1hdERhdGEgPT4gXCVGb3JtYXREYXRhLAogICAgVXNlcklEICAgICA9PiAxLAopOwoKJFNlbGYtPlRydWUoCiAgICAkU3VjY2VzcywKICAgICJGb3JtYXREYXRhIGZvciB0ZXN0IHRlbXBsYXRlIGlzIGFkZGVkIiwKKTsKCiMgc2F2ZSB0aGUgc2VhcmNoIGRhdGEgb2YgYSB0ZW1wbGF0ZQpteSAlU2VhcmNoRGF0YSA9ICgKICAgIE5hbWUgPT4gJ1Rlc3RDb25maWdJdGVtKicsCik7CiRTdWNjZXNzID0gJEltcG9ydEV4cG9ydE9iamVjdC0+U2VhcmNoRGF0YVNhdmUoCiAgICBUZW1wbGF0ZUlEID0+ICRUZW1wbGF0ZUlELAogICAgU2VhcmNoRGF0YSA9PiBcJVNlYXJjaERhdGEsCiAgICBVc2VySUQgICAgID0+IDEsCik7CgojIGFkZCBtYXBwaW5nIGRhdGEgZm9yIHRlc3QgdGVtcGxhdGUKZm9yIG15ICRPYmplY3REYXRhVmFsdWUgKHF3KCBOYW1lIERlcGxTdGF0ZSBJbmNpU3RhdGUgKSkgewoKICAgIG15ICRNYXBwaW5nSUQgPSAkSW1wb3J0RXhwb3J0T2JqZWN0LT5NYXBwaW5nQWRkKAogICAgICAgIFRlbXBsYXRlSUQgPT4gJFRlbXBsYXRlSUQsCiAgICAgICAgVXNlcklEICAgICA9PiAxLAogICAgKTsKCiAgICBteSAlTWFwcGluZ09iamVjdERhdGEgPSAoIEtleSA9PiAkT2JqZWN0RGF0YVZhbHVlICk7CiAgICBteSAkU3VjY2VzcyA9ICRJbXBvcnRFeHBvcnRPYmplY3QtPk1hcHBpbmdPYmplY3REYXRhU2F2ZSgKICAgICAgICBNYXBwaW5nSUQgICAgICAgICA9PiAkTWFwcGluZ0lELAogICAgICAgIE1hcHBpbmdPYmplY3REYXRhID0+IFwlTWFwcGluZ09iamVjdERhdGEsCiAgICAgICAgVXNlcklEICAgICAgICAgICAgPT4gMSwKICAgICk7CgogICAgJFNlbGYtPlRydWUoCiAgICAgICAgJFN1Y2Nlc3MsCiAgICAgICAgIk9iamVjdERhdGEgZm9yIHRlc3QgdGVtcGxhdGUgaXMgbWFwcGVkIC0gJE9iamVjdERhdGFWYWx1ZSIsCiAgICApOwp9CgojIG1ha2UgZGlyZWN0b3J5IGZvciBleHBvcnQgZmlsZQpteSAkU291cmNlUGF0aAogICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OkNvbmZpZycpLT5HZXQoJ0hvbWUnKSAuICIvc2NyaXB0cy90ZXN0L3NhbXBsZS9JbXBvcnRFeHBvcnQvVGVtcGxhdGVFeHBvcnQuY3N2IjsKCiMgdGVzdCBjb21tYW5kIHdpdGggd3JvbmcgdGVtcGxhdGUgbnVtYmVyCiRFeGl0Q29kZSA9ICRDb21tYW5kT2JqZWN0LT5FeGVjdXRlKCAnLS10ZW1wbGF0ZS1udW1iZXInLCAkSGVscGVyLT5HZXRSYW5kb21JRCgpLCAkU291cmNlUGF0aCAuICdUZW1wbGF0ZUV4cG9ydC5jc3YnICk7CgokU2VsZi0+SXMoCiAgICAkRXhpdENvZGUsCiAgICAxLAogICAgIkNvbW1hbmQgd2l0aCB3cm9uZyB0ZW1wbGF0ZSBudW1iZXIgLSBleGl0IGNvZGUiLAopOwoKIyB0ZXN0IGNvbW1hbmQgd2l0aG91dCBTb3VyY2UgYXJndW1lbnQKJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoICctLXRlbXBsYXRlLW51bWJlcicsICRUZW1wbGF0ZUlEICk7CgokU2VsZi0+SXMoCiAgICAkRXhpdENvZGUsCiAgICAxLAogICAgIk5vIFNvdXJjZSBhcmd1bWVudCAtIGV4aXQgY29kZSIsCik7CgojIHRlc3QgY29tbWFuZCB3aXRoIC0tdGVtcGxhdGUtbnVtYmVyIG9wdGlvbiBhbmQgU291cmNlIGFyZ3VtZW50CiRFeGl0Q29kZSA9ICRDb21tYW5kT2JqZWN0LT5FeGVjdXRlKCAnLS10ZW1wbGF0ZS1udW1iZXInLCAkVGVtcGxhdGVJRCwgJFNvdXJjZVBhdGggKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDAsCiAgICAiT3B0aW9uIC0gLS10ZW1wbGF0ZS1udW1iZXIgb3B0aW9uIGFuZCBTb3VyY2UgYXJndW1lbnQiLAopOwoKIyBnZXQgY29uZmlnIGl0ZW0gSURzCm15ICRDb25maWdJdGVtSURzID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OklUU01Db25maWdJdGVtJyktPkNvbmZpZ0l0ZW1TZWFyY2hFeHRlbmRlZCgKICAgIE5hbWUgPT4gJ1Rlc3RDb25maWdJdGVtKicKKTsKbXkgJE51bUNvbmZpZ0l0ZW1JbXBvcnRlZCA9IHNjYWxhciBAeyRDb25maWdJdGVtSURzfTsKCiMgY2hlY2sgaWYgdGhlIGNvbmZpZyBpdGVtcyBhcmUgaW1wb3J0ZWQKJFNlbGYtPlRydWUoCiAgICAkTnVtQ29uZmlnSXRlbUltcG9ydGVkLAogICAgIlRoZXJlIGFyZSAkTnVtQ29uZmlnSXRlbUltcG9ydGVkIGltcG9ydGVkIGNvbmZpZyBpdGVtcyIsCik7CgojIGNsZWFudXAgaXMgZG9uZSBieSBSZXN0b3JlRGF0YWJhc2UuCgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE4IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLmNvbS8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCiMjIG5vIGNyaXRpYyAoTW9kdWxlczo6UmVxdWlyZUV4cGxpY2l0UGFja2FnZSkKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwp1c2UgdXRmODsKCnVzZSB2YXJzIChxdygkU2VsZikpOwp1c2UgRmlsZTo6UGF0aCBxdyhta3BhdGggcm10cmVlKTsKCiMgZ2V0IG5lZWRlZCBvYmplY3RzCm15ICRDb21tYW5kT2JqZWN0ICAgICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpDb25zb2xlOjpDb21tYW5kOjpBZG1pbjo6SVRTTTo6SW1wb3J0RXhwb3J0OjpFeHBvcnQnKTsKbXkgJEdlbmVyYWxDYXRhbG9nT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkdlbmVyYWxDYXRhbG9nJyk7Cm15ICRDb25maWdJdGVtT2JqZWN0ICAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpJVFNNQ29uZmlnSXRlbScpOwoKIyBnZXQgaGVscGVyIG9iamVjdAokS2VybmVsOjpPTS0+T2JqZWN0UGFyYW1BZGQoCiAgICAnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInID0+IHsKICAgICAgICBSZXN0b3JlRGF0YWJhc2UgPT4gMSwKICAgIH0sCik7Cm15ICRIZWxwZXIgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6VW5pdFRlc3Q6OkhlbHBlcicpOwoKIyB0ZXN0IGNvbW1hbmQgd2l0aG91dCAtLXRlbXBsYXRlLW51bWJlciBvcHRpb24KbXkgJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDEsCiAgICAiTm8gLS10ZW1wbGF0ZS1udW1iZXIgIC0gZXhpdCBjb2RlIiwKKTsKCiMgZ2V0ICdIYXJkd2FyZScgY2F0YWxvZyBjbGFzcyBJRApteSAkQ29uZmlnSXRlbURhdGFSZWYgPSAkR2VuZXJhbENhdGFsb2dPYmplY3QtPkl0ZW1HZXQoCiAgICBDbGFzcyA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6Q2xhc3MnLAogICAgTmFtZSAgPT4gJ0hhcmR3YXJlJywKKTsKbXkgJEhhcmR3YXJlQ29uZmlnSXRlbUlEID0gJENvbmZpZ0l0ZW1EYXRhUmVmLT57SXRlbUlEfTsKCiMgZ2V0ICdQcm9kdWN0aW9uJyBkZXBsb3ltZW50IHN0YXRlIElEcwpteSAkUHJvZHVjdGlvbkRlcGxTdGF0ZURhdGFSZWYgPSAkR2VuZXJhbENhdGFsb2dPYmplY3QtPkl0ZW1HZXQoCiAgICBDbGFzcyA9PiAnSVRTTTo6Q29uZmlnSXRlbTo6RGVwbG95bWVudFN0YXRlJywKICAgIE5hbWUgID0+ICdQcm9kdWN0aW9uJywKKTsKbXkgJFByb2R1Y3Rpb25EZXBsU3RhdGVJRCA9ICRQcm9kdWN0aW9uRGVwbFN0YXRlRGF0YVJlZi0+e0l0ZW1JRH07CgpteSBAQ29uZmlnSXRlbUlEczsKCiMgYWRkIHRlc3QgY29uZmlnIGl0ZW1zCmZvciAoIDEgLi4gMTAgKSB7CgogICAgIyBjcmVhdGUgQ29uZmlnSXRlbSBudW1iZXIKICAgIG15ICRDb25maWdJdGVtTnVtYmVyID0gJENvbmZpZ0l0ZW1PYmplY3QtPkNvbmZpZ0l0ZW1OdW1iZXJDcmVhdGUoCiAgICAgICAgVHlwZSAgICA9PiAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OkNvbmZpZycpLT5HZXQoJ0lUU01Db25maWdJdGVtOjpOdW1iZXJHZW5lcmF0b3InKSwKICAgICAgICBDbGFzc0lEID0+ICRIYXJkd2FyZUNvbmZpZ0l0ZW1JRCwKICAgICk7CgogICAgIyBhZGQgdGVzdCBDb25maWdJdGVtCiAgICBteSAkQ29uZmlnSXRlbUlEID0gJENvbmZpZ0l0ZW1PYmplY3QtPkNvbmZpZ0l0ZW1BZGQoCiAgICAgICAgTnVtYmVyICA9PiAkQ29uZmlnSXRlbU51bWJlciwKICAgICAgICBDbGFzc0lEID0+ICRIYXJkd2FyZUNvbmZpZ0l0ZW1JRCwKICAgICAgICBVc2VySUQgID0+IDEsCiAgICApOwoKICAgICRTZWxmLT5UcnVlKAogICAgICAgICRDb25maWdJdGVtSUQsCiAgICAgICAgIkNvbmZpZyBpdGVtIGlzIGNyZWF0ZWQgLSAkQ29uZmlnSXRlbUlEIiwKICAgICk7CgogICAgbXkgJENvbmZpZ0l0ZW1OYW1lID0gJ1Rlc3RDb25maWdJdGVtJyAuICRIZWxwZXItPkdldFJhbmRvbUlEKCk7CiAgICBteSAkVmVyc2lvbklEICAgICAgPSAkQ29uZmlnSXRlbU9iamVjdC0+VmVyc2lvbkFkZCgKICAgICAgICBOYW1lICAgICAgICAgPT4gJENvbmZpZ0l0ZW1OYW1lLAogICAgICAgIERlZmluaXRpb25JRCA9PiAxLAogICAgICAgIERlcGxTdGF0ZUlEICA9PiAkUHJvZHVjdGlvbkRlcGxTdGF0ZUlELAogICAgICAgIEluY2lTdGF0ZUlEICA9PiAxLAogICAgICAgIFVzZXJJRCAgICAgICA9PiAxLAogICAgICAgIENvbmZpZ0l0ZW1JRCA9PiAkQ29uZmlnSXRlbUlELAogICAgKTsKCiAgICAkU2VsZi0+VHJ1ZSgKICAgICAgICAkVmVyc2lvbklELAogICAgICAgICJWZXJzaW9uIGZvciBjb25maWcgaXRlbSAkQ29uZmlnSXRlbUlEIGlzIGNyZWF0ZWQgLSAkQ29uZmlnSXRlbU5hbWUiLAogICAgKTsKCiAgICBwdXNoIEBDb25maWdJdGVtSURzLCAkQ29uZmlnSXRlbUlEOwp9CgojIGdldCBJbXBvcnRFeHBvcnQgb2JqZWN0Cm15ICRJbXBvcnRFeHBvcnRPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SW1wb3J0RXhwb3J0Jyk7CgojIGFkZCB0ZXN0IHRlbXBsYXRlCm15ICRUZW1wbGF0ZUlEID0gJEltcG9ydEV4cG9ydE9iamVjdC0+VGVtcGxhdGVBZGQoCiAgICBPYmplY3QgID0+ICdJVFNNQ29uZmlnSXRlbScsCiAgICBGb3JtYXQgID0+ICdDU1YnLAogICAgTmFtZSAgICA9PiAnVGVtcGxhdGUnIC4gJEhlbHBlci0+R2V0UmFuZG9tSUQoKSwKICAgIFZhbGlkSUQgPT4gMSwKICAgIENvbW1lbnQgPT4gJ0NvbW1lbnQnLAogICAgVXNlcklEICA9PiAxLAopOwoKJFNlbGYtPlRydWUoCiAgICAkVGVtcGxhdGVJRCwKICAgICJJbXBvcnQvRXhwb3J0IHRlbXBsYXRlIGlzIGNyZWF0ZWQgLSAkVGVtcGxhdGVJRCIsCik7CgojIGdldCBvYmplY3QgZGF0YSBmb3IgdGVzdCB0ZW1wbGF0ZQpteSAlVGVtcGxhdGVSZWYgPSAoCiAgICAnQ2xhc3NJRCcgID0+ICRIYXJkd2FyZUNvbmZpZ0l0ZW1JRCwKICAgICdDb3VudE1heCcgPT4gMTAsCik7Cm15ICRTdWNjZXNzID0gJEltcG9ydEV4cG9ydE9iamVjdC0+T2JqZWN0RGF0YVNhdmUoCiAgICBUZW1wbGF0ZUlEID0+ICRUZW1wbGF0ZUlELAogICAgT2JqZWN0RGF0YSA9PiBcJVRlbXBsYXRlUmVmLAogICAgVXNlcklEICAgICA9PiAxLAopOwoKJFNlbGYtPlRydWUoCiAgICAkU3VjY2VzcywKICAgICJPYmplY3REYXRhIGZvciB0ZXN0IHRlbXBsYXRlIGlzIGFkZGVkIiwKKTsKCiMgYWRkIHRoZSBmb3JtYXQgZGF0YSBvZiB0aGUgdGVzdCB0ZW1wbGF0ZQpteSAlRm9ybWF0RGF0YSA9ICgKICAgIENoYXJzZXQgICAgICAgICAgICAgID0+ICdVVEYtOCcsCiAgICBDb2x1bW5TZXBhcmF0b3IgICAgICA9PiAnQ29tbWEnLAogICAgSW5jbHVkZUNvbHVtbkhlYWRlcnMgPT4gMSwKKTsKJFN1Y2Nlc3MgPSAkSW1wb3J0RXhwb3J0T2JqZWN0LT5Gb3JtYXREYXRhU2F2ZSgKICAgIFRlbXBsYXRlSUQgPT4gJFRlbXBsYXRlSUQsCiAgICBGb3JtYXREYXRhID0+IFwlRm9ybWF0RGF0YSwKICAgIFVzZXJJRCAgICAgPT4gMSwKKTsKCiRTZWxmLT5UcnVlKAogICAgJFN1Y2Nlc3MsCiAgICAiRm9ybWF0RGF0YSBmb3IgdGVzdCB0ZW1wbGF0ZSBpcyBhZGRlZCIsCik7CgojIHNhdmUgdGhlIHNlYXJjaCBkYXRhIG9mIGEgdGVtcGxhdGUKbXkgJVNlYXJjaERhdGEgPSAoCiAgICBOYW1lID0+ICdUZXN0Q29uZmlnSXRlbSonLAopOwokU3VjY2VzcyA9ICRJbXBvcnRFeHBvcnRPYmplY3QtPlNlYXJjaERhdGFTYXZlKAogICAgVGVtcGxhdGVJRCA9PiAkVGVtcGxhdGVJRCwKICAgIFNlYXJjaERhdGEgPT4gXCVTZWFyY2hEYXRhLAogICAgVXNlcklEICAgICA9PiAxLAopOwoKIyBhZGQgbWFwcGluZyBkYXRhIGZvciB0ZXN0IHRlbXBsYXRlCmZvciBteSAkT2JqZWN0RGF0YVZhbHVlIChxdyggTmFtZSBEZXBsU3RhdGUgSW5jaVN0YXRlICkpIHsKCiAgICBteSAkTWFwcGluZ0lEID0gJEltcG9ydEV4cG9ydE9iamVjdC0+TWFwcGluZ0FkZCgKICAgICAgICBUZW1wbGF0ZUlEID0+ICRUZW1wbGF0ZUlELAogICAgICAgIFVzZXJJRCAgICAgPT4gMSwKICAgICk7CgogICAgbXkgJU1hcHBpbmdPYmplY3REYXRhID0gKCBLZXkgPT4gJE9iamVjdERhdGFWYWx1ZSApOwogICAgbXkgJFN1Y2Nlc3MgPSAkSW1wb3J0RXhwb3J0T2JqZWN0LT5NYXBwaW5nT2JqZWN0RGF0YVNhdmUoCiAgICAgICAgTWFwcGluZ0lEICAgICAgICAgPT4gJE1hcHBpbmdJRCwKICAgICAgICBNYXBwaW5nT2JqZWN0RGF0YSA9PiBcJU1hcHBpbmdPYmplY3REYXRhLAogICAgICAgIFVzZXJJRCAgICAgICAgICAgID0+IDEsCiAgICApOwoKICAgICRTZWxmLT5UcnVlKAogICAgICAgICRTdWNjZXNzLAogICAgICAgICJPYmplY3REYXRhIGZvciB0ZXN0IHRlbXBsYXRlIGlzIG1hcHBlZCAtICRPYmplY3REYXRhVmFsdWUiLAogICAgKTsKfQoKIyBtYWtlIGRpcmVjdG9yeSBmb3IgZXhwb3J0IGZpbGUKbXkgJERlc3RpbmF0aW9uUGF0aCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyktPkdldCgnSG9tZScpIC4gIi92YXIvdG1wL0ltcG9ydEV4cG9ydC8iOwpta3BhdGgoIFskRGVzdGluYXRpb25QYXRoXSwgMCwgMDc3MCApOyAgICAjIyBubyBjcml0aWMKCiMgdGVzdCBjb21tYW5kIHdpdGggd3JvbmcgdGVtcGxhdGUgbnVtYmVyCiRFeGl0Q29kZSA9ICRDb21tYW5kT2JqZWN0LT5FeGVjdXRlKAogICAgJy0tdGVtcGxhdGUtbnVtYmVyJywKICAgICRIZWxwZXItPkdldFJhbmRvbU51bWJlcigpLAogICAgJERlc3RpbmF0aW9uUGF0aCAuICdUZW1wbGF0ZUV4cG9ydC5jc3YnCik7CgokU2VsZi0+SXMoCiAgICAkRXhpdENvZGUsCiAgICAxLAogICAgIkNvbW1hbmQgd2l0aCB3cm9uZyB0ZW1wbGF0ZSBudW1iZXIgLSBleGl0IGNvZGUiLAopOwoKIyB0ZXN0IGNvbW1hbmQgd2l0aG91dCBkZXN0aW5hdGlvbiBhcmd1bWVudAokRXhpdENvZGUgPSAkQ29tbWFuZE9iamVjdC0+RXhlY3V0ZSggJy0tdGVtcGxhdGUtbnVtYmVyJywgJFRlbXBsYXRlSUQgKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDEsCiAgICAiTm8gZGVzdGluYXRpb24gYXJndW1lbnQgLSBleGl0IGNvZGUiLAopOwoKIyB0ZXN0IGNvbW1hbmQgd2l0aCAtLXRlbXBsYXRlLW51bWJlciBvcHRpb24gYW5kIGRlc3RpbmF0aW9uIGFyZ3VtZW50CiRFeGl0Q29kZSA9ICRDb21tYW5kT2JqZWN0LT5FeGVjdXRlKCAnLS10ZW1wbGF0ZS1udW1iZXInLCAkVGVtcGxhdGVJRCwgJERlc3RpbmF0aW9uUGF0aCAuICdUZW1wbGF0ZUV4cG9ydC5jc3YnICk7CgokU2VsZi0+SXMoCiAgICAkRXhpdENvZGUsCiAgICAwLAogICAgIk9wdGlvbiAtIC0tdGVtcGxhdGUtbnVtYmVyIG9wdGlvbiBhbmQgZGVzdGluYXRpb24gYXJndW1lbnQiLAopOwoKIyByZW1vdmUgdGVzdCBkZXN0aW5hdGlvbiBwYXRoCiRTdWNjZXNzID0gcm10cmVlKCBbJERlc3RpbmF0aW9uUGF0aF0gKTsKJFNlbGYtPlRydWUoCiAgICAkU3VjY2VzcywKICAgICJUZXN0IGRpcmVjdG9yeSBkZWxldGVkIC0gJERlc3RpbmF0aW9uUGF0aCIsCik7CgojIGNsZWFudXAgaXMgZG9uZSBieSBSZXN0b3JlRGF0YWJhc2UuCgoxOwo=
# --
# Copyright (C) 2001-2018 OTRS AG, http://otrs.com/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

# get selenium object
my $Selenium = $Kernel::OM->Get('Kernel::System::UnitTest::Selenium');

$Selenium->RunTest(
    sub {

        # get needed objects
        my $Helper               = $Kernel::OM->Get('Kernel::System::UnitTest::Helper');
        my $GeneralCatalogObject = $Kernel::OM->Get('Kernel::System::GeneralCatalog');

        # get 'Location' catalog class IDs
        my $ConfigItemDataRef = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::Class',
            Name  => 'Location',
        );
        my $LocationConfigItemID = $ConfigItemDataRef->{ItemID};

        # get 'Production' deployment state ID
        my $DeplStateDataRef = $GeneralCatalogObject->ItemGet(
            Class => 'ITSM::ConfigItem::DeploymentState',
            Name  => 'Production',
        );
        my $ProductionDeplStateID = $DeplStateDataRef->{ItemID};

        # get needed objects
        my $ConfigItemObject = $Kernel::OM->Get('Kernel::System::ITSMConfigItem');
        my $ConfigObject     = $Kernel::OM->Get('Kernel::Config');

        # create ConfigItem number
        my $ConfigItemNumber = $ConfigItemObject->ConfigItemNumberCreate(
            Type    => $ConfigObject->Get('ITSMConfigItem::NumberGenerator'),
            ClassID => $LocationConfigItemID,
        );

        $Self->True(
            $ConfigItemNumber,
            "ConfigItem number is created - $ConfigItemNumber",
        );

        # add 'Location' test ConfigItem
        my $ConfigItemID = $ConfigItemObject->ConfigItemAdd(
            Number  => $ConfigItemNumber,
            ClassID => $LocationConfigItemID,
            UserID  => 1,
        );

        $Self->True(
            $ConfigItemID,
            "ConfigItem 'Location' is created - ID $ConfigItemID",
        );

        # add a new version
        my $VersionName = "Selenium" . $Helper->GetRandomID();
        my $VersionID   = $ConfigItemObject->VersionAdd(
            Name         => $VersionName,
            DefinitionID => 1,
            DeplStateID  => $ProductionDeplStateID,
            InciStateID  => 1,
            UserID       => 1,
            ConfigItemID => $ConfigItemID,
        );

        $Self->True(
            $VersionID,
            "Test version of the ConfigItem is created - ID $VersionID",
        );

        # create test user and login
        my $TestUserLogin = $Helper->TestUserCreate(
            Groups => [ 'admin', 'itsm-configitem' ],
        ) || die "Did not get test user";

        $Selenium->Login(
            Type     => 'Agent',
            User     => $TestUserLogin,
            Password => $TestUserLogin,
        );

        # navigate to AdminImportExport screen
        my $ScriptAlias = $ConfigObject->Get('ScriptAlias');
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AdminImportExport");

        # check screen
        $Selenium->find_element( "table",             'css' );
        $Selenium->find_element( "table thead tr th", 'css' );
        $Selenium->find_element( "table tbody tr td", 'css' );

        # click on 'Add template'
        $Selenium->find_element("//a[contains(\@href, \'Action=AdminImportExport;Subaction=TemplateEdit' )]")
            ->VerifiedClick();

        # check and input step 1 of 5 screen
        for my $StepOneID (
            qw(Name Object Format ValidID Comment)
            )
        {
            my $Element = $Selenium->find_element( "#$StepOneID", 'css' );
            $Element->is_enabled();
            $Element->is_displayed();
        }
        my $ImportExportName = "ImportExport" . $Helper->GetRandomID();
        $Selenium->find_element( "#Name", 'css' )->send_keys($ImportExportName);
        $Selenium->execute_script(
            "\$('#Object').val('ITSMConfigItem').trigger('redraw.InputField').trigger('change');"
        );
        $Selenium->execute_script("\$('#Format').val('CSV').trigger('redraw.InputField').trigger('change');");
        $Selenium->find_element( "#Comment", 'css' )->send_keys('SeleniumTest');
        $Selenium->find_element("//button[\@class='Primary CallForAction'][\@type='submit']")->VerifiedClick();

        # check and input step 2 of 5 screen
        for my $StepTwoID (
            qw(ClassID CountMax EmptyFieldsLeaveTheOldValues)
            )
        {
            my $Element = $Selenium->find_element( "#$StepTwoID", 'css' );
            $Element->is_enabled();
            $Element->is_displayed();
        }
        $Selenium->execute_script(
            "\$('#ClassID').val('$LocationConfigItemID').trigger('redraw.InputField').trigger('change');"
        );
        $Selenium->find_element("//button[\@class='Primary CallForAction'][\@type='submit']")->VerifiedClick();

        # check and input step 3 of 5 screen
        for my $StepThreeID (
            qw(ColumnSeparator Charset IncludeColumnHeaders)
            )
        {
            my $Element = $Selenium->find_element( "#$StepThreeID", 'css' );
            $Element->is_enabled();
            $Element->is_displayed();
        }
        $Selenium->execute_script(
            "\$('#ColumnSeparator').val('Comma').trigger('redraw.InputField').trigger('change');"
        );
        $Selenium->find_element("//button[\@class='Primary CallForAction'][\@type='submit']")->VerifiedClick();

        # check and input step 4 of 5 screen
        $Selenium->find_element( "#MappingAddButton", 'css' )->VerifiedClick();
        for my $StepFourID (
            qw(Key Identifier)
            )
        {
            my $Element = $Selenium->find_element(".//*[\@id='Object::0::$StepFourID']");

            $Element->is_enabled();
            $Element->is_displayed();
        }

        for my $StepFourClass (
            qw(ArrowUp ArrowDown DeleteColumn)
            )
        {
            my $Element = $Selenium->find_element( ".$StepFourClass", 'css' );
            $Element->is_enabled();
            $Element->is_displayed();
        }
        $Selenium->find_element( "table",             'css' );
        $Selenium->find_element( "table thead tr th", 'css' );
        $Selenium->find_element( "table tbody tr td", 'css' );

        # select 'Number' mapping element
        $Selenium->find_element(".//*[\@id='Object::0::Key']/option[2]")->click();

        # add and select 'Name' mapping element
        $Selenium->find_element( "#MappingAddButton", 'css' )->VerifiedClick();
        $Selenium->find_element(".//*[\@id='Object::1::Key']/option[3]")->click();

        # add and select 'Deployment State' mapping element
        $Selenium->find_element( "#MappingAddButton", 'css' )->VerifiedClick();
        $Selenium->find_element(".//*[\@id='Object::2::Key']/option[4]")->click();

        # add and select 'Incident State' mapping element
        $Selenium->find_element( "#MappingAddButton", 'css' )->VerifiedClick();
        $Selenium->find_element(".//*[\@id='Object::3::Key']/option[5]")->click();
        $Selenium->find_element( "#SubmitNextButton", 'css' )->VerifiedClick();

        # check step 5 of 5 screen
        for my $StepFourID (
            qw(RestrictExport Number Name DeplStateIDs InciStateIDs Type Phone1 Phone2
            Fax E-Mail Address Note)
            )
        {
            my $Element = $Selenium->find_element( "#$StepFourID", 'css' );
            $Element->is_enabled();
            $Element->is_displayed();
        }

        # search ConfigItem by number and deployment state
        $Selenium->find_element( "#RestrictExport", 'css' )->click();
        $Selenium->find_element( "#Number",         'css' )->send_keys($ConfigItemNumber);
        $Selenium->find_element( "#Name",           'css' )->send_keys($VersionName);
        $Selenium->execute_script(
            "\$('#DeplStateIDs').val('$ProductionDeplStateID').trigger('redraw.InputField').trigger('change');"
        );
        $Selenium->execute_script("\$('#InciStateIDs').val('1').trigger('redraw.InputField').trigger('change');");
        $Selenium->find_element("//button[\@class='Primary CallForAction'][\@type='submit']")->VerifiedClick();

        # get needed objects
        my $ImportExportObject = $Kernel::OM->Get('Kernel::System::ImportExport');
        my $DBObject           = $Kernel::OM->Get('Kernel::System::DB');

        # get TemplateID of created test template
        $DBObject->Prepare(
            SQL   => 'SELECT id FROM imexport_template WHERE name = ?',
            Bind  => [ \$ImportExportName ],
            Limit => 1,
        );

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

        # navigate to test created ConfigItem and verify it
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AgentITSMConfigItemZoom;ConfigItemID=$ConfigItemID");
        $Self->True(
            index( $Selenium->get_page_source(), $VersionName ) > -1,
            "Test ConfigItem name $VersionName - found",
        );

        # export created test template
        my $ExportResultRef = $ImportExportObject->Export(
            TemplateID => $TemplateID,
            UserID     => 1,
        );

        # delete created test ConfigItem, so it can be imported back
        $ConfigItemObject->ConfigItemDelete(
            ConfigItemID => $ConfigItemID,
            UserID       => 1,
        );

        my $ConfigItem = $ConfigItemObject->ConfigItemGet(
            ConfigItemID => $ConfigItemID,
            Cache        => 0,
        );

        # check if ConfigItem is deleted
        $Self->False(
            $ConfigItem,
            "ConfigItem is deleted - ID $ConfigItemID",
        );

        # refresh screen and verify that test ConfigItem does not exist anymore
        $Selenium->VerifiedRefresh();
        $Self->True(
            index( $Selenium->get_page_source(), "Can\'t show item, no access rights for ConfigItem are given!" ) > -1,
            "Test ConfigItem name $VersionName is not found",
        );

        # get main object
        my $MainObject = $Kernel::OM->Get('Kernel::System::Main');

        # create test Exported file to a system
        my $ExportFileName = "ITSMExport" . $Helper->GetRandomID() . ".csv";
        my $ExportLocation = $ConfigObject->Get('Home') . "/var/tmp/" . $ExportFileName;
        my $Success        = $MainObject->FileWrite(
            Location   => $ExportLocation,
            Content    => \$ExportResultRef->{DestinationContent}->[0],
            Mode       => 'utf8',
            Type       => 'Attachment',
            Permission => '664',
        );
        $Self->True(
            $Success,
            "Export file $ExportFileName '$ExportLocation' is created",
        );

        # navigate to AdminImportExport screen
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AdminImportExport");

        # click on 'Import'
        $Selenium->find_element("//a[contains(\@href, \'Subaction=ImportInformation;TemplateID=$TemplateID' )]")
            ->VerifiedClick();

        # select Exported file and start importing
        $Selenium->find_element("//input[contains(\@name, \'SourceFile' )]")->send_keys($ExportLocation);

        $Selenium->find_element("//button[\@value='Start Import'][\@type='submit']")->VerifiedClick();

        # check for expected outcome
        $Self->True(
            index( $Selenium->get_page_source(), '(Created: 1)' ) > -1,
            "Import test ConfigItem - success",
        );

        # navigate to imported test created ConfigItem and verify it
        my $ImportedConfigItemID = $ConfigItemID + 1;
        $Selenium->VerifiedGet(
            "${ScriptAlias}index.pl?Action=AgentITSMConfigItemZoom;ConfigItemID=$ImportedConfigItemID"
        );
        $Self->True(
            index( $Selenium->get_page_source(), $VersionName ) > -1,
            "Test ConfigItem name $VersionName is found",
        );

        # navigate to AdminImportExport screen
        $Selenium->VerifiedGet("${ScriptAlias}index.pl?Action=AdminImportExport");

        # click to delete test template
        $Selenium->find_element( "#DeleteTemplateID$TemplateID", 'css' )->VerifiedClick();

        # delete test imported ConfigItem
        $Success = $ConfigItemObject->ConfigItemDelete(
            ConfigItemID => $ImportedConfigItemID,
            UserID       => 1,
        );
        $Self->True(
            $Success,
            "ConfigItem is deleted - ID $ImportedConfigItemID",
        );

        # delete test Exported file from system
        $Success = $MainObject->FileDelete(
            Location => $ExportLocation,
            Type     => 'Attachment',
        );
        $Self->True(
            $Success,
            "Export file $ExportFileName is deleted",
        );

    }
);

1;

Um93MS1Db2wxO1JvdzEtQ29sMjtSb3cxLUNvbDMNClJvdzItQ29sMTtSb3cyLUNvbDI7Um93Mi1Db2wzDQpSb3czLUNvbDE7Um93My1Db2wyO1JvdzMtQ29sMw0K
Um93MS1Db2wxCVJvdzEtQ29sMglSb3cxLUNvbDMNClJvdzItQ29sMQlSb3cyLUNvbDIJUm93Mi1Db2wzDQpSb3czLUNvbDEJUm93My1Db2wyCVJvdzMtQ29sMw0K
IlJvdzEtQ29sMSI6IlJvdzEtQ29sMiI6IlJvdzEtQ29sMyINCiJSb3cyLUNvbDEiOiJSb3cyLUNvbDIiOiJSb3cyLUNvbDMiDQoiUm93My1Db2wxIjoiUm93My1Db2wyIjoiUm93My1Db2wzIg0K
IlJvdzEtQ29sMSI7IlJvdzEtQ29sMiI7IlJvdzEtQ29sMyINCiJSb3cyLUNvbDEiOyJSb3cyLUNvbDIiOyJSb3cyLUNvbDMiDQoiUm93My1Db2wxIjsiUm93My1Db2wyIjsiUm93My1Db2wzIg0K
IlJvdzEtQ29sMSIJIlJvdzEtQ29sMiIJIlJvdzEtQ29sMyINCiJSb3cyLUNvbDEiCSJSb3cyLUNvbDIiCSJSb3cyLUNvbDMiDQoiUm93My1Db2wxIgkiUm93My1Db2wyIgkiUm93My1Db2wzIg0K
IgpUZXN0IDEgLSAxIjtUZXN0IDEgLSAyOyJUZXN0IDEKLSAzIjtUZXN0IFxuXHRcclxzDQoiVGVzdCAyIAotIDEiOyJUZQpzdCAyIC0gMiI7IlRlc3QgMiAtIDMKIjsNCg==
IgpUZXN0IDEgLSAxIglUZXN0IDEgLSAyCSJUZXN0IDEKLSAzIglUZXN0IFxuXHRcclxzDQoiVGVzdCAyIAotIDEiCSJUZQpzdCAyIC0gMiIJIlRlc3QgMiAtIDMKIgkNCg==
IgpUZXN0IDEgLSAxIjoiVGVzdCAxIC0gMiI6IlRlc3QgMQotIDMiOiJUZXN0IFxuXHRcclxzIg0KIlRlc3QgMiAKLSAxIjoiVGUKc3QgMiAtIDIiOiJUZXN0IDIgLSAzCiI6DQo=
IgpUZXN0IDEgLSAxIjsiVGVzdCAxIC0gMiI7IlRlc3QgMQotIDMiOyJUZXN0IFxuXHRcclxzIg0KIlRlc3QgMiAKLSAxIjsiVGUKc3QgMiAtIDIiOyJUZXN0IDIgLSAzCiI7DQo=
IgpUZXN0IDEgLSAxIgkiVGVzdCAxIC0gMiIJIlRlc3QgMQotIDMiCSJUZXN0IFxuXHRcclxzIg0KIlRlc3QgMiAKLSAxIgkiVGUKc3QgMiAtIDIiCSJUZXN0IDIgLSAzCiIJDQo=
ICBUZXN0ICA7ICAgIDtUZXN0ICANCiAgICBUZXN0OztUZXN0DQo7OyANCg==
ICBUZXN0ICAJICAgIAlUZXN0ICANCiAgICBUZXN0CQlUZXN0DQoJCSANCg==
IiAgVGVzdCAgIjoiICAgICI6IlRlc3QgICINCiIgICAgVGVzdCI6OiJUZXN0Ig0KOjoiICINCg==
IiAgVGVzdCAgIjsiICAgICI7IlRlc3QgICINCiIgICAgVGVzdCI7OyJUZXN0Ig0KOzsiICINCg==
IiAgVGVzdCAgIgkiICAgICIJIlRlc3QgICINCiIgICAgVGVzdCIJCSJUZXN0Ig0KCQkiICINCg==
IlRlc3Q7Ol/CsF4hIiLCpyQlJi8oKT0/wrRgKitUZXN0Ijs+PEB+J317W11cDQoiIiIiIjs7OjouLi0tX18jIyI7DQo=
IlRlc3Q7Ol/CsF4hIiLCpyQlJi8oKT0/wrRgKitUZXN0Igk+PEB+J317W11cDQoiIiIiIjs7OjouLi0tX18jIyIJDQo=
IlRlc3Q7Ol/CsF4hIiLCpyQlJi8oKT0/wrRgKitUZXN0IjoiPjxAfid9e1tdXCINCiIiIiIiOzs6Oi4uLS1fXyMjIjoNCg==
IlRlc3Q7Ol/CsF4hIiLCpyQlJi8oKT0/wrRgKitUZXN0IjsiPjxAfid9e1tdXCINCiIiIiIiOzs6Oi4uLS1fXyMjIjsNCg==
IlRlc3Q7Ol/CsF4hIiLCpyQlJi8oKT0/wrRgKitUZXN0IgkiPjxAfid9e1tdXCINCiIiIiIiOzs6Oi4uLS1fXyMjIgkNCg==
w7zDtsOkw587w5zDlsOEDQrDn8Okw7bDvDvDhMOWw5wNCg==
w7zDtsOkw58Jw5zDlsOEDQrDn8Okw7bDvAnDhMOWw5wNCg==
IsO8w7bDpMOfIjoiw5zDlsOEIg0KIsOfw6TDtsO8Ijoiw4TDlsOcIg0K
IsO8w7bDpMOfIjsiw5zDlsOEIg0KIsOfw6TDtsO8Ijsiw4TDlsOcIg0K
IsO8w7bDpMOfIgkiw5zDlsOEIg0KIsOfw6TDtsO8Igkiw4TDlsOcIg0K
IsqpIMqsIMquIjsiIMqhIMukIM6MICINCiIgIM6XIM+XIM+gICAiOyLOhiDOmyDOniINCg==
IsqpIMqsIMquIjoiIMqhIMukIM6MICINCiIgIM6XIM+XIM+gICAiOiLOhiDOmyDOniINCg==
IsqpIMqsIMquIgkiIMqhIMukIM6MICINCiIgIM6XIM+XIM+gICAiCSLOhiDOmyDOniINCg==
IlJvdzEtQ29sMSI7IlJvdzEtQ29sMiI7IlJvdzEtQ29sMyINCiJSb3cyLUNvbDEiOyJSb3cyLUNvbDIiOyJSb3cyLUNvbDMiDQoiUm93My1Db2wxIjsiMFJvdzMtQ29sMiI7IlJvdzMtQ29sMyINCg==
IlJvdzEtQ29sMSI7IlJvdzEtQ29sMiI7IlJvdzEtQ29sMyINCiJSb3cyLUNvbDEiOyIwUm93Mi1Db2wyIjsiUm93Mi1Db2wzIg0KIlJvdzMtQ29sMSI7IlJvdzMtQ29sMiI7IlJvdzMtQ29sMyINCg==
Ik5hbWUiLCJEZXBsb3ltZW50IFN0YXRlIiwiSW5jaWRlbnQgU3RhdGUiCiJUZXN0Q29uZmlnSXRlbXRlc3QxNDM1ODU2ODk1MTQ3OTYyMzc2IiwiUHJvZHVjdGlvbiIsIk9wZXJhdGlvbmFsIgoiVGVzdENvbmZpZ0l0ZW10ZXN0MTQzNTg1Njg5NTE0MTgwMjc3IiwiUHJvZHVjdGlvbiIsIk9wZXJhdGlvbmFsIgoiVGVzdENvbmZpZ0l0ZW10ZXN0MTQzNTg1Njg5NTE4MDg3MzMxMCIsIlByb2R1Y3Rpb24iLCJPcGVyYXRpb25hbCIKIlRlc3RDb25maWdJdGVtdGVzdDE0MzU4NTY4OTU1MTM5MjQ0MzkiLCJQcm9kdWN0aW9uIiwiT3BlcmF0aW9uYWwiCiJUZXN0Q29uZmlnSXRlbXRlc3QxNDM1ODU2ODk1MzAwNzMwMzMiLCJQcm9kdWN0aW9uIiwiT3BlcmF0aW9uYWwiCiJUZXN0Q29uZmlnSXRlbXRlc3QxNDM1ODU2ODk1NDkzMjczNjg4IiwiUHJvZHVjdGlvbiIsIk9wZXJhdGlvbmFsIgoiVGVzdENvbmZpZ0l0ZW10ZXN0MTQzNTg1Njg5NTE4ODIxNjA2NSIsIlByb2R1Y3Rpb24iLCJPcGVyYXRpb25hbCIKIlRlc3RDb25maWdJdGVtdGVzdDE0MzU4NTY4OTU5ODY2OTMzOSIsIlByb2R1Y3Rpb24iLCJPcGVyYXRpb25hbCIKIlRlc3RDb25maWdJdGVtdGVzdDE0MzU4NTY4OTU2MjE5MTAwODYiLCJQcm9kdWN0aW9uIiwiT3BlcmF0aW9uYWwiCiJUZXN0Q29uZmlnSXRlbXRlc3QxNDM1ODU2ODk1NTQ0MTk2NzI0IiwiUHJvZHVjdGlvbiIsIk9wZXJhdGlvbmFsIg==
IyAtLQojIHZhci9wYWNrYWdlc2V0dXAvSW1wb3J0RXhwb3J0LnBtCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMTggT1RSUyBBRywgaHR0cDovL290cnMuY29tLwojIENvcHlyaWdodCAoQykgMjAxMC0yMDE4IE9GT1JLLCBodHRwczovL28tZm9yay5kZQojIC0tCiMgJElkOiBJbXBvcnRFeHBvcnQucG0sdiAxLjEuMS4xIDIwMTgvMTAvMDIgMTU6MTY6MDQgdWQgRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCgpwYWNrYWdlIHZhcjo6cGFja2FnZXNldHVwOjpJbXBvcnRFeHBvcnQ7ICAgICMjIG5vIGNyaXRpYwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06OkRCJywKICAgICdLZXJuZWw6OkNvbmZpZycsCiAgICAnS2VybmVsOjpTeXN0ZW06OlN5c0NvbmZpZycsCik7Cgo9aGVhZDEgTkFNRQoKdmFyOjpwYWNrYWdlc2V0dXA6OkltcG9ydEV4cG9ydCAtIGNvZGUgdG8gZXhlY3V0ZSBkdXJpbmcgcGFja2FnZSBpbnN0YWxsYXRpb24KCj1oZWFkMSBQVUJMSUMgSU5URVJGQUNFCgo9Y3V0Cgo9aGVhZDIgbmV3KCkKCkNyZWF0ZSBhbiBvYmplY3QKCiAgICB1c2UgS2VybmVsOjpTeXN0ZW06Ok9iamVjdE1hbmFnZXI7CiAgICBsb2NhbCAkS2VybmVsOjpPTSA9IEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyLT5uZXcoKTsKICAgIG15ICRDb2RlT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgndmFyOjpwYWNrYWdlc2V0dXA6OkltcG9ydEV4cG9ydCcpOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICAjIGFsd2F5cyBkaXNjYXJkIHRoZSBjb25maWcgb2JqZWN0IGJlZm9yZSBwYWNrYWdlIGNvZGUgaXMgZXhlY3V0ZWQsCiAgICAjIHRvIG1ha2Ugc3VyZSB0aGF0IHRoZSBjb25maWcgb2JqZWN0IHdpbGwgYmUgY3JlYXRlZCBuZXdseSwgc28gdGhhdCBpdAogICAgIyB3aWxsIHVzZSB0aGUgcmVjZW50bHkgd3JpdHRlbiBuZXcgY29uZmlnIGZyb20gdGhlIHBhY2thZ2UKICAgICRLZXJuZWw6Ok9NLT5PYmplY3RzRGlzY2FyZCgKICAgICAgICBPYmplY3RzID0+IFsnS2VybmVsOjpDb25maWcnXSwKICAgICk7CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aGVhZDIgQ29kZUluc3RhbGwoKQoKUnVuIHRoZSBjb2RlIGluc3RhbGwgcGFydAoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+Q29kZUluc3RhbGwoKTsKCj1jdXQKCnN1YiBDb2RlSW5zdGFsbCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiAxOwp9Cgo9aGVhZDIgQ29kZVJlaW5zdGFsbCgpCgpSdW4gdGhlIGNvZGUgcmVpbnN0YWxsIHBhcnQKCiAgICBteSAkUmVzdWx0ID0gJENvZGVPYmplY3QtPkNvZGVSZWluc3RhbGwoKTsKCj1jdXQKCnN1YiBDb2RlUmVpbnN0YWxsIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIDE7Cn0KCj1oZWFkMiBDb2RlVXBncmFkZSgpCgpSdW4gdGhlIGNvZGUgdXBncmFkZSBwYXJ0CgogICAgbXkgJFJlc3VsdCA9ICRDb2RlT2JqZWN0LT5Db2RlVXBncmFkZSgpOwoKPWN1dAoKc3ViIENvZGVVcGdyYWRlIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIDE7Cn0KCj1oZWFkMiBDb2RlVXBncmFkZUZyb21CZWZvcmVfMl8wXzMoKQoKVGhpcyBmdW5jdGlvbiBpcyBvbmx5IGV4ZWN1dGVkIGlmIHRoZSBpbnN0YWxsZWQgbW9kdWxlIHZlcnNpb24gaXMgc21hbGxlciB0aGFuIDIuMC4zLgoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+Q29kZVVwZ3JhZGVGcm9tQmVmb3JlXzJfMF8zKCk7Cgo9Y3V0CgpzdWIgQ29kZVVwZ3JhZGVGcm9tQmVmb3JlXzJfMF8zIHsgICAgIyMgbm8gY3JpdGljCiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgZml4IGEgdHlwbyBpbiB0aGUgZGF0YWJhc2UKICAgICRTZWxmLT5fRml4RGF0YWJhc2VUeXBvKCk7CgogICAgcmV0dXJuIDE7Cn0KCj1oZWFkMiBDb2RlVXBncmFkZUZyb21Mb3dlclRoYW5fNF8wXzkxKCkKClRoaXMgZnVuY3Rpb24gaXMgb25seSBleGVjdXRlZCBpZiB0aGUgaW5zdGFsbGVkIG1vZHVsZSB2ZXJzaW9uIGlzIHNtYWxsZXIgdGhhbiA0LjAuOTEuCgpteSAkUmVzdWx0ID0gJENvZGVPYmplY3QtPkNvZGVVcGdyYWRlRnJvbUxvd2VyVGhhbl80XzBfOTEoKTsKCj1jdXQKCnN1YiBDb2RlVXBncmFkZUZyb21Mb3dlclRoYW5fNF8wXzkxIHsgICAgIyMgbm8gY3JpdGljCiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hhbmdlIGNvbmZpZ3VyYXRpb25zIHRvIG1hdGNoIHRoZSBuZXcgbW9kdWxlIGxvY2F0aW9uLgogICAgJFNlbGYtPl9NaWdyYXRlQ29uZmlncygpOwoKICAgIHJldHVybiAxOwp9Cgo9aGVhZDIgQ29kZVVuaW5zdGFsbCgpCgpydW4gdGhlIGNvZGUgdW5pbnN0YWxsIHBhcnQKCiAgICBteSAkUmVzdWx0ID0gJENvZGVPYmplY3QtPkNvZGVVbmluc3RhbGwoKTsKCj1jdXQKCnN1YiBDb2RlVW5pbnN0YWxsIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgcmV0dXJuIDE7Cn0KCj1oZWFkMSBQUklWQVRFIElOVEVSRkFDRQoKPWhlYWQyIF9GaXhEYXRhYmFzZVR5cG8oKQoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+X0ZpeERhdGFiYXNlVHlwbygpOwoKPWN1dAoKc3ViIF9GaXhEYXRhYmFzZVR5cG8gewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGZpeCB0aGUgQ29sdW1uU2VwZXJhdG9yIHR5cG8gKGNvcnJlY3QgaXMgQ29sdW1uU2VwYXJhdG9yKQogICAgcmV0dXJuIGlmICEkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6REInKS0+RG8oCiAgICAgICAgU1FMID0+ICJVUERBVEUgaW1leHBvcnRfZm9ybWF0ICIKICAgICAgICAgICAgLiAiU0VUIGRhdGFfa2V5ID0gJ0NvbHVtblNlcGFyYXRvcicgIgogICAgICAgICAgICAuICJXSEVSRSBkYXRhX2tleSA9ICdDb2x1bW5TZXBlcmF0b3InIiwKICAgICk7CgogICAgcmV0dXJuIDE7Cn0KCj1oZWFkMiBfTWlncmF0ZUNvbmZpZ3MoKQoKY2hhbmdlIGNvbmZpZ3VyYXRpb25zIHRvIG1hdGNoIHRoZSBuZXcgbW9kdWxlIGxvY2F0aW9uLgoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+X01pZ3JhdGVDb25maWdzKCk7Cgo9Y3V0CgpzdWIgX01pZ3JhdGVDb25maWdzIHsKCiAgICAjIGNyZWF0ZSBuZWVkZWQgb2JqZWN0cwogICAgbXkgJENvbmZpZ09iamVjdCAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyk7CiAgICBteSAkU3lzQ29uZmlnT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlN5c0NvbmZpZycpOwoKICAgIG15IEBOZXdTZXR0aW5nczsKCiAgICAjIG1pZ3JhdGUgTmF2QmFyIG1vZHVsZXMKICAgICMgZ2V0IHNldHRpbmcgY29udGVudCBmb3IgTmF2QmFyIG1vZHVsZXMKICAgIG15ICRTZXR0aW5nID0gJENvbmZpZ09iamVjdC0+R2V0KCdGcm9udGVuZDo6TmF2aWdhdGlvbk1vZHVsZScpOwoKICAgICMgdXBkYXRlIG1vZHVsZSBsb2NhdGlvbgogICAgJFNldHRpbmctPnsnQWRtaW5JbXBvcnRFeHBvcnQnfS0+e01vZHVsZX0gPSAiS2VybmVsOjpPdXRwdXQ6OkhUTUw6Ok5hdkJhcjo6TW9kdWxlQWRtaW4iOwoKICAgICMgQnVpbGQgbmV3IHNldHRpbmcuCiAgICBwdXNoIEBOZXdTZXR0aW5ncywgewogICAgICAgIE5hbWUgICAgICAgICAgID0+ICdGcm9udGVuZDo6TmF2aWdhdGlvbk1vZHVsZSMjI0FkbWluSW1wb3J0RXhwb3J0JywKICAgICAgICBFZmZlY3RpdmVWYWx1ZSA9PiAkU2V0dGluZy0+eydBZG1pbkltcG9ydEV4cG9ydCd9LAogICAgfTsKCiAgICAjIFdyaXRlIG5ldyBzZXR0aW5nLgogICAgJFN5c0NvbmZpZ09iamVjdC0+U2V0dGluZ3NTZXQoCiAgICAgICAgVXNlcklEICAgPT4gMSwKICAgICAgICBDb21tZW50cyA9PiAnSW1wb3J0RXhwb3J0IC0gcGFja2FnZSBzZXR1cCBmdW5jdGlvbjogX01pZ3JhdGVDb25maWdzJywKICAgICAgICBTZXR0aW5ncyA9PiBcQE5ld1NldHRpbmdzLAogICAgKTsKCiAgICByZXR1cm4gMTsKfQoKMTsKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPRk9SSyBwcm9qZWN0IChMPGh0dHBzOi8vby1mb3JrLmRlLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQ+LgoKPWN1dAo=
Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDAxLTIwMTggT1RSUyBBRywgaHR0cDovL290cnMuY29tLwovLyBDb3B5cmlnaHQgKEMpIDIwMTAtMjAxOCBPRk9SSywgaHR0cHM6Ly9vLWZvcmsuZGUKLy8gLS0KLy8gVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKLy8gdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQovLyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2FncGwudHh0LgovLyAtLQoKInVzZSBzdHJpY3QiOwoKdmFyIElUU00gPSBJVFNNIHx8IHt9OwpJVFNNLkFkbWluID0gSVRTTS5BZG1pbiB8fCB7fTsKCgovKioKICogQG5hbWVzcGFjZSBBZG1pbgogKiBAYXV0aG9yIE9UUlMgQUcKICogQGRlc2NyaXB0aW9uCiAqICAgICAgVGhpcyBuYW1lc3BhY2UgY29udGFpbnMgdGhlIHNwZWNpYWwgbW9kdWxlIGJlaGF2aW91cnMgZm9yIElUU00gU2VydmljZSBab29tLgogKi8KIElUU00uQWRtaW4uSW1wb3J0RXhwb3J0ID0gKGZ1bmN0aW9uIChUYXJnZXROUykgewoKICAgIC8qKgogICAgICogQG5hbWUgSW5pdAogICAgICogQG1lbWJlcm9mIEFkbWluLkltcG9ydEV4cG9ydAogICAgICogQGZ1bmN0aW9uCiAgICAgKiBAZGVzY3JpcHRpb24KICAgICAqICAgICAgVGhpcyBmdW5jdGlvbiBpbml0aWFsaXplcyBhY3Rpb25zIGZvciBJVFNNIFNlcnZpY2UgWm9vbS4KICAgICAqLwogICAgVGFyZ2V0TlMuSW5pdCA9IGZ1bmN0aW9uKCkgewoKICAgICAgICB2YXIgJE5leHRCdXR0b24sICRGaXJzdENvbHVtbjsKCiAgICAgICAgaWYgKENvcmUuQ29uZmlnLkdldCgnVGVtcGxhdGVFZGl0NCcpKSB7CgogICAgICAgICAgICAvLyBmaW5kIHRoZSBuZXh0IGJ1dHRvbiBhbmQgZ2V0IHRoZSBmaXJzdCBjb2x1bW4gZHJvcGRvd24KICAgICAgICAgICAgJE5leHRCdXR0b24gPSAkKCJidXR0b24uUHJpbWFyeVtuYW1lPSdTdWJtaXROZXh0QnV0dG9uJ10iKS5maXJzdCgpOwogICAgICAgICAgICAkRmlyc3RDb2x1bW4gPSAkKCcjT2JqZWN0XFw6XFw6MFxcOlxcOktleScpOwoKICAgICAgICAgICAgLy8gaGFuZGxlIGNoYW5nZXMgdG8gdGhlIGZpcnN0IGNvbHVtbiBzZWxlY3RvcgogICAgICAgICAgICAkRmlyc3RDb2x1bW4uYmluZCgnY2hhbmdlJywgZnVuY3Rpb24gKCkgewoKICAgICAgICAgICAgICAgIC8vIGNoZWNrIGlmIHRoZXJlIGlzIGF0IGxlYXN0IG9uZSBjb2x1bW4gd2l0aCBhIHZhbHVlCiAgICAgICAgICAgICAgICBpZiAoJEZpcnN0Q29sdW1uLnZhbCgpKSB7CiAgICAgICAgICAgICAgICAgICAgLy8gd2UgcmVtb3ZlIHRoZSBkaXNhYmxlZCBhdHRyaWJ1dGUKICAgICAgICAgICAgICAgICAgICAkTmV4dEJ1dHRvbi5yZW1vdmVBdHRyKCJkaXNhYmxlZCIpOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgZWxzZSB7CiAgICAgICAgICAgICAgICAgICAgLy8gd2UgYWRkIHRoZSBkaXNhYmxlZCBhdHRyaWJ1dGUKICAgICAgICAgICAgICAgICAgICAkTmV4dEJ1dHRvbi5hdHRyKCJkaXNhYmxlZCIsICJkaXNhYmxlZCIpOwogICAgICAgICAgICAgICAgfQoKICAgICAgICAgICAgfSkudHJpZ2dlcignY2hhbmdlJyk7CgogICAgICAgICAgICAvLyBzZXQgdGhlIGhpZGRlbiBmaWVsZCB0byBkZWxldGUgdGhpcyBjb2x1bW4gYW5kIHN1Ym1pdCB0aGUgZm9ybQogICAgICAgICAgICAkKCcuRGVsZXRlQ29sdW1uJykudW5iaW5kKCdjbGljaycpLmJpbmQoJ2NsaWNrJywgZnVuY3Rpb24oKSB7CiAgICAgICAgICAgICAgICAkKHRoaXMpLmNsb3Nlc3QoJ3RkJykuZmluZCgnaW5wdXRbdHlwZT0iaGlkZGVuIl0nKS52YWwoMSk7CiAgICAgICAgICAgICAgICAkKHRoaXMpLmNsb3Nlc3QoJ2Zvcm0nKS5zdWJtaXQoKTsKICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwogICAgICAgICAgICB9KTsKCiAgICAgICAgICAgICQoJyNNYXBwaW5nQWRkQnV0dG9uJykuYmluZCgnY2xpY2snLCBmdW5jdGlvbiAoKSB7CiAgICAgICAgICAgICAgICAkKCdpbnB1dFtuYW1lPU1hcHBpbmdBZGRdJykudmFsKCcxJyk7CiAgICAgICAgICAgICAgICAkKCdpbnB1dFtuYW1lPVN1Ym1pdE5leHRdJykudmFsKCcwJyk7CiAgICAgICAgICAgIH0pOwoKICAgICAgICAgICAgJCgnI1N1Ym1pdE5leHRCdXR0b24nKS5iaW5kKCdjbGljaycsIGZ1bmN0aW9uICgpIHsKICAgICAgICAgICAgICAgICQoJ2lucHV0W25hbWU9TWFwcGluZ0FkZF0nKS52YWwoJzAnKTsKICAgICAgICAgICAgICAgICQoJ2lucHV0W25hbWU9U3VibWl0TmV4dF0nKS52YWwoJzEnKTsKICAgICAgICAgICAgfSk7CiAgICAgICAgfQoKICAgICAgICBpZiAoQ29yZS5Db25maWcuR2V0KCdUZW1wbGF0ZU92ZXJ2aWV3JykpIHsKCiAgICAgICAgICAgICQoJ2J1dHRvbi5CYWNrJykuYmluZCgnY2xpY2snLCBmdW5jdGlvbiAoKSB7CiAgICAgICAgICAgICAgICBsb2NhdGlvbi5ocmVmID0gQ29yZS5Db25maWcuR2V0KCdCYXNlbGluaycpICsgQ29yZS5Db25maWcuR2V0KCdCYWNrVVJMJyk7CiAgICAgICAgICAgIH0pOwoKICAgICAgICAgICAgQ29yZS5Gb3JtLlZhbGlkYXRlLkFkZE1ldGhvZCgiVmFsaWRhdGVfTnVtYmVyQmlnZ2VyVGhhblplcm8iLCBmdW5jdGlvbihWYWx1ZSkgewogICAgICAgICAgICAgICAgdmFyIE51bWJlciA9IHBhcnNlSW50KFZhbHVlLCAxMCk7CiAgICAgICAgICAgICAgICBpZiAoaXNOYU4oTnVtYmVyKSkgewogICAgICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTsKICAgICAgICAgICAgICAgIH0KCiAgICAgICAgICAgICAgICBpZiAoTnVtYmVyID4gMCkgewogICAgICAgICAgICAgICAgICAgIHJldHVybiB0cnVlOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgcmV0dXJuIGZhbHNlOwoKICAgICAgICAgICAgfSk7CgogICAgICAgICAgICBDb3JlLkZvcm0uVmFsaWRhdGUuQWRkTWV0aG9kKCJWYWxpZGF0ZV9OdW1iZXJJbnRlZ2VyIiwgZnVuY3Rpb24oVmFsdWUpIHsKICAgICAgICAgICAgICAgIHJldHVybiAoVmFsdWUubWF0Y2goL15bMC05XSskLykpID8gdHJ1ZSA6IGZhbHNlOwoKICAgICAgICAgICAgfSk7CgogICAgICAgICAgICBDb3JlLkZvcm0uVmFsaWRhdGUuQWRkUnVsZSgiVmFsaWRhdGVfTnVtYmVyQmlnZ2VyVGhhblplcm8iLCB7IFZhbGlkYXRlX051bWJlckJpZ2dlclRoYW5aZXJvOiB0cnVlIH0pOwogICAgICAgICAgICBDb3JlLkZvcm0uVmFsaWRhdGUuQWRkUnVsZSgiVmFsaWRhdGVfTnVtYmVySW50ZWdlciIsIHsgVmFsaWRhdGVfTnVtYmVySW50ZWdlcjogdHJ1ZSB9KTsKICAgICAgICAgICAgQ29yZS5Gb3JtLlZhbGlkYXRlLkFkZFJ1bGUoIlZhbGlkYXRlX051bWJlckludGVnZXJCaWdnZXJUaGFuWmVybyIsIHsgVmFsaWRhdGVfTnVtYmVySW50ZWdlcjogdHJ1ZSwgVmFsaWRhdGVfTnVtYmVyQmlnZ2VyVGhhblplcm86IHRydWUgfSk7CgogICAgICAgIH0KCiAgICB9OwoKICAgIENvcmUuSW5pdC5SZWdpc3Rlck5hbWVzcGFjZShUYXJnZXROUywgJ0FQUF9NT0RVTEUnKTsKCiAgICByZXR1cm4gVGFyZ2V0TlM7Cn0oSVRTTS5BZG1pbi5JbXBvcnRFeHBvcnQgfHwge30pKTsK
LyoqCiAqIEBwcm9qZWN0ICAgICBPRk9SSyAoaHR0cHM6Ly9vLWZvcmsuZGUpIC0gQWdlbnQgRnJvbnRlbmQKICogQGNvcHlyaWdodCAgIE9GT1JLCiAqIEBsaWNlbnNlICAgICBBR1BMIChodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQpCiAqLwoKLyoqCiAqIEBwYWNrYWdlICAgICBTa2luICJEZWZhdWx0IgogKiBAc2VjdGlvbiAgICAgSW1wb3J0IEV4cG9ydCBTY3JlZW4KICovCgpAbWVkaWEgc2NyZWVuLHByb2plY3Rpb24sdHYsaGFuZGhlbGQgewoKLyoqCiAqIEBzdWJzZWN0aW9uCiAqLwoKCi5NYXBIZWFkZXJSb3cgbGFiZWwgewogICAgY29sb3I6ICM5MjkyOTI7Cn0KCi5NYXBIZWFkZXJSb3cgLkhlYWRlciwKLk1hcEhlYWRlclJvdyAuRmllbGQgewogICAgZGlzcGxheTogaW5saW5lOwogICAgbWFyZ2luLXJpZ2h0OiAxNXB4OwogICAgcGFkZGluZy1sZWZ0OiAycHg7Cn0KCmJ1dHRvbi5BcnJvd1VwLApidXR0b24uQXJyb3dEb3duIHsKICAgIGhlaWdodDogMTZweDsKICAgIHdpZHRoOiAxNnB4OwogICAgcGFkZGluZzogMXB4OwogICAgbWFyZ2luLXRvcDogMDsKICAgIG1hcmdpbi1ib3R0b206IDFweDsKICAgIGJvcmRlci1zdHlsZTogbm9uZTsKICAgIHRleHQtaW5kZW50OiAtOTk5OXB4OwogICAgZGlzcGxheTogaW5saW5lLWJsb2NrOwogICAgdmVydGljYWwtYWxpZ246IG1pZGRsZTsKICAgIGN1cnNvcjogcG9pbnRlcjsKfQoKYnV0dG9uLkFycm93VXA6YWN0aXZlLApidXR0b24uQXJyb3dEb3duOmFjdGl2ZSB7CiAgICBtYXJnaW4tdG9wOiAxcHg7CiAgICBtYXJnaW4tYm90dG9tOiAwOwp9CgpidXR0b24uQXJyb3dVcFtkaXNhYmxlZD0iZGlzYWJsZWQiXTphY3RpdmUsCmJ1dHRvbi5BcnJvd0Rvd25bZGlzYWJsZWQ9ImRpc2FibGVkIl06YWN0aXZlIHsKICAgIG1hcmdpbi10b3A6IDBweDsKICAgIG1hcmdpbi1ib3R0b206IDFweDsKfQoKYnV0dG9uLkFycm93VXAgewogICAgYmFja2dyb3VuZDogdXJsKC4uL2ltZy9pY29ucy9pbXBvcnRleHBvcnRfYXJyb3dfdXAucG5nKTsKfQoKYnV0dG9uLkFycm93VXBbZGlzYWJsZWQ9ImRpc2FibGVkIl0gewogICAgYmFja2dyb3VuZDogdXJsKC4uL2ltZy9pY29ucy9pbXBvcnRleHBvcnRfYXJyb3dfdXBfZGlzYWJsZWQucG5nKTsKICAgIGN1cnNvcjogZGVmYXVsdDsKfQoKYnV0dG9uLkFycm93RG93biB7CiAgICBiYWNrZ3JvdW5kOiB1cmwoLi4vaW1nL2ljb25zL2ltcG9ydGV4cG9ydF9hcnJvd19kb3duLnBuZyk7Cn0KCmJ1dHRvbi5BcnJvd0Rvd25bZGlzYWJsZWQ9ImRpc2FibGVkIl17CiAgICBiYWNrZ3JvdW5kOiB1cmwoLi4vaW1nL2ljb25zL2ltcG9ydGV4cG9ydF9hcnJvd19kb3duX2Rpc2FibGVkLnBuZyk7CiAgICBjdXJzb3I6IGRlZmF1bHQ7Cn0KCn0gLyogZW5kIEBtZWRpYSAqLwo=
LyoqCiAqIEBwcm9qZWN0ICAgICBPRk9SSyAoaHR0cHM6Ly9vLWZvcmsuZGUpIC0gQWdlbnQgRnJvbnRlbmQKICogQGNvcHlyaWdodCAgIE9GT1JLCiAqIEBsaWNlbnNlICAgICBBR1BMIChodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQpCiAqLwoKLyoqCiAqIEBwYWNrYWdlICAgICBTa2luICJEZWZhdWx0IgogKiBAc2VjdGlvbiAgICAgSW1wb3J0IEV4cG9ydCBTY3JlZW4KICovCgpAbWVkaWEgc2NyZWVuLHByb2plY3Rpb24sdHYsaGFuZGhlbGQgewoKLyoqCiAqIEBzdWJzZWN0aW9uCiAqLwoKCi5NYXBIZWFkZXJSb3cgbGFiZWwgewogICAgY29sb3I6ICM5MjkyOTI7Cn0KCi5NYXBIZWFkZXJSb3cgLkhlYWRlciwKLk1hcEhlYWRlclJvdyAuRmllbGQgewogICAgZGlzcGxheTogaW5saW5lOwogICAgbWFyZ2luLXJpZ2h0OiAxNXB4OwogICAgcGFkZGluZy1sZWZ0OiAycHg7Cn0KCmJ1dHRvbi5BcnJvd1VwLApidXR0b24uQXJyb3dEb3duIHsKICAgIGhlaWdodDogMTZweDsKICAgIHdpZHRoOiAxNnB4OwogICAgcGFkZGluZzogMXB4OwogICAgbWFyZ2luLXRvcDogMDsKICAgIG1hcmdpbi1ib3R0b206IDFweDsKICAgIGJvcmRlci1zdHlsZTogbm9uZTsKICAgIHRleHQtaW5kZW50OiAtOTk5OXB4OwogICAgZGlzcGxheTogaW5saW5lLWJsb2NrOwogICAgdmVydGljYWwtYWxpZ246IG1pZGRsZTsKICAgIGN1cnNvcjogcG9pbnRlcjsKfQoKYnV0dG9uLkFycm93VXA6YWN0aXZlLApidXR0b24uQXJyb3dEb3duOmFjdGl2ZSB7CiAgICBtYXJnaW4tdG9wOiAxcHg7CiAgICBtYXJnaW4tYm90dG9tOiAwOwp9CgpidXR0b24uQXJyb3dVcFtkaXNhYmxlZD0iZGlzYWJsZWQiXTphY3RpdmUsCmJ1dHRvbi5BcnJvd0Rvd25bZGlzYWJsZWQ9ImRpc2FibGVkIl06YWN0aXZlIHsKICAgIG1hcmdpbi10b3A6IDBweDsKICAgIG1hcmdpbi1ib3R0b206IDFweDsKfQoKYnV0dG9uLkFycm93VXAgewogICAgYmFja2dyb3VuZDogdXJsKC4uL2ltZy9pY29ucy9pbXBvcnRleHBvcnRfYXJyb3dfdXAucG5nKTsKfQoKYnV0dG9uLkFycm93VXBbZGlzYWJsZWQ9ImRpc2FibGVkIl0gewogICAgYmFja2dyb3VuZDogdXJsKC4uL2ltZy9pY29ucy9pbXBvcnRleHBvcnRfYXJyb3dfdXBfZGlzYWJsZWQucG5nKTsKICAgIGN1cnNvcjogZGVmYXVsdDsKfQoKYnV0dG9uLkFycm93RG93biB7CiAgICBiYWNrZ3JvdW5kOiB1cmwoLi4vaW1nL2ljb25zL2ltcG9ydGV4cG9ydF9hcnJvd19kb3duLnBuZyk7Cn0KCmJ1dHRvbi5BcnJvd0Rvd25bZGlzYWJsZWQ9ImRpc2FibGVkIl17CiAgICBiYWNrZ3JvdW5kOiB1cmwoLi4vaW1nL2ljb25zL2ltcG9ydGV4cG9ydF9hcnJvd19kb3duX2Rpc2FibGVkLnBuZyk7CiAgICBjdXJzb3I6IGRlZmF1bHQ7Cn0KCn0gLyogZW5kIEBtZWRpYSAqLwo=
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==