ImportExport
5.0.36
OTRS AG
https://otrs.com/
GNU GENERAL PUBLIC LICENSE Version 3, 29 June 2007
Build for OTRS::ITSM 5 patch level 36.
Build for OTRS::ITSM 5 patch level 35.
Build for OTRS::ITSM 5 patch level 34.
Build for OTRS::ITSM 5 patch level 33.
Build for OTRS::ITSM 5 patch level 32.
Build for OTRS::ITSM 5 patch level 31.
Build for OTRS::ITSM 5 patch level 30.
Build for OTRS::ITSM 5 patch level 29.
Build for OTRS::ITSM 5 patch level 28.
Build for OTRS::ITSM 5 patch level 27.
Build for OTRS::ITSM 5 patch level 26.
Build for OTRS::ITSM 5 patch level 25.
Build for OTRS::ITSM 5 patch level 24.
Build for OTRS::ITSM 5 patch level 23.
Build for OTRS::ITSM 5 patch level 22.
Build for OTRS::ITSM 5 patch level 21.
Build for OTRS::ITSM 5 patch level 20.
Build for OTRS::ITSM 5 patch level 19.
Build for OTRS::ITSM 5 patch level 18.
Build for OTRS::ITSM 5 patch level 17.
Build for OTRS::ITSM 5 patch level 16.
Build for OTRS::ITSM 5 patch level 15.
Build for OTRS::ITSM 5 patch level 14.
Build for OTRS::ITSM 5 patch level 13.
Build for OTRS::ITSM 5 patch level 12.
Build for OTRS::ITSM 5 patch level 11.
Build for OTRS::ITSM 5 patch level 10.
Build for OTRS::ITSM 5 patch level 9.
Build for OTRS::ITSM 5 patch level 8.
Build for OTRS::ITSM 5 patch level 7.
Build for OTRS::ITSM 5 patch level 6.
Build for OTRS::ITSM 5 patch level 5.
Build for OTRS::ITSM 5 patch level 4.
Build for OTRS::ITSM 5 patch level 3.
Build for OTRS::ITSM 5 patch level 2.
Build for OTRS::ITSM 5.
Build for OTRS::ITSM 5 rc1.
Build for OTRS::ITSM 5 beta3.
Build for OTRS::ITSM 5 beta2.
Build for OTRS::ITSM 5 beta1.
The ImportExport package.
Das ImportExport Paket.
El paquete ImportExport.
Az ImportExport csomag.
5.0.x
</br>
<strong>ATTENTION</strong>
</br>
</br>
If you uninstall this package, all database tables that were created during installation will be deleted.
All data from these tables will be irrevocably lost!
</br>
</br>
((enjoy))</br>
</br>
</br>
<strong>ACHTUNG</strong>
</br>
</br>
Bei der Deinstallation werden die von diesem Paket angelegten Datenbank-Tabellen gelöscht.
Alle darin enthaltenen Daten gehen unwiderruflich verloren!
</br>
</br>
((enjoy))</br>
</br>
<br/>
<strong>ATENCIÓN</strong>
<br/>
<br/>
Si usted desinstala este paquete, todas las tablas de la base de datos que se crearon durante la instalación
del mismo, se eliminarán. ¡La totalidad de los datos que contenían dichas tablas, se perderá irrevocablemente!
<br/>
<br/>
((enjoy))<br/>
<br/>
</br>
<strong>FIGYELEM</strong>
</br>
</br>
Ha eltávolítja ezt a csomagot, akkor a telepítés során létrehozott összes adatbázistábla törlésre kerül.
Az ezekben a táblákban lévő összes adat visszavonhatatlanul el fog veszni!
</br>
</br>
((enjoy))</br>
</br>
# create the package name
my $CodeModule = 'var::packagesetup::' . $Param{Structure}->{Name}->{Content};
# discard internally stored object, so that the next access to object creates them newly
$Kernel::OM->ObjectsDiscard(
Objects => [$CodeModule],
ForcePackageReload => 1,
);
$Kernel::OM->Get($CodeModule)->CodeUpgradeFromBefore_2_0_3();
# create the package name
my $CodeModule = 'var::packagesetup::' . $Param{Structure}->{Name}->{Content};
$Kernel::OM->Get($CodeModule)->CodeUpgradeFromLowerThan_4_0_91();
11612f2cd74086ddb8f5150c18a5eb9b72adba5e
2019-05-20 11:27:22
opms.otrs.com
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acqUKUKIixcv1tbWBgUFWSyW2traixcvDh48WFyJNMl/+srKyt27d1sslvHjx0+dOnXgwIF6vb6goOD777/ftWtXbGxsREREh+fB2k8//bR9+/bGxsaIiIhf//rX1ou4A4Armc1m6TLa5nMMAHpcZWWlTcmuXbukq1P5zkeKO7z00kurV6/+9NNPjx079h//8R9vvvnm3LlzdTrdzJkzDx06FBER8cADD7z44otSmCkiIqK1tbW6uloIoVKp/P39L1y4IFf12Wef/frXvx46dOj7778vXQ0ajcb/9//+X35+vsVimTNnztq1a6VnQEv719TUPPnkk9u3b//DH/6wfv16qc76+no5INJhLGzYsGHnzp0TQsybN8/6kE616+A8CCGWLVu2Z8+e8vLyKVOmrFy58o477uiwn4mJiUII67vxNtkcbrPZZrvtnbfm5uann376nXfe8fHxWbdu3V/+8pcTJ07IDe3bty8jI6Oqqso6Btep+mWbN29OTU2VN8eNGzdu3Dj7obW0tMiX3FOmTGloaHBcLoRITk7etGnTCy+84PikoVd1+fbQuoYJEybcdtttoaGhly9f/vLLLwsLCz/55JPhw4eHh4eLnrjXk58I352kHl9f3/T09Ndee+3cuXO7du269957hRAff/xxWVlZaGhoenq6fcqLtwzNBbxg3aXGxkZhl0lkQ3pX2rMLnr3ixRdf3LZtm16vv+mmm26//XbrfaZPn37zzTe3trYePnz4n//85/r161966SUp2ORkK999911ZWdntt9/enSjGs3bs95k0aVJqaurgwYOtQyqLFy++6aabpBWjAgMDR40atWDBAiGE9ReM7LbbbktJSRk0aJC/v39MTMzMmTOFEHq9vs1y62yjb775pqWl5eabb77//vvVanVAQEBoaOiUKVOkRabkr1IpYHT27FlpJrYcRZLKpU2TyXT27FlhFV369ttvzWZzXFzcfffdN2jQIF9f30GDBs2ZM+e6664zmUzffPONM+dBdujQoTfffLOxsTE+Pt7m+YAA4GLSh2GbH1YA0NtmzpxpEwSRNk0m05YtW2bNmlVUVLRly5b09HQhRFZWVk5OjsFgOH36dHJyclZWlnTId999N3ny5PPnz5eVlSUkJMhpDlJVly5dKisre/rpp//0pz9J5atWrYqOji4pKTlx4kR0dPSqVaus9583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AUCAyMzMlAYgIQUAQcH5R0YBAAgryjo1rOoNAKprbm5WOwQAgIf4qhYAENbILgEAAACXiewSACB8ybKs/HYqM+MAAAAAjzEzDgAQvuw/3+63sUvd3d29vb1JSUmJiYn+aREAAADwNb6qBQCEL/8v6d3X12cwGMxms99aBAAAAHyNsUsAgPDl/0WXkpKSYmNj4+Li/NYiAHhAkiQhhDJ3WHX+DEaSpAC5agAILoxdAgCEL2Xskj+zSwkJCampqWSXAKglKyvLncP8k2Hp7e115zB/pnvcbyspKcmnkQBAcCG7BAAIX8rYJZb0BhA+mpqaHD/29fXdd999mZmZWVlZe/bsmThxouPeo0eP5ubmxsTElJeX2wvXrl07adIkjUYzf/78d955RymUJEmSpB07djgd39vbe+edd2q12ry8vFOnTimHCSFaW1tnzJih0WgyMzPvueced9JM586dmzlzplarXbVqldFodGz38OHDQojDhw9LknTFFVfYTykoKCgoKHDzziinK+HZDXVdOp3OYDBIX3KzCQAIYTxPAwDCl/9nxgFAQNm4caPZbK6urj5z5oxGo+no6HDcW1FRcerUqX379hUXF9sLt2zZcurUqa6urg0bNixatEgpVIb86PX6mpoax+PXr1+v0+mampoqKyuPHDliP7KkpGTbtm1ms7m+vn7u3LklJSXDhrpnz54DBw7U19cnJib+8pe/VArff//9mTNnFhUVCSEKCwvz8/P//Oc/20+x/zCoO4qKigYePNR1KTdK/pKbTQBACGNeMQAgfLW3t3d0dKSkpGRkZKgdS9C44YYbPvvss+zsbLUDwVecPHmyt7d3+vTpageCAGIwGGRZPnXqlBAiJydHr9c77s3Ozm5sbMzKyjpx4oROpxt4uiRJFoslKipKOKxGVFdX98ADD1RVVZlMpry8vMrKSvvbhOOKRfbtjIyMzz//XKm/tbU1IyNDKR81apRjJis9Pb2lpcXFtUiS1NDQkJOTI4To6OiYPHmy/fglS5bcdtttS5YsOXDgwMGDB//0pz95dLf+1ZDT+9Gg1zXokfC/mTNnlpaWzp07V+1AALCqNwAgjDF2yQM9PT06ne7RRx9VOxB8xd13311fX8/fCxx99tlnu3fvVrYbGxuVDaecSH9/v4salNSSo6VLly5btmzXrl06na6trW3YVZwcZ405tjt9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bzcYDPYKe3p6li1bNn78+Nzc3MLCwosXLzoF09nZuXLlysTExPvuu08pv+qqq5SwJQfuXLKLtoQQ586dmzlzplarXbVqldFodIxhqFaGuhtCiIMHD+bl5SUkJHzjG9946aWXJk6ceJnBAwErDDuTMWPGOJ3ocRjD3ighxKpVqzIzMydNmnTo0CF7IZ1JUJIBAIBvnD9/vra29uLFi2oH4k2zZs269tpr1Y4CzoqKimJiYtSOYkiJiYkXLlzwf7sDn3WFEFu3bn3xxRfT0tKKi4v3798fHR1t37tixYp3333XbDb39PQcOnTo5z//uX3X6dOnS0pKenp6rFbr888/39DQ4Fjnf/zHfzQ1Ncmy/OSTT/70pz+177rnnns++eQTZfvtt99es2aNUzDXX3/9nj17uru7h418WC7aEkI88sgj58+f7+zs/NWvfuV4XS5aHOpu7N69++abb66pqenr62tra/vv//5vnU53mcF7y8KFC1944QW1WsewKisrr776ao9Pr6iomDNnjhfjcV9YdSYuTvQgDNc3SpZlo9Fos9k++OCDhIQEbwV/mZTvBc1msyqtBy+ySwAA+Mq5c+dqa2sNBoPagXgT2aXARHZpUIO+ENo3bDab0zE6nc7xW9j09HTHcz/44INp06YtXLiwurraqc7m5mZlu7OzMzU11b4rMzPT/n7S19eXmZnpdKLRaHQz8mG5aEsIYX+D7erqcrquoVoc6m6MGzfO9d8m2SUMJcSyS/aNEOtMXJzoQRiub9T7778/derU+Pj4GTNmuLjJfkZ2yTPMjAMAwFdsNpsQIiKC/22BQDRwnsXkyZMdH5RbWloc9yYkJGg0moaGhkuXLg1VlSzLMTEx9nJZlpV+QAyxKnBCQoI7oZpMpmGPGbYtRXR0tJudkou7YW/IHe4EDwS1EOtMXLjMMAbeqBUrVtx7773t7e2HDx8etnU6kwDH8y4AAL5itVoF2SUgeCxatOiZZ545f/682Wz+4osvVq9ebd915syZNWvWvPjii8eOHdu/f399fb3jia+//npzc7MQYs+ePbfeequ9fPHixe+9956y/corr9x8881uRhIdHa0sTXLhwoUdO3bMnj172FNct7Vz586GhobOzs69e/cuXbrUnRiGuhtbt2699957u7q6bDbb2bNn9+7du2LFissMHggxQd2ZuOBxGENpaWmZ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ukYoCMJLL700YcIEvV5/zTXXvPzyy2PGjAnd8KOPPvryl7+cmZmp0+lCTy+AVEF2CQAA9AudTldSUiKKosPhaGpqUjscAOjG2LFju9Zayi2BQODpp5/+9re//dRTT+3atWvRokVKhzVr1jz++ONer/fcuXPTp09fs2aN3F5VVXX27Nnvfve7TqczGAxu3Lixqqrq4sWLypgVFRXHjh1rbW294YYbHnjgAWXADRs2bNiw4eLFi/X19WvWrNmwYUNYMLfddtvXv/71mpqa7du3y+1nz56VV0kh4jnkJUuWzJ071+fz2e32CRMmNDQ0dHtcsieffLKuru6555674447up6urhE+88wzzz333PPPP2+3248fP26325ubm0M3fPTRR59//nm/319WVhZ6egGkCpFidQAA0H/a29vl2qXCwkKz2ax2OGlr3759e/fufe2119QOBOgbTz/99KFDh/bt2zfwuxbF8EskpUUUxWAwKJfnKH0KCgpaWlqUzlarNTRH89577/3whz8cOnTopk2brr322tAx6+vrbTabIAitra0jRoyw2+3yquLi4kuXLmm1WkEQPB7P8OHD6+rqQjd0uVx6vT6eyLt14MCB2267zefzFRYWWq3WvXv3Tpo0qdvj+utf//qjH/2osrLyuuuue++997qerq4Rjhw58sSJE8XFxRHDEEXR6/XKh5zYgSTss88+mzRpUmNj48DsDkhj1C4BAIB+lJubK18+NTU1tbW1qR0OAPRK1zvOrrrqqtCKodDUkiAIer3eYDBUVVV1dnZGG0qSJJ1Op7RLkqTMghQxyRIxtdRVPE/tnDdvntfrdTqdf//73++4445Zs2bFc1x33nnnqlWrmpubDx48GHHYiBFGm9pJpqSWAKQosksAAKB/5eXlWSwWQRAaGhpC5xYBgDQwf/78J5988tKlS16v99NPP12xYoWy6vz58/fee++LL75YUVGxb9++c+fOhW74xhtvyKWdzzzzzHe+8x2lfcGCBW+//ba8/Mc//vGWW26JMxKtVivPWFRbW7tjx46bbrqp202sVuu7776r0WiGDRt21VVXuVyueI6roaFh7NixDofjmWeeiTO2LVu2rFq1qrW1NRgMXrhw4fnnn7/zzjvj3BZASiC7BAAA+t3/b+/O46Oo8/yPV/WZTnc6SeciF3ggojgqp6DgunI46HiAKKis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6kiXAAAAAAAAkDrSJQAAAAAAAKSOdAkAAAAAAACpI10CAAAAAABA6kiXAAAAAAAAkDrSJQAAAAAAAKSOdAkAAAAAAACpI10CAAAAAABA6kiXAAAAAAAAkDrSJQAAAAAAAKSOdAkAAAAAAACpI10CAAAAAGBw+e677/71X/915MiRNpvNarWeeOKJN95443fffRfTTUxCvM5Wq3XUqFF33HFHQ0ND5wJ8Pt/q1asnTZqUl5dnMBgKCgrOOuuse++99+uvv07tjqKL6VHZSVainnjVVVd1vvpVV13VeeRkuN3uYcOGpXBifyPKsqx1DQAAAAAAoI+sWrXq3nvvDYfDMe16vX7FihV333232pJM5KGmCvE6l5SU7Nq169hjj1Vbmpqazj333D179iQesEeUqyvnJl928pUoY4qiaDab6+vr8/Ly1ENtbW2lpaWBQEDp36P6b7nllueeey4YDPb0xP6GuUsAAAAAAAwWjz322NKlSyORyKJFiz755BOXy+XxeD799NPf/e53kUhkyZIljz/+uNpZ/kcJGjuf0tTU9Oabbx533HGNjY333XdfdJ+77757z5495eXl69evr66uDgQCbW1tu3btWrFixamnnnr095h82T2t5Nxzz/X7/a+88kp048svv+z3+6dOndrTOnft2vWnP/0pOs7LXMxdAgAAAABgUKiurh4+fHgwGHz++ed/+9vfxhxdv379woULTSbTjz/+WFFR0fn06PlBSR7929/+Nnny5OLi4sbGRrWxoKCgtbV1165dEydOPMo7Sqa2BIeSr0QZ5JVXXrnyyivHjBnzxRdfqIfGjBnz5Zdfvvrqq1dccUW8C3UWDAZHjx4tSdI333xjsViSP7F/Yu4SAAAAAACDwjPPPBMMBufMmdM5WhIE4be//e2ll14aDAafeeaZdF3xtNNOEwShvb09utHj8QiCcMIJJ3R7epcbEqVxl6LkK1HMnj3b4XD8/e9//+qrr5SWL7/88ssvv3Q4HLNmzerRpVesWLF3796nn346KyurRyf2T6RLAAAAAAAMClu2bBEE4dprr43XQTn0wQcfpOuKf//73wVBiJkJNXr0aEEQFi1adODAgXRdKDU9rcRsNiu7eq9bt05pUV7Mnz/fbDYnf929e/euXLly/vz5Kayn659YGQcAAAAAwKDgcDicTmdLS4vD4eiyw5EjR4qKigoKCo4cOdL5aI9Wxh05cmTHjh133HHHoUOHli1bdv/996s9d+zYMWPGDL/fLwjCcccdd/rpp59yyilnn332Oeeco9fru71i8o3dHkqhkq+//vr000/Pz8+vq6sTBKG0tLStre3rr78+9dRTEz8fVSQSmTx58g8//PD9998XFRUlrjBTkC4BAAAAADAoGI3GUCgkSZLBYOiygyRJJpPJYDBIktT5aDLpUmdXXHHFiy++aDQaoxu///77FStWvPnmmy6XS22sqKh49NFHL7300iRvJ8naEpedZCXRg4wfP/7zzz9/5ZVXZFm+6qqrxo8f/+mnn3Z7IdUTTzxx6623rl27dtGiRclUmBFIlwAAAAAAGBT6YO5SjIqKirfeektZgNZZOBzevXv3nj17Pv300/fee2///v2iKFZVVV188cXJ3lIStSWT3XRbSfQgzzzzzI033jh16lRZlrdt2/bMM89cf/31SV7o8OHDo0aNGj169Mcff6w+MdIlAAAAAACQGSZMmPDZZ59t3rx5+vTpXXbYvHnzeeedN2HChE8++aTz0eRXxkmS9NNPPy1fvvyll14qKir69ttvS0pKEtcWiUTuvffeBx98MOYb2ZJ0lOlSt5VED9Le3l5WVubz+QRBsFgs9fX1drs9yQtdcMEFH3zwwVdffXXSSSelXGE/xK7eAAAAAAAMCtOmTRMEYc2aNfE6KIeUbkfDaDSOGDHihRdeOO+885qbm++9995uT9HpdEuWLBEEYc+ePUd59aPUbSW5ublz5syRZVmW5csuu0yJlpL03nvvBYPBk08+WYyiHErjd+H1PdIlAAAAAAAGhRtuuMFoNP75z3/+r//6r85HX3rppddff91kMt1www1puZwoio8//rjBYHjxxRf37dvXbX/li9tsNltarn40uq1E3TJJfTHIkS4BAAAAADAoDB06dOXKlYIgXHPNNdddd91nn33m9Xp9Pt/nn39+/fXXL1iwQBCEBx98sLKyMl1XHDFixMKFC0OhUPT0pVGjRt17773vv/9+dXV1IBDw+/0//fTTn/70J2WTo5kzZ6o9u5zOk8Y5PslXEuPss89W5i5NmTKlR1eUuxJ96GhuR0PsuwQAAAAAwCCyfPnyZcuWRSKRmHadTvfAAw/8/ve/j3di8vsuRaurqxs+fLjf7//000/HjRsnxP92OUEQTjrppG3btg0ZMiTBmMk3Jnko5UqSv1BiA2DfJdIlAAAAAAAGlz179jz55JNbt26tra2VZbmiouLcc8+95ZZbRo0aleCs1NIlQRCWLl26atWqadOmbdmyRRCEvXv3vvnmm9u3b9+zZ09TU5MsywUFBaNGjbrkkkt+97vfZWVlJR4zjenSUVaS/IUSI10CAAAAAADAoMa+SwAAAAAAAEgd6RIAAAAAAABSR7oEAAAAAACA1JEuAQAAAAAAIHWkSwAAAAAAAEgd6RIAAAAAAABSR7oEAAAAAACA1JEuAQAAAAAAIHUGrQsAAAAAAAB9QRRFWZa1ruL/iKLYubFfVYgkkS4BAAAAAIBe12W2pW2W1N/itszFyjgAAAAAAACkjnQJAAAAAIDBRRTFV155Zfz48dnZ2Tabbfr06bt37960adNZZ51ltVoLCwuvvPLKlpYWtfOzzz57wgknmEym448//oUXXoge6tlnnx0+fLjJZBo+fPiaNWtiLnHGGWfYbDa9Xq8sghN/0W2FN9xwwxVXXBHdMm/evBtuuEHDkpCIDAAAAAAABgE1BBAEYeTIkR988IHL5aqvr7/mmmscDsfJJ5+8ZcsWteXKK69UOw8bNuyjjz5yuVwffvjh0KFD33//feXQxo0bKyoqtm7d2tHRsXXr1oqKirfeeks9a8SIEVu3bvV4PDFX71xPZ8FgcPLkyU899ZTy9sknn5wyZUowGOztkpAaVhgCAAAAADAoqNsMiaL4+eefjx07Vmlvbm4uLi7+4osvxowZo7acdNJJR44cUTq/9dZbF154oXLozTfffPTRRz/66CNBEM4666zFixdfcsklyqGqqqqHH374r3/9q3LW9u3bp0yZ0vnq0S2di1T7NDU1nXXWWf/zP/8jCMK8efP+9re/FRcXqyf2UklIDc8RAAAAAIBBITpdCofDOp0u+lDnFrWz0+nMy8tT2p1O5/Dhw5V1cw6H48cff8zPz1cPHX/88a2trcpZPp8vKyur84AJWmJ88cUX8+bNMxgML7/8spp89WpJSA37LgEAAAAAMOhEB0nxWo5SdI6Tmk8++cRutwcCgcOHD/eTktAl0iUAAAAAAJDIjh071Ncff/zxKaecorw+6aSTtm/fHn3o5JNPjjeIwWAIh8PJX/Trr7/+wx/+8Nprr23YsOH666//8ccfNS8J8ZAuAQAAAACARG655ZaPP/7Y7XZ/9NFHt9122z333KO033XXXbfeeutHH32kHlqyZEm8QYYOHfrBBx9EIpFkrujxeObOnfvoo48ef/zx48eP/3//7//NmTPH7/drWBISYIUhAAAAAACDQvRWSt1ugRTd+emnn169evXPP/9cWVl53333LVy4UO32zDPPKIeGDRu2ZMmSa6+9Nt6AGzduvOuuuw4fPhwOh9WROxepHLrmmmv0ev3zzz+vtl955ZUWi2XdunW9WhJSQ7oEAAAAAADi6odbX/fDkgY5VsYBAAAAAAAgdaRLAAAAAAAASB1zyQAAAAAAAJA65i4BAAAAAAAgdaRLAAAAAAAASB3pEgAAAAAAAFJHugQAAAAAAIDUkS4BAAAAADDo/Pzzz9XV1cprt9t96NAh5XUgEGhpaVG7tbe3+3y+6BOjO0Sf2NLScvjw4ZirqJ0DgUBNTU1NTY3L5Ur7vUBzBq0LAAAAAAAAfeqtt9765ptvWltbjz322Dlz5tx+++2nnXaa2Wy+6KKL/uM//qO5ufnNN98UBMHlcp177rn33HPPnDlzlBP379+vdqivr1dPnDRp0nPPPXfCCSfk5ubefPPNnTvv2rXrscceGzt27LnnnnvmmWdqdufoHaIsy1rXAAAAAAAA+lpHR8eNN944adKkUaNG/fM///Ps2bM3btwoCMIll1yipEvLli1zOBwVFRVquqRQOjz55JPqiSNHjrzyyitPOeWUadOmffDBB507f/TRRxs2bLj44otnzJixZcuWN954w263L1u2zG639+Uto5cwdwkAAAAAgEFHluW777578eLFVVVVRUVFnTt89NFHJ554ojIl5eDBg62traNGjbJYLGqH5uZm9cQpU6Y8/fTTo0ePPnDgQJedR4wYcdlllx06dOj2228vKCg444wzZs2aRbQ0YJAuAQAAAAAw6Nx1112XXXbZ6aef/umnn9bV1Z1yyimiKEZ32LZtW2tr6759+/R6fW5ubmNj4/HHHx8dGJWXl6snzpw5c8SIEX6/f+zYsXV1dYcOHYrpXFZWVlZWJgjCxRdf/Mgjj3z44YeLFi3693//91GjRvXZLaP3kC4BAAAAADC4PPfcc9u2bQsEArt37/6Xf/mXG2644d13373gggtaW1uXLVv2zTffPPjgg3/84x8FQfjv//7vrKys6dOnKydGd7juuuvUEw8cOLB27dq6urrbbrtt8uTJkydPjun8q1/9aseOHXV1dRdffPFLL73U1NSk0+mys7O1fApIH/ZdAgAAAABgUJNl2e/3R081SuFEl8tltVp1urjfTd/e3p6dnW00GgVBcDqdOTk5BgNTXgaI/w9jSRXEjZRl2QAAAABJRU5ErkJggg==
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0idXRmLTgiID8+CjxvdHJzX2NvbmZpZyB2ZXJzaW9uPSIxLjAiIGluaXQ9IkZyYW1ld29yayI+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJGcm9udGVuZDo6TW9kdWxlIyMjQWRtaW5JbXBvcnRFeHBvcnQiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIFRyYW5zbGF0YWJsZT0iMSI+RnJvbnRlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGFnZW50IGludGVyZmFjZS48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5JbXBvcnRFeHBvcnQ8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Gcm9udGVuZDo6QWRtaW46Ok1vZHVsZVJlZ2lzdHJhdGlvbjwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxGcm9udGVuZE1vZHVsZVJlZz4KICAgICAgICAgICAgICAgIDxHcm91cD5hZG1pbjwvR3JvdXA+CiAgICAgICAgICAgICAgICA8RGVzY3JpcHRpb24gVHJhbnNsYXRhYmxlPSIxIj5JbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uPC9EZXNjcmlwdGlvbj4KICAgICAgICAgICAgICAgIDxUaXRsZSBUcmFuc2xhdGFibGU9IjEiPkltcG9ydC9FeHBvcnQ8L1RpdGxlPgogICAgICAgICAgICAgICAgPE5hdkJhck5hbWU+QWRtaW48L05hdkJhck5hbWU+CiAgICAgICAgICAgICAgICA8TmF2QmFyTW9kdWxlPgogICAgICAgICAgICAgICAgICAgIDxNb2R1bGU+S2VybmVsOjpPdXRwdXQ6OkhUTUw6Ok5hdkJhcjo6TW9kdWxlQWRtaW48L01vZHVsZT4KICAgICAgICAgICAgICAgICAgICA8TmFtZSBUcmFuc2xhdGFibGU9IjEiPkltcG9ydC9FeHBvcnQ8L05hbWU+CiAgICAgICAgICAgICAgICAgICAgPEJsb2NrPlN5c3RlbTwvQmxvY2s+CiAgICAgICAgICAgICAgICAgICAgPFByaW8+NzEwPC9QcmlvPgogICAgICAgICAgICAgICAgPC9OYXZCYXJNb2R1bGU+CiAgICAgICAgICAgICAgICA8TG9hZGVyPgogICAgICAgICAgICAgICAgICAgIDxDU1M+SVRTTS5JbXBvcnRFeHBvcnQuY3NzPC9DU1M+CiAgICAgICAgICAgICAgICA8L0xvYWRlcj4KICAgICAgICAgICAgPC9Gcm9udGVuZE1vZHVsZVJlZz4KICAgICAgICA8L1NldHRpbmc+CiAgICA8L0NvbmZpZ0l0ZW0+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJJbXBvcnRFeHBvcnQ6OkZvcm1hdEJhY2tlbmRSZWdpc3RyYXRpb24jIyNDU1YiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIFRyYW5zbGF0YWJsZT0iMSI+Rm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPEdyb3VwPkltcG9ydEV4cG9ydDwvR3JvdXA+CiAgICAgICAgPFN1Ykdyb3VwPkZvcm1hdEJhY2tlbmQ6Ok1vZHVsZVJlZ2lzdHJhdGlvbjwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJNb2R1bGUiPktlcm5lbDo6U3lzdGVtOjpJbXBvcnRFeHBvcnQ6OkZvcm1hdEJhY2tlbmQ6OkNTVjwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iTmFtZSI+Q1NWPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgo8L290cnNfY29uZmlnPgo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmNzX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdJbXBvcnQvRXhwb3J0IFNwcsOhdmEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdaYWjDoWppdCBJbXBvcnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAnWmFow6FqaXQgRXhwb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMSBvZiA1IC0gRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMiBvZiA1IC0gRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAzIG9mIDUgLSBFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA0IG9mIDUgLSBFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNSBvZiA1IC0gRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdPbWV6aXQgRXhwb3J0IHZ5aGxlZMOhdsOhbsOtbSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICdJbmZvcm1hY2UgbyBJbXBvcnR1JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnWmRyb2pvdsO9IFNvdWJvcic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgc3VtbWFyeSBmb3InfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ1rDoXpuYW3Fryc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAnw5pzcMSbxaFuxJsnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnTmXDunNwxJvFoW7Emyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICdEdXBsaWNpdG7DrSBqbcOpbmEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPayc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW52YWxpZCBkYXRhLCBwbGVhc2UgaW5zZXJ0IGEgdmFsaWQgJXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IG5vdCBmb3VuZCEnfSA9ICcnOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9TeXN0ZW0vSW1wb3J0RXhwb3J0L0Zvcm1hdEJhY2tlbmQvQ1NWLnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnT2RkxJtsb3ZhxI0gU2xvdXBjxa8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGFidWxhdG9yIChUQUIpJ30gPSAnVGFidWzDoXRvciAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTZW1pY29sb24gKDspJ30gPSAnU3TFmWVkbsOtayAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sb24gKDopJ30gPSAnRHZvanRlxI1rYSAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ1RlxI1rYSAoLiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29tbWEgKCwpJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdabmFrb3bDoSBzYWRhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnU2xvdXBlYyc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICdJbXBvcnQvRXhwb3J0JzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmRhX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdJbXBvcnQvRWtwb3J0IHN0eXJpbmcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdTdGFydCBpbXBvcnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAnU3RhcnQgZWtwb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMSBvZiA1IC0gRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMiBvZiA1IC0gRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAzIG9mIDUgLSBFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA0IG9mIDUgLSBFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNSBvZiA1IC0gRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdCZWdyw6ZucyBla3BvcnQgcHIuIHPDuGduaW5nJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0ltcG9ydCBpbmZvcm1hdGlvbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ0tpbGRlIGZpbCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgc3VtbWFyeSBmb3InfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPayc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW52YWxpZCBkYXRhLCBwbGVhc2UgaW5zZXJ0IGEgdmFsaWQgJXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IG5vdCBmb3VuZCEnfSA9ICcnOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9TeXN0ZW0vSW1wb3J0RXhwb3J0L0Zvcm1hdEJhY2tlbmQvQ1NWLnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYXRvciAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTZW1pY29sb24gKDspJ30gPSAnU2VtaWtvbG9uICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdLb2xvbiAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ1B1bmt0dW0gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbW1hICgsKSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAnVGVnbnPDpnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW5jbHVkZSBDb2x1bW4gSGVhZGVycyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdLb2xvbm5lJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ0ltcG9ydC9Fa3BvcnQnOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmRlX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdJbXBvcnQvRXhwb3J0LVZlcndhbHR1bmcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0Vyc3RlbGxlbiBlaW5lciBWb3JsYWdlIHp1bSBJbXBvcnRpZXJlbiB1bmQgRXhwb3J0aWVyZW4gdm9uIE9iamVrdC1JbmZvcm1hdGlvbmVuLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdJbXBvcnQgc3RhcnRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdFeHBvcnQgc3RhcnRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDEgb2YgNSAtIEVkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnU2Nocml0dCAxIHZvbiA1IC0gQWxsZ2VtZWluZSBJbmZvcm1hdGlvbmVuIGJlYXJiZWl0ZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICdOYW1lIHdpcmQgYmVuw7Z0aWd0ISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnT2JqZWt0IGlzdCBlcmZvcmRlcmxpY2ghJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICdGb3JtYXQgaXN0IGVyZm9yZGVybGljaCEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAyIG9mIDUgLSBFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJ1NjaHJpdHQgMiB2b24gNSAtIE9iamVrdC1JbmZvcm1hdGlvbmVuIGJlYXJiZWl0ZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAzIG9mIDUgLSBFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJ1NjaHJpdHQgMyB2b24gNSAtIEZvcm1hdGluZm9ybWF0aW9uZW4gYmVhcmJlaXRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpcyByZXF1aXJlZCEnfSA9ICd3aXJkIGJlbsO2dGlndCEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA0IG9mIDUgLSBFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICdTY2hyaXR0IDQgdm9uIDUgLSBNYXBwaW5nLUluZm9ybWF0aW9uZW4gYmVhcmJlaXRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnS2VpbmUgTWFwcGluZy1FbGVtZW50ZSBnZWZ1bmRlbi4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ01hcHBpbmctRWxlbWVudCBoaW56dWbDvGdlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDUgb2YgNSAtIEVkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnU2Nocml0dCA1IHZvbiA1IC0gU3VjaC1JbmZvcm1hdGlvbmVuIGJlYXJiZWl0ZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdFeHBvcnQgcGVyIFN1Y2hlIGVpbnNjaHLDpG5rZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW1wb3J0LUluZm9ybWF0aW9uZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdRdWVsbC1EYXRlaSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgc3VtbWFyeSBmb3InfSA9ICdJbXBvcnQtQmVyaWNodCBmw7xyJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlY29yZHMnfSA9ICdEYXRlbnPDpHR6ZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAnRXJmb2xncmVpY2gnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnTmljaHQgZXJmb2xncmVpY2gnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnRG9wcGVsdGUgTmFtZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICdadWxldHp0IHZlcmFyYmVpdGV0ZSBaZWlsZSBkZXIgSW1wb3J0LURhdGVpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnT2snOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9Nb2R1bGVzL0FkbWluSW1wb3J0RXhwb3J0LnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBvYmplY3QgYmFja2VuZCBmb3VuZCEnfSA9ICdLZWluIEJhY2tlbmQgZsO8ciBkYXMgT2JqZWt0IHZlcmbDvGdiYXIhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIGZvcm1hdCBiYWNrZW5kIGZvdW5kISd9ID0gJ0tlaW4gQmFja2VuZCBmw7xyIGRhcyBGb3JtYXQgdmVyZsO8Z2JhciEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJ1ZvcmxhZ2UgbmljaHQgdmVyZsO8Z2JhciEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2FuXCd0IGluc2VydC91cGRhdGUgdGVtcGxhdGUhJ30gPSAnRGVpIFZvcmxhZ2Uga29ubnRlIG5pY2h0IGVyc3RlbGx0IG9kZXIgYWt0dWFsaXNpZXJ0IHdlcmRlbiEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnRGllIElEIGRlciBWb3JsYWdlIHdpcmQgYmVuw7Z0aWd0ISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFcnJvciBvY2N1cnJlZC4gSW1wb3J0IGltcG9zc2libGUhIFNlZSBTeXNsb2cgZm9yIGRldGFpbHMuJ30gPSAnRmVobGVyIGF1ZmdldHJldGVuLiBJbXBvcnRpZXJlbiB1bm3DtmdsaWNoISBGw7xyIERldGFpbHMgc2llaGUgU3lzbG9nLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFcnJvciBvY2N1cnJlZC4gRXhwb3J0IGltcG9zc2libGUhIFNlZSBTeXNsb2cgZm9yIGRldGFpbHMuJ30gPSAnRmVobGVyIGF1ZmdldHJldGVuLiBFeHBvcnRpZXJlbiB1bm3DtmdsaWNoISBGw7xyIERldGFpbHMgc2llaGUgU3lzbG9nLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAnVm9ybGFnZW5saXN0ZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXInfSA9ICdaYWhsJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnWmFobCBncsO2w59lciBhbHMgTnVsbCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyJ30gPSAnR2FuenphaGwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaW50ZWdlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnR2FuenphaGwgZ3LDtsOfZXIgYWxzIE51bGwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWxlbWVudCByZXF1aXJlZCwgcGxlYXNlIGluc2VydCBkYXRhJ30gPSAnRWxlbWVudCBlcmZvcmRlcmxpY2gsIGJpdHRlIGbDvGdlbiBTaWUgRGF0ZW4gZWluJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ludmFsaWQgZGF0YSwgcGxlYXNlIGluc2VydCBhIHZhbGlkICVzJ30gPSAnVW5nw7xsdGlnZSBEYXRlbiwgYml0dGUgZ2ViZW4gU2llIGVpbiBnw7xsdGlnZXMgJXMgZWluJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBub3QgZm91bmQhJ30gPSAnRm9ybWF0IG5pY2h0IGdlZnVuZGVuISc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL1N5c3RlbS9JbXBvcnRFeHBvcnQvRm9ybWF0QmFja2VuZC9DU1YucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdTcGFsdGVudHJlbm5lcic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bGF0b3IgKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ1NlbWljb2xvbiAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sb24gKDopJ30gPSAnRG9wcGVscHVua3QgKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICdQdW5rdCAoLiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29tbWEgKCwpJ30gPSAnS29tbWEgKCwpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdaZWljaGVuc2F0eic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnTWl0IFNwYWx0ZW7DvGJlcnNjaHJpZnRlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdTcGFsdGUnOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICdGb3JtYXQtQmFja2VuZCBNb2R1bC1SZWdpc3RyYXRpb24gZGVzIEltcG9ydC9FeHBvcnQgTW9kdWxzLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnSW1wb3J0aWVyZW4gdW5kIEV4cG9ydGllcmVuIHZvbiBPYmpla3QtSW5mb3JtYXRpb25lbi4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ0ltcG9ydC9FeHBvcnQnOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmVzX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdHZXN0acOzbiBkZSBJbXBvcnRhY2nDs24vRXhwb3J0YWNpw7NuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdDcmVhciB1bmEgcGxhbnRpbGxhIHBhcmEgaW1wb3J0YXIgeSBleHBvcnRhciBpbmZvcm1hY2lvbiBkZWwgb2JqZXRvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdJbmljaWFyIEltcG9ydGFjacOzbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdJbmljaWFyIEV4cG9ydGFjacOzbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDEgb2YgNSAtIEVkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnUGFzbyAxIGRlIDUgLSBFZGl0YXIgaW5mb3JtYWNpw7NuIGNvbcO6bic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJ8KhRWwgbm9tYnJlIGVzIHJlcXVlcmlkbyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ8KhRGViZSBlc3BlY2lmaWNhciBPYmpldG8hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICfCoURlYmUgZXNwZWNpZmljYXIgRm9ybWF0byEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAyIG9mIDUgLSBFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJ1Bhc28gMiBkZSA1IC0gRWRpdGFyIGluZm9ybWFjacOzbiBkZSBvYmpldG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAzIG9mIDUgLSBFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJ1Bhc28gMyBkZSA1IC0gRWRpdGFyIGluZm9ybWFjacOzbiBkZSBmb3JtYXRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJ8KhZXMgcmVxdWVyaWRvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDQgb2YgNSAtIEVkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ1Bhc28gNCBkZSA1IC0gRWRpdGFyIGluZm9ybWFjacOzbiBkZSBtYXBlbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnTm8gc2UgZW5jb250cmFyb24gZWxlbWVudG9zIGRlIG1hcGVvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnQcOxYWRpciBNYXBlbyBkZSBFbGVtZW50b3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA1IG9mIDUgLSBFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJ1Bhc28gNSBkZSA1IC0gRWRpdGFyIGluZm9ybWFjacOzbiBkZSBiw7pzcXVlZGEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdSZXN0cmluZ2lyIGV4cG9ydGFjacOzbiBwb3IgYsO6c3F1ZWRhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0ltcG9ydGFyIGluZm9ybWFjacOzbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ0FyY2hpdm8gb3JpZ2VuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ0ltcG9ydGFyIHJlc3VtZW4gcGFyYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnUmVnaXN0cm9zJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICfDiXhpdG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnRnJhY2FzYWRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ05vbWJyZXMgZHVwbGljYWRvcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJ8OabHRpbWEgbsO6bWVybyBkZSBsw61uZWEgcHJvY2VzYWRhIGRlbCBhcmNoaXZvIGltcG9ydGFyJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnT2snOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9Nb2R1bGVzL0FkbWluSW1wb3J0RXhwb3J0LnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBvYmplY3QgYmFja2VuZCBmb3VuZCEnfSA9ICfCoU5vIHNlIGVuY29udHLDsyBuaW5nw7puIG9iamV0byBiYWNrZW5kISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICdObyBzZSBlbmNvbnRyw7MgbmluZ8O6biBmb3JtYXRvIGJhY2tlbmQhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RlbXBsYXRlIG5vdCBmb3VuZCEnfSA9ICfCoVBsYW50aWxsYSBubyBlbmNvbnRyYWRhISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICfCoU5vIHNlIHB1ZWRlIGluc2VydGFyL2FjdHVhbGl6YXIgbGEgcGxhbnRpbGxhISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOZWVkZWQgVGVtcGxhdGVJRCEnfSA9ICdTZSBuZWNlc2l0YSBJRFBsYW50aWxsYSEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEltcG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJ09jdXJyacOzIHVuIGVycm9yIS4gSW1wb3NpYmxlIGltcG9ydGFyISBWZWEgU3lzbG9nIHBhcmEgZGV0YWxsZXMuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBFeHBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICdPY3VycmnDsyB1biBlcnJvciEuIEltcG9zaWJsZSBleHBvcnRhciEgVmVhIFN5c2xvZyBwYXJhIGRldGFsbGVzLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlcid9ID0gJ07Dum1lcm8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnbnVtYmVyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICduw7ptZXJvIG1heW9yIHF1ZSBjZXJvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXInfSA9ICdlbnRlcm8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaW50ZWdlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnRW50ZXJvIG1heW8gcXVlIGNlcm8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWxlbWVudCByZXF1aXJlZCwgcGxlYXNlIGluc2VydCBkYXRhJ30gPSAnRWxlbWVudG8gcmVxdWVyaWRvLCBwb3IgZmF2b3IgaW5zZXJ0YXIgZGF0b3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW52YWxpZCBkYXRhLCBwbGVhc2UgaW5zZXJ0IGEgdmFsaWQgJXMnfSA9ICdEYXRvcyBubyB2w6FsaWRvLCBwb3IgZmF2b3IgaW5zZXJ0ZSB1biAlcyB2w6FsaWRvICc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgbm90IGZvdW5kISd9ID0gJ8KhRm9ybWF0byBubyBlbmNvbnRyYWRvISc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL1N5c3RlbS9JbXBvcnRFeHBvcnQvRm9ybWF0QmFja2VuZC9DU1YucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdTZXBhcmFkb3IgZGUgQ29sdW1uYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bGFkb3IgKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ1B1bnRvIHkgQ29tYSAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sb24gKDopJ30gPSAnRG9zIHB1bnRvcyAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ1B1bnRvICguKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb21tYSAoLCknfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ0p1ZWdvIGRlIGNhcmFjdGVyZXMnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW5jbHVkZSBDb2x1bW4gSGVhZGVycyd9ID0gJ0luY2x1aXIgQ2FiZWNlcmEgZGUgbGEgQ29sdW1uYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdDb2x1bW5hJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnUmVnaXN0cm8gZGUgbcOzZHVsbyBkZSBmb3JtYXRvIGJhY2tlbmQgcGFyYSBlbCBtw7NkdWxvIGltcG9ydC9leHBvcnQuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdJbXBvcnRhciB5IGV4cG9ydGFyIGluZm9ybWFjacOzbiBkZSBvYmpldG9zLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3J0YXIvRXhwb3J0YXInOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmVzX01YX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdHZXN0acOzbiBkZSBJbXBvcnRhY2nDs24vRXhwb3J0YWNpw7NuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdDcmVhciB1bmEgcGxhbnRpbGxhIHBhcmEgaW1wb3J0YXIgeSBleHBvcnRhciBpbmZvcm1hY2nDs24gZGUgb2JqZXRvcy4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAnSW5pY2lhciBJbXBvcnRhY2nDs24nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAnSW5pY2lhciBFeHBvcnRhY2nDs24nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAxIG9mIDUgLSBFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJ0VzIHJlcXVlcmlkbyB1biBub21icmUhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICdFcyByZXF1ZXJpZG8gdW4gb2JqZXRvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnRXMgcmVxdWVyaWRvIHVuIGZvcm1hdG8uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMiBvZiA1IC0gRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAzIG9mIDUgLSBFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpcyByZXF1aXJlZCEnfSA9ICdlcyByZXF1ZXJpZG8hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNCBvZiA1IC0gRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICdObyBzZSBoYW4gZW5jb250cmFkbyBtYXBhcy4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ0FncmVnYXIgdW4gRWxlbWVudG8gZGUgTWFwZW8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA1IG9mIDUgLSBFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ1Jlc3RyaW5naXIgbGEgZXhwb3J0YWNpw7NuIGEgbGEgYsO6c3F1ZWRhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0luZm9ybWFjacOzbiBkZSBpbXBvcnRhY2nDs24nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdBcmNoaXZvIEZ1ZW50ZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgc3VtbWFyeSBmb3InfSA9ICdJbXBvcnRhciByZXN1bWVuIHBhcmEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ1JlZ2lzdHJvcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAnw4l4aXRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJ0ZhbGzDsyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICdOb21icmVzIGR1cGxpY2Fkb3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICfDmmx0aW1vIG7Dum1lcm8gZGUgbMOtbmVhIHByb2Nlc2FkYSBkZWwgYXJjaGl2byBhIGltcG9ydGFyJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnQWNlcHRhcic7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW52YWxpZCBkYXRhLCBwbGVhc2UgaW5zZXJ0IGEgdmFsaWQgJXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IG5vdCBmb3VuZCEnfSA9ICcnOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9TeXN0ZW0vSW1wb3J0RXhwb3J0L0Zvcm1hdEJhY2tlbmQvQ1NWLnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnU2VwYXJhZG9yIGRlIENvbHVtbmEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGFidWxhdG9yIChUQUIpJ30gPSAnVGFidWxhZG9yIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdQdW50byB5IGNvbWEgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ0RvcyBQdW50b3MgKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICdQdW50byAoLiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29tbWEgKCwpJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdDb25qdW50byBkZSBDYXJhY3RlcmVzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICdJbmNsdWlyIENhYmVjZXJhIGRlIENvbHVtbmFzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ0NhbHVtbmEnOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICdGb3JtYXRvIGRlbCBtw7NkdWxvIGFkbWluaXN0cmF0aXZvIGRlIHJlZ2lzdHJvIHBhcmEgZWwgbcOzZHVsbyBkZSBpbXBvcnRhY2nDs24vZXhwb3J0YWNpw7NuLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnSW1wb3J0YXIgeSBleHBvcnRhciBpbmZvcm1hY2nDs24gZGUgdW4gb2JqZXRvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3J0YXIvRXhwb3J0YXInOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmZhX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICfZhdiv24zYsduM2Kog2YjYsdmI2K8v2LXYr9mI2LEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ9iz2KfYrtiqINmC2KfZhNio24wg2KjYsdin24wg2YjYsdmI2K8g2Ygg2LXYr9mI2LEg2KfYt9mE2KfYudin2Kog2KLYqNis2qnYqic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICfYtNix2YjYuSDYudmF2YTbjNin2Kog2YjYsdmI2K8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAn2LTYsdmI2Lkg2LnZhdmE24zYp9iqINi12K/ZiNixJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMSBvZiA1IC0gRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ9ii2KjYrNqp2Kog2YXZiNix2K8g2YbbjNin2LIg2KfYs9iqISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAn2YLYp9mE2KjigIzYqNmG2K/bjCDZhdmI2LHYryDZhtuM2KfYsiDYp9iz2KohJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMiBvZiA1IC0gRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAzIG9mIDUgLSBFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA0IG9mIDUgLSBFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJ9mH24zahiDYudmG2LXYsSDZhtqv2KfYtNiq24wg24zYp9mB2Kog2YbYtNivLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAn2KfZgdiy2YjYr9mGINi52YbYtdixINmG2q/Yp9i02KonOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA1IG9mIDUgLSBFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ9mF2K3Yr9mI2K/Ys9in2LLbjCDYudmF2YTbjNin2Kog2LXYr9mI2LEg2KjZhyDYp9iy2KfbjCDYrNiz2KrYrNmIJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ9mI2LHZiNivINin2LfZhNin2LnYp9iqJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAn2YHYp9uM2YQg2YXZhtio2LknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlY29yZHMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAn2KrYp9uM24zYryc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW52YWxpZCBkYXRhLCBwbGVhc2UgaW5zZXJ0IGEgdmFsaWQgJXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IG5vdCBmb3VuZCEnfSA9ICcnOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9TeXN0ZW0vSW1wb3J0RXhwb3J0L0Zvcm1hdEJhY2tlbmQvQ1NWLnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAn2KzYr9in2qnZhtmG2K/ZhyDYs9iq2YjZhuKAjNmH2KcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGFidWxhdG9yIChUQUIpJ30gPSAn2KzYr9mI2YQg2LPYp9iyIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICfYs9mF24wg2qnYp9mE2YYgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ9iv2YjZhtmC2LfZhyAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ9mG2YLYt9mHICguKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb21tYSAoLCknfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ9qp2K/YqNmG2K/bjCDYp9i32YTYp9i52KfYqic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ9iz2KrZiNmGJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAn2YLYp9mE2KjigIzYqNmG2K/bjCDYq9io2Kog2YXYp9qY2YjZhCDYqNix2KfbjCDZhdin2pjZiNmEINmI2LHZiNivL9i12K/ZiNixJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICfZiNix2YjYryDZiCDYtdiv2YjYsSDYp9i32YTYp9i52KfYqiDYotio2KzaqdiqJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICfZiNix2YjYry/Ytdiv2YjYsSc7Cgp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmZyX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdHZXN0aW9uIGRlIGxcJ0ltcG9ydC9FeHBvcnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0Nyw6llciB1biBtb2TDqGxlIHBvdXIgaW1wb3J0ZXIgZXQgZXhwb3J0ZXIgbGVzIGluZm9ybWF0aW9ucyBkXCdvYmpldCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdEw6ltYXJyZXIgSW1wb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ0TDqW1hcnJlciBFeHBvcnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAxIG9mIDUgLSBFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJ0V0YXBlIDEgc3VyIDUgLSBFZGl0ZXIgbFwnaW5mb3JtYXRpb24gY29tbXVuZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJ1VuIE5vbSBlc3QgcmVxdWlzISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnVW4gT2JqZXQgZXN0IHJlcXVpcyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJ1VuIEZvcm1hdCBlc3QgcmVxdWlzISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDIgb2YgNSAtIEVkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnRXRhcGUgMiBzdXIgNSAtIEVkaXRlciBsXCdpbmZvcm1hdGlvbiBkXCdvYmpldCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDMgb2YgNSAtIEVkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJ2VzdCByZXF1aXMgISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDQgb2YgNSAtIEVkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ0V0YXBlIDQgc3VyIDUgLSBFZGl0ZXIgbFwnaW5mb3JtYXRpb24gZGUgbWFwcGFnZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnQXVjdW4gw6lsw6ltZW50IGRlIG1hcHBhZ2UgdHJvdXbDqSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnQWpvdXRlciB1biDDqWzDqW1lbnQgZGUgbWFwcGFnZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDUgb2YgNSAtIEVkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnRXRhcGUgNSBzdXIgNSAtIEVkaXRlciBsXCdpbmZvcm1hdGlvbiBkZSByZWNoZXJjaGUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdSZXN0cmVpbmRyZSBsXCdleHBvcnQgcGFyIHJlY2hlcmNoZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICdJbmZvcm1hdGlvbnMgZFwnaW1wb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnRmljaGllciBTb3VyY2UnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnUsOpc3Vtw6kgZGUgbFwnaW1wb3J0IHBvdXInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ0VucmVnaXN0cmVtZW50cyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAnUsOpdXNzaSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICdFY2hvdcOpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ05vbXMgZW4gZG91YmxlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnTnVtw6lybyBkZSBsYSBkZXJuacOocmUgbGlnbmUgdHJhaXTDqWUgZGFucyBsZSBmaWNoaWVyIGRcJ2ltcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJ09rJzsKCiAgICAjIFBlcmwgTW9kdWxlOiBLZXJuZWwvTW9kdWxlcy9BZG1pbkltcG9ydEV4cG9ydC5wbQogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gb2JqZWN0IGJhY2tlbmQgZm91bmQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIGZvcm1hdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBub3QgZm91bmQhJ30gPSAnVGVtcGxhdGUgbm9uIHRyb3V2w6khJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NhblwndCBpbnNlcnQvdXBkYXRlIHRlbXBsYXRlISd9ID0gJ0ltcG9zc2libGUgZFwnaW5zw6lyZXIvbWV0dHJlIMOgIGpvdXIgbGUgdGVtcGxhdGUhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05lZWRlZCBUZW1wbGF0ZUlEISd9ID0gJ0lkZW50aWZpYW50IGRlIHRlbXBsYXRlIHJlcXVpcyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEltcG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJ1VuZSBlcnJldXIgZXN0IHN1cnZlbnVlLiBJbXBvcnQgaW1wb3NzaWJsZSEgVm95ZXogbGUgbG9nIHN5c3TDqG1lIHBvdXIgbGVzIGTDqXRhaWxzLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFcnJvciBvY2N1cnJlZC4gRXhwb3J0IGltcG9zc2libGUhIFNlZSBTeXNsb2cgZm9yIGRldGFpbHMuJ30gPSAnVW5lIGVycmV1ciBlc3Qgc3VydmVudWUuIEV4cG9ydCBpbXBvc3NpYmxlISBWb3lleiBsZSBsb2cgc3lzdMOobWUgcG91ciBsZXMgZMOpdGFpbHMuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RlbXBsYXRlIExpc3QnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnbnVtYmVyJ30gPSAnbm9tYnJlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnbm9tYnJlIHBsdXMgZ3JhbmQgcXVlIHrDqXJvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXInfSA9ICdlbnRpZXInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaW50ZWdlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnZW50aWVyIHBsdXMgZ3JhbmQgcXVlIHrDqXJvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VsZW1lbnQgcmVxdWlyZWQsIHBsZWFzZSBpbnNlcnQgZGF0YSd9ID0gJ0Vsw6ltZW50IHJlcXVpcywgdmV1aWxsZXogZW50cmVyIHVuZSBkb25uw6llJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ludmFsaWQgZGF0YSwgcGxlYXNlIGluc2VydCBhIHZhbGlkICVzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBub3QgZm91bmQhJ30gPSAnRm9ybWF0IG5vbiB0cm91dsOpISc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL1N5c3RlbS9JbXBvcnRFeHBvcnQvRm9ybWF0QmFja2VuZC9DU1YucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdTw6lwYXJhdGV1ciBkZSBjb2xvbm5lJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYXRpb24gKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ1BvaW50IHZpcmd1bGUgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ0RldXggcG9pbnRzICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUG9pbnQgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbW1hICgsKSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAnSmV1IGRlIGNhcmFjdMOocmVzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICdJbmNsdXJlIGxlcyBlbi10w6p0ZXMgZGUgY29sb25uZXMnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnQ29sb25uZSc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ0VucmVnaXN0cmVtZW50IGVuIGFycmnDqHJlIHBsYW4gZHUgbW9kdWxlIGRcJ2ltcG9ydC9leHBvcnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0ltcG9ydGVyIGV0IGV4cG9ydGVyIGRlcyBpbmZvcm1hdGlvbnMgZFwnb2JqZXQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ0ltcG9ydGVyL0V4cG9ydGVyJzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmdsX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdYZXN0acOzbiBkYSBpbXBvcnRhY2nDs24vZXhwb3J0YWNpw7NuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdDcmVhciB1biBtb2RlbG8gcGFyYSBpbXBvcnRhciBlIGV4cG9ydGFyIGEgaW5mb3JtYWNpw7NuIHNvYnJlIG9zIG9ieGVjdG9zLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdJbmljaWFyIGEgaW1wb3J0YWNpw7NuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ0luaWNpYXIgYSBleHBvcnRhY2nDs24nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAxIG9mIDUgLSBFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJ1JlcXXDrXJlc2UgdW4gbm9tZSEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ1JlcXXDrXJlc2UgdW4gb2J4ZWN0byEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJ1JlcXXDrXJlc2UgdW4gZm9ybWF0byEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAyIG9mIDUgLSBFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDMgb2YgNSAtIEVkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJ0UgbmVjZXNhcmlvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDQgb2YgNSAtIEVkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnRWxlbWVudG9zIGRvIG1hcGEgbm9uIGF0b3BhZG9zJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICdFbmdhZGlyIGVsZW1lbnRvIGRlIG1hcGVvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNSBvZiA1IC0gRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdSZXN0cmlueGlyIGEgZXhwb3J0YWNpw7NuIGRhIGJ1c2NhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0luZm9ybWFjacOzbiBkZSBpbXBvcnRhcic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ0ZpY2hlaXJvIG9yaXhlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ0ltcG9ydGFyIG8gcmVzdW1vIGRlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlY29yZHMnfSA9ICdSZXhpc3Ryb3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJ0NvcnJlY3RvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJ0ZhbGxvdSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICdOb21lcyBkdXBsaWNhZG9zJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnTsO6bWVybyBkYSBkZXJyYWRlaXJhIGxpw7FhIHByb2Nlc2FkYSBkbyBhcnF1aXZvIGltcG9ydGFyJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnQWNlcHRhcic7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW52YWxpZCBkYXRhLCBwbGVhc2UgaW5zZXJ0IGEgdmFsaWQgJXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IG5vdCBmb3VuZCEnfSA9ICcnOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9TeXN0ZW0vSW1wb3J0RXhwb3J0L0Zvcm1hdEJhY2tlbmQvQ1NWLnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnU2VwYXJhZG9yIGRlIGNvbHVtbmFzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYWNpw7NuIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdQdW50byBlIGNvbWEgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ0RvdXMgcHVudG9zIDonOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ1B1bnRvICguKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb21tYSAoLCknfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ0Nvbnh1bnRvIGRlIGNhcmFjdGVyZXMnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW5jbHVkZSBDb2x1bW4gSGVhZGVycyd9ID0gJ0luY2x1w61yIGFzIGNhYmVjZWlyYXMgZGFzIGNvbHVtbmFzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ0NvbHVtbmEnOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICdNw7NkdWxvIGRlIHJleGlzdHJvIGRvIGZvcm1hdG8gZGUgYmFja2VuZCBwYXJhIG8gbcOzZHVsbyBpbXBvcnRhci9leHBvcnRhcic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnSW5mb3JtYWNpw7NuIHBhcmEgaW1wb3J0YXIgZSBleHBvcnRhciBvIG9ieGV0byc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3J0YWNpw7NuL0V4cG9ydGFjacOzbic7Cgp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6Omh1X0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdJbXBvcnTDoWzDoXMvZXhwb3J0w6Fsw6FzIGtlemVsw6lzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdTYWJsb24gbMOpdHJlaG96w6FzYSBvYmpla3R1bWluZm9ybcOhY2nDs2sgaW1wb3J0w6Fsw6Fzw6Fob3ogw6lzIGV4cG9ydMOhbMOhc8OhaG96Lic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdJbXBvcnTDoWzDoXMgaW5kw610w6FzYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdFeHBvcnTDoWzDoXMgaW5kw610w6FzYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDEgb2YgNSAtIEVkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnMS4gbMOpcMOpcyBheiA1LWLFkWwgLSBrw7Z6w7ZzIGluZm9ybcOhY2nDs2sgc3plcmtlc3p0w6lzZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJ0EgbsOpdiBrw7Z0ZWxlesWRISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnQXogb2JqZWt0dW0ga8O2dGVsZXrFkSEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJ0EgZm9ybcOhdHVtIGvDtnRlbGV6xZEhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMiBvZiA1IC0gRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICcyLiBsw6lww6lzIGF6IDUtYsWRbCAtIG9iamVrdHVtaW5mb3Jtw6FjacOzayBzemVya2VzenTDqXNlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMyBvZiA1IC0gRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICczLiBsw6lww6lzIGF6IDUtYsWRbCAtIGZvcm3DoXR1bWluZm9ybcOhY2nDs2sgc3plcmtlc3p0w6lzZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpcyByZXF1aXJlZCEnfSA9ICdrw7Z0ZWxlesWRISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDQgb2YgNSAtIEVkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJzQuIGzDqXDDqXMgYXogNS1ixZFsIC0gbGVrw6lwZXrDqXNpIGluZm9ybcOhY2nDs2sgc3plcmtlc3p0w6lzZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnTmVtIHRhbMOhbGhhdMOzayB0w6lya8OpcGVsZW1lay4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ0xla8OpcGV6w6lzaSBlbGVtZWsgaG96esOhYWTDoXNhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNSBvZiA1IC0gRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICc1LiBsw6lww6lzIGF6IDUtYsWRbCAtIGtlcmVzw6lzaSBpbmZvcm3DoWNpw7NrIHN6ZXJrZXN6dMOpc2UnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdFeHBvcnTDoWzDoXMga29ybMOhdG96w6FzYSBrZXJlc8Opc2Vua8OpbnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW1wb3J0w6Fsw6FzaSBpbmZvcm3DoWNpw7NrJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnRm9ycsOhc2bDoWpsJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ0ltcG9ydMOhbMOhc2kgw7Zzc3plZ3rDqXMgZWhoZXonOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ1Jla29yZG9rJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICdTaWtlcmVzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJ1Npa2VydGVsZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnTmV2ZWsga2V0dMWResOpc2UnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICdBeiBpbXBvcnRmw6FqbCB1dG9sc8OzIGZlbGRvbGdvem90dCBzb3JzesOhbWEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPSyc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJ05lbSB0YWzDoWxoYXTDsyBvYmpla3R1bS1ow6F0dMOpcnByb2dyYW0hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIGZvcm1hdCBiYWNrZW5kIGZvdW5kISd9ID0gJ05lbSB0YWzDoWxoYXTDsyBmb3Jtw6F0dW0taMOhdHTDqXJwcm9ncmFtISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBub3QgZm91bmQhJ30gPSAnU2FibG9uIG5lbSB0YWzDoWxoYXTDsyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2FuXCd0IGluc2VydC91cGRhdGUgdGVtcGxhdGUhJ30gPSAnTmVtIGxlaGV0IGJlc3rDunJuaSB2YWd5IGZyaXNzw610ZW5pIGEgc2FibG9udCEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnU2FibG9uLWF6b25vc8OtdMOzIHN6w7xrc8OpZ2VzISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFcnJvciBvY2N1cnJlZC4gSW1wb3J0IGltcG9zc2libGUhIFNlZSBTeXNsb2cgZm9yIGRldGFpbHMuJ30gPSAnSGliYSB0w7ZydMOpbnQuIEF6IGltcG9ydMOhbMOhcyBsZWhldGV0bGVuISBOw6l6emUgbWVnIGEgcmVuZHN6ZXJuYXBsw7N0IGEgcsOpc3psZXRla8OpcnQuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBFeHBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICdIaWJhIHTDtnJ0w6ludC4gQXogZXhwb3J0w6Fsw6FzIGxlaGV0ZXRsZW4hIE7DqXp6ZSBtZWcgYSByZW5kc3plcm5hcGzDs3QgYSByw6lzemxldGVrw6lydC4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgTGlzdCd9ID0gJ1NhYmxvbmxpc3RhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlcid9ID0gJ3N6w6FtJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnbnVsbMOhbsOhbCBuYWd5b2JiIHN6w6FtJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXInfSA9ICdlZ8Opc3onOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaW50ZWdlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnbnVsbMOhbsOhbCBuYWd5b2JiIGVnw6lzeic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICdFbGVtIHN6w7xrc8OpZ2VzLCBzesO6cmpvbiBiZSBhZGF0b2thdCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbnZhbGlkIGRhdGEsIHBsZWFzZSBpbnNlcnQgYSB2YWxpZCAlcyd9ID0gJ8OJcnbDqW55dGVsZW4gYWRhdG9rLCBlZ3kgw6lydsOpbnllcyAlcyBiZXN6w7pyw6FzYSBzesO8a3PDqWdlcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgbm90IGZvdW5kISd9ID0gJ0Zvcm3DoXR1bSBuZW0gdGFsw6FsaGF0w7MhJzsKCiAgICAjIFBlcmwgTW9kdWxlOiBLZXJuZWwvU3lzdGVtL0ltcG9ydEV4cG9ydC9Gb3JtYXRCYWNrZW5kL0NTVi5wbQogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uIFNlcGFyYXRvcid9ID0gJ09zemxvcGVsdsOhbGFzenTDsyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bMOhdG9yIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdQb250b3N2ZXNzesWRICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdLZXR0xZFzcG9udCAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ1BvbnQgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbW1hICgsKSd9ID0gJ1Zlc3N6xZEgKCwpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdLYXJha3RlcmvDqXN6bGV0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICdPc3psb3BmZWpsw6ljZWsgZmVsdsOpdGVsZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdPc3psb3AnOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICdGb3Jtw6F0dW0gaMOhdHTDqXJwcm9ncmFtIG1vZHVsIHJlZ2lzenRyw6FjacOzIGF6IGltcG9ydMOhbMOhcy9leHBvcnTDoWzDoXMgbW9kdWxob3ouJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdPYmpla3R1bWluZm9ybcOhY2nDs2sgaW1wb3J0w6Fsw6FzYSDDqXMgZXhwb3J0w6Fsw6FzYS4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ0ltcG9ydMOhbMOhcy9leHBvcnTDoWzDoXMnOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmlkX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdNYW5hamFtZW4gSW1wb3IvRWtzcG9yJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdCdWF0IHNlYnVhaCB0ZW1wbGF0ZSB1bnR1ayBtZW5naW1wb3IgZGFuIGVrc3BvciBvYnllayBpbmZvcm1hc2kuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ011bGFpIEltcG9yJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ011bGFpIEVrc3Bvcic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDEgb2YgNSAtIEVkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnTmFtYSBkaWJ1dHVoa2FuISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnT2J5ZWsgZGlidXR1aGthbiEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJ0Zvcm1hdCBkaWJ1dHVoa2FuISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDIgb2YgNSAtIEVkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMyBvZiA1IC0gRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAnZGlidXR1aGthbiEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA0IG9mIDUgLSBFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJ1RpZGFrIGRpdGVtdWthbiBlbGVtZW4gcGV0YS4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ1RhbWJhaCBFbGVtZW4gUGVtZXRhYW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA1IG9mIDUgLSBFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ0JhdGFzIGVrc3BvciBwZXIgcGVuY2FyaWFuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0luZm9ybWFzaSBpbXBvciAnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdCZXJrYXMgU3VtYmVyJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ0ltcG9yIHJpbmdrYXNhbiB1bnR1ayc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnRGF0YSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAnQmVyaGFzaWwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnR2FnYWwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnTmFtYSBkdXBsaWthdCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJ05vbW9yIGJhcmlzIHRlcmFraGlyIGJlcmthcyBpbXBvciB5YW5nIGRpcHJvc2VzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnT2snOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9Nb2R1bGVzL0FkbWluSW1wb3J0RXhwb3J0LnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBvYmplY3QgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gZm9ybWF0IGJhY2tlbmQgZm91bmQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RlbXBsYXRlIG5vdCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2FuXCd0IGluc2VydC91cGRhdGUgdGVtcGxhdGUhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05lZWRlZCBUZW1wbGF0ZUlEISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFcnJvciBvY2N1cnJlZC4gSW1wb3J0IGltcG9zc2libGUhIFNlZSBTeXNsb2cgZm9yIGRldGFpbHMuJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBFeHBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgTGlzdCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXInfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnbnVtYmVyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaW50ZWdlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWxlbWVudCByZXF1aXJlZCwgcGxlYXNlIGluc2VydCBkYXRhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ludmFsaWQgZGF0YSwgcGxlYXNlIGluc2VydCBhIHZhbGlkICVzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBub3QgZm91bmQhJ30gPSAnJzsKCiAgICAjIFBlcmwgTW9kdWxlOiBLZXJuZWwvU3lzdGVtL0ltcG9ydEV4cG9ydC9Gb3JtYXRCYWNrZW5kL0NTVi5wbQogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uIFNlcGFyYXRvcid9ID0gJ1BlbWlzYWggS29sb20nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGFidWxhdG9yIChUQUIpJ30gPSAnVGFiIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdUaXRpa2tvbWEgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ1RpdGlrZHVhICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnVGl0aWsgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbW1hICgsKSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAnS2FyYWt0ZXInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW5jbHVkZSBDb2x1bW4gSGVhZGVycyd9ID0gJ1NlcnRha2FuIEtlcGFsYSBLb2xvbSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdLb2xvbSc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ0Zvcm1hdCBtb2R1bCBwZW5kYWZ0YXJhbiBiYWNrZW5kIHVudHVrIG1vZHVsIGltcG9yL2Vrc3Bvci4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0luZm9ybWFzaSBvYnllayBpbXBvciBkYW4gZWtzcG9yLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3IvRWtzcG9yJzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6Oml0X0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdHZXN0aW9uZSBJbXBvcnRhemlvbmUvRXNwb3J0YXppb25lJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdDcmVhIHVuIHRlbXBsYXRlIHBlciBpbXBvcnRhcmUgZWQgZXNwb3J0YXJlIGxlIGluZm9ybWF6aW9uaSBkZWdsaSBvZ2dldHRpLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdJbml6aWEgSW1wb3J0YXppb25lJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ0luaXppYSBFc3BvcnRhemlvbmUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAxIG9mIDUgLSBFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJ0lsIG5vbWUgw6ggb2JibGlnYXRvcmlvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnTFwnb2dnZXR0byDDqCBvYmJsaWdhdG9yaW8hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICdJbCBmb3JtYXRvIMOoIG9iYmxpZ2F0b3JpbyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAyIG9mIDUgLSBFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDMgb2YgNSAtIEVkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJ8OoIG9iYmxpZ2F0b3JpbyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA0IG9mIDUgLSBFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJ05lc3N1biBlbGVtZW50byBtYXBwYSB0cm92YXRvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnQWdnaXVuZ2kgdW4gZWxlbWVudG8gZGkgbWFwcGF0dXJhLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDUgb2YgNSAtIEVkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnUmVzdHJpbmdlcmUgZXNwb3J0YXppb25lIHBlciByaWNlcmNhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0ltcG9ydGFyZSBpbmZvcm1hemlvbmUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdBcmNoaXZpbyBvcmlnaW5lJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ0ltcG9ydGEgaWwgc29tbWFyaW8gcGVyJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlY29yZHMnfSA9ICdWb2NpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICdTdWNjZXNzbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICdGYWxsaXRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ0R1cGxpY2EgaSBub21pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnTnVtZXJvIGRlbGxcJ3VsdGltYSByaWdhIHByb2Nlc3NhdGEgZGVsIGZpbGUgZGEgaW1wb3J0YXJlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnT2snOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9Nb2R1bGVzL0FkbWluSW1wb3J0RXhwb3J0LnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBvYmplY3QgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gZm9ybWF0IGJhY2tlbmQgZm91bmQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RlbXBsYXRlIG5vdCBmb3VuZCEnfSA9ICdNb2RlbGxvIG5vbiB0cm92YXRvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnTmVjZXNzYXJpbyBUZW1wbGF0ZUlEISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFcnJvciBvY2N1cnJlZC4gSW1wb3J0IGltcG9zc2libGUhIFNlZSBTeXNsb2cgZm9yIGRldGFpbHMuJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBFeHBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgTGlzdCd9ID0gJ0xpc3RhIE1vZGVsbGknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnbnVtYmVyJ30gPSAnbnVtZXJvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnbnVtZXJvIHBpw7kgZ3JhbmRlIGRpIHplcm8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaW50ZWdlcid9ID0gJ2ludGVybyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICdpbnRlcm8gcGnDuSBncmFuZGUgZGkgemVybyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICdFbGVtZW50byByaWNoaWVzdG8sIGluc2VyaXNjaSBkYXRpIGdlbnRpbG1lbnRlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ludmFsaWQgZGF0YSwgcGxlYXNlIGluc2VydCBhIHZhbGlkICVzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBub3QgZm91bmQhJ30gPSAnRm9ybWF0byBub24gdHJvdmF0byEnOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9TeXN0ZW0vSW1wb3J0RXhwb3J0L0Zvcm1hdEJhY2tlbmQvQ1NWLnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnU2VwYXJhdG9yZSBkaSBjb2xvbm5hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYXRvcmUgKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ1B1bnRvIGUgdmlyZ29sYSAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sb24gKDopJ30gPSAnRHVlIHB1bnRpICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUHVudG8gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbW1hICgsKSd9ID0gJ1ZpcmdvbGEgKCwpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdDaGFyc2V0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICdJbmNsdWRpIGxlIENvbG9ubmUgZGkgSW50ZXN0YXppb25lJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ0NvbG9ubmEnOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICdSZWdpc3RyYXppb25lIGRlbCBtb2R1bG8gZGkgYmFja2VuZCBkZWwgZm9ybWF0byBwZXIgaWwgbW9kdWxvIGRpIGltcG9ydGF6aW9uZS9lc3BvcnRhemlvbmUuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdJbXBvcnRhIGVkIGVzcG9ydGEgbGUgaW5mb3JtYXppb25pIHN1bGxcJ29nZ2V0dG8uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICdJbXBvcnRhcmUvRXNwb3J0YXJlJzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmphX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdJbXBvcnQvRXhwb3J0IOODnuODjeODvOOCuOODoyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAn44Kq44OW44K444Kn44Kv44OI44Gu44Kk44Oz44Od44O844OI44O744Ko44Kv44K544Od44O844OI55So44Gu44OG44Oz44OX44Os44O844OI44KS5L2c5oiQ44GZ44KLJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ+OCpOODs+ODneODvOODiOmWi+Wniyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICfjgqjjgq/jgrnjg53jg7zjg4jplovlp4snOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAxIG9mIDUgLSBFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJ1N0ZXAgMSBvZiA1IC0g5Z+65pys5oOF5aCx44Gu6Kit5a6aJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAn5ZCN56ew44Gv5b+F6aCI44Gn44GZISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAn44Kq44OW44K444Kn44Kv44OI44Gv5b+F6aCI44Gn44GZISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAn44OV44Kp44O844Oe44OD44OI44Gv5b+F6aCI44Gn44GZISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDIgb2YgNSAtIEVkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnU3RlcCAyIG9mIDUgLSDjgqrjg5bjgrjjgqfjgq/jg4jmg4XloLHjga7oqK3lrponOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAzIG9mIDUgLSBFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJ1N0ZXAgMyBvZiA1IC0g44OV44Kp44O844Oe44OD44OI5oOF5aCx44Gu6Kit5a6aJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJ+W/hemgiOOBp+OBmSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDQgb2YgNSAtIEVkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ1N0ZXAgNCBvZiA1IC0g44Oe44OD44OU44Oz44Kw5oOF5aCx44Gu6Kit5a6aJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICfjg57jg4Pjg5Tjg7PjgrDopoHntKDjgYzopovjgaTjgYvjgorjgb7jgZvjgpPjgafjgZfjgZ8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ+ODnuODg+ODlOODs+OCsOimgee0oOOBrui/veWKoCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDUgb2YgNSAtIEVkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnU3RlcCA1IG9mIDUgLSDmpJzntKLmg4XloLHjga7oqK3lrponOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICfmpJzntKLjgYLjgZ/jgorjga7lh7rlipvjgpLliLbpmZAnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAn5oOF5aCx44Gu6Kqt44G/6L6844G/JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAn44K944O844K544OV44Kh44Kk44OrJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ+amguimgeOCkuiqreOBv+i+vOOCgCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAn44Os44Kz44O844OJJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICfmiJDlip8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAn5aSx5pWXJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ+WQjeWJjeOBjOmHjeikh+OBl+OBpuOBhOOBvuOBmSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJ+iqreOBv+i+vOOBv+a4iOOBv+ODleOCoeOCpOODq+OBruacgOe1guWun+ihjOihjOaVsCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJ09rJzsKCiAgICAjIFBlcmwgTW9kdWxlOiBLZXJuZWwvTW9kdWxlcy9BZG1pbkltcG9ydEV4cG9ydC5wbQogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gb2JqZWN0IGJhY2tlbmQgZm91bmQhJ30gPSAn5oyH5a6a44GV44KM44Gf44Kq44OW44K444Kn44Kv44OI44Gu44OQ44OD44Kv44Ko44Oz44OJ44GM6KaL44Gk44GL44KK44G+44Gb44KT77yBJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIGZvcm1hdCBiYWNrZW5kIGZvdW5kISd9ID0gJ+aMh+WumuOBleOCjOOBn+ODleOCqeODvOODnuODg+ODiOOBruODkOODg+OCr+OCqOODs+ODieOBjOimi+OBpOOBi+OCiuOBvuOBm+OCk++8gSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBub3QgZm91bmQhJ30gPSAn44OG44Oz44OX44Os44O844OI44GM6KaL44Gk44GL44KK44G+44Gb44KT77yBJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NhblwndCBpbnNlcnQvdXBkYXRlIHRlbXBsYXRlISd9ID0gJ+ODhuODs+ODl+ODrOODvOODiOOBruaMv+WFpeODu+abtOaWsOOBjOOBp+OBjeOBvuOBm+OCk++8gSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOZWVkZWQgVGVtcGxhdGVJRCEnfSA9ICfjg4bjg7Pjg5fjg6zjg7zjg4hJROOBruWFpeWKm+OBjOW/heimgeOBp+OBme+8gSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFcnJvciBvY2N1cnJlZC4gSW1wb3J0IGltcG9zc2libGUhIFNlZSBTeXNsb2cgZm9yIGRldGFpbHMuJ30gPSAn44Kk44Oz44Od44O844OI44GM5Ye65p2l44G+44Gb44KT77yI44Ko44Op44O844GM55m655Sf44GX44G+44GX44Gf44CC6Kmz57Sw44Gv44K344K544OG44Og44Ot44Kw44KS44GU56K66KqN44GP44Gg44GV44GE77yJ77yBJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBFeHBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICfjgqjjgq/jgrnjg53jg7zjg4jjgYzlh7rmnaXjgb7jgZvjgpPvvIjjgqjjg6njg7zjgYznmbrnlJ/jgZfjgb7jgZfjgZ/jgILoqbPntLDjga/jgrfjgrnjg4bjg6Djg63jgrDjgpLjgZTnorroqo3jgY/jgaDjgZXjgYTvvInvvIEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgTGlzdCd9ID0gJ+ODhuODs+ODl+ODrOODvOODiOS4gOimpyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXInfSA9ICfmlbDlgKQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnbnVtYmVyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICcw5Lul5LiK44Gu5pWw5YCkJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXInfSA9ICfmlbTmlbDlgKQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaW50ZWdlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnMOS7peS4iuOBruaVtOaVsOWApCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICfjg4fjg7zjgr/jgpLlhaXlipvjgZnjgovlv4XopoHjgYzjgYLjgorjgb7jgZnjgIInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW52YWxpZCBkYXRhLCBwbGVhc2UgaW5zZXJ0IGEgdmFsaWQgJXMnfSA9ICfnhKHlirnjgarjg4fjg7zjgr/jgafjgZnjgILmnInlirnjgaogJXMg44KS5YWl5Yqb44GX44Gm44GP44Gg44GV44GE44CCJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBub3QgZm91bmQhJ30gPSAn5oyH5a6a44GV44KM44Gf44OV44Kp44O844Oe44OD44OI44GM6KaL44Gk44GL44KK44G+44Gb44KT77yBJzsKCiAgICAjIFBlcmwgTW9kdWxlOiBLZXJuZWwvU3lzdGVtL0ltcG9ydEV4cG9ydC9Gb3JtYXRCYWNrZW5kL0NTVi5wbQogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uIFNlcGFyYXRvcid9ID0gJ+ahgeOBruOCu+ODkeODrOODvOOCvyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICfjgr/jg5YgKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ+OCu+ODn+OCs+ODreODsyAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sb24gKDopJ30gPSAn44Kz44Ot44OzICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAn44OJ44OD44OIKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbW1hICgsKSd9ID0gJ+OCq+ODs+ODniAoLCknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ+OCreODo+ODqeOCr+OCv+OCu+ODg+ODiCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAn44OY44OD44OA5oOF5aCx44KS5ZCr44KAJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ+ahgSc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ2ltcG9ydC9leHBvcnTjg6Ljgrjjg6Xjg7zjg6vjga7jg5Djg4Pjgq/jgqjjg7Pjg4njg6Ljgrjjg6Xjg7zjg6vjgpLnmbvpjLInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ+OCquODluOCuOOCp+OCr+ODiOaDheWgseOBruOCpOODs+ODneODvOODiOODu+OCqOOCr+OCueODneODvOODiCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAn44Kk44Oz44Od44O844OIL+OCqOOCr+OCueODneODvOODiCc7Cgp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6Om1rX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdBw7FhZGlyIHBsYW50aWxsYSBkZSBtYXBlbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAnSnVlZ28gZGUgY2FyYWN0ZXJlcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEb3MgcHVudG9zICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdDb2x1bW5hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdTZXBhcmFkb3IgZGUgQ29sdW1uYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUHVudG8gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdQdW50byB5IENvbWEgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RhYnVsYWRvciAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnSW5jbHVpciBDYWJlY2VyYSBkZSBsYSBDb2x1bW5hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ0ltcG9ydGFyIHJlc3VtZW4gcGFyYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnUmVnaXN0cm9zIGltcG9ydGFkb3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXhwb3J0ZWQgcmVjb3Jkcyd9ID0gJ1JlZ2lzdHJvcyBleHBvcnRhZG9zJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlY29yZHMnfSA9ICdSZWdpc3Ryb3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2tpcHBlZCd9ID0gJ1NhbHRhZG8nOwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdHZXN0acOzbiBkZSBJbXBvcnRhY2nDs24vRXhwb3J0YWNpw7NuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdDcmVhciB1bmEgcGxhbnRpbGxhIHBhcmEgaW1wb3J0YXIgeSBleHBvcnRhciBpbmZvcm1hY2lvbiBkZWwgb2JqZXRvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdJbmljaWFyIEltcG9ydGFjacOzbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdJbmljaWFyIEV4cG9ydGFjacOzbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDEgb2YgNSAtIEVkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnRWwgbm9tYnJlIGVzIHJlcXVlcmlkbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnwqFEZWJlIGVzcGVjaWZpY2FyIE9iamV0byEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJ8KhRGViZSBlc3BlY2lmaWNhciBGb3JtYXRvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDIgb2YgNSAtIEVkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMyBvZiA1J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJ8KhZXMgcmVxdWVyaWRvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDQgb2YgNSAtIEVkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnTm8gc2UgZW5jb250cmFyb24gZWxlbWVudG9zIGRlIG1hcGVvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnQcOxYWRpciBNYXBlbyBkZSBFbGVtZW50b3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA1IG9mIDUgLSBFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ1Jlc3RyaW5naXIgZXhwb3J0YWNpw7NuIHBvciBiw7pzcXVlZGEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW1wb3J0YXIgaW5mb3JtYWNpw7NuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnQXJjaGl2byBvcmlnZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJ8OJeGl0byc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICdGcmFjYXNhZG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnTm9tYnJlcyBkdXBsaWNhZG9zJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnw5psdGltYSBuw7ptZXJvIGRlIGzDrW5lYSBwcm9jZXNhZGEgZGVsIGFyY2hpdm8gaW1wb3J0YXInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPayc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXInfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnbnVtYmVyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaW50ZWdlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWxlbWVudCByZXF1aXJlZCwgcGxlYXNlIGluc2VydCBkYXRhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ludmFsaWQgZGF0YSwgcGxlYXNlIGluc2VydCBhIHZhbGlkICVzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBub3QgZm91bmQhJ30gPSAnJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnUmVnaXN0cm8gZGUgbcOzZHVsbyBkZSBmb3JtYXRvIGJhY2tlbmQgcGFyYSBlbCBtw7NkdWxvIGltcG9ydC9leHBvcnQuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdJbXBvcnRhciB5IGV4cG9ydGFyIGluZm9ybWFjacOzbiBkZSBvYmpldG9zLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3J0YXIvRXhwb3J0YXInOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6Om1zX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdQZW5ndXJ1c2FuIEltcG9ydC9Fa3Nwb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdDaXB0YSB0ZW1wbGF0IHVudHVrIGltcG9ydCBkYW4gZWtzcG9ydCBpbmZvcm1hc2kgb2JqZWsuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ011bGEgaW1wb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ011bGEgZWtzcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDEgb2YgNSAtIEVkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnTmFtYSBkaWtlaGVuZGFraSEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ09iamVrIGFkYWxhaCBkaXBlcmx1a2FuISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnRm9ybWF0IGFkYWxhaCBkaXBlcmx1a2FuISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDIgb2YgNSAtIEVkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMyBvZiA1IC0gRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAnZGlwZXJsdWthbiEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA0IG9mIDUgLSBFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJ1RpYWRhIGVsZW1lbiBwZXRhIGRpdGVtdWkuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICdUYW1iYWggZWxlbWVuIHBldGEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA1IG9mIDUgLSBFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ0RpbGFyYW5nIGVrc3BvcnQgcGVyIGNhcmlhbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICdJbmZvcm1hc2kgaW1wb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnU3VtYmVyIGZhaWwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnSW1wb3J0IHJpbmdrYXNhbiB1bnR1ayc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnUmVrb2QtcmVrb2QnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJ0JlcmpheWEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnR2FnYWwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnTWVueWFsaW4gbmFtYS1uYW1hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnTm9tYm9yIGJhcmlzIHRlcmFraGlyIGRpcHJvc2VzIGZhaWwgaW1wb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnT2snOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9Nb2R1bGVzL0FkbWluSW1wb3J0RXhwb3J0LnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBvYmplY3QgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gZm9ybWF0IGJhY2tlbmQgZm91bmQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RlbXBsYXRlIG5vdCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2FuXCd0IGluc2VydC91cGRhdGUgdGVtcGxhdGUhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05lZWRlZCBUZW1wbGF0ZUlEISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFcnJvciBvY2N1cnJlZC4gSW1wb3J0IGltcG9zc2libGUhIFNlZSBTeXNsb2cgZm9yIGRldGFpbHMuJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBFeHBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgTGlzdCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXInfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnbnVtYmVyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaW50ZWdlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWxlbWVudCByZXF1aXJlZCwgcGxlYXNlIGluc2VydCBkYXRhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ludmFsaWQgZGF0YSwgcGxlYXNlIGluc2VydCBhIHZhbGlkICVzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBub3QgZm91bmQhJ30gPSAnJzsKCiAgICAjIFBlcmwgTW9kdWxlOiBLZXJuZWwvU3lzdGVtL0ltcG9ydEV4cG9ydC9Gb3JtYXRCYWNrZW5kL0NTVi5wbQogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uIFNlcGFyYXRvcid9ID0gJ0tvbHVtIHBlbWlzYWgnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGFidWxhdG9yIChUQUIpJ30gPSAnVGFidWxhdG9yIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdTZW1pa29sb24gKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ1RpdGlrIGJlcnRpbmRpaCAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ1RpdGlrICguKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb21tYSAoLCknfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ1NldCBrYXJha3Rlcic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnVGVybWFzdWsgS29sdW0gS2VwYWxhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ0tvbHVtJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnRm9ybWF0IGJhY2tlbmQgbW9kdWwgcGVuZGFmdGFyYW4gdW50dWsgbW9kdWwgaW1wb3J0L2Vrc3BvcnQuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdJbmZvcm1hc2kgb2JqZWsgaW1wb3J0IGRhbiBla3Nwb3J0Lic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3J0L0Vrc3BvcnQnOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6Om5iX05PX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdBZG1pbmlzdHJhc2pvbiBhdiBJbXBvcnQvRWtzcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnT3BwcmV0dCBlbiBtYWwgZm9yIMOlIGVrc3BvcnRlcmUgb2cgaW1wb3J0ZXJlIGluZm9ybWFzam9uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ1N0YXJ0IGltcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdTdGFydCBla3Nwb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMSBvZiA1IC0gRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICdOYXZuIGVyIHDDpWtyZXZkISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnT2JqZWt0IGVyIHDDpWtyZXZkISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnRm9ybWF0IGVyIHDDpWtyZXZkISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDIgb2YgNSAtIEVkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMyBvZiA1IC0gRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAnZXIgcMOla3JldmQhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNCBvZiA1IC0gRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICdJbmdlbiBlbGVtZW50ZXIgZnVubmV0Lic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnTGVnZyB0aWwgbWFwcGluZy1lbGVtZW50JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNSBvZiA1IC0gRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdCZWdyZW5zIGVrc3BvcnQgcGVyIHPDuGsnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW1wb3J0LWluZm9ybWFzam9uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnS2lsZGVmaWwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnT3Bwc3VtbWVyaW5nZXIgZm9yIGltcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnUmFkZXInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJ1ZlbGx5a2tldCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICdGZWlsZXQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnRHVwbGlrYXRlIG5hdm4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICdTaXN0ZSBwcm9zZXNzZXJ0ZSBsaW5qZW51bW1lciBhdiBpbXBvcnRmaWwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPayc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW52YWxpZCBkYXRhLCBwbGVhc2UgaW5zZXJ0IGEgdmFsaWQgJXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IG5vdCBmb3VuZCEnfSA9ICcnOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9TeXN0ZW0vSW1wb3J0RXhwb3J0L0Zvcm1hdEJhY2tlbmQvQ1NWLnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnS29sb25uZXNlcGFyYXRvcic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bGF0b3IgKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ1NlbWlrb2xvbiAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sb24gKDopJ30gPSAnS29sb24gKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICdQdW5rdHVtICguKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb21tYSAoLCknfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ1RlZ25zZXR0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICdJbmtsdWRlciBrb2xvbm5lb3ZlcnNrcmlmdGVyJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ0tvbG9ubmUnOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICdCYWtzaWRlbW9kdWwtcmVnaXN0cmVyaW5nIGZvciBmb3JtYXRldCB0aWwgaW1wb3J0L2Vrc3BvcnQtbW9kdWxlbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnSW5mb3JtYXNqb24gZm9yIGltcG9ydC0gb2cgZWtzcG9ydC1vYmpla3QnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ0ltcG9ydC9Fa3Nwb3J0JzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6Om5sX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdJbXBvcnQvRXhwb3J0IGJlaGVlcic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ0ltcG9ydCBzdGFydGVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ0V4cG9ydCBzdGFydGVuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMSBvZiA1IC0gRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ09iamVjdCBpcyB2ZXJwbGljaHQuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICdGb3JtYWF0IGlzIHZlcnBsaWNodC4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAyIG9mIDUgLSBFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDMgb2YgNSAtIEVkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJ2lzIHZlcnBsaWNodCEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA0IG9mIDUgLSBFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJ0dlZW4gZWxlbWVudGVuIGdldm9uZGVuLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNSBvZiA1IC0gRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdCZXBlcmsgZXhwb3J0IHRvdCB6b2Vrb3BkcmFjaHQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW1wb3J0LWluZm9ybWF0aWUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdCcm9uYmVzdGFuZCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgc3VtbWFyeSBmb3InfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAnU3VjY2VzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPSyc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW52YWxpZCBkYXRhLCBwbGVhc2UgaW5zZXJ0IGEgdmFsaWQgJXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IG5vdCBmb3VuZCEnfSA9ICcnOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9TeXN0ZW0vSW1wb3J0RXhwb3J0L0Zvcm1hdEJhY2tlbmQvQ1NWLnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnS29sb21zY2hlaWRpbmdzdGVrZW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGFidWxhdG9yIChUQUIpJ30gPSAnVGFiJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdQdW50a29tbWEgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ0R1YmJlbGUgcHVudCAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ1B1bnQgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbW1hICgsKSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAnS2FyYWt0ZXJzZXQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW5jbHVkZSBDb2x1bW4gSGVhZGVycyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdLb2xvbSc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnSW1wb3J0IGVuIGV4cG9ydCBvYmplY3RpbmZvcm1hdGllJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICdJbXBvcnQvRXhwb3J0JzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnBsX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdaYXJ6xIVkemFuaWUgSW1wb3J0ZW0vRXhwb3J0ZW0nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ1V0d8Ozcnogc3phYmxvbiBkbyBpbXBvcnR1IGkgZWtzcG9ydHUgZGFueWNoIG9iaWVrdMOzdy4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAnUm96cG9jem5paiBpbXBvcnQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAnUm96cG9jem5paiBla3Nwb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMSBvZiA1IC0gRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICdOYXp3YSBqZXN0IHd5bWFnYW5hISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAnT2JpZWt0IGplc3Qgd3ltYWdhbnkhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICdGb3JtYXQgamVzdCB3eW1hZ2FueSEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAyIG9mIDUgLSBFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDMgb2YgNSAtIEVkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJ2plc3Qgd3ltYWdhbmEoeSkhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNCBvZiA1IC0gRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICdOaWUgem5hbGV6aW9ubyBlbGVtZW50w7N3IG1hcHkuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICdEb2RhaiBlbGVtZW50IG1hcG93YW5pYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDUgb2YgNSAtIEVkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnT2dyYW5pY3ogZWtzcG9ydCBwcnpleiB3eXN6dWthbmllJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0ltcG9ydHVqIGluZm9ybWFjamUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdQbGlrIMW6csOzZMWCb3d5JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ1BvZHN1bW93YW5pZSBpbXBvcnR1IGRsYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnUmVrb3JkeSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAnUG93b2R6ZW5pZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICdOaWVwb3dvZHplbmllJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ0R1cGxpa2F0eSBuYXp3JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnTnIgb3N0YW5pZWogcHJ6ZXR3b3J6b25laiBsaW5paSBwbGlrdSBpbXBvcnRvd2Vnbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJ09rJzsKCiAgICAjIFBlcmwgTW9kdWxlOiBLZXJuZWwvTW9kdWxlcy9BZG1pbkltcG9ydEV4cG9ydC5wbQogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gb2JqZWN0IGJhY2tlbmQgZm91bmQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIGZvcm1hdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBub3QgZm91bmQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NhblwndCBpbnNlcnQvdXBkYXRlIHRlbXBsYXRlISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOZWVkZWQgVGVtcGxhdGVJRCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEltcG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFcnJvciBvY2N1cnJlZC4gRXhwb3J0IGltcG9zc2libGUhIFNlZSBTeXNsb2cgZm9yIGRldGFpbHMuJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RlbXBsYXRlIExpc3QnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnbnVtYmVyJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXInfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaW50ZWdlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VsZW1lbnQgcmVxdWlyZWQsIHBsZWFzZSBpbnNlcnQgZGF0YSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbnZhbGlkIGRhdGEsIHBsZWFzZSBpbnNlcnQgYSB2YWxpZCAlcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgbm90IGZvdW5kISd9ID0gJyc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL1N5c3RlbS9JbXBvcnRFeHBvcnQvRm9ybWF0QmFja2VuZC9DU1YucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdTZXBhcmF0b3Iga29sdW1ueSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJ1bGF0b3IgKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ8WacmVkbmlrICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICdEd3Vrcm9wZWsgKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICdLcm9wa2EgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbW1hICgsKSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAnS29kb3dhbmllICc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnVW1pZcWbxIcgbmFnxYLDs3draSBrb2x1bW4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnS29sdW1uYSc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ01vZHXFgiBmb3JtYXRvd2FuaWEgYmFja2VuZCBkbGEgbW9kdcWCdSBpbXBvcnQvZWtzcG9ydC4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0ltcG9ydHVqIGkgZWtzcG9ydHVqIGluZm9ybWFjamUgb2JpZWt0w7N3Lic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3J0L2Vrc3BvcnQnOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnB0X0JSX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdHZXJlbmNpYW1lbnRvIGRlIEltcG9ydGHDp8Ojby9FeHBvcnRhw6fDo28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0NyaWFyIHVtIG1vZGVsbyBwYXJhIGltcG9ydGFyIGUgZXhwb3J0YXIgaW5mb3JtYcOnw7VlcyBkZSBvYmpldG8uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ0luaWNpYXIgSW1wb3J0YcOnw6NvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ0luaWNpYXIgRXhwb3J0YcOnw6NvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMSBvZiA1IC0gRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICdQYXNzbyAxIGRlIDUgLSBFZGl0YXIgaW5mb3JtYcOnw7VlcyBjb211bnMnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICdOb21lIMOpIG9icmlnYXTDs3JpbyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ09iamV0byDDqSBvYnJpZ2F0w7NyaW8hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICdPIGZvcm1hdG8gw6kgb2JyaWdhdMOzcmlvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDIgb2YgNSAtIEVkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnUGFzc28gMiBkZSA1IC0gRWRpdGFyIGluZm9ybWHDp8O1ZXMgZG8gb2JqZXRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMyBvZiA1IC0gRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICdQYXNzbyAzIGRlIDUgLSBFZGl0YXIgaW5mb3JtYcOnw7VlcyBkZSBmb3JtYXRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJ8OpIG9icmlnYXTDs3JpbyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA0IG9mIDUgLSBFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICdQYXNzbyA0IGRlIDUgLSBFZGl0YXIgaW5mb3JtYcOnw7VlcyBkZSBtYXBlYW1lbnRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICdOw6NvIGjDoSBlbGVtZW50b3MgbWFwYSBlbmNvbnRyYWRvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnQWRpY2lvbmFyIGVsZW1lbnRvIGRlIG1hcGVhbWVudG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA1IG9mIDUgLSBFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJ1Bhc3NvIDUgZGUgNSAtIEVkaXRhciBpbmZvcm1hw6fDtWVzIGRlIHBlc3F1aXNhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnUmVzdHJpbmdpciBleHBvcnRhw6fDo28gcG9yIHBlc3F1aXNhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0luZm9ybWHDp8O1ZXMgZGUgaW1wb3J0YcOnw6NvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAnQXJxdWl2byBkZSBPcmlnZW0nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnUmVzdW1vIGRlIGltcG9ydGHDp8OjbyBwYXJhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlY29yZHMnfSA9ICdSZWdpc3Ryb3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJ1N1Y2Vzc28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnRmFsaG91JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ05vbWVzIGR1cGxpY2Fkb3MnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICfDmmx0aW1vIG7Dum1lcm8gZGUgbGluaGEgcHJvY2Vzc2FkYSBkbyBhcnF1aXZvIGRlIGltcG9yYXRhw6fDo28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPayc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJ05lbmh1bSBvYmpldG8gYmFja2VuZCBlbmNvbnRyYWRvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICdOZW5odW0gZm9ybWF0byBiYWNrZW5kIGVuY29udHJhZG8hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RlbXBsYXRlIG5vdCBmb3VuZCEnfSA9ICdNb2RlbG8gbsOjbyBlbmNvbnRyYWRvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICdOw6NvIMOpIHBvc3PDrXZlbCBpbnNlcmlyL2F0dWFsaXphciBtb2RlbG8hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05lZWRlZCBUZW1wbGF0ZUlEISd9ID0gJ05lY2Vzc8OhcmlvIFRlbXBsYXRlSUQhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICdPY29ycmV1IHVtIGVycm8uIEZvaSBpbXBvc3PDrXZlbCByZWFsaXphciBhIGltcG9ydGHDp8OjbyEgVmVyaWZpcXVlIG8gU3lzbG9nIHBhcmEgbWFpcyBkZXRhbGhlcy4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJ09jb3JyZXUgdW0gZXJyby4gRm9pIGltcG9zc8OtdmVsIHJlYWxpemFyIGEgZXhwb3J0YcOnw6NvISBWZXJpZmlxdWUgbyBTeXNsb2cgcGFyYSBtYWlzIGRldGFsaGVzLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAnTGlzdGEgZGUgbW9kZWxvcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXInfSA9ICduw7ptZXJvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnbsO6bWVybyBtYWlvciBxdWUgemVybyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyJ30gPSAnaW50ZWlybyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICdpbnRlaXJvIG1haW9yIHF1ZSB6ZXJvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VsZW1lbnQgcmVxdWlyZWQsIHBsZWFzZSBpbnNlcnQgZGF0YSd9ID0gJ0VsZW1lbnRvIG5lY2Vzc8OhcmlvLCBwb3IgZmF2b3IgaW5zaXJhIG8gZGFkbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbnZhbGlkIGRhdGEsIHBsZWFzZSBpbnNlcnQgYSB2YWxpZCAlcyd9ID0gJ0RhZG8gaW52w6FsaWRvLCBwb3IgZmF2b3IgaW5zaXJhIHVtICVzIHbDoWxpZG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IG5vdCBmb3VuZCEnfSA9ICdGb3JtYXRvIG7Do28gZW5jb250cmFkbyEnOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9TeXN0ZW0vSW1wb3J0RXhwb3J0L0Zvcm1hdEJhY2tlbmQvQ1NWLnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnU2VwYXJhZG9yIGRlIENvbHVuYXMnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGFidWxhdG9yIChUQUIpJ30gPSAnVGFidWxhw6fDo28gKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ1BvbnRvIGUgVsOtcmd1bGEgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ0RvaXMgUG9udG9zICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUG9udG8gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbW1hICgsKSd9ID0gJ1Zpcmd1bGEgKCwpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdDb2RpZmljYcOnw6NvIGRlIENhcmFjdGVyZXMnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW5jbHVkZSBDb2x1bW4gSGVhZGVycyd9ID0gJ0luY2x1aXIgQ2FiZcOnYWxob3MgZGUgQ29sdW5hcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdDb2x1bmEnOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICdGb3JtYXRvIGRlIHJlZ2lzdHJvIGJhY2tlbmQgZG8gbcOzZHVsbyBkZSBpbXBvcnRhw6fDo28gLyBleHBvcnRhw6fDo28gbcOzZHVsby4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0ltcG9ydGFyIGUgZXhwb3J0YXIgaW5mb3JtYcOnw7VlcyBkZSBvYmpldG8uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICdJbXBvcnRhw6fDo28vRXhwb3J0YcOnw6NvJzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnJvX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICflop7liqDmmKDlsITmqKHniYgnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ+Wtl+espumbhic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICflhpLlj7cgKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ+WIlyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAn5YiX5YiG6ZqU56ymJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICflj6Xlj7cgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICfliIblj7cgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RBQumUriAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAn5YyF5ous5YiX5qCH6aKYJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ+WvvOWFpeaAu+e7kyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAn5a+85YWl6K6w5b2VJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0V4cG9ydGVkIHJlY29yZHMnfSA9ICflr7zlh7rorrDlvZUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ+iusOW9lSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTa2lwcGVkJ30gPSAn6Lez6L+H55qEJzsKCiAgICAjIFRlbXBsYXRlOiBBZG1pbkltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCBNYW5hZ2VtZW50J30gPSAn5a+85YWlL+WvvOWHuueuoeeQhic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAn5Yib5bu65qih5p2/5a+85YWl5ZKM5a+85Ye65a+56LGh5L+h5oGv44CCJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ+W8gOWni+WvvOWFpSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICflvIDlp4vlr7zlh7onOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAxIG9mIDUgLSBFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJ+WRveWQjeaYr+W/hemcgOeahO+8gSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAn5a+56LGh5piv5b+F6ZyA55qE77yBJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICfmoLzlvI/mmK/lv4XpnIDnmoTvvIEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAyIG9mIDUgLSBFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDMgb2YgNSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpcyByZXF1aXJlZCEnfSA9ICfmmK/lv4XpnIDnmoTvvIEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA0IG9mIDUgLSBFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJ+ayoeacieaJvuWIsOaYoOWwhOeahOWtl+autSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAn5re75Yqg5pig5bCE5a2X5q61JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNSBvZiA1IC0gRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICfmjInmkJzntKLpmZDliLblr7zlh7onOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAn5a+85YWl5L+h5oGvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAn5rqQ5paH5Lu2JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICfmiJDlip8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAn5aSx6LSlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ+mHjeWkjeeahOWQjeensCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJ+WvvOWFpeaWh+S7tuacgOWQjuWkhOeQhueahOihjOaVsCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJ+ehruWumic7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXInfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnbnVtYmVyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaW50ZWdlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWxlbWVudCByZXF1aXJlZCwgcGxlYXNlIGluc2VydCBkYXRhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ludmFsaWQgZGF0YSwgcGxlYXNlIGluc2VydCBhIHZhbGlkICVzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBub3QgZm91bmQhJ30gPSAnJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAn5Li65a+85YWlL+WvvOWHuuaooeWdl+eahOagvOW8j+WQjuerr+aooeWdl+azqOWGjOOAgic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAn5a+85YWl5ZKM5a+85Ye65a+56LGh5L+h5oGv44CCJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICflr7zlhaUv5a+85Ye6JzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnB0X0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdHZXN0w6NvIGRlIEltcG9ydGHDp8Ojby9FeHBvcnRhw6fDo28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0NyaWFyIHVtIG1vZGVsbyBwYXJhIGltcG9ydGFyIGUgZXhwb3J0YXIgaW5mb3JtYcOnw7VlcyBkZSBvYmpldG8uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ0luaWNpYXIgSW1wb3J0YcOnw6NvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ0luaWNpYXIgRXhwb3J0YcOnw6NvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMSBvZiA1IC0gRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICdOb21lIMOpIG9icmlnYXTDs3JpbyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ09iamV0byDDqSBvYnJpZ2F0w7NyaW8hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICdPIGZvcm1hdG8gw6kgb2JyaWdhdMOzcmlvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDIgb2YgNSAtIEVkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMyBvZiA1IC0gRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAnw6kgb2JyaWdhdMOzcmlvISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDQgb2YgNSAtIEVkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJ1Bhc3NvIDQgZGUgNSAtIEVkaXRhciBpbmZvcm1hw6fDo28gZGUgbWFwZWFtZW50byc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnTsOjbyBow6EgZWxlbWVudG9zIG1hcGEgZW5jb250cmFkby4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ0FkaWNpb25hciBlbGVtZW50byBkZSBtYXBlYW1lbnRvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNSBvZiA1IC0gRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICdQYXNzbyA1IGRlIDUgLSBFZGl0YXIgaW5mb3JtYcOnw6NvIGRlIHByb2N1cmEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdSZXN0cmluZ2lyIGV4cG9ydGHDp8OjbyBwb3IgcGVzcXVpc2EnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnSW5mb3JtYcOnw7VlcyBkZSBpbXBvcnRhw6fDo28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdBcnF1aXZvIGRlIE9yaWdlbSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgc3VtbWFyeSBmb3InfSA9ICdSZXN1bW8gZGUgaW1wb3J0YcOnw6NvIHBhcmEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ1JlZ2lzdG9zJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICdTdWNlc3NvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJ0ZhbGhvdSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICdOb21lcyBkdXBsaWNhZG9zJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnw5psdGltbyBuw7ptZXJvIGRlIGxpbmhhIHByb2Nlc3NhZGEgZG8gYXJxdWl2byBkZSBpbXBvcnRhw6fDo28nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPayc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICdPY29ycmV1IHVtIGVycm8uSW1wb3J0YcOnw6NvIGltcG9zc8OtdmVsIVZlcmlmaWNhciBTeXNsb2cgcGFyYSBkZXRhbGhlcy4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJ09jb3JyZXUgdW0gZXJyby5FeHBvcnRhw6fDo28gaW1wb3Nzw612ZWwhVmVyaWZpY2FyIFN5c2xvZyBwYXJhIGRldGFsaGVzLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW52YWxpZCBkYXRhLCBwbGVhc2UgaW5zZXJ0IGEgdmFsaWQgJXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IG5vdCBmb3VuZCEnfSA9ICcnOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9TeXN0ZW0vSW1wb3J0RXhwb3J0L0Zvcm1hdEJhY2tlbmQvQ1NWLnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnU2VwYXJhZG9yIGRlIENvbHVuYXMnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGFidWxhdG9yIChUQUIpJ30gPSAnVGFidWxhw6fDo28gKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ1BvbnRvIGUgVsOtcmd1bGEgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ0RvaXMgUG9udG9zICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAnUG9udG8gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbW1hICgsKSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAnQ29kaWZpY2HDp8OjbyBkZSBDYXJhY3RlcmVzJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICdJbmNsdWlyIENhYmXDp2FsaG9zIGRlIENvbHVuYXMnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnQ29sdW5hJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnRm9ybWF0YXIgbcOzZHVsbyBkZSByZWdpc3RvIGJhY2tlbmQgZG8gbcOzZHVsbyBkZSBpbXBvcnRhw6fDo28gLyBleHBvcnRhw6fDo28uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdJbXBvcnRhciBlIGV4cG9ydGFyIGluZm9ybWHDp8O1ZXMgZGUgb2JqZXRvLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnSW1wb3J0YXIvRXhwb3J0YXInOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnJ1X0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICfQo9C/0YDQsNCy0LvQtdC90LjQtSDQmNC80L/QvtGA0YLQvtC8L9Ct0LrRgdC/0L7RgNGC0L7QvCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAn0KHQvtC30LTQsNC50YLQtSDRiNCw0LHQu9C+0L0g0LTQu9GPINC40LzQv9C+0YDRgtCwINC4INGN0LrRgdC/0L7RgNGC0LAg0LjQvdGE0L7RgNC80LDRhtC40Lgg0L7QsdGK0LXQutGC0L7Qsic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICfQndCw0YfQsNGC0Ywg0LjQvNC/0L7RgNGCJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ9Cd0LDRh9Cw0YLRjCDRjdC60YHQv9C+0YDRgic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDEgb2YgNSAtIEVkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAn0KjQsNCzIDEg0LjQtyA1IC0g0KDQtdC00LDQutGC0LjRgNC+0LLQsNGC0Ywg0L7QsdGJ0YPRjiDQuNC90YTQvtGA0LzQsNGG0LjRjic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJ9Ci0YDQtdCx0YPQtdGC0YHRjyDQuNC80Y8hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICfQntCx0YrQtdC60YIg0L7QsdGP0LfQsNGC0LXQu9C10L0hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICfQpNC+0YDQvNCw0YIg0L7QsdGP0LfQsNGC0LXQu9C10L0hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMiBvZiA1IC0gRWRpdCBvYmplY3QgaW5mb3JtYXRpb24nfSA9ICfQqNCw0LMgMiDQuNC3IDUgLSDQoNC10LTQsNC60YLQuNGA0L7QstCw0YLRjCDQuNC90YTQvtGA0LzQsNGG0LjRjiDQvtCxINC+0LHRitC10LrRgtC1JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMyBvZiA1IC0gRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICfQqNCw0LMgMyDQuNC3IDUgLSDQoNC10LTQsNC60YLQuNGA0L7QstCw0YLRjCDRhNC+0YDQvNCw0YIg0LjQvdGE0L7RgNC80LDRhtC40LgnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAn0L7QsdGP0LfQsNGC0LXQu9C10L0hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNCBvZiA1IC0gRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gPSAn0KjQsNCzIDQg0LjQtyA1IC0g0KDQtdC00LDQutGC0LjRgNC+0LLQsNGC0Ywg0LjQvdGE0L7RgNC80LDRhtC40Y4g0YHQvtC/0L7RgdGC0LDQstC70LXQvdC40Y8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJ9Cd0LXRgiDRjdC70LXQvNC10L3RgtC+0LIg0YHQvtC/0L7RgdGC0LDQstC70LXQvdC40Y8uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICfQlNC+0LHQsNCy0YzRgtC1INGN0LvQtdC80LXQvdGCINGB0L7Qv9C+0YHRgtCw0LLQu9C10L3QuNGPJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNSBvZiA1IC0gRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICfQqNCw0LMgNSDQuNC3IDUgLSDQoNC10LTQsNC60YLQuNGA0L7QstCw0YLRjCDQuNC90YTQvtGA0LzQsNGG0LjRjiDQv9C+0LjRgdC60LAnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICfQntCz0YDQsNC90LjRh9C40YLRjCDRjdC60YHQv9C+0YDRgiDQv9C+0LjRgdC60L7QvCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICfQmNC90YTQvtGA0LzQsNGG0LjRjyDQvtCxINC40LzQv9C+0YDRgtC1JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAn0JjRgdGF0L7QtNC90YvQuSDRhNCw0LnQuyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgc3VtbWFyeSBmb3InfSA9ICfQntGC0YfQtdGCINC+0LEg0LjQvNC/0L7RgNGC0LUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ9CX0LDQv9C40YHQtdC5JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICfQo9GB0L/QtdGI0L3Qvic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICfQndC1INGD0LTQsNC70L7RgdGMINCy0YvQv9C+0LvQvdC40YLRjCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICfQlNGD0LHQu9C40YDRg9GO0YnQuNC1INC40LzQtdC90LAnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICfQndC+0LzQtdGAINC/0L7RgdC70LXQtNC90LXQuSDQvtCx0YDQsNCx0L7RgtCw0L3QvdC+0Lkg0YHRgtGA0L7QutC4INC40LzQv9C+0YDRgtC40YDRg9C10LzQvtCz0L4g0YTQsNC50LvQsCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJ09rJzsKCiAgICAjIFBlcmwgTW9kdWxlOiBLZXJuZWwvTW9kdWxlcy9BZG1pbkltcG9ydEV4cG9ydC5wbQogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gb2JqZWN0IGJhY2tlbmQgZm91bmQhJ30gPSAn0J7QsdGK0LXQutGCINC90LUg0L3QsNC50LTQtdC9ISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICfQkdGN0LrRjdC90LQg0L3QtSDQvdCw0LnQtNC10L0hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RlbXBsYXRlIG5vdCBmb3VuZCEnfSA9ICfQqNCw0LHQu9C+0L0g0L3QtSDQvdCw0LnQtNC10L0hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NhblwndCBpbnNlcnQvdXBkYXRlIHRlbXBsYXRlISd9ID0gJ9Cd0LXQstC+0LfQvNC+0LbQvdC+INC00L7QsdCw0LLQuNGC0Ywv0L7QsdC90L7QstC40YLRjCDRiNCw0LHQu9C+0L0hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05lZWRlZCBUZW1wbGF0ZUlEISd9ID0gJ9Ci0YDQtdCx0YPQtdGC0YHRjyBJRCDQqNCw0LHQu9C+0L3QsCEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEltcG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJ9Ce0YjQuNCx0LrQsC4g0JjQvNC/0L7RgNGCINC90LXQstC+0LfQvNC+0LbQtdC9ISDQodC80L7RgtGA0Lgg0L/QvtC00YDQvtCx0L3QvtGB0YLQuCDQsiBTeXNsb2cuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBFeHBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICfQntGI0LjQsdC60LAuINCt0LrRgdC/0L7RgNGCINC90LXQstC+0LfQvNC+0LbQtdC9ISDQodC80L7RgtGA0Lgg0L/QvtC00YDQvtCx0L3QvtGB0YLQuCDQsiBTeXNsb2cuJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RlbXBsYXRlIExpc3QnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnbnVtYmVyJ30gPSAn0L3QvtC80LXRgCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJ9C90L7QvNC10YAg0LHQvtC70YzRiNC1INC90YPQu9GPJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXInfSA9ICfRhtC10LvQvtC1INGH0LjRgdC70L4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaW50ZWdlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAn0YbQtdC70L7QtSDRh9C40YHQu9C+INCx0L7Qu9GM0YjQtSDQvdGD0LvRjyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICfQntCx0Y/Qt9Cw0YLQtdC70YzQvdGL0LUg0Y3Qu9C10LzQtdC90YIsINC/0L7QttCw0LvRg9C50YHRgtCwINCy0LLQtdC00LjRgtC1INC00LDQvdC90YvQtS4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW52YWxpZCBkYXRhLCBwbGVhc2UgaW5zZXJ0IGEgdmFsaWQgJXMnfSA9ICfQndC10LLQtdGA0L3Ri9C1INC00LDQvdC90YvQtSwg0L/QvtC20LDQu9GD0LnRgdGC0LAsINGD0LrQsNC20LjRgtC1INCy0LXRgNC90YvQtSAlcyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgbm90IGZvdW5kISd9ID0gJ9Ck0L7RgNC80LDRgiDQvdC1INC90LDQudC00LXQvSEnOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9TeXN0ZW0vSW1wb3J0RXhwb3J0L0Zvcm1hdEJhY2tlbmQvQ1NWLnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAn0KDQsNC30LTQtdC70LjRgtC10LvRjCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICfQotCw0LHRg9C70Y/RhtC40Y8gKFRBQiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ9Ci0L7Rh9C60LAg0YEg0LfQsNC/0Y/RgtC+0LkgKDspJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbG9uICg6KSd9ID0gJ9CU0LLQvtC10YLQvtGH0LjQtSAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ9Ci0L7Rh9C60LAgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbW1hICgsKSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAn0JrQvtC00LjRgNC+0LLQutCwJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICfQktC60LvRjtGH0LjRgtGMINC30LDQs9C+0LvQvtCy0LrQuCDRgdGC0L7Qu9Cx0YbQvtCyJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ9Ch0YLQvtC70LHQtdGGICc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24g0LTQu9GPINC80L7QtNGD0LvRjyBpbXBvcnQvZXhwb3J0Lic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAn0JjQvNC/0L7RgNGCINC4INGN0LrRgdC/0L7RgNGCINC40L3RhNC+0YDQvNCw0YbQuNC4INC+0LEg0L7QsdGK0LXQutGC0LDRhSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAn0JjQvNC/0L7RgNGCL9Ct0LrRgdC/0L7RgNGCJzsKCn0KCjE7Cg==
# --
# Copyright (C) 2001-2019 OTRS AG, https://otrs.com/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Language::sr_Cyrl_ImportExport;

use strict;
use warnings;
use utf8;

sub Data {
    my $Self = shift;

    # Template: AdminImportExport
    $Self->{Translation}->{'Import/Export Management'} = 'Увоз/Извоз управљање';
    $Self->{Translation}->{'Create a template to import and export object information.'} = 'Креирај шаблон за увоз и извоз информација о објекту.';
    $Self->{Translation}->{'Start Import'} = 'Почни увоз';
    $Self->{Translation}->{'Start Export'} = 'Почни извоз';
    $Self->{Translation}->{'Step 1 of 5 - Edit common information'} = 'Корак 1 од 5 - Уреди заједничке информације';
    $Self->{Translation}->{'Name is required!'} = 'Име је обавезно!';
    $Self->{Translation}->{'Object is required!'} = 'Објект је обавезан!';
    $Self->{Translation}->{'Format is required!'} = 'Формат је обавезан!';
    $Self->{Translation}->{'Step 2 of 5 - Edit object information'} = 'Корак 2 од 5 - Уреди информације о објекту';
    $Self->{Translation}->{'Step 3 of 5 - Edit format information'} = 'Корак 3 од 5 - Уреди информације о формату';
    $Self->{Translation}->{'is required!'} = 'је обавезно!';
    $Self->{Translation}->{'Step 4 of 5 - Edit mapping information'} = 'Корак 4 од 5 - Уреди информације о мапирању';
    $Self->{Translation}->{'No map elements found.'} = 'Ниједан елемент мапе није пронађен.';
    $Self->{Translation}->{'Add Mapping Element'} = 'Додај елемент за мапирање';
    $Self->{Translation}->{'Step 5 of 5 - Edit search information'} = 'Корак 5 од 5 - Уреди информације за претрагу';
    $Self->{Translation}->{'Restrict export per search'} = 'Ограничење извоза по претрази';
    $Self->{Translation}->{'Import information'} = 'Увоз информација';
    $Self->{Translation}->{'Source File'} = 'Изворна датотека';
    $Self->{Translation}->{'Import summary for'} = 'Резиме увоза за';
    $Self->{Translation}->{'Records'} = 'Записи';
    $Self->{Translation}->{'Success'} = 'Успешно';
    $Self->{Translation}->{'Failed'} = 'Неуспешно';
    $Self->{Translation}->{'Duplicate names'} = 'Дупликат имена';
    $Self->{Translation}->{'Last processed line number of import file'} = 'Број последње обрађене линије увезене датотеке';
    $Self->{Translation}->{'Ok'} = 'У реду';

    # Perl Module: Kernel/Modules/AdminImportExport.pm
    $Self->{Translation}->{'No object backend found!'} = 'Није пронађен позадински модул објекта!';
    $Self->{Translation}->{'No format backend found!'} = 'Није пронађен позадински модул формата!';
    $Self->{Translation}->{'Template not found!'} = 'Шаблон није пронађен!';
    $Self->{Translation}->{'Can\'t insert/update template!'} = 'Шаблон се не може унети/ажурирати!';
    $Self->{Translation}->{'Needed TemplateID!'} = 'Потребан ИД шаблона!';
    $Self->{Translation}->{'Error occurred. Import impossible! See Syslog for details.'} = 'Догодила се грешка. Увоз је немогућ! За детаље погледајте у системски дневник.';
    $Self->{Translation}->{'Error occurred. Export impossible! See Syslog for details.'} = 'Догодила се грешка. Извоз је немогућ! За детаље погледајте у системски дневник.';
    $Self->{Translation}->{'Template List'} = 'Листа шаблона';
    $Self->{Translation}->{'number'} = 'број';
    $Self->{Translation}->{'number bigger than zero'} = 'број већи од нула';
    $Self->{Translation}->{'integer'} = 'цео број';
    $Self->{Translation}->{'integer bigger than zero'} = 'цео број већи од нула';
    $Self->{Translation}->{'Element required, please insert data'} = 'Неопходан елемент, молимо унесите податке';
    $Self->{Translation}->{'Invalid data, please insert a valid %s'} = 'Неисправни подаци, унесете важећи %s';
    $Self->{Translation}->{'Format not found!'} = 'Формат није пронађен!';

    # Perl Module: Kernel/System/ImportExport/FormatBackend/CSV.pm
    $Self->{Translation}->{'Column Separator'} = 'Сепаратор колона';
    $Self->{Translation}->{'Tabulator (TAB)'} = 'Табулатор (TAB)';
    $Self->{Translation}->{'Semicolon (;)'} = 'Тачка и запета (;)';
    $Self->{Translation}->{'Colon (:)'} = 'Двотачка (:)';
    $Self->{Translation}->{'Dot (.)'} = 'Тачка (.)';
    $Self->{Translation}->{'Comma (,)'} = 'Зарез (,)';
    $Self->{Translation}->{'Charset'} = 'Карактерсет';
    $Self->{Translation}->{'Include Column Headers'} = 'Укључи наслове колона';
    $Self->{Translation}->{'Column'} = 'Колона';

    # SysConfig
    $Self->{Translation}->{'Format backend module registration for the import/export module.'} =
        'Формат регистрације „backend” модула за увоз/извоз модул.';
    $Self->{Translation}->{'Import and export object information.'} = 'Информације о увозу и извозу објеката';
    $Self->{Translation}->{'Import/Export'} = 'Увоз/Извоз';

}

1;

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnNyX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFBQUltcG9ydEV4cG9ydAogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIG1hcHBpbmcgdGVtcGxhdGUnfSA9ICdPbmdlemEga2llbGV6byBjaGEga3V3ZWthIHJhbWFuaSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAnU2V0aSB5YSBoZXJ1ZmknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sb24gKDopJ30gPSAnQ29sb24gKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ1NhZnV3aW1hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdLaXRlbmdhbmlzaGkgc2FmdSB3aW1hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICdOdWt0YSAoLiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ0FsYW1hIHlhIG51a3RhIG1rYXRvICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJvIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICdBbWJhdGlzaGEgdmljaHdhIHZ5YSBoYWJhcmkgdnlhIHNhZnV3aW1hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ0xldGEgbXVodGFzYXJpIGt3YSBhamlsaSB5YSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnRlZCByZWNvcmRzJ30gPSAnS3VtYnVrdW1idSB6aWxpem9sZXR3YSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFeHBvcnRlZCByZWNvcmRzJ30gPSAnS3VtYnVrdW1idSB6aWxpem9oYW1pc2h3YSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnS3VtYnVrdW1idSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTa2lwcGVkJ30gPSAnUnVrd2EnOwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdVc2ltYW1pemkgd2Ega3VsZXRhL0hhbWlzaGEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ1RlbmdlbmV6YSBraW9sZXpvIGt1bGV0YSBuYSBrdWhhbWlzaGEgdGFhcmlmYSB6YSBraXBlbmdlZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdBbnphIGt1bGV0YSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdBbnphIGt1aGFtaXNoYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDEgb2YgNSAtIEVkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnSmluYSBsaW5haGl0YWppa2EhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICdLaXBlbmdlZSBraW5haGl0YWppa2EhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICdVbWJpem8gdW5haGl0YWppa2EnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAyIG9mIDUgLSBFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDMgb2YgNSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpcyByZXF1aXJlZCEnfSA9ICdJbmFoaXRhamlrYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDQgb2YgNSAtIEVkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnSGFrdW5hIGVsZW1lbnRpIHppbGl6b3BhdGlrYW5hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICdPbmdlemEgZWxlbWVudGkgemEga3V3ZWthIHJhbWFuaSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDUgb2YgNSAtIEVkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAnWnVpYSB1aGFtaXNoYWppIGt3YSBraWxhIHV0YWZ1dGFqaSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICdMZXRhIHRhYXJpZmEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU291cmNlIEZpbGUnfSA9ICdGYWlsaSBsYSBjaGFuem8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJ01hZmFuaWtpbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICdTaGluZHdhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ0ppbmEgbGltZWppcnVkaWE6JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnTmFtYmEgeWEgbXN0YXJpIHVuYW9rYXRpa2EgbWNoYWthdG8gd2EgbXdpc2hvIHdhIGZhaWxpIGxpbGlsb2xldHdhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnU2F3YSc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXInfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnbnVtYmVyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaW50ZWdlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyIGJpZ2dlciB0aGFuIHplcm8nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWxlbWVudCByZXF1aXJlZCwgcGxlYXNlIGluc2VydCBkYXRhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ludmFsaWQgZGF0YSwgcGxlYXNlIGluc2VydCBhIHZhbGlkICVzJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBub3QgZm91bmQhJ30gPSAnJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnVW1iaXphIHVzYWppbGkgd2EgbW9kdWxpIHdhIG1hemluZ2lyYSB5YSBueXVtYSBrd2EgYWppbGkgeWEgbW9kdWxpIHlhIGt1bGV0YS9rdWhhbWlzaGEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ0xldGEgbmEgaGFtaXNoYSB0YWFyaWZhIHphIGtpcGVuZ2VlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydC9FeHBvcnQnfSA9ICdMZXRhL0hhbWlzaGEnOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnN2X0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdIYW50ZXJpbmcgYXYgSW1wb3J0L0V4cG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAnU2thcGEgZW4gbWFsbCBmw7ZyIGF0dCBpbXBvcnRlcmEgb2NoIGV4cG9ydGVyYSBvYmpla3RpbmZvcm1hdGlvbi4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAnU3RhcnRhIGltcG9ydCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdTdGFydGEgZXhwb3J0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMSBvZiA1IC0gRWRpdCBjb21tb24gaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmFtZSBpcyByZXF1aXJlZCEnfSA9ICdOYW1uIGtyw6R2cyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ09iamVrdCBrcsOkdnMhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICdGb3JtYXQga3LDpHZzISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDIgb2YgNSAtIEVkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMyBvZiA1IC0gRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAna3LDpHZzISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDQgb2YgNSAtIEVkaXQgbWFwcGluZyBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA1IG9mIDUgLSBFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgaW5mb3JtYXRpb24nfSA9ICdJbXBvcnRpbmZvcm1hdGlvbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ0vDpGxsZmlsJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ0ltcG9ydHN1bW1lcmluZyBmw7ZyJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1JlY29yZHMnfSA9ICdQb3N0ZXInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJ0x5Y2thZCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICdNaXNzbHlja2FkJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ1VwcHJlcGEgbmFtbic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJ09rJzsKCiAgICAjIFBlcmwgTW9kdWxlOiBLZXJuZWwvTW9kdWxlcy9BZG1pbkltcG9ydEV4cG9ydC5wbQogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gb2JqZWN0IGJhY2tlbmQgZm91bmQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIGZvcm1hdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBub3QgZm91bmQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NhblwndCBpbnNlcnQvdXBkYXRlIHRlbXBsYXRlISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOZWVkZWQgVGVtcGxhdGVJRCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEltcG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFcnJvciBvY2N1cnJlZC4gRXhwb3J0IGltcG9zc2libGUhIFNlZSBTeXNsb2cgZm9yIGRldGFpbHMuJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RlbXBsYXRlIExpc3QnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnbnVtYmVyJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXInfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaW50ZWdlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VsZW1lbnQgcmVxdWlyZWQsIHBsZWFzZSBpbnNlcnQgZGF0YSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbnZhbGlkIGRhdGEsIHBsZWFzZSBpbnNlcnQgYSB2YWxpZCAlcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgbm90IGZvdW5kISd9ID0gJyc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL1N5c3RlbS9JbXBvcnRFeHBvcnQvRm9ybWF0QmFja2VuZC9DU1YucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdLb2x1bW4tYXZza2lsamFyZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU2VtaWNvbG9uICg7KSd9ID0gJ1NlbWlrb2xvbiAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sb24gKDopJ30gPSAnS29sb24gKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICdQdW5rdCAoLiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29tbWEgKCwpJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdUZWNrZW5rb2RuaW5nJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnS29sdW1uJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdJbXBvcnRlcmEgb2NoIGV4cG9ydGVyYSBvYmpla3RpbmZvcm1hdGlvbi4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ0ltcG9ydC9FeHBvcnQnOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnN3X0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdVc2ltYW1pemkgd2Ega3VsZXRhL0hhbWlzaGEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ1RlbmdlbmV6YSBraW9sZXpvIGt1bGV0YSBuYSBrdWhhbWlzaGEgdGFhcmlmYSB6YSBraXBlbmdlZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdBbnphIGt1bGV0YSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICdBbnphIGt1aGFtaXNoYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDEgb2YgNSAtIEVkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnSmluYSBsaW5haGl0YWppa2EhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICdLaXBlbmdlZSBraW5haGl0YWppa2EhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICdVbWJpem8gdW5haGl0YWppa2EnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAyIG9mIDUgLSBFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDMgb2YgNSAtIEVkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJ0luYWhpdGFqaWthJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNCBvZiA1IC0gRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICdIYWt1bmEgZWxlbWVudGkgemlsaXpvcGF0aWthbmEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ09uZ2V6YSBlbGVtZW50aSB6YSBrdXdla2EgcmFtYW5pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNSBvZiA1IC0gRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICdadWlhIHVoYW1pc2hhamkga3dhIGtpbGEgdXRhZnV0YWppJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ0xldGEgdGFhcmlmYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ0ZhaWxpIGxhIGNoYW56byc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgc3VtbWFyeSBmb3InfSA9ICdMZXRhIG11aHRhc2FyaSBrd2EgYWppbGkgeWEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ0t1bWJ1a3VtYnUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3VjY2Vzcyd9ID0gJ01hZmFuaWtpbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGYWlsZWQnfSA9ICdTaGluZHdhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ0ppbmEgbGltZWppcnVkaWE6JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0xhc3QgcHJvY2Vzc2VkIGxpbmUgbnVtYmVyIG9mIGltcG9ydCBmaWxlJ30gPSAnTmFtYmEgeWEgbXN0YXJpIHVuYW9rYXRpa2EgbWNoYWthdG8gd2EgbXdpc2hvIHdhIGZhaWxpIGxpbGlsb2xldHdhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09rJ30gPSAnU2F3YSc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW52YWxpZCBkYXRhLCBwbGVhc2UgaW5zZXJ0IGEgdmFsaWQgJXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IG5vdCBmb3VuZCEnfSA9ICcnOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9TeXN0ZW0vSW1wb3J0RXhwb3J0L0Zvcm1hdEJhY2tlbmQvQ1NWLnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAnS2l0ZW5nYW5pc2hpIHNhZnUgd2ltYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdUYWJvIChUQUIpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICdBbGFtYSB5YSBudWt0YSBta2F0byAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sb24gKDopJ30gPSAnQ29sb24gKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICdOdWt0YSAoLiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29tbWEgKCwpJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdTZXRpIHlhIGhlcnVmaSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnQW1iYXRpc2hhIHZpY2h3YSB2eWEgaGFiYXJpIHZ5YSBzYWZ1d2ltYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4nfSA9ICdTYWZ1d2ltYSc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ1VtYml6YSB1c2FqaWxpIHdhIG1vZHVsaSB3YSBtYXppbmdpcmEgeWEgbnl1bWEga3dhIGFqaWxpIHlhIG1vZHVsaSB5YSBrdWxldGEva3VoYW1pc2hhJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdMZXRhIG5hIGhhbWlzaGEgdGFhcmlmYSB6YSBraXBlbmdlZSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAnTGV0YS9IYW1pc2hhJzsKCn0KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnRoX1RIX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICfguIHguLLguKPguIjguLHguJTguIHguLLguKMg4LiZ4Liz4LmA4LiC4LmJ4LiyL+C4quC5iOC4h+C4reC4reC4gSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDcmVhdGUgYSB0ZW1wbGF0ZSB0byBpbXBvcnQgYW5kIGV4cG9ydCBvYmplY3QgaW5mb3JtYXRpb24uJ30gPSAn4Liq4Lij4LmJ4Liy4LiH4LmB4Lih4LmI4LmB4Lia4Lia4LmA4Lie4Li34LmI4Lit4LiZ4Liz4LmA4LiC4LmJ4Liy4LmB4Lil4Liw4Liq4LmI4LiH4Lit4Lit4LiB4LiC4LmJ4Lit4Lih4Li54Lil4Lit4Lit4Lia4LmA4LiI4LiB4LiV4LmMJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEltcG9ydCd9ID0gJ+C5gOC4o+C4tOC5iOC4oeC4meC4s+C5gOC4guC5ieC4sic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBFeHBvcnQnfSA9ICfguYDguKPguLTguYjguKHguKrguYjguIfguK3guK3guIEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAxIG9mIDUgLSBFZGl0IGNvbW1vbiBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJ+C4leC5ieC4reC4h+C4o+C4sOC4muC4uOC4iuC4t+C5iOC4rSEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2JqZWN0IGlzIHJlcXVpcmVkISd9ID0gJ+C4leC5ieC4reC4h+C4o+C4sOC4muC4uOC4reC4reC4muC5gOC4iOC4geC4leC5jCEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJ+C4leC5ieC4reC4h+C4o+C4sOC4muC4uOC4o+C4ueC4m+C5geC4muC4miEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAyIG9mIDUgLSBFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDMgb2YgNSAtIEVkaXQgZm9ybWF0IGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2lzIHJlcXVpcmVkISd9ID0gJ+C4leC5ieC4reC4h+C4o+C4sOC4muC4uCEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA0IG9mIDUgLSBFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJ+C5hOC4oeC5iOC4nuC4muC4reC4h+C4hOC5jOC4m+C4o+C4sOC4geC4reC4muC5geC4nOC4meC4l+C4teC5iCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAn4LmA4Lie4Li04LmI4Lih4Lit4LiH4LiE4LmM4Lib4Lij4Liw4LiB4Lit4Lia4LiC4Lit4LiH4LiB4Liy4Lij4LiX4Liz4LmB4Lic4LiZ4LiX4Li14LmIJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNSBvZiA1IC0gRWRpdCBzZWFyY2ggaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICfguIHguLLguKPguIjguLPguIHguLHguJTguIHguLLguKPguKrguYjguIfguK3guK3guIHguJXguLLguKHguIHguLLguKPguITguYnguJnguKvguLInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAn4LiC4LmJ4Lit4Lih4Li54Lil4LiZ4Liz4LmA4LiC4LmJ4LiyJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAn4LmB4Lir4Lil4LmI4LiH4LmE4Lif4Lil4LmMJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ+C4meC4s+C5gOC4guC5ieC4suC4muC4l+C4quC4o+C4uOC4m+C4quC4s+C4q+C4o+C4seC4mic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAn4LiB4Liy4Lij4Lia4Lix4LiZ4LiX4Li24LiBJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICfguITguKfguLLguKHguKrguLPguYDguKPguYfguIgnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAn4Lil4LmJ4Lih4LmA4Lir4Lil4LinJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ+C4iuC4t+C5iOC4reC4i+C5ieC4syc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJ+C4q+C4oeC4suC4ouC5gOC4peC4guC4guC4reC4h+C4geC4suC4o+C4m+C4o+C4sOC4oeC4p+C4peC4hOC4o+C4seC5ieC4h+C4quC4uOC4lOC4l+C5ieC4suC4ouC4guC4reC4h+C4geC4suC4o+C4meC4s+C5hOC4n+C4peC5jOC5gOC4guC5ieC4sic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJ+C5guC4reC5gOC4hCc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW52YWxpZCBkYXRhLCBwbGVhc2UgaW5zZXJ0IGEgdmFsaWQgJXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IG5vdCBmb3VuZCEnfSA9ICcnOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9TeXN0ZW0vSW1wb3J0RXhwb3J0L0Zvcm1hdEJhY2tlbmQvQ1NWLnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAn4LiV4Lix4Lin4LiE4Lix4LmI4LiZ4LiE4Lit4Lil4Lix4Lih4LiZ4LmMJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ+C4leC4seC4p+C4l+C4s+C4leC4suC4o+C4suC4hyjguYHguJfguYfguJopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NlbWljb2xvbiAoOyknfSA9ICfguYDguITguKPguLfguYjguK3guIfguKvguKHguLLguKLguK3guLHguJLguKDguLLguIQoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sb24gKDopJ30gPSAn4LmA4LiE4Lij4Li34LmI4Lit4LiH4Lir4Lih4Liy4Lii4LiX4Lin4Li04Lig4Liy4LiEICg6KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEb3QgKC4pJ30gPSAn4LiI4Li44LiUKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbW1hICgsKSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAn4LmA4LiL4LiV4LiE4Liy4LmA4Lij4LiB4LmA4LiV4Lit4Lij4LmMJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICfguKPguKfguKHguJfguLHguYnguIfguKrguYjguKfguJnguKvguLHguKfguILguK3guIfguITguK3guKXguLHguKHguJnguYwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAn4LiE4Lit4Lil4Lix4Lih4LiZ4LmMJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAn4Lij4Li54Lib4LmB4Lia4Lia4LiB4Liy4Lij4Lil4LiH4LiX4Liw4LmA4Lia4Li14Lii4LiZ4LmC4Lih4LiU4Li54LilIGJhY2tlbmQg4Liq4Liz4Lir4Lij4Lix4Lia4LmC4Lih4LiU4Li54Lil4LiB4Liy4Lij4LiZ4Liz4LmA4LiC4LmJ4LiyIC8g4Liq4LmI4LiH4Lit4Lit4LiBJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICfguIHguLLguKPguJnguLPguYDguILguYnguLLguKvguKPguLfguK3guKrguYjguIfguK3guK3guIHguILguYnguK3guKHguLnguKXguK3guK3guJrguYDguIjguIHguJXguYwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ+C4geC4suC4o+C4meC4s+C5gOC4guC5ieC4si/guKrguYjguIfguK3guK3guIEnOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnZpX1ZOX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICdRdeG6o24gbMO9IE5o4bqtcC9YdeG6pXQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ3JlYXRlIGEgdGVtcGxhdGUgdG8gaW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ1ThuqFvIG3hu5l0IG3huqt1IMSR4buDIG5o4bqtcCB2w6AgeHXhuqV0IHRow7RuZyB0aW4gxJHhu5FpIHTGsOG7o25nLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGFydCBJbXBvcnQnfSA9ICdC4bqvdCDEkeG6p3Ugbmjhuq1wIHbDoG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgRXhwb3J0J30gPSAnQuG6r3QgxJHhuqd1IHh14bqldCByYSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDEgb2YgNSAtIEVkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAnWcOqdSBj4bqndSBwaOG6o2kgY8OzIHTDqm4hJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICdZw6p1IGPhuqd1IHBo4bqjaSBjw7MgxJHhu5FpIHTGsOG7o25nISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgaXMgcmVxdWlyZWQhJ30gPSAnWcOqdSBj4bqndSBwaOG6o2kgY8OzIMSR4buLbmggZOG6oW5nISc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDIgb2YgNSAtIEVkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMyBvZiA1IC0gRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAnbMOgIGLhuq90IGJ14buZYyEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA0IG9mIDUgLSBFZGl0IG1hcHBpbmcgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gbWFwIGVsZW1lbnRzIGZvdW5kLid9ID0gJ0tow7RuZyB0w6xtIHRo4bqleSDEkeG7kWkgdMaw4bujbmcgbWFwLic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydBZGQgTWFwcGluZyBFbGVtZW50J30gPSAnVGjDqm0gxJHhu5FpIHTGsOG7o25nIG1hcHBpbmcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA1IG9mIDUgLSBFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZXN0cmljdCBleHBvcnQgcGVyIHNlYXJjaCd9ID0gJ0dp4bubaSBo4bqhbiBk4buvIGxp4buHdSB4deG6pXQgbeG7l2kgbOG6p24gdMOsbSBraeG6v20nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAnVGjDtG5nIHRpbiBuaOG6rXAgdsOgbyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ1Thu4dwIG5ndeG7k24nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IHN1bW1hcnkgZm9yJ30gPSAnVMOzbSBsxrDhu6NjIG5o4bqtcCB2w6BvIGNobyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAnQuG6o24gZ2hpJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICdUaMOgbmggY8O0bmcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAnTOG7l2knOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRHVwbGljYXRlIG5hbWVzJ30gPSAnVHLDuW5nIHTDqm4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICdEw7JuZyBjdeG7kWkgY8O5bmcgxJHGsOG7o2MgeOG7rSBsw70gdHJvbmcgdOG7h3Agbmjhuq1wIHbDoG8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICfEkOG7k25nIMO9JzsKCiAgICAjIFBlcmwgTW9kdWxlOiBLZXJuZWwvTW9kdWxlcy9BZG1pbkltcG9ydEV4cG9ydC5wbQogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTm8gb2JqZWN0IGJhY2tlbmQgZm91bmQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIGZvcm1hdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBub3QgZm91bmQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NhblwndCBpbnNlcnQvdXBkYXRlIHRlbXBsYXRlISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOZWVkZWQgVGVtcGxhdGVJRCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEltcG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFcnJvciBvY2N1cnJlZC4gRXhwb3J0IGltcG9zc2libGUhIFNlZSBTeXNsb2cgZm9yIGRldGFpbHMuJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RlbXBsYXRlIExpc3QnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnbnVtYmVyJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXInfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaW50ZWdlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0VsZW1lbnQgcmVxdWlyZWQsIHBsZWFzZSBpbnNlcnQgZGF0YSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbnZhbGlkIGRhdGEsIHBsZWFzZSBpbnNlcnQgYSB2YWxpZCAlcyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgbm90IGZvdW5kISd9ID0gJyc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL1N5c3RlbS9JbXBvcnRFeHBvcnQvRm9ybWF0QmFja2VuZC9DU1YucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbiBTZXBhcmF0b3InfSA9ICdQaMOibiBjw6FjaCBj4buZdCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUYWJ1bGF0b3IgKFRBQiknfSA9ICdC4buZIGzhuq1wIGLhuqNuZyAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTZW1pY29sb24gKDspJ30gPSAnROG6pXUgY2jhuqVtIHBo4bqpeSAoOyknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sb24gKDopJ30gPSAnROG6pXUgaGFpIGNo4bqlbSAoOiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRG90ICguKSd9ID0gJ0ThuqV1IGNo4bqlbSAoLiknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29tbWEgKCwpJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NoYXJzZXQnfSA9ICdC4buZIG3DoyBrw70gdOG7sSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAnR+G7k20gdGnDqnUgxJHhu4EgY+G7mXQnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAnQ+G7mXQnOwoKICAgICMgU3lzQ29uZmlnCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydGb3JtYXQgYmFja2VuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgaW1wb3J0L2V4cG9ydCBtb2R1bGUuJ30gPQogICAgICAgICfEkMSDbmcga8O9IG3DtC3EkXVuIMSR4buLbmggZOG6oW5nIG7hu4FuIGNobyBtw7QtxJF1biBuaOG6rXAveHXhuqV0JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICdUaMO0bmcgdGluIG5o4bqtcCB2w6AgeHXhuqV0IMSR4buRaSB0xrDhu6NuZy4nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ05o4bqtcC9YdeG6pXQnOwoKfQoKMTsK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnpoX0NOX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICflr7zlhaUv5a+85Ye6566h55CGJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICfliJvlu7rmqKHmnb/lr7zlhaXlkozlr7zlh7rlr7nosaHkv6Hmga/jgIInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAn5byA5aeL5a+85YWlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ+W8gOWni+WvvOWHuic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDEgb2YgNSAtIEVkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAn5q2l6aqkIDEvNSAtIOe8lui+kemAmueUqOS/oeaBryc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydOYW1lIGlzIHJlcXVpcmVkISd9ID0gJ+WRveWQjeaYr+W/hemcgOeahO+8gSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPYmplY3QgaXMgcmVxdWlyZWQhJ30gPSAn5a+56LGh5piv5b+F6ZyA55qE77yBJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBpcyByZXF1aXJlZCEnfSA9ICfmoLzlvI/mmK/lv4XpnIDnmoTvvIEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAyIG9mIDUgLSBFZGl0IG9iamVjdCBpbmZvcm1hdGlvbid9ID0gJ+atpemqpCAyLzUgLSDnvJbovpHnm67moIfkv6Hmga8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCAzIG9mIDUgLSBFZGl0IGZvcm1hdCBpbmZvcm1hdGlvbid9ID0gJ+atpemqpCAzLzUgLSDnvJbovpHmoLzlvI/kv6Hmga8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAn5piv5b+F6ZyA55qE77yBJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNCBvZiA1IC0gRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gPSAn5q2l6aqkIDQvNSAtIOe8lui+keaYoOWwhOS/oeaBryc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBtYXAgZWxlbWVudHMgZm91bmQuJ30gPSAn5rKh5pyJ5om+5Yiw5pig5bCE55qE5a2X5q61JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0FkZCBNYXBwaW5nIEVsZW1lbnQnfSA9ICfmt7vliqDmmKDlsITlrZfmrrUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RlcCA1IG9mIDUgLSBFZGl0IHNlYXJjaCBpbmZvcm1hdGlvbid9ID0gJ+atpemqpCA1LzUgLSDnvJbovpHmkJzntKLkv6Hmga8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVzdHJpY3QgZXhwb3J0IHBlciBzZWFyY2gnfSA9ICfmjInmkJzntKLpmZDliLblr7zlh7onOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGluZm9ybWF0aW9uJ30gPSAn5a+85YWl5L+h5oGvJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1NvdXJjZSBGaWxlJ30gPSAn5rqQ5paH5Lu2JzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBzdW1tYXJ5IGZvcid9ID0gJ+WvvOWFpeaAu+e7kyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydSZWNvcmRzJ30gPSAn6K6w5b2VJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N1Y2Nlc3MnfSA9ICfmiJDlip8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRmFpbGVkJ30gPSAn5aSx6LSlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0R1cGxpY2F0ZSBuYW1lcyd9ID0gJ+mHjeWkjeeahOWQjeensCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydMYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZSd9ID0gJ+WvvOWFpeaWh+S7tuacgOWQjuWkhOeQhueahOihjOaVsCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydPayd9ID0gJ+ehruWumic7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJ+ayoeacieaJvuWIsOWvueixoeeahOWQjuerr++8gSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICfmsqHmnInmib7liLDmoLzlvI/nmoTlkI7nq6/vvIEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJ+ayoeacieaJvuWIsOaooeadv++8gSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICfkuI3og73mj5LlhaUv5pu05paw5qih5p2/77yBJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05lZWRlZCBUZW1wbGF0ZUlEISd9ID0gJ+mcgOimgVRlbXBsYXRlSUTvvIEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEltcG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJ+WHuueOsOmUmeivr+OAguS4jeiDveWvvOWFpe+8geafpeeci1N5c2xvZ+iOt+WPluivpue7huS/oeaBr+OAgic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFcnJvciBvY2N1cnJlZC4gRXhwb3J0IGltcG9zc2libGUhIFNlZSBTeXNsb2cgZm9yIGRldGFpbHMuJ30gPSAn5Ye6546w6ZSZ6K+v44CC5LiN6IO95a+85Ye677yB5p+l55yLU3lzbG9n6I635Y+W6K+m57uG5L+h5oGv44CCJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RlbXBsYXRlIExpc3QnfSA9ICfmqKHmnb/muIXljZUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnbnVtYmVyJ30gPSAn5pWw5a2XJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlciBiaWdnZXIgdGhhbiB6ZXJvJ30gPSAn5pWw5a2X5aSn5LqOMCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyJ30gPSAn5pW05pWwJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJ+aVtOaVsOWkp+S6jjAnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRWxlbWVudCByZXF1aXJlZCwgcGxlYXNlIGluc2VydCBkYXRhJ30gPSAn6ZyA6KaB5YWD57Sg77yM6K+35o+S5YWl5pWw5o2uJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ludmFsaWQgZGF0YSwgcGxlYXNlIGluc2VydCBhIHZhbGlkICVzJ30gPSAn5peg5pWI55qE5pWw5o2u77yM6K+35o+S5YWl5LiA5Liq5pyJ5pWI55qEJXMnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IG5vdCBmb3VuZCEnfSA9ICfmsqHmnInmib7liLDmoLzlvI/vvIEnOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9TeXN0ZW0vSW1wb3J0RXhwb3J0L0Zvcm1hdEJhY2tlbmQvQ1NWLnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAn5YiX5YiG6ZqU56ymJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ1RBQumUriAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTZW1pY29sb24gKDspJ30gPSAn5YiG5Y+3ICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICflhpLlj7cgKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICflj6Xlj7cgKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbW1hICgsKSd9ID0gJ+mAl+WPt++8iCzvvIknOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ2hhcnNldCd9ID0gJ+Wtl+espumbhic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbmNsdWRlIENvbHVtbiBIZWFkZXJzJ30gPSAn5YyF5ous5YiX5qCH6aKYJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbHVtbid9ID0gJ+WIlyc7CgogICAgIyBTeXNDb25maWcKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Zvcm1hdCBiYWNrZW5kIG1vZHVsZSByZWdpc3RyYXRpb24gZm9yIHRoZSBpbXBvcnQvZXhwb3J0IG1vZHVsZS4nfSA9CiAgICAgICAgJ+S4uuWvvOWFpS/lr7zlh7rmqKHlnZfnmoTmoLzlvI/lkI7nq6/mqKHlnZfms6jlhozjgIInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0IGFuZCBleHBvcnQgb2JqZWN0IGluZm9ybWF0aW9uLid9ID0gJ+WvvOWFpeWSjOWvvOWHuuWvueixoeS/oeaBr+OAgic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0J30gPSAn5a+85YWlL+WvvOWHuic7Cgp9CgoxOwo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OnpoX1RXX0ltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgICMgVGVtcGxhdGU6IEFkbWluSW1wb3J0RXhwb3J0CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQvRXhwb3J0IE1hbmFnZW1lbnQnfSA9ICflsI7lhaUv5bCO5Ye6566h55CGJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NyZWF0ZSBhIHRlbXBsYXRlIHRvIGltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICflibXlu7rmqKHmnb/lsI7lhaXlkozlsI7lh7rlsI3osaHkv6Hmga/jgIInOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnU3RhcnQgSW1wb3J0J30gPSAn6ZaL5aeL5bCO5YWlJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0YXJ0IEV4cG9ydCd9ID0gJ+mWi+Wni+WwjuWHuic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDEgb2YgNSAtIEVkaXQgY29tbW9uIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05hbWUgaXMgcmVxdWlyZWQhJ30gPSAn5ZCN56ix5piv5b+F6ZyA55qE77yBJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J09iamVjdCBpcyByZXF1aXJlZCEnfSA9ICflsI3osaHmmK/lv4XpnIDnmoTvvIEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGlzIHJlcXVpcmVkISd9ID0gJ+agvOW8j+aYr+W/hemcgOeahCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDIgb2YgNSAtIEVkaXQgb2JqZWN0IGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgMyBvZiA1IC0gRWRpdCBmb3JtYXQgaW5mb3JtYXRpb24nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnaXMgcmVxdWlyZWQhJ30gPSAn5b+F6ZyA55qEJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1N0ZXAgNCBvZiA1IC0gRWRpdCBtYXBwaW5nIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG1hcCBlbGVtZW50cyBmb3VuZC4nfSA9ICfmspLmnInmib7liLDmmKDlsITnmoTlrZfmrrUnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQWRkIE1hcHBpbmcgRWxlbWVudCd9ID0gJ+a3u+WKoOaYoOWwhOWtl+autSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdGVwIDUgb2YgNSAtIEVkaXQgc2VhcmNoIGluZm9ybWF0aW9uJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1Jlc3RyaWN0IGV4cG9ydCBwZXIgc2VhcmNoJ30gPSAn6ZmQ5Yi25bCO5Ye65q+P5YCL5pCc5bCLJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBpbmZvcm1hdGlvbid9ID0gJ+WwjuWFpeS/oeaBryc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTb3VyY2UgRmlsZSd9ID0gJ+a6kOaWh+S7tic7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydJbXBvcnQgc3VtbWFyeSBmb3InfSA9ICflsI7lhaXnuL3ntZAnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnUmVjb3Jkcyd9ID0gJ+e0gOmMhCc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTdWNjZXNzJ30gPSAn5oiQ5YqfJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ZhaWxlZCd9ID0gJ+WkseaVlyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydEdXBsaWNhdGUgbmFtZXMnfSA9ICfph43opIfnmoTlkI3nqLEnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTGFzdCBwcm9jZXNzZWQgbGluZSBudW1iZXIgb2YgaW1wb3J0IGZpbGUnfSA9ICflsI7lhaXmlofku7bmnIDlvozomZXnkIbnmoTooYzmlbgnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnT2snfSA9ICdPayc7CgogICAgIyBQZXJsIE1vZHVsZTogS2VybmVsL01vZHVsZXMvQWRtaW5JbXBvcnRFeHBvcnQucG0KICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J05vIG9iamVjdCBiYWNrZW5kIGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydObyBmb3JtYXQgYmFja2VuZCBmb3VuZCEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnVGVtcGxhdGUgbm90IGZvdW5kISd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDYW5cJ3QgaW5zZXJ0L3VwZGF0ZSB0ZW1wbGF0ZSEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnTmVlZGVkIFRlbXBsYXRlSUQhJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0Vycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy4nfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRXJyb3Igb2NjdXJyZWQuIEV4cG9ydCBpbXBvc3NpYmxlISBTZWUgU3lzbG9nIGZvciBkZXRhaWxzLid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydUZW1wbGF0ZSBMaXN0J30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J251bWJlcid9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydudW1iZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydpbnRlZ2VyJ30gPSAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J2ludGVnZXIgYmlnZ2VyIHRoYW4gemVybyd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydFbGVtZW50IHJlcXVpcmVkLCBwbGVhc2UgaW5zZXJ0IGRhdGEnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW52YWxpZCBkYXRhLCBwbGVhc2UgaW5zZXJ0IGEgdmFsaWQgJXMnfSA9ICcnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IG5vdCBmb3VuZCEnfSA9ICcnOwoKICAgICMgUGVybCBNb2R1bGU6IEtlcm5lbC9TeXN0ZW0vSW1wb3J0RXhwb3J0L0Zvcm1hdEJhY2tlbmQvQ1NWLnBtCiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2x1bW4gU2VwYXJhdG9yJ30gPSAn5YiX5YiG6ZqU56ymJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J1RhYnVsYXRvciAoVEFCKSd9ID0gJ+WItuihqOmNtSAoVEFCKSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydTZW1pY29sb24gKDspJ30gPSAn5YiG6JmfICg7KSc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDb2xvbiAoOiknfSA9ICflhpLomZ8gKDopJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0RvdCAoLiknfSA9ICflj6XomZ8gKC4pJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0NvbW1hICgsKSd9ID0gJyc7CiAgICAkU2VsZi0+e1RyYW5zbGF0aW9ufS0+eydDaGFyc2V0J30gPSAn5a2X56ym6ZuGJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0luY2x1ZGUgQ29sdW1uIEhlYWRlcnMnfSA9ICfljIXmi6zliJfmqJnpoYwnOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnQ29sdW1uJ30gPSAn5YiXJzsKCiAgICAjIFN5c0NvbmZpZwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnRm9ybWF0IGJhY2tlbmQgbW9kdWxlIHJlZ2lzdHJhdGlvbiBmb3IgdGhlIGltcG9ydC9leHBvcnQgbW9kdWxlLid9ID0KICAgICAgICAnJzsKICAgICRTZWxmLT57VHJhbnNsYXRpb259LT57J0ltcG9ydCBhbmQgZXhwb3J0IG9iamVjdCBpbmZvcm1hdGlvbi4nfSA9ICflsI7lhaXlkozlsI7lh7rlsI3osaHkv6Hmga8nOwogICAgJFNlbGYtPntUcmFuc2xhdGlvbn0tPnsnSW1wb3J0L0V4cG9ydCd9ID0gJ+WwjuWFpS/lsI7lh7onOwoKfQoKMTsK
# --
# Copyright (C) 2001-2019 OTRS AG, https://otrs.com/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

package Kernel::Modules::AdminImportExport;

use strict;
use warnings;

use Kernel::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' );

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

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

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

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnQ6OkxheW91dENoZWNrYm94OwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06OkxvZycsCiAgICAnS2VybmVsOjpTeXN0ZW06OldlYjo6UmVxdWVzdCcsCik7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydDo6TGF5b3V0Q2hlY2tib3ggLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKQWxsIGxheW91dCBmdW5jdGlvbnMgZm9yIGNoZWNrYm94IGVsZW1lbnRzIGluIGltcG9ydC9leHBvcnQuCgo9b3ZlciA0Cgo9Y3V0Cgo9aXRlbSBuZXcoKQoKY3JlYXRlIGFuIG9iamVjdAoKICAgICRCYWNrZW5kT2JqZWN0ID0gS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydDo6TGF5b3V0Q2hlY2tib3gtPm5ldygKICAgICAgICAlUGFyYW0sCiAgICApOwoKPWN1dAoKc3ViIG5ldyB7CiAgICBteSAoICRUeXBlLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgYWxsb2NhdGUgbmV3IGhhc2ggZm9yIG9iamVjdAogICAgbXkgJFNlbGYgPSB7fTsKICAgIGJsZXNzKCAkU2VsZiwgJFR5cGUgKTsKCiAgICByZXR1cm4gJFNlbGY7Cn0KCj1pdGVtIEZvcm1JbnB1dENyZWF0ZSgpCgpjcmVhdGUgYSBpbnB1dCBzdHJpbmcKCiAgICBteSAkVmFsdWUgPSAkQmFja2VuZE9iamVjdC0+Rm9ybUlucHV0Q3JlYXRlKAogICAgICAgIEl0ZW0gICA9PiAkSXRlbVJlZiwKICAgICAgICBQcmVmaXggPT4gJ1ByZWZpeDo6JywgICMgKG9wdGlvbmFsKQogICAgICAgIFZhbHVlICA9PiAnVmFsdWUnLCAgICAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1JbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpLT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIEl0ZW0hJywKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAkUGFyYW17UHJlZml4fSB8fD0gJyc7CgogICAgbXkgJENoZWNrZWQgPSAkUGFyYW17VmFsdWV9ID8gJ2NoZWNrZWQ9ImNoZWNrZWQiJyA6ICcnOwoKICAgIHJldHVybgogICAgICAgIHFxezxpbnB1dCBpZD0iJFBhcmFte1ByZWZpeH0kUGFyYW17SXRlbX0tPntLZXl9IiB0eXBlPSJjaGVja2JveCIgbmFtZT0iJFBhcmFte1ByZWZpeH0kUGFyYW17SXRlbX0tPntLZXl9IiAkQ2hlY2tlZCAvPn07Cn0KCj1pdGVtIEZvcm1EYXRhR2V0KCkKCmdldCBmb3JtIGRhdGEKCiAgICBteSAkRm9ybURhdGEgPSAkQmFja2VuZE9iamVjdC0+Rm9ybURhdGFHZXQoCiAgICAgICAgSXRlbSAgID0+ICRJdGVtUmVmLAogICAgICAgIFByZWZpeCA9PiAnUHJlZml4OjonLCAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1EYXRhR2V0IHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgSXRlbSEnLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICRQYXJhbXtQcmVmaXh9IHx8PSAnJzsKCiAgICAjIGdldCBmb3JtIGRhdGEKICAgIG15ICRGb3JtRGF0YSA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpXZWI6OlJlcXVlc3QnKS0+R2V0UGFyYW0oCiAgICAgICAgUGFyYW0gPT4gJFBhcmFte1ByZWZpeH0gLiAkUGFyYW17SXRlbX0tPntLZXl9LAogICAgKTsKCiAgICByZXR1cm4gJEZvcm1EYXRhOwp9CgoxOwoKPWJhY2sKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKEw8aHR0cHM6Ly9vdHJzLm9yZy8+KS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0Pi4KCj1jdXQK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnQ6OkxheW91dFRUOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06OkxvZycsCik7Cgo9aGVhZDEgTkFNRQoKS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydDo6TGF5b3V0VFQgLSBsYXlvdXQgYmFja2VuZCBtb2R1bGUKCj1oZWFkMSBTWU5PUFNJUwoKQWxsIGxheW91dCBmdW5jdGlvbnMgZm9yIGRpc3BsYXkgVFQgY29kZQoKPW92ZXIgNAoKPWN1dAoKPWl0ZW0gbmV3KCkKCmNyZWF0ZSBhbiBvYmplY3QKCiAgICAkQmFja2VuZE9iamVjdCA9IEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnQ6OkxheW91dFRULT5uZXcoCiAgICAgICAgJVBhcmFtLAogICAgKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aXRlbSBGb3JtSW5wdXRDcmVhdGUoKQoKY3JlYXRlIGEgaW5wdXQgc3RyaW5nCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkZvcm1JbnB1dENyZWF0ZSgKICAgICAgICBJdGVtID0+ICRJdGVtUmVmLAogICAgKTsKCj1jdXQKCnN1YiBGb3JtSW5wdXRDcmVhdGUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgaWYgKCAhJFBhcmFte0l0ZW19ICkgewogICAgICAgICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAnTmVlZCBJdGVtIScKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICByZXR1cm4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57RGF0YX07Cn0KCj1pdGVtIEZvcm1EYXRhR2V0KCkKCmdldCBmb3JtIGRhdGEKCiAgICBteSAkRm9ybURhdGEgPSAkQmFja2VuZE9iamVjdC0+Rm9ybURhdGFHZXQoKTsKCj1jdXQKCnN1YiBGb3JtRGF0YUdldCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybjsKfQoKMTsKCj1iYWNrCgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKClRoaXMgc29mdHdhcmUgaXMgcGFydCBvZiB0aGUgT1RSUyBwcm9qZWN0IChMPGh0dHBzOi8vb3Rycy5vcmcvPikuCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIEw8aHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dD4uCgo9Y3V0Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnQ6OkxheW91dFNlbGVjdGlvbjsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCm91ciBAT2JqZWN0RGVwZW5kZW5jaWVzID0gKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpMb2cnLAogICAgJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpXZWI6OlJlcXVlc3QnLAopOwoKPWhlYWQxIE5BTUUKCktlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnQ6OkxheW91dFNlbGVjdGlvbiAtIGxheW91dCBiYWNrZW5kIG1vZHVsZQoKPWhlYWQxIFNZTk9QU0lTCgpBbGwgbGF5b3V0IGZ1bmN0aW9ucyBmb3Igc2VsZWN0aW9uIGVsZW1lbnRzCgo9b3ZlciA0Cgo9Y3V0Cgo9aXRlbSBuZXcoKQoKY3JlYXRlIGFuIG9iamVjdAoKICAgICRCYWNrZW5kT2JqZWN0ID0gS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydDo6TGF5b3V0U2VsZWN0aW9uLT5uZXcoCiAgICAgICAgJVBhcmFtLAogICAgKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgcmV0dXJuICRTZWxmOwp9Cgo9aXRlbSBGb3JtSW5wdXRDcmVhdGUoKQoKY3JlYXRlIGEgaW5wdXQgc3RyaW5nCgogICAgbXkgJFZhbHVlID0gJEJhY2tlbmRPYmplY3QtPkZvcm1JbnB1dENyZWF0ZSgKICAgICAgICBJdGVtICAgPT4gJEl0ZW1SZWYsCiAgICAgICAgUHJlZml4ID0+ICdQcmVmaXg6OicsICAjIChvcHRpb25hbCkKICAgICAgICBWYWx1ZSAgPT4gJ1ZhbHVlJywgICAgICMgKG9wdGlvbmFsKQogICAgICAgIENsYXNzICA9PiAnTW9kZXJuaXplJyAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1JbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpLT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIEl0ZW0hJwogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgICMgc2V0IGRlZmF1bHQgdmFsdWUKICAgICRQYXJhbXtQcmVmaXh9IHx8PSAnJzsKICAgICRQYXJhbXtWYWx1ZX0gIHx8PSAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntWYWx1ZURlZmF1bHR9OwoKICAgIGlmICggJFBhcmFte1ZhbHVlfSAmJiAkUGFyYW17VmFsdWV9ID1+IG17ICMjIyMjIH14bXMgKSB7CiAgICAgICAgbXkgQFZhbHVlcyA9IHNwbGl0ICcjIyMjIycsICRQYXJhbXtWYWx1ZX07CiAgICAgICAgJFBhcmFte1ZhbHVlfSA9IFxAVmFsdWVzOwogICAgfQoKICAgICMgZ2VuZXJhdGUgb3B0aW9uIHN0cmluZwogICAgbXkgJFN0cmluZyA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnKS0+QnVpbGRTZWxlY3Rpb24oCiAgICAgICAgSUQgICAgICAgICAgID0+ICRQYXJhbXtQcmVmaXh9IC4gJFBhcmFte0l0ZW19LT57S2V5fSwKICAgICAgICBDbGFzcyAgICAgICAgPT4gJFBhcmFte0NsYXNzfSwKICAgICAgICBOYW1lICAgICAgICAgPT4gJFBhcmFte1ByZWZpeH0gLiAkUGFyYW17SXRlbX0tPntLZXl9LAogICAgICAgIERhdGEgICAgICAgICA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntEYXRhfSB8fCB7fSwKICAgICAgICBTZWxlY3RlZElEICAgPT4gJFBhcmFte1ZhbHVlfSwKICAgICAgICBUcmFuc2xhdGlvbiAgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VHJhbnNsYXRpb259LAogICAgICAgIFRyZWVWaWV3ICAgICA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntUcmVlVmlld30gfHwgMCwKICAgICAgICBQb3NzaWJsZU5vbmUgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57UG9zc2libGVOb25lfSwKICAgICAgICBNdWx0aXBsZSAgICAgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57TXVsdGlwbGV9LAogICAgICAgIFNpemUgICAgICAgICA9PiAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntTaXplfSwKICAgICk7CgogICAgcmV0dXJuICRTdHJpbmc7Cn0KCj1pdGVtIEZvcm1EYXRhR2V0KCkKCmdldCBmb3JtIGRhdGEKCiAgICBteSAkRm9ybURhdGEgPSAkQmFja2VuZE9iamVjdC0+Rm9ybURhdGFHZXQoCiAgICAgICAgSXRlbSAgID0+ICRJdGVtUmVmLAogICAgICAgIFByZWZpeCA9PiAnUHJlZml4OjonLCAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1EYXRhR2V0IHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgSXRlbSEnCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgJFBhcmFte1ByZWZpeH0gfHw9ICcnOwoKICAgICMgZ2V0IGZvcm0gZGF0YQogICAgbXkgQEZvcm1EYXRhcyA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpXZWI6OlJlcXVlc3QnKS0+R2V0QXJyYXkoCiAgICAgICAgUGFyYW0gPT4gJFBhcmFte1ByZWZpeH0gLiAkUGFyYW17SXRlbX0tPntLZXl9LAogICAgKTsKCiAgICBteSAkRm9ybURhdGEgPSBqb2luICcjIyMjIycsIEBGb3JtRGF0YXM7CgogICAgcmV0dXJuICRGb3JtRGF0YSBpZiAkRm9ybURhdGE7CiAgICByZXR1cm4gJEZvcm1EYXRhIGlmICEkUGFyYW17SXRlbX0tPntJbnB1dH0tPntSZXF1aXJlZH07CgogICAgIyBzZXQgaW52YWxpZCBwYXJhbQogICAgJFBhcmFte0l0ZW19LT57Rm9ybX0tPntJbnZhbGlkfSA9IDE7CgogICAgcmV0dXJuICRGb3JtRGF0YTsKfQoKMTsKCj1iYWNrCgo9aGVhZDEgVEVSTVMgQU5EIENPTkRJVElPTlMKClRoaXMgc29mdHdhcmUgaXMgcGFydCBvZiB0aGUgT1RSUyBwcm9qZWN0IChMPGh0dHBzOi8vb3Rycy5vcmcvPikuCgpUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQp0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIEw8aHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dD4uCgo9Y3V0Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpJbXBvcnRFeHBvcnQ6OkxheW91dFRleHQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7CgpvdXIgQE9iamVjdERlcGVuZGVuY2llcyA9ICgKICAgICdLZXJuZWw6OlN5c3RlbTo6TG9nJywKICAgICdLZXJuZWw6OlN5c3RlbTo6V2ViOjpSZXF1ZXN0JywKICAgICdLZXJuZWw6Ok91dHB1dDo6SFRNTDo6TGF5b3V0JywKKTsKCj1oZWFkMSBOQU1FCgpLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0OjpMYXlvdXRUZXh0IC0gbGF5b3V0IGJhY2tlbmQgbW9kdWxlCgo9aGVhZDEgU1lOT1BTSVMKCkFsbCBsYXlvdXQgZnVuY3Rpb25zIGZvciB0ZXh0IGVsZW1lbnRzCgo9b3ZlciA0Cgo9Y3V0Cgo9aXRlbSBuZXcoKQoKY3JlYXRlIGFuIG9iamVjdAoKICAgICRCYWNrZW5kT2JqZWN0ID0gS2VybmVsOjpPdXRwdXQ6OkhUTUw6OkltcG9ydEV4cG9ydDo6TGF5b3V0VGV4dC0+bmV3KAogICAgICAgICVQYXJhbSwKICAgICk7Cgo9Y3V0CgpzdWIgbmV3IHsKICAgIG15ICggJFR5cGUsICVQYXJhbSApID0gQF87CgogICAgIyBhbGxvY2F0ZSBuZXcgaGFzaCBmb3Igb2JqZWN0CiAgICBteSAkU2VsZiA9IHt9OwogICAgYmxlc3MoICRTZWxmLCAkVHlwZSApOwoKICAgIHJldHVybiAkU2VsZjsKfQoKPWl0ZW0gRm9ybUlucHV0Q3JlYXRlKCkKCmNyZWF0ZSBhIGlucHV0IHN0cmluZwoKICAgIG15ICRWYWx1ZSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtSW5wdXRDcmVhdGUoCiAgICAgICAgSXRlbSAgICAgPT4gJEl0ZW1SZWYsCiAgICAgICAgUHJlZml4ICAgPT4gJ1ByZWZpeDo6JywgICMgKG9wdGlvbmFsKQogICAgICAgIFZhbHVlICAgID0+ICdWYWx1ZScsICAgICAjIChvcHRpb25hbCkKICAgICAgICBSZWFkb25seSA9PiAxLCAgICAgICAgICAgIyAob3B0aW9uYWwpCiAgICApOwoKPWN1dAoKc3ViIEZvcm1JbnB1dENyZWF0ZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgY2hlY2sgbmVlZGVkIHN0dWZmCiAgICBpZiAoICEkUGFyYW17SXRlbX0gKSB7CiAgICAgICAgJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpLT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIEl0ZW0hJywKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAkUGFyYW17UHJlZml4fSB8fD0gJyc7CgogICAgbXkgJFZhbHVlID0gJFBhcmFte1ZhbHVlfSB8fCAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntWYWx1ZURlZmF1bHR9OwogICAgbXkgJFNpemUgPSAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntTaXplfSB8fCA0MDsKICAgIG15ICRTaXplQ2xhc3M7CiAgICBpZiAoICRTaXplIDwgMTUgKSB7CiAgICAgICAgJFNpemVDbGFzcyA9ICdXMTBwYyc7CiAgICB9CiAgICBlbHNpZiAoICRTaXplIDwgMzUgKSB7CiAgICAgICAgJFNpemVDbGFzcyA9ICdXMzNwYyc7CiAgICB9CiAgICBlbHNpZiAoICRTaXplIDwgNTAgKSB7CiAgICAgICAgJFNpemVDbGFzcyA9ICdXNTBwYyc7CiAgICB9CiAgICBlbHNlIHsKICAgICAgICAkU2l6ZUNsYXNzID0gJ1c3NXBjJzsKICAgIH0KCiAgICAjIHByZXBhcmUgZGF0YQogICAgbXkgJElEICAgPSAoICRQYXJhbXtQcmVmaXh9IHx8ICcnICkgLiAoICRQYXJhbXtJdGVtfS0+e0tleX0gKTsKICAgIG15ICROYW1lID0gKCAkUGFyYW17UHJlZml4fSB8fCAnJyApIC4gKCAkUGFyYW17TmFtZX0gfHwgJElEICk7CiAgICBteSAkQ2xhc3MgPSAoICRTaXplQ2xhc3MgfHwgJycgKSAuICggJFBhcmFte0NsYXNzfSB8fCAnJyApOwoKICAgIG15ICRTdHJpbmcgPSAiPGlucHV0IGlkPVwiJElEXCIgdHlwZT1cInRleHRcIiBuYW1lPVwiJE5hbWVcIiBjbGFzcz1cIiRDbGFzc1wiICI7CgogICAgaWYgKCRWYWx1ZSkgewoKICAgICAgICAjIGdldCBsYXlvdXQgb2JqZWN0CiAgICAgICAgbXkgJExheW91dE9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQnKTsKCiAgICAgICAgIyB0cmFuc2xhdGUKICAgICAgICBpZiAoICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1RyYW5zbGF0aW9ufSApIHsKICAgICAgICAgICAgJFZhbHVlID0gJExheW91dE9iamVjdC0+e0xhbmd1YWdlT2JqZWN0fS0+VHJhbnNsYXRlKCRWYWx1ZSk7CiAgICAgICAgfQoKICAgICAgICAjIHRyYW5zZm9ybSBhc2NpaSB0byBodG1sCiAgICAgICAgJFZhbHVlID0gJExheW91dE9iamVjdC0+QXNjaWkySHRtbCgKICAgICAgICAgICAgVGV4dCAgICAgICAgICAgPT4gJFZhbHVlLAogICAgICAgICAgICBIVE1MUmVzdWx0TW9kZSA9PiAxLAogICAgICAgICk7CgogICAgICAgICRTdHJpbmcgLj0gInZhbHVlPVwiJFZhbHVlXCIgIjsKICAgIH0KCiAgICAjIGFkZCBtYXhpbXVtIGxlbmd0aAogICAgaWYgKCAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntNYXhMZW5ndGh9ICkgewogICAgICAgICRTdHJpbmcgLj0gIm1heGxlbmd0aD1cIiRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e01heExlbmd0aH1cIiAiOwogICAgfQoKICAgICMgYWRkIHJlYWRvbmx5CiAgICBpZiAoICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1JlYWRvbmx5fSApIHsKICAgICAgICAkU3RyaW5nIC49ICdyZWFkb25seT0icmVhZG9ubHkiICc7CiAgICB9CgogICAgJFN0cmluZyAuPSAiLz4gIjsKCiAgICByZXR1cm4gJFN0cmluZzsKfQoKPWl0ZW0gRm9ybURhdGFHZXQoKQoKZ2V0IGZvcm0gZGF0YQoKICAgIG15ICRGb3JtRGF0YSA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtRGF0YUdldCgKICAgICAgICBJdGVtICAgPT4gJEl0ZW1SZWYsCiAgICAgICAgUHJlZml4ID0+ICdQcmVmaXg6OicsICAjIChvcHRpb25hbCkKICAgICk7Cgo9Y3V0CgpzdWIgRm9ybURhdGFHZXQgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgaWYgKCAhJFBhcmFte0l0ZW19ICkgewogICAgICAgICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAnTmVlZCBJdGVtIScKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAkUGFyYW17UHJlZml4fSB8fD0gJyc7CgogICAgIyBnZXQgZm9ybSBkYXRhCiAgICBteSAkRm9ybURhdGEgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6V2ViOjpSZXF1ZXN0JyktPkdldFBhcmFtKAogICAgICAgIFBhcmFtID0+ICRQYXJhbXtQcmVmaXh9IC4gJFBhcmFte0l0ZW19LT57S2V5fSwKICAgICk7CgogICAgIyByZWdleCBjaGVjawogICAgaWYgKCAkUGFyYW17SXRlbX0tPntJbnB1dH0tPntSZWdleH0gJiYgJEZvcm1EYXRhICF+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1JlZ2V4fSApIHsKCiAgICAgICAgJFBhcmFte0l0ZW19LT57Rm9ybX0tPntJbnZhbGlkfSA9IDE7CiAgICAgICAgcmV0dXJuICRGb3JtRGF0YTsKICAgIH0KCiAgICByZXR1cm4gJEZvcm1EYXRhIGlmICRGb3JtRGF0YTsKICAgIHJldHVybiAkRm9ybURhdGEgaWYgISRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1JlcXVpcmVkfTsKCiAgICAjIHNldCBpbnZhbGlkIHBhcmFtCiAgICAkUGFyYW17SXRlbX0tPntGb3JtfS0+e0ludmFsaWR9ID0gMTsKCiAgICByZXR1cm4gJEZvcm1EYXRhOwp9CgoxOwoKPWJhY2sKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKEw8aHR0cHM6Ly9vdHJzLm9yZy8+KS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0Pi4KCj1jdXQK
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6T3V0cHV0OjpIVE1MOjpMYXlvdXQ6OkltcG9ydEV4cG9ydDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCm91ciAkT2JqZWN0TWFuYWdlckRpc2FibGVkID0gMTsKCj1vdmVyCgo9aXRlbSBJbXBvcnRFeHBvcnRGb3JtSW5wdXRDcmVhdGUoKQoKcmV0dXJucyBhIGlucHV0IGZpZWxkIGh0bWwgc3RyaW5nCgogICAgbXkgJFN0cmluZyA9ICRMYXlvdXRPYmplY3QtPkltcG9ydEV4cG9ydEZvcm1JbnB1dENyZWF0ZSgKICAgICAgICBJdGVtICA9PiAkSXRlbVJlZiwKICAgICAgICBWYWx1ZSA9PiAnVmFsdWUnLCAgICMgKG9wdGlvbmFsKQogICAgKTsKCj1jdXQKCnN1YiBJbXBvcnRFeHBvcnRGb3JtSW5wdXRDcmVhdGUgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGNoZWNrIG5lZWRlZCBzdHVmZgogICAgaWYgKCAhJFBhcmFte0l0ZW19ICkgewogICAgICAgICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpMb2cnKS0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAnTmVlZCBJdGVtIScKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICAjIGxvYWQgYmFja2VuZAogICAgbXkgJEJhY2tlbmRPYmplY3QgPSAkU2VsZi0+X0ltcG9ydEV4cG9ydExvYWRMYXlvdXRCYWNrZW5kKAogICAgICAgIFR5cGUgPT4gJFBhcmFte0l0ZW19LT57SW5wdXR9LT57VHlwZX0sCiAgICApOwoKICAgIHJldHVybiAnJyBpZiAhJEJhY2tlbmRPYmplY3Q7CgogICAgIyBsb29rdXAgaXRlbSB2YWx1ZQogICAgbXkgJFN0cmluZyA9ICRCYWNrZW5kT2JqZWN0LT5Gb3JtSW5wdXRDcmVhdGUoJVBhcmFtKTsKCiAgICByZXR1cm4gJFN0cmluZzsKfQoKPWl0ZW0gSW1wb3J0RXhwb3J0Rm9ybURhdGFHZXQoKQoKcmV0dXJucyB0aGUgdmFsdWVzIGZyb20gdGhlIGh0bWwgZm9ybSBhcyBoYXNoIHJlZmVyZW5jZQoKICAgIG15ICRGb3JtRGF0YSA9ICRMYXlvdXRPYmplY3QtPkltcG9ydEV4cG9ydEZvcm1EYXRhR2V0KAogICAgICAgIEl0ZW0gPT4gJEl0ZW1SZWYsCiAgICApOwoKPWN1dAoKc3ViIEltcG9ydEV4cG9ydEZvcm1EYXRhR2V0IHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGVjayBuZWVkZWQgc3R1ZmYKICAgIGlmICggISRQYXJhbXtJdGVtfSApIHsKICAgICAgICAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6TG9nJyktPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gJ05lZWQgSXRlbSEnCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBsb2FkIGJhY2tlbmQKICAgIG15ICRCYWNrZW5kT2JqZWN0ID0gJFNlbGYtPl9JbXBvcnRFeHBvcnRMb2FkTGF5b3V0QmFja2VuZCgKICAgICAgICBUeXBlID0+ICRQYXJhbXtJdGVtfS0+e0lucHV0fS0+e1R5cGV9LAogICAgKTsKCiAgICByZXR1cm4gaWYgISRCYWNrZW5kT2JqZWN0OwoKICAgICMgZ2V0IGZvcm0gZGF0YQogICAgbXkgJEZvcm1EYXRhID0gJEJhY2tlbmRPYmplY3QtPkZvcm1EYXRhR2V0KCVQYXJhbSk7CgogICAgcmV0dXJuICRGb3JtRGF0YTsKfQoKPWl0ZW0gX0ltcG9ydEV4cG9ydExvYWRMYXlvdXRCYWNrZW5kKCkKCnRvIGxvYWQgYSBpbXBvcnQvZXhwb3J0IGxheW91dCBiYWNrZW5kIG1vZHVsZQoKICAgIG15ICRCYWNrZW5kID0gJExheW91dE9iamVjdC0+X0ltcG9ydEV4cG9ydExvYWRMYXlvdXRCYWNrZW5kKAogICAgICAgIFR5cGUgPT4gJ1NlbGVjdGlvbicsCiAgICApOwoKQW4gaW5zdGFuY2Ugb2YgdGhlIGxvYWRlZCBiYWNrZW5kIG1vZHVsZSBpcyByZXR1cm5lZC4KCj1jdXQKCnN1YiBfSW1wb3J0RXhwb3J0TG9hZExheW91dEJhY2tlbmQgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGdldCBsb2cgb2JqZWN0CiAgICBteSAkTG9nT2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkxvZycpOwoKICAgIGlmICggISRQYXJhbXtUeXBlfSApIHsKICAgICAgICAkTG9nT2JqZWN0LT5Mb2coCiAgICAgICAgICAgIFByaW9yaXR5ID0+ICdlcnJvcicsCiAgICAgICAgICAgIE1lc3NhZ2UgID0+ICdOZWVkIFR5cGUhJywKICAgICAgICApOwogICAgICAgIHJldHVybjsKICAgIH0KCiAgICBteSAkR2VuZXJpY01vZHVsZSA9ICJLZXJuZWw6Ok91dHB1dDo6SFRNTDo6SW1wb3J0RXhwb3J0OjpMYXlvdXQkUGFyYW17VHlwZX0iOwoKICAgICMgbG9hZCB0aGUgYmFja2VuZCBtb2R1bGUKICAgIGlmICggISRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpNYWluJyktPlJlcXVpcmUoJEdlbmVyaWNNb2R1bGUpICkgewogICAgICAgICRMb2dPYmplY3QtPkxvZygKICAgICAgICAgICAgUHJpb3JpdHkgPT4gJ2Vycm9yJywKICAgICAgICAgICAgTWVzc2FnZSAgPT4gIkNhbid0IGxvYWQgYmFja2VuZCBtb2R1bGUgJFBhcmFte1R5cGV9ISIsCiAgICAgICAgKTsKICAgICAgICByZXR1cm47CiAgICB9CgogICAgIyBjcmVhdGUgbmV3IGluc3RhbmNlCiAgICBteSAkQmFja2VuZE9iamVjdCA9ICRHZW5lcmljTW9kdWxlLT5uZXcoCiAgICAgICAgJXskU2VsZn0sCiAgICAgICAgJVBhcmFtLAogICAgICAgIExheW91dE9iamVjdCA9PiAkU2VsZiwKICAgICk7CgogICAgaWYgKCAhJEJhY2tlbmRPYmplY3QgKSB7CiAgICAgICAgJExvZ09iamVjdC0+TG9nKAogICAgICAgICAgICBQcmlvcml0eSA9PiAnZXJyb3InLAogICAgICAgICAgICBNZXNzYWdlICA9PiAiQ2FuJ3QgY3JlYXRlIGEgbmV3IGluc3RhbmNlIG9mIGJhY2tlbmQgbW9kdWxlICRQYXJhbXtUeXBlfSEiLAogICAgICAgICk7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIHJldHVybiAkQmFja2VuZE9iamVjdDsKfQoKMTsKCj1iYWNrCg==
# --
# Copyright (C) 2001-2019 OTRS AG, https://otrs.com/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

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

    <div class="SidebarColumn">

[% RenderBlockStart("ActionList") %]
        <div class="WidgetSimple">
            <div class="Header">
                <h2>[% Translate("Actions") | html %]</h2>
            </div>
            <div class="Content">
                <ul class="ActionList">
[% RenderBlockStart("ActionOverview") %]
                    <li>
                        <a href="[% Env("Baselink") %]Action=[% Env("Action") %]" class="CallForAction 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>
[% 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>
[% 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>

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

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

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

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

    }).trigger('change');

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

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

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

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

[% RenderBlockEnd("TemplateEdit4") %]

[% RenderBlockStart("TemplateEdit5") %]
            <div class="Header">
                <h2>[% Translate("Step 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>
[% 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>
[% 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>

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

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

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

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

    });

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

    });

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

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

</div>

[% RenderBlockEnd("Overview") %]

[% RenderBlockStart("ImportResult") %]
<div class="MainBox AriaRoleMain">
    <div class="W50pc SpacingTopLarge SpacingBottomLarge CenterBox">
        <div class="WidgetSimple">
            <div class="Header">
                <h2>[% Translate("Import summary for %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") %]

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

package Kernel::System::ImportExport;

use strict;
use warnings;

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

=head1 NAME

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

=head1 SYNOPSIS

All import and export functions.

=head1 PUBLIC INTERFACE

=over 4

=cut

=item new()

create an object

    use Kernel::System::ObjectManager;
    local $Kernel::OM = Kernel::System::ObjectManager->new();
    my $ImportExportObject = $Kernel::OM->Get('Kernel::System::ImportExport');

=cut

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

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

    return $Self;
}

=item TemplateList()

return a list of templates as array reference

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

=cut

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

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

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

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

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

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

=item TemplateGet()

get a import export template

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

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

=cut

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

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

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

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

=item TemplateAdd()

add a new import/export template

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

=cut

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

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

=item TemplateUpdate()

update a existing import/export template

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

=cut

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

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

=item TemplateDelete()

delete existing import/export templates

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

    or

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

=cut

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

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

=item ObjectList()

return a list of available objects as hash reference

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

=cut

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

    # get config
    my $ModuleList = $Kernel::OM->Get('Kernel::Config')->Get('ImportExport::ObjectBackendRegistration');

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

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

    return $ObjectList;
}

=item ObjectAttributesGet()

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

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

=cut

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

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

=item ObjectDataGet()

get the object data from a template

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

=cut

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

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

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

=item ObjectDataSave()

save the object data of a template

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

=cut

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

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

=item ObjectDataDelete()

delete the existing object data of a template

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

    or

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

=cut

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

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

=item FormatList()

return a list of available formats as hash reference

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

=cut

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

    # get config
    my $ModuleList = $Kernel::OM->Get('Kernel::Config')->Get('ImportExport::FormatBackendRegistration');

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

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

    return $FormatList;
}

=item FormatAttributesGet()

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

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

=cut

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

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

=item FormatDataGet()

get the format data from a template

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

=cut

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

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

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

=item FormatDataSave()

save the format data of a template

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

=cut

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

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

=item FormatDataDelete()

delete the existing format data of a template

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

    or

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

=cut

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

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

=item MappingList()

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

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

=cut

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

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

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

=item MappingAdd()

add a new mapping data row

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

=cut

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

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

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

=item MappingDelete()

delete existing mapping data rows

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

    or

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

=cut

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

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

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

=item MappingUp()

move an mapping data row up

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

=cut

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

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

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

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

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

=item MappingDown()

move an mapping data row down

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

=cut

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

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

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

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

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

=item MappingPositionRebuild()

rebuild the positions of a mapping list

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

=cut

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

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

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

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

    return 1;
}

=item MappingObjectAttributesGet()

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

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

=cut

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

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

=item MappingObjectDataDelete()

delete the existing object data of a mapping

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

    or

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

=cut

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

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

=item MappingObjectDataSave()

save the object data of a mapping

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

=cut

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

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

=item MappingObjectDataGet()

get the object data of a mapping

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

=cut

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

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

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

=item MappingFormatAttributesGet()

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

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

=cut

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

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

=item MappingFormatDataDelete()

delete the existing format data of a mapping

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

    or

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

=cut

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

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

=item MappingFormatDataSave()

save the format data of a mapping

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

=cut

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

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

=item MappingFormatDataGet()

get the format data of a mapping

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

=cut

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

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

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

=item SearchAttributesGet()

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

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

=cut

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

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

=item SearchDataGet()

get the search data from a template

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

=cut

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

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

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

=item SearchDataSave()

save the search data of a template

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

=cut

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

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

=item SearchDataDelete()

delete the existing search data of a template

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

    or

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

=cut

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

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

=item Export()

export function

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

returns something like

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

=cut

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

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

=item Import()

import function

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

=cut

sub Import {
    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 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}",
        );
    }

    return \%Result;
}

1;

=back

=head1 TERMS AND CONDITIONS

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

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

=cut

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

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

use strict;
use warnings;

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 SYNOPSIS

All functions to import and export a csv format

=over 4

=cut

=item new()

create an object

    use Kernel::System::ObjectManager;
    local $Kernel::OM = Kernel::System::ObjectManager->new();
    my $ImportExportCSVBackendObject = $Kernel::OM->Get('Kernel::System::ImportExport::FormatBackend::CSV');

=cut

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

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

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

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

    return $Self;
}

=item FormatAttributesGet()

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

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

=cut

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

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

    my $Attributes = [
        {
            Key   => 'ColumnSeparator',
            Name  => 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;
}

=item MappingFormatAttributesGet()

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

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

=cut

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

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

    my $Attributes = [
        {
            Key   => 'Column',
            Name  => Translatable('Column'),
            Input => {
                Type     => 'TT',
                Data     => '',
                Required => 0,
            },
        },
    ];

    return $Attributes;
}

=item ImportDataGet()

get import data as 2D-array reference

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

=cut

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

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

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

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

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

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

    # get charset
    my $Charset = $FormatData->{Charset} ||= '';

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

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

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

    # close the in memory file handle
    close $FH;

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

    return \@ImportData;
}

=item ExportDataSave()

export one row of the export data

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

=cut

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

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

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

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

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

    # get charset
    my $Charset = $FormatData->{Charset} ||= '';

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

=back

=head1 TERMS AND CONDITIONS

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

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

=cut

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6U3lzdGVtOjpDb25zb2xlOjpDb21tYW5kOjpBZG1pbjo6SVRTTTo6SW1wb3J0RXhwb3J0OjpJbXBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgYmFzZSBxdyhLZXJuZWw6OlN5c3RlbTo6Q29uc29sZTo6QmFzZUNvbW1hbmQpOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06Ok1haW4nLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpJbXBvcnRFeHBvcnQnLAopOwoKc3ViIENvbmZpZ3VyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICRTZWxmLT5EZXNjcmlwdGlvbignVGhlIHRvb2wgZm9yIGltcG9ydGluZyBjb25maWcgaXRlbXMnKTsKICAgICRTZWxmLT5BZGRPcHRpb24oCiAgICAgICAgTmFtZSAgICAgICAgPT4gJ3RlbXBsYXRlLW51bWJlcicsCiAgICAgICAgRGVzY3JpcHRpb24gPT4gIlNwZWNpZnkgYSB0ZW1wbGF0ZSBudW1iZXIgdG8gYmUgaW1wb2VydGVkLiIsCiAgICAgICAgUmVxdWlyZWQgICAgPT4gMSwKICAgICAgICBIYXNWYWx1ZSAgICA9PiAxLAogICAgICAgIFZhbHVlUmVnZXggID0+IHFyL1xkKy9zbXgsCiAgICApOwogICAgJFNlbGYtPkFkZEFyZ3VtZW50KAogICAgICAgIE5hbWUgICAgICAgID0+ICdzb3VyY2UnLAogICAgICAgIERlc2NyaXB0aW9uID0+ICJTcGVjaWZ5IHRoZSBwYXRoIHRvIHRoZSBmaWxlIHdoaWNoIGNvbnRhaW5pbmcgdGhlIGNvbmZpZyBpdGVtIGRhdGEgZm9yIGltcG9ydGluZy4iLAogICAgICAgIFJlcXVpcmVkICAgID0+IDEsCiAgICAgICAgVmFsdWVSZWdleCAgPT4gcXIvLiovc214LAogICAgKTsKCiAgICByZXR1cm47Cn0KCnN1YiBQcmVSdW4gewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICBteSAkU291cmNlUGF0aCA9ICRTZWxmLT5HZXRBcmd1bWVudCgnc291cmNlJyk7CiAgICBpZiAoICRTb3VyY2VQYXRoICYmICEtciAkU291cmNlUGF0aCApIHsKICAgICAgICBkaWUgIkZpbGUgJFNvdXJjZVBhdGggZG9lcyBub3QgZXhpc3QsIGNhbiBub3QgYmUgcmVhZC5cbiI7CiAgICB9CgogICAgcmV0dXJuOwp9CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJFRlbXBsYXRlSUQgPSAkU2VsZi0+R2V0T3B0aW9uKCd0ZW1wbGF0ZS1udW1iZXInKTsKCiAgICAjIGdldCB0ZW1wbGF0ZSBkYXRhCiAgICBteSAkVGVtcGxhdGVEYXRhID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkltcG9ydEV4cG9ydCcpLT5UZW1wbGF0ZUdldCgKICAgICAgICBUZW1wbGF0ZUlEID0+ICRUZW1wbGF0ZUlELAogICAgICAgIFVzZXJJRCAgICAgPT4gMSwKICAgICk7CgogICAgaWYgKCAhJFRlbXBsYXRlRGF0YS0+e1RlbXBsYXRlSUR9ICkgewogICAgICAgICRTZWxmLT5QcmludEVycm9yKCJUZW1wbGF0ZSAkVGVtcGxhdGVJRCBub3QgZm91bmQhLlxuIik7CiAgICAgICAgJFNlbGYtPlByaW50RXJyb3IoIkV4cG9ydCBhYm9ydGVkLi5cbiIpOwogICAgICAgIHJldHVybiAkU2VsZi0+RXhpdENvZGVFcnJvcigpOwogICAgfQoKICAgICRTZWxmLT5QcmludCgiPHllbGxvdz5JbXBvcnRpbmcgY29uZmlnIGl0ZW1zLi4uPC95ZWxsb3c+XG4iKTsKICAgICRTZWxmLT5QcmludCggIjx5ZWxsb3c+IiAuICggJz0nIHggNjkgKSAuICI8L3llbGxvdz5cbiIgKTsKCiAgICBteSAkU291cmNlQ29udGVudDsKICAgIG15ICRTb3VyY2VGaWxlID0gJFNlbGYtPkdldEFyZ3VtZW50KCdzb3VyY2UnKTsKCiAgICBpZiAoJFNvdXJjZUZpbGUpIHsKCiAgICAgICAgJFNlbGYtPlByaW50KCI8eWVsbG93PlJlYWQgRmlsZSAkU291cmNlRmlsZS48L3llbGxvdz5cbiIpOwoKICAgICAgICAjIHJlYWQgc291cmNlIGZpbGUKICAgICAgICAkU291cmNlQ29udGVudCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpNYWluJyktPkZpbGVSZWFkKAogICAgICAgICAgICBMb2NhdGlvbiA9PiAkU291cmNlRmlsZSwKICAgICAgICAgICAgUmVzdWx0ICAgPT4gJ1NDQUxBUicsCiAgICAgICAgICAgIE1vZGUgICAgID0+ICdiaW5tb2RlJywKICAgICAgICApOwoKICAgICAgICBpZiAoICEkU291cmNlQ29udGVudCApIHsKICAgICAgICAgICAgJFNlbGYtPlByaW50RXJyb3IoIkNhbid0IHJlYWQgZmlsZSAkU291cmNlRmlsZS5cbkltcG9ydCBhYm9ydGVkLlxuIikgaWYgISRTb3VyY2VDb250ZW50OwogICAgICAgICAgICByZXR1cm4gJFNlbGYtPkV4aXRDb2RlRXJyb3IoKTsKICAgICAgICB9CgogICAgfQoKICAgICMgaW1wb3J0IGRhdGEKICAgIG15ICRSZXN1bHQgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SW1wb3J0RXhwb3J0JyktPkltcG9ydCgKICAgICAgICBUZW1wbGF0ZUlEICAgID0+ICRUZW1wbGF0ZUlELAogICAgICAgIFNvdXJjZUNvbnRlbnQgPT4gJFNvdXJjZUNvbnRlbnQsCiAgICAgICAgVXNlcklEICAgICAgICA9PiAxLAogICAgKTsKCiAgICBpZiAoICEkUmVzdWx0ICkgewogICAgICAgICRTZWxmLT5QcmludEVycm9yKCJcbkVycm9yIG9jY3VycmVkLiBJbXBvcnQgaW1wb3NzaWJsZSEgU2VlIHRoZSBPVFJTIGxvZyBmb3IgZGV0YWlscy5cbiIpOwogICAgICAgIHJldHVybiAkU2VsZi0+RXhpdENvZGVFcnJvcigpOwogICAgfQoKICAgICMgcHJpbnQgcmVzdWx0CiAgICAkU2VsZi0+UHJpbnQoIlxuPGdyZWVuPkltcG9ydCBvZiAkUmVzdWx0LT57Q291bnRlcn0gJFJlc3VsdC0+e09iamVjdH0gcmVjb3Jkczo8L2dyZWVuPlxuIik7CiAgICAkU2VsZi0+UHJpbnQoICI8Z3JlZW4+IiAuICggJy0nIHggNjkgKSAuICI8L2dyZWVuPlxuIiApOwogICAgJFNlbGYtPlByaW50KCI8Z3JlZW4+U3VjY2VzczogJFJlc3VsdC0+e1N1Y2Nlc3N9IHN1Y2NlZWRlZDwvZ3JlZW4+XG4iKTsKICAgIGlmICggJFJlc3VsdC0+e0ZhaWxlZH0gKSB7CiAgICAgICAgJFNlbGYtPlByaW50RXJyb3IoIiRSZXN1bHQtPntGYWlsZWR9IGZhaWxlZC5cbiIpOwogICAgfQogICAgZWxzZSB7CiAgICAgICAgJFNlbGYtPlByaW50KCI8Z3JlZW4+RXJyb3I6ICRSZXN1bHQtPntGYWlsZWR9IGZhaWxlZC48L2dyZWVuPlxuIik7CiAgICB9CgogICAgZm9yIG15ICRSZXRDb2RlICggc29ydCBrZXlzICV7ICRSZXN1bHQtPntSZXRDb2RlfSB9ICkgewogICAgICAgIG15ICRDb3VudCA9ICRSZXN1bHQtPntSZXRDb2RlfS0+eyRSZXRDb2RlfSB8fCAwOwogICAgICAgICRTZWxmLT5QcmludCgiPGdyZWVuPkltcG9ydCBvZiAkUmVzdWx0LT57Q291bnRlcn0gJFJlc3VsdC0+e09iamVjdH0gcmVjb3JkczogJENvdW50ICRSZXRDb2RlPC9ncmVlbj5cbiIpOwogICAgfQogICAgaWYgKCAkUmVzdWx0LT57RmFpbGVkfSApIHsKICAgICAgICAkU2VsZi0+UHJpbnQoIjxncmVlbj5MYXN0IHByb2Nlc3NlZCBsaW5lIG51bWJlciBvZiBpbXBvcnQgZmlsZTogJFJlc3VsdC0+e0NvdW50ZXJ9PC9ncmVlbj5cbiIpOwogICAgfQoKICAgICRTZWxmLT5QcmludCgiPGdyZWVuPkltcG9ydCBjb21wbGV0ZS48L2dyZWVuPlxuIik7CiAgICAkU2VsZi0+UHJpbnQoICI8Z3JlZW4+IiAuICggJy0nIHggNjkgKSAuICI8L2dyZWVuPlxuIiApOwogICAgJFNlbGYtPlByaW50KCI8Z3JlZW4+RG9uZS48L2dyZWVuPlxuIik7CgogICAgcmV0dXJuICRTZWxmLT5FeGl0Q29kZU9rKCk7Cn0KCjE7Cgo9YmFjawoKPWhlYWQxIFRFUk1TIEFORCBDT05ESVRJT05TCgpUaGlzIHNvZnR3YXJlIGlzIHBhcnQgb2YgdGhlIE9UUlMgcHJvamVjdCAoTDxodHRwczovL290cnMub3JnLz4pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQ+LgoKPWN1dAo=
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6U3lzdGVtOjpDb25zb2xlOjpDb21tYW5kOjpBZG1pbjo6SVRTTTo6SW1wb3J0RXhwb3J0OjpFeHBvcnQ7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgYmFzZSBxdyhLZXJuZWw6OlN5c3RlbTo6Q29uc29sZTo6QmFzZUNvbW1hbmQpOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06Ok1haW4nLAogICAgJ0tlcm5lbDo6U3lzdGVtOjpJbXBvcnRFeHBvcnQnLAopOwoKc3ViIENvbmZpZ3VyZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICRTZWxmLT5EZXNjcmlwdGlvbignVGhlIHRvb2wgZm9yIGV4cG9ydGluZyBjb25maWcgaXRlbXMnKTsKICAgICRTZWxmLT5BZGRPcHRpb24oCiAgICAgICAgTmFtZSAgICAgICAgPT4gJ3RlbXBsYXRlLW51bWJlcicsCiAgICAgICAgRGVzY3JpcHRpb24gPT4gIlNwZWNpZnkgYSB0ZW1wbGF0ZSBudW1iZXIgdG8gYmUgZXhwb3J0ZWQuIiwKICAgICAgICBSZXF1aXJlZCAgICA9PiAxLAogICAgICAgIEhhc1ZhbHVlICAgID0+IDEsCiAgICAgICAgVmFsdWVSZWdleCAgPT4gcXIvXGQrL3NteCwKICAgICk7CiAgICAkU2VsZi0+QWRkQXJndW1lbnQoCiAgICAgICAgTmFtZSAgICAgICAgPT4gJ2Rlc3RpbmF0aW9uJywKICAgICAgICBEZXNjcmlwdGlvbiA9PiAiU3BlY2lmeSB0aGUgcGF0aCB0byBhIGZpbGUgd2hlcmUgY29uZmlnIGl0ZW0gZGF0YSBzaG91bGQgYmUgZXhwb3J0ZWQuIiwKICAgICAgICBSZXF1aXJlZCAgICA9PiAxLAogICAgICAgIFZhbHVlUmVnZXggID0+IHFyLy4qL3NteCwKICAgICk7CgogICAgcmV0dXJuOwp9CgpzdWIgUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgbXkgJFRlbXBsYXRlSUQgPSAkU2VsZi0+R2V0T3B0aW9uKCd0ZW1wbGF0ZS1udW1iZXInKTsKCiAgICAjIGdldCB0ZW1wbGF0ZSBkYXRhCiAgICBteSAkVGVtcGxhdGVEYXRhID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkltcG9ydEV4cG9ydCcpLT5UZW1wbGF0ZUdldCgKICAgICAgICBUZW1wbGF0ZUlEID0+ICRUZW1wbGF0ZUlELAogICAgICAgIFVzZXJJRCAgICAgPT4gMSwKICAgICk7CgogICAgaWYgKCAhJFRlbXBsYXRlRGF0YS0+e1RlbXBsYXRlSUR9ICkgewogICAgICAgICRTZWxmLT5QcmludEVycm9yKCJUZW1wbGF0ZSAkVGVtcGxhdGVJRCBub3QgZm91bmQhLlxuIik7CiAgICAgICAgJFNlbGYtPlByaW50RXJyb3IoIkV4cG9ydCBhYm9ydGVkLi5cbiIpOwogICAgICAgIHJldHVybiAkU2VsZi0+RXhpdENvZGVFcnJvcigpOwogICAgfQoKICAgICRTZWxmLT5QcmludCgiPHllbGxvdz5FeHBvcnRpbmcgY29uZmlnIGl0ZW1zLi4uPC95ZWxsb3c+XG4iKTsKICAgICRTZWxmLT5QcmludCggIjx5ZWxsb3c+IiAuICggJz0nIHggNjkgKSAuICI8L3llbGxvdz5cbiIgKTsKCiAgICAjIGV4cG9ydCBkYXRhCiAgICBteSAkUmVzdWx0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkltcG9ydEV4cG9ydCcpLT5FeHBvcnQoCiAgICAgICAgVGVtcGxhdGVJRCA9PiAkVGVtcGxhdGVJRCwKICAgICAgICBVc2VySUQgICAgID0+IDEsCiAgICApOwoKICAgIGlmICggISRSZXN1bHQgKSB7CiAgICAgICAgJFNlbGYtPlByaW50RXJyb3IoIkVycm9yIG9jY3VycmVkLiBFeHBvcnQgaW1wb3NzaWJsZSEgU2VlIFN5c2xvZyBmb3IgZGV0YWlscy5cbiIpOwogICAgICAgIHJldHVybiAkU2VsZi0+RXhpdENvZGVFcnJvcigpOwogICAgfQoKICAgICRTZWxmLT5QcmludCggIjxncmVlbj4iIC4gKCAnLScgeCA2OSApIC4gIjwvZ3JlZW4+XG4iICk7CiAgICAkU2VsZi0+UHJpbnQoIjxncmVlbj5TdWNjZXNzOiAkUmVzdWx0LT57U3VjY2Vzc30gc3VjY2VlZGVkPC9ncmVlbj5cbiIpOwogICAgaWYgKCAkUmVzdWx0LT57RmFpbGVkfSApIHsKICAgICAgICAkU2VsZi0+UHJpbnRFcnJvcigiJFJlc3VsdC0+e0ZhaWxlZH0gZmFpbGVkLlxuIik7CiAgICB9CiAgICBlbHNlIHsKICAgICAgICAkU2VsZi0+UHJpbnQoIjxncmVlbj5FcnJvcjogJFJlc3VsdC0+e0ZhaWxlZH0gZmFpbGVkLjwvZ3JlZW4+XG4iKTsKICAgIH0KCiAgICBteSAkRGVzdGluYXRpb25GaWxlID0gJFNlbGYtPkdldEFyZ3VtZW50KCdkZXN0aW5hdGlvbicpOwoKICAgIGlmICgkRGVzdGluYXRpb25GaWxlKSB7CgogICAgICAgIG15ICRGaWxlQ29udGVudCA9IGpvaW4gIlxuIiwgQHsgJFJlc3VsdC0+e0Rlc3RpbmF0aW9uQ29udGVudH0gfTsKCiAgICAgICAgIyBzYXZlIGRlc3RpbmF0aW9uIGNvbnRlbnQgdG8gZmlsZQogICAgICAgIG15ICRTdWNjZXNzID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06Ok1haW4nKS0+RmlsZVdyaXRlKAogICAgICAgICAgICBMb2NhdGlvbiA9PiAkRGVzdGluYXRpb25GaWxlLAogICAgICAgICAgICBDb250ZW50ICA9PiBcJEZpbGVDb250ZW50LAogICAgICAgICk7CgogICAgICAgIGlmICggISRTdWNjZXNzICkgewogICAgICAgICAgICAkU2VsZi0+UHJpbnRFcnJvcigiQ2FuJ3Qgd3JpdGUgZmlsZSAkRGVzdGluYXRpb25GaWxlLlxuRXhwb3J0IGFib3J0ZWQuXG4iKTsKICAgICAgICAgICAgcmV0dXJuICRTZWxmLT5FeGl0Q29kZUVycm9yKCk7CiAgICAgICAgfQoKICAgICAgICAkU2VsZi0+UHJpbnQoIjxncmVlbj5GaWxlICREZXN0aW5hdGlvbkZpbGUgc2F2ZWQuPC9ncmVlbj5cbiIpOwoKICAgIH0KCiAgICAkU2VsZi0+UHJpbnQoIjxncmVlbj5FeHBvcnQgY29tcGxldGUuPC9ncmVlbj5cbiIpOwogICAgJFNlbGYtPlByaW50KCAiPGdyZWVuPiIgLiAoICctJyB4IDY5ICkgLiAiPC9ncmVlbj5cbiIgKTsKICAgICRTZWxmLT5QcmludCgiPGdyZWVuPkRvbmUuPC9ncmVlbj5cbiIpOwoKICAgIHJldHVybiAkU2VsZi0+RXhpdENvZGVPaygpOwp9CgoxOwoKPWJhY2sKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBzb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKEw8aHR0cHM6Ly9vdHJzLm9yZy8+KS4KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgTDxodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0Pi4KCj1jdXQK
# --
# Copyright (C) 2001-2019 OTRS AG, https://otrs.com/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

# 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-2019 OTRS AG, https://otrs.com/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

use Data::Dumper;

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

IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgojIyBubyBjcml0aWMgKE1vZHVsZXM6OlJlcXVpcmVFeHBsaWNpdFBhY2thZ2UpCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7Cgp1c2UgdmFycyAocXcoJFNlbGYpKTsKdXNlIEZpbGU6OlBhdGggcXcocm10cmVlKTsKCiMgZ2V0IG5lZWRlZCBvYmplY3RzCm15ICRDb21tYW5kT2JqZWN0ICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6Q29uc29sZTo6Q29tbWFuZDo6QWRtaW46OklUU006OkltcG9ydEV4cG9ydDo6SW1wb3J0Jyk7Cm15ICRJbXBvcnRFeHBvcnRPYmplY3QgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SW1wb3J0RXhwb3J0Jyk7CgojIGdldCBoZWxwZXIgb2JqZWN0CiRLZXJuZWw6Ok9NLT5PYmplY3RQYXJhbUFkZCgKICAgICdLZXJuZWw6OlN5c3RlbTo6VW5pdFRlc3Q6OkhlbHBlcicgPT4gewogICAgICAgIFJlc3RvcmVEYXRhYmFzZSA9PiAxLAogICAgfSwKKTsKbXkgJEhlbHBlciA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyk7CgojIHRlc3QgY29tbWFuZCB3aXRob3V0IC0tdGVtcGxhdGUtbnVtYmVyIG9wdGlvbgpteSAkRXhpdENvZGUgPSAkQ29tbWFuZE9iamVjdC0+RXhlY3V0ZSgpOwoKJFNlbGYtPklzKAogICAgJEV4aXRDb2RlLAogICAgMSwKICAgICJObyAtLXRlbXBsYXRlLW51bWJlciAgLSBleGl0IGNvZGUiLAopOwoKIyBhZGQgdGVzdCB0ZW1wbGF0ZQpteSAkVGVtcGxhdGVJRCA9ICRJbXBvcnRFeHBvcnRPYmplY3QtPlRlbXBsYXRlQWRkKAogICAgT2JqZWN0ICA9PiAnSVRTTUNvbmZpZ0l0ZW0nLAogICAgRm9ybWF0ICA9PiAnQ1NWJywKICAgIE5hbWUgICAgPT4gJ1RlbXBsYXRlJyAuICRIZWxwZXItPkdldFJhbmRvbUlEKCksCiAgICBWYWxpZElEID0+IDEsCiAgICBDb21tZW50ID0+ICdDb21tZW50JywKICAgIFVzZXJJRCAgPT4gMSwKKTsKCiRTZWxmLT5UcnVlKAogICAgJFRlbXBsYXRlSUQsCiAgICAiSW1wb3J0L0V4cG9ydCB0ZW1wbGF0ZSBpcyBjcmVhdGVkIC0gJFRlbXBsYXRlSUQiLAopOwoKIyBnZXQgJ0hhcmR3YXJlJyBjYXRhbG9nIGNsYXNzIElECm15ICRDb25maWdJdGVtRGF0YVJlZiA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpLT5JdGVtR2V0KAogICAgQ2xhc3MgPT4gJ0lUU006OkNvbmZpZ0l0ZW06OkNsYXNzJywKICAgIE5hbWUgID0+ICdIYXJkd2FyZScsCik7Cm15ICRIYXJkd2FyZUNvbmZpZ0l0ZW1JRCA9ICRDb25maWdJdGVtRGF0YVJlZi0+e0l0ZW1JRH07CgojIGdldCBvYmplY3QgZGF0YSBmb3IgdGVzdCB0ZW1wbGF0ZQpteSAlVGVtcGxhdGVSZWYgPSAoCiAgICAnQ2xhc3NJRCcgID0+ICRIYXJkd2FyZUNvbmZpZ0l0ZW1JRCwKICAgICdDb3VudE1heCcgPT4gMTAsCik7Cm15ICRTdWNjZXNzID0gJEltcG9ydEV4cG9ydE9iamVjdC0+T2JqZWN0RGF0YVNhdmUoCiAgICBUZW1wbGF0ZUlEID0+ICRUZW1wbGF0ZUlELAogICAgT2JqZWN0RGF0YSA9PiBcJVRlbXBsYXRlUmVmLAogICAgVXNlcklEICAgICA9PiAxLAopOwoKJFNlbGYtPlRydWUoCiAgICAkU3VjY2VzcywKICAgICJPYmplY3REYXRhIGZvciB0ZXN0IHRlbXBsYXRlIGlzIGFkZGVkIiwKKTsKCiMgYWRkIHRoZSBmb3JtYXQgZGF0YSBvZiB0aGUgdGVzdCB0ZW1wbGF0ZQpteSAlRm9ybWF0RGF0YSA9ICgKICAgIENoYXJzZXQgICAgICAgICAgICAgID0+ICdVVEYtOCcsCiAgICBDb2x1bW5TZXBhcmF0b3IgICAgICA9PiAnQ29tbWEnLAogICAgSW5jbHVkZUNvbHVtbkhlYWRlcnMgPT4gMSwKKTsKJFN1Y2Nlc3MgPSAkSW1wb3J0RXhwb3J0T2JqZWN0LT5Gb3JtYXREYXRhU2F2ZSgKICAgIFRlbXBsYXRlSUQgPT4gJFRlbXBsYXRlSUQsCiAgICBGb3JtYXREYXRhID0+IFwlRm9ybWF0RGF0YSwKICAgIFVzZXJJRCAgICAgPT4gMSwKKTsKCiRTZWxmLT5UcnVlKAogICAgJFN1Y2Nlc3MsCiAgICAiRm9ybWF0RGF0YSBmb3IgdGVzdCB0ZW1wbGF0ZSBpcyBhZGRlZCIsCik7CgojIHNhdmUgdGhlIHNlYXJjaCBkYXRhIG9mIGEgdGVtcGxhdGUKbXkgJVNlYXJjaERhdGEgPSAoCiAgICBOYW1lID0+ICdUZXN0Q29uZmlnSXRlbSonLAopOwokU3VjY2VzcyA9ICRJbXBvcnRFeHBvcnRPYmplY3QtPlNlYXJjaERhdGFTYXZlKAogICAgVGVtcGxhdGVJRCA9PiAkVGVtcGxhdGVJRCwKICAgIFNlYXJjaERhdGEgPT4gXCVTZWFyY2hEYXRhLAogICAgVXNlcklEICAgICA9PiAxLAopOwoKIyBhZGQgbWFwcGluZyBkYXRhIGZvciB0ZXN0IHRlbXBsYXRlCmZvciBteSAkT2JqZWN0RGF0YVZhbHVlIChxdyggTmFtZSBEZXBsU3RhdGUgSW5jaVN0YXRlICkpIHsKCiAgICBteSAkTWFwcGluZ0lEID0gJEltcG9ydEV4cG9ydE9iamVjdC0+TWFwcGluZ0FkZCgKICAgICAgICBUZW1wbGF0ZUlEID0+ICRUZW1wbGF0ZUlELAogICAgICAgIFVzZXJJRCAgICAgPT4gMSwKICAgICk7CgogICAgbXkgJU1hcHBpbmdPYmplY3REYXRhID0gKCBLZXkgPT4gJE9iamVjdERhdGFWYWx1ZSApOwogICAgbXkgJFN1Y2Nlc3MgICAgICAgICAgID0gJEltcG9ydEV4cG9ydE9iamVjdC0+TWFwcGluZ09iamVjdERhdGFTYXZlKAogICAgICAgIE1hcHBpbmdJRCAgICAgICAgID0+ICRNYXBwaW5nSUQsCiAgICAgICAgTWFwcGluZ09iamVjdERhdGEgPT4gXCVNYXBwaW5nT2JqZWN0RGF0YSwKICAgICAgICBVc2VySUQgICAgICAgICAgICA9PiAxLAogICAgKTsKCiAgICAkU2VsZi0+VHJ1ZSgKICAgICAgICAkU3VjY2VzcywKICAgICAgICAiT2JqZWN0RGF0YSBmb3IgdGVzdCB0ZW1wbGF0ZSBpcyBtYXBwZWQgLSAkT2JqZWN0RGF0YVZhbHVlIiwKICAgICk7Cn0KCiMgbWFrZSBkaXJlY3RvcnkgZm9yIGV4cG9ydCBmaWxlCm15ICRTb3VyY2VQYXRoCiAgICA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6Q29uZmlnJyktPkdldCgnSG9tZScpIC4gIi9zY3JpcHRzL3Rlc3Qvc2FtcGxlL0ltcG9ydEV4cG9ydC9UZW1wbGF0ZUV4cG9ydC5jc3YiOwoKIyB0ZXN0IGNvbW1hbmQgd2l0aCB3cm9uZyB0ZW1wbGF0ZSBudW1iZXIKJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoICctLXRlbXBsYXRlLW51bWJlcicsICRIZWxwZXItPkdldFJhbmRvbUlEKCksICRTb3VyY2VQYXRoIC4gJ1RlbXBsYXRlRXhwb3J0LmNzdicgKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDEsCiAgICAiQ29tbWFuZCB3aXRoIHdyb25nIHRlbXBsYXRlIG51bWJlciAtIGV4aXQgY29kZSIsCik7CgojIHRlc3QgY29tbWFuZCB3aXRob3V0IFNvdXJjZSBhcmd1bWVudAokRXhpdENvZGUgPSAkQ29tbWFuZE9iamVjdC0+RXhlY3V0ZSggJy0tdGVtcGxhdGUtbnVtYmVyJywgJFRlbXBsYXRlSUQgKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDEsCiAgICAiTm8gU291cmNlIGFyZ3VtZW50IC0gZXhpdCBjb2RlIiwKKTsKCiMgdGVzdCBjb21tYW5kIHdpdGggLS10ZW1wbGF0ZS1udW1iZXIgb3B0aW9uIGFuZCBTb3VyY2UgYXJndW1lbnQKJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoICctLXRlbXBsYXRlLW51bWJlcicsICRUZW1wbGF0ZUlELCAkU291cmNlUGF0aCApOwoKJFNlbGYtPklzKAogICAgJEV4aXRDb2RlLAogICAgMCwKICAgICJPcHRpb24gLSAtLXRlbXBsYXRlLW51bWJlciBvcHRpb24gYW5kIFNvdXJjZSBhcmd1bWVudCIsCik7CgojIGdldCBjb25maWcgaXRlbSBJRHMKbXkgJENvbmZpZ0l0ZW1JRHMgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nKS0+Q29uZmlnSXRlbVNlYXJjaEV4dGVuZGVkKAogICAgTmFtZSA9PiAnVGVzdENvbmZpZ0l0ZW0qJwopOwpteSAkTnVtQ29uZmlnSXRlbUltcG9ydGVkID0gc2NhbGFyIEB7JENvbmZpZ0l0ZW1JRHN9OwoKIyBjaGVjayBpZiB0aGUgY29uZmlnIGl0ZW1zIGFyZSBpbXBvcnRlZAokU2VsZi0+VHJ1ZSgKICAgICROdW1Db25maWdJdGVtSW1wb3J0ZWQsCiAgICAiVGhlcmUgYXJlICROdW1Db25maWdJdGVtSW1wb3J0ZWQgaW1wb3J0ZWQgY29uZmlnIGl0ZW1zIiwKKTsKCiMgY2xlYW51cCBpcyBkb25lIGJ5IFJlc3RvcmVEYXRhYmFzZS4KCjE7Cg==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgojIyBubyBjcml0aWMgKE1vZHVsZXM6OlJlcXVpcmVFeHBsaWNpdFBhY2thZ2UpCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKdXNlIHV0Zjg7Cgp1c2UgdmFycyAocXcoJFNlbGYpKTsKdXNlIEZpbGU6OlBhdGggcXcobWtwYXRoIHJtdHJlZSk7CgojIGdldCBuZWVkZWQgb2JqZWN0cwpteSAkQ29tbWFuZE9iamVjdCAgICAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6Q29uc29sZTo6Q29tbWFuZDo6QWRtaW46OklUU006OkltcG9ydEV4cG9ydDo6RXhwb3J0Jyk7Cm15ICRHZW5lcmFsQ2F0YWxvZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpHZW5lcmFsQ2F0YWxvZycpOwpteSAkQ29uZmlnSXRlbU9iamVjdCAgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OlN5c3RlbTo6SVRTTUNvbmZpZ0l0ZW0nKTsKCiMgZ2V0IGhlbHBlciBvYmplY3QKJEtlcm5lbDo6T00tPk9iamVjdFBhcmFtQWRkKAogICAgJ0tlcm5lbDo6U3lzdGVtOjpVbml0VGVzdDo6SGVscGVyJyA9PiB7CiAgICAgICAgUmVzdG9yZURhdGFiYXNlID0+IDEsCiAgICB9LAopOwpteSAkSGVscGVyID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OlVuaXRUZXN0OjpIZWxwZXInKTsKCiMgdGVzdCBjb21tYW5kIHdpdGhvdXQgLS10ZW1wbGF0ZS1udW1iZXIgb3B0aW9uCm15ICRFeGl0Q29kZSA9ICRDb21tYW5kT2JqZWN0LT5FeGVjdXRlKCk7CgokU2VsZi0+SXMoCiAgICAkRXhpdENvZGUsCiAgICAxLAogICAgIk5vIC0tdGVtcGxhdGUtbnVtYmVyICAtIGV4aXQgY29kZSIsCik7CgojIGdldCAnSGFyZHdhcmUnIGNhdGFsb2cgY2xhc3MgSUQKbXkgJENvbmZpZ0l0ZW1EYXRhUmVmID0gJEdlbmVyYWxDYXRhbG9nT2JqZWN0LT5JdGVtR2V0KAogICAgQ2xhc3MgPT4gJ0lUU006OkNvbmZpZ0l0ZW06OkNsYXNzJywKICAgIE5hbWUgID0+ICdIYXJkd2FyZScsCik7Cm15ICRIYXJkd2FyZUNvbmZpZ0l0ZW1JRCA9ICRDb25maWdJdGVtRGF0YVJlZi0+e0l0ZW1JRH07CgojIGdldCAnUHJvZHVjdGlvbicgZGVwbG95bWVudCBzdGF0ZSBJRHMKbXkgJFByb2R1Y3Rpb25EZXBsU3RhdGVEYXRhUmVmID0gJEdlbmVyYWxDYXRhbG9nT2JqZWN0LT5JdGVtR2V0KAogICAgQ2xhc3MgPT4gJ0lUU006OkNvbmZpZ0l0ZW06OkRlcGxveW1lbnRTdGF0ZScsCiAgICBOYW1lICA9PiAnUHJvZHVjdGlvbicsCik7Cm15ICRQcm9kdWN0aW9uRGVwbFN0YXRlSUQgPSAkUHJvZHVjdGlvbkRlcGxTdGF0ZURhdGFSZWYtPntJdGVtSUR9OwoKbXkgQENvbmZpZ0l0ZW1JRHM7CgojIGFkZCB0ZXN0IGNvbmZpZyBpdGVtcwpmb3IgKCAxIC4uIDEwICkgewoKICAgICMgY3JlYXRlIENvbmZpZ0l0ZW0gbnVtYmVyCiAgICBteSAkQ29uZmlnSXRlbU51bWJlciA9ICRDb25maWdJdGVtT2JqZWN0LT5Db25maWdJdGVtTnVtYmVyQ3JlYXRlKAogICAgICAgIFR5cGUgICAgPT4gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpDb25maWcnKS0+R2V0KCdJVFNNQ29uZmlnSXRlbTo6TnVtYmVyR2VuZXJhdG9yJyksCiAgICAgICAgQ2xhc3NJRCA9PiAkSGFyZHdhcmVDb25maWdJdGVtSUQsCiAgICApOwoKICAgICMgYWRkIHRlc3QgQ29uZmlnSXRlbQogICAgbXkgJENvbmZpZ0l0ZW1JRCA9ICRDb25maWdJdGVtT2JqZWN0LT5Db25maWdJdGVtQWRkKAogICAgICAgIE51bWJlciAgPT4gJENvbmZpZ0l0ZW1OdW1iZXIsCiAgICAgICAgQ2xhc3NJRCA9PiAkSGFyZHdhcmVDb25maWdJdGVtSUQsCiAgICAgICAgVXNlcklEICA9PiAxLAogICAgKTsKCiAgICAkU2VsZi0+VHJ1ZSgKICAgICAgICAkQ29uZmlnSXRlbUlELAogICAgICAgICJDb25maWcgaXRlbSBpcyBjcmVhdGVkIC0gJENvbmZpZ0l0ZW1JRCIsCiAgICApOwoKICAgIG15ICRDb25maWdJdGVtTmFtZSA9ICdUZXN0Q29uZmlnSXRlbScgLiAkSGVscGVyLT5HZXRSYW5kb21JRCgpOwogICAgbXkgJFZlcnNpb25JRCAgICAgID0gJENvbmZpZ0l0ZW1PYmplY3QtPlZlcnNpb25BZGQoCiAgICAgICAgTmFtZSAgICAgICAgID0+ICRDb25maWdJdGVtTmFtZSwKICAgICAgICBEZWZpbml0aW9uSUQgPT4gMSwKICAgICAgICBEZXBsU3RhdGVJRCAgPT4gJFByb2R1Y3Rpb25EZXBsU3RhdGVJRCwKICAgICAgICBJbmNpU3RhdGVJRCAgPT4gMSwKICAgICAgICBVc2VySUQgICAgICAgPT4gMSwKICAgICAgICBDb25maWdJdGVtSUQgPT4gJENvbmZpZ0l0ZW1JRCwKICAgICk7CgogICAgJFNlbGYtPlRydWUoCiAgICAgICAgJFZlcnNpb25JRCwKICAgICAgICAiVmVyc2lvbiBmb3IgY29uZmlnIGl0ZW0gJENvbmZpZ0l0ZW1JRCBpcyBjcmVhdGVkIC0gJENvbmZpZ0l0ZW1OYW1lIiwKICAgICk7CgogICAgcHVzaCBAQ29uZmlnSXRlbUlEcywgJENvbmZpZ0l0ZW1JRDsKfQoKIyBnZXQgSW1wb3J0RXhwb3J0IG9iamVjdApteSAkSW1wb3J0RXhwb3J0T2JqZWN0ID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkltcG9ydEV4cG9ydCcpOwoKIyBhZGQgdGVzdCB0ZW1wbGF0ZQpteSAkVGVtcGxhdGVJRCA9ICRJbXBvcnRFeHBvcnRPYmplY3QtPlRlbXBsYXRlQWRkKAogICAgT2JqZWN0ICA9PiAnSVRTTUNvbmZpZ0l0ZW0nLAogICAgRm9ybWF0ICA9PiAnQ1NWJywKICAgIE5hbWUgICAgPT4gJ1RlbXBsYXRlJyAuICRIZWxwZXItPkdldFJhbmRvbUlEKCksCiAgICBWYWxpZElEID0+IDEsCiAgICBDb21tZW50ID0+ICdDb21tZW50JywKICAgIFVzZXJJRCAgPT4gMSwKKTsKCiRTZWxmLT5UcnVlKAogICAgJFRlbXBsYXRlSUQsCiAgICAiSW1wb3J0L0V4cG9ydCB0ZW1wbGF0ZSBpcyBjcmVhdGVkIC0gJFRlbXBsYXRlSUQiLAopOwoKIyBnZXQgb2JqZWN0IGRhdGEgZm9yIHRlc3QgdGVtcGxhdGUKbXkgJVRlbXBsYXRlUmVmID0gKAogICAgJ0NsYXNzSUQnICA9PiAkSGFyZHdhcmVDb25maWdJdGVtSUQsCiAgICAnQ291bnRNYXgnID0+IDEwLAopOwpteSAkU3VjY2VzcyA9ICRJbXBvcnRFeHBvcnRPYmplY3QtPk9iamVjdERhdGFTYXZlKAogICAgVGVtcGxhdGVJRCA9PiAkVGVtcGxhdGVJRCwKICAgIE9iamVjdERhdGEgPT4gXCVUZW1wbGF0ZVJlZiwKICAgIFVzZXJJRCAgICAgPT4gMSwKKTsKCiRTZWxmLT5UcnVlKAogICAgJFN1Y2Nlc3MsCiAgICAiT2JqZWN0RGF0YSBmb3IgdGVzdCB0ZW1wbGF0ZSBpcyBhZGRlZCIsCik7CgojIGFkZCB0aGUgZm9ybWF0IGRhdGEgb2YgdGhlIHRlc3QgdGVtcGxhdGUKbXkgJUZvcm1hdERhdGEgPSAoCiAgICBDaGFyc2V0ICAgICAgICAgICAgICA9PiAnVVRGLTgnLAogICAgQ29sdW1uU2VwYXJhdG9yICAgICAgPT4gJ0NvbW1hJywKICAgIEluY2x1ZGVDb2x1bW5IZWFkZXJzID0+IDEsCik7CiRTdWNjZXNzID0gJEltcG9ydEV4cG9ydE9iamVjdC0+Rm9ybWF0RGF0YVNhdmUoCiAgICBUZW1wbGF0ZUlEID0+ICRUZW1wbGF0ZUlELAogICAgRm9ybWF0RGF0YSA9PiBcJUZvcm1hdERhdGEsCiAgICBVc2VySUQgICAgID0+IDEsCik7CgokU2VsZi0+VHJ1ZSgKICAgICRTdWNjZXNzLAogICAgIkZvcm1hdERhdGEgZm9yIHRlc3QgdGVtcGxhdGUgaXMgYWRkZWQiLAopOwoKIyBzYXZlIHRoZSBzZWFyY2ggZGF0YSBvZiBhIHRlbXBsYXRlCm15ICVTZWFyY2hEYXRhID0gKAogICAgTmFtZSA9PiAnVGVzdENvbmZpZ0l0ZW0qJywKKTsKJFN1Y2Nlc3MgPSAkSW1wb3J0RXhwb3J0T2JqZWN0LT5TZWFyY2hEYXRhU2F2ZSgKICAgIFRlbXBsYXRlSUQgPT4gJFRlbXBsYXRlSUQsCiAgICBTZWFyY2hEYXRhID0+IFwlU2VhcmNoRGF0YSwKICAgIFVzZXJJRCAgICAgPT4gMSwKKTsKCiMgYWRkIG1hcHBpbmcgZGF0YSBmb3IgdGVzdCB0ZW1wbGF0ZQpmb3IgbXkgJE9iamVjdERhdGFWYWx1ZSAocXcoIE5hbWUgRGVwbFN0YXRlIEluY2lTdGF0ZSApKSB7CgogICAgbXkgJE1hcHBpbmdJRCA9ICRJbXBvcnRFeHBvcnRPYmplY3QtPk1hcHBpbmdBZGQoCiAgICAgICAgVGVtcGxhdGVJRCA9PiAkVGVtcGxhdGVJRCwKICAgICAgICBVc2VySUQgICAgID0+IDEsCiAgICApOwoKICAgIG15ICVNYXBwaW5nT2JqZWN0RGF0YSA9ICggS2V5ID0+ICRPYmplY3REYXRhVmFsdWUgKTsKICAgIG15ICRTdWNjZXNzICAgICAgICAgICA9ICRJbXBvcnRFeHBvcnRPYmplY3QtPk1hcHBpbmdPYmplY3REYXRhU2F2ZSgKICAgICAgICBNYXBwaW5nSUQgICAgICAgICA9PiAkTWFwcGluZ0lELAogICAgICAgIE1hcHBpbmdPYmplY3REYXRhID0+IFwlTWFwcGluZ09iamVjdERhdGEsCiAgICAgICAgVXNlcklEICAgICAgICAgICAgPT4gMSwKICAgICk7CgogICAgJFNlbGYtPlRydWUoCiAgICAgICAgJFN1Y2Nlc3MsCiAgICAgICAgIk9iamVjdERhdGEgZm9yIHRlc3QgdGVtcGxhdGUgaXMgbWFwcGVkIC0gJE9iamVjdERhdGFWYWx1ZSIsCiAgICApOwp9CgojIG1ha2UgZGlyZWN0b3J5IGZvciBleHBvcnQgZmlsZQpteSAkRGVzdGluYXRpb25QYXRoID0gJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpDb25maWcnKS0+R2V0KCdIb21lJykgLiAiL3Zhci90bXAvSW1wb3J0RXhwb3J0LyI7Cm1rcGF0aCggWyREZXN0aW5hdGlvblBhdGhdLCAwLCAwNzcwICk7ICAgICMjIG5vIGNyaXRpYwoKIyB0ZXN0IGNvbW1hbmQgd2l0aCB3cm9uZyB0ZW1wbGF0ZSBudW1iZXIKJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoCiAgICAnLS10ZW1wbGF0ZS1udW1iZXInLAogICAgJEhlbHBlci0+R2V0UmFuZG9tTnVtYmVyKCksCiAgICAkRGVzdGluYXRpb25QYXRoIC4gJ1RlbXBsYXRlRXhwb3J0LmNzdicKKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDEsCiAgICAiQ29tbWFuZCB3aXRoIHdyb25nIHRlbXBsYXRlIG51bWJlciAtIGV4aXQgY29kZSIsCik7CgojIHRlc3QgY29tbWFuZCB3aXRob3V0IGRlc3RpbmF0aW9uIGFyZ3VtZW50CiRFeGl0Q29kZSA9ICRDb21tYW5kT2JqZWN0LT5FeGVjdXRlKCAnLS10ZW1wbGF0ZS1udW1iZXInLCAkVGVtcGxhdGVJRCApOwoKJFNlbGYtPklzKAogICAgJEV4aXRDb2RlLAogICAgMSwKICAgICJObyBkZXN0aW5hdGlvbiBhcmd1bWVudCAtIGV4aXQgY29kZSIsCik7CgojIHRlc3QgY29tbWFuZCB3aXRoIC0tdGVtcGxhdGUtbnVtYmVyIG9wdGlvbiBhbmQgZGVzdGluYXRpb24gYXJndW1lbnQKJEV4aXRDb2RlID0gJENvbW1hbmRPYmplY3QtPkV4ZWN1dGUoICctLXRlbXBsYXRlLW51bWJlcicsICRUZW1wbGF0ZUlELCAkRGVzdGluYXRpb25QYXRoIC4gJ1RlbXBsYXRlRXhwb3J0LmNzdicgKTsKCiRTZWxmLT5JcygKICAgICRFeGl0Q29kZSwKICAgIDAsCiAgICAiT3B0aW9uIC0gLS10ZW1wbGF0ZS1udW1iZXIgb3B0aW9uIGFuZCBkZXN0aW5hdGlvbiBhcmd1bWVudCIsCik7CgojIHJlbW92ZSB0ZXN0IGRlc3RpbmF0aW9uIHBhdGgKJFN1Y2Nlc3MgPSBybXRyZWUoIFskRGVzdGluYXRpb25QYXRoXSApOwokU2VsZi0+VHJ1ZSgKICAgICRTdWNjZXNzLAogICAgIlRlc3QgZGlyZWN0b3J5IGRlbGV0ZWQgLSAkRGVzdGluYXRpb25QYXRoIiwKKTsKCiMgY2xlYW51cCBpcyBkb25lIGJ5IFJlc3RvcmVEYXRhYmFzZS4KCjE7Cg==
# --
# Copyright (C) 2001-2019 OTRS AG, https://otrs.com/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars (qw($Self));

# 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==
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE5IE9UUlMgQUcsIGh0dHBzOi8vb3Rycy5jb20vCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjAudHh0LgojIC0tCgpwYWNrYWdlIHZhcjo6cGFja2FnZXNldHVwOjpJbXBvcnRFeHBvcnQ7ICAgICMjIG5vIGNyaXRpYwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKb3VyIEBPYmplY3REZXBlbmRlbmNpZXMgPSAoCiAgICAnS2VybmVsOjpTeXN0ZW06OkRCJywKICAgICdLZXJuZWw6OkNvbmZpZycsCiAgICAnS2VybmVsOjpTeXN0ZW06OlN5c0NvbmZpZycsCik7Cgo9aGVhZDEgTkFNRQoKSW1wb3J0RXhwb3J0LnBtIC0gY29kZSB0byBleGNlY3V0ZSBkdXJpbmcgcGFja2FnZSBpbnN0YWxsYXRpb24KCj1oZWFkMSBTWU5PUFNJUwoKQWxsIGZ1bmN0aW9ucwoKPWhlYWQxIFBVQkxJQyBJTlRFUkZBQ0UKCj1vdmVyIDQKCj1jdXQKCj1pdGVtIG5ldygpCgpjcmVhdGUgYW4gb2JqZWN0CgogICAgdXNlIEtlcm5lbDo6U3lzdGVtOjpPYmplY3RNYW5hZ2VyOwogICAgbG9jYWwgJEtlcm5lbDo6T00gPSBLZXJuZWw6OlN5c3RlbTo6T2JqZWN0TWFuYWdlci0+bmV3KCk7CiAgICBteSAkQ29kZU9iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ3Zhcjo6cGFja2FnZXNldHVwOjpJbXBvcnRFeHBvcnQnKTsKCj1jdXQKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0ge307CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgIyBhbHdheXMgZGlzY2FyZCB0aGUgY29uZmlnIG9iamVjdCBiZWZvcmUgcGFja2FnZSBjb2RlIGlzIGV4ZWN1dGVkLAogICAgIyB0byBtYWtlIHN1cmUgdGhhdCB0aGUgY29uZmlnIG9iamVjdCB3aWxsIGJlIGNyZWF0ZWQgbmV3bHksIHNvIHRoYXQgaXQKICAgICMgd2lsbCB1c2UgdGhlIHJlY2VudGx5IHdyaXR0ZW4gbmV3IGNvbmZpZyBmcm9tIHRoZSBwYWNrYWdlCiAgICAkS2VybmVsOjpPTS0+T2JqZWN0c0Rpc2NhcmQoCiAgICAgICAgT2JqZWN0cyA9PiBbJ0tlcm5lbDo6Q29uZmlnJ10sCiAgICApOwoKICAgIHJldHVybiAkU2VsZjsKfQoKPWl0ZW0gQ29kZUluc3RhbGwoKQoKcnVuIHRoZSBjb2RlIGluc3RhbGwgcGFydAoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+Q29kZUluc3RhbGwoKTsKCj1jdXQKCnN1YiBDb2RlSW5zdGFsbCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiAxOwp9Cgo9aXRlbSBDb2RlUmVpbnN0YWxsKCkKCnJ1biB0aGUgY29kZSByZWluc3RhbGwgcGFydAoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+Q29kZVJlaW5zdGFsbCgpOwoKPWN1dAoKc3ViIENvZGVSZWluc3RhbGwgewogICAgbXkgKCAkU2VsZiwgJVBhcmFtICkgPSBAXzsKCiAgICByZXR1cm4gMTsKfQoKPWl0ZW0gQ29kZVVwZ3JhZGUoKQoKcnVuIHRoZSBjb2RlIHVwZ3JhZGUgcGFydAoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+Q29kZVVwZ3JhZGUoKTsKCj1jdXQKCnN1YiBDb2RlVXBncmFkZSB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiAxOwp9Cgo9aXRlbSBDb2RlVXBncmFkZUZyb21CZWZvcmVfMl8wXzMoKQoKVGhpcyBmdW5jdGlvbiBpcyBvbmx5IGV4ZWN1dGVkIGlmIHRoZSBpbnN0YWxsZWQgbW9kdWxlIHZlcnNpb24gaXMgc21hbGxlciB0aGFuIDIuMC4zLgoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+Q29kZVVwZ3JhZGVGcm9tQmVmb3JlXzJfMF8zKCk7Cgo9Y3V0CgpzdWIgQ29kZVVwZ3JhZGVGcm9tQmVmb3JlXzJfMF8zIHsgICAgIyMgbm8gY3JpdGljCiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgICMgZml4IGEgdHlwbyBpbiB0aGUgZGF0YWJhc2UKICAgICRTZWxmLT5fRml4RGF0YWJhc2VUeXBvKCk7CgogICAgcmV0dXJuIDE7Cn0KCj1pdGVtIENvZGVVcGdyYWRlRnJvbUxvd2VyVGhhbl80XzBfOTEoKQoKVGhpcyBmdW5jdGlvbiBpcyBvbmx5IGV4ZWN1dGVkIGlmIHRoZSBpbnN0YWxsZWQgbW9kdWxlIHZlcnNpb24gaXMgc21hbGxlciB0aGFuIDQuMC45MS4KCm15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+Q29kZVVwZ3JhZGVGcm9tTG93ZXJUaGFuXzRfMF85MSgpOwoKPWN1dAoKc3ViIENvZGVVcGdyYWRlRnJvbUxvd2VyVGhhbl80XzBfOTEgeyAgICAjIyBubyBjcml0aWMKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBjaGFuZ2UgY29uZmlndXJhdGlvbnMgdG8gbWF0Y2ggdGhlIG5ldyBtb2R1bGUgbG9jYXRpb24uCiAgICAkU2VsZi0+X01pZ3JhdGVDb25maWdzKCk7CgogICAgcmV0dXJuIDE7Cn0KCj1pdGVtIENvZGVVbmluc3RhbGwoKQoKcnVuIHRoZSBjb2RlIHVuaW5zdGFsbCBwYXJ0CgogICAgbXkgJFJlc3VsdCA9ICRDb2RlT2JqZWN0LT5Db2RlVW5pbnN0YWxsKCk7Cgo9Y3V0CgpzdWIgQ29kZVVuaW5zdGFsbCB7CiAgICBteSAoICRTZWxmLCAlUGFyYW0gKSA9IEBfOwoKICAgIHJldHVybiAxOwp9Cgo9YmVnaW4gSW50ZXJuYWw6Cgo9aXRlbSBfRml4RGF0YWJhc2VUeXBvKCkKCiAgICBteSAkUmVzdWx0ID0gJENvZGVPYmplY3QtPl9GaXhEYXRhYmFzZVR5cG8oKTsKCj1jdXQKCnN1YiBfRml4RGF0YWJhc2VUeXBvIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBmaXggdGhlIENvbHVtblNlcGVyYXRvciB0eXBvIChjb3JyZWN0IGlzIENvbHVtblNlcGFyYXRvcikKICAgIHJldHVybiBpZiAhJEtlcm5lbDo6T00tPkdldCgnS2VybmVsOjpTeXN0ZW06OkRCJyktPkRvKAogICAgICAgIFNRTCA9PiAiVVBEQVRFIGltZXhwb3J0X2Zvcm1hdCAiCiAgICAgICAgICAgIC4gIlNFVCBkYXRhX2tleSA9ICdDb2x1bW5TZXBhcmF0b3InICIKICAgICAgICAgICAgLiAiV0hFUkUgZGF0YV9rZXkgPSAnQ29sdW1uU2VwZXJhdG9yJyIsCiAgICApOwoKICAgIHJldHVybiAxOwp9Cgo9aXRlbSBfTWlncmF0ZUNvbmZpZ3MoKQoKY2hhbmdlIGNvbmZpZ3VyYXRpb25zIHRvIG1hdGNoIHRoZSBuZXcgbW9kdWxlIGxvY2F0aW9uLgoKICAgIG15ICRSZXN1bHQgPSAkQ29kZU9iamVjdC0+X01pZ3JhdGVDb25maWdzKCk7Cgo9Y3V0CgpzdWIgX01pZ3JhdGVDb25maWdzIHsKCiAgICAjIGNyZWF0ZSBuZWVkZWQgb2JqZWN0cwogICAgbXkgJFN5c0NvbmZpZ09iamVjdCA9ICRLZXJuZWw6Ok9NLT5HZXQoJ0tlcm5lbDo6U3lzdGVtOjpTeXNDb25maWcnKTsKICAgIG15ICRDb25maWdPYmplY3QgICAgPSAkS2VybmVsOjpPTS0+R2V0KCdLZXJuZWw6OkNvbmZpZycpOwoKICAgICMgbWlncmF0ZSBJbXBvcnRFeHBvcnQgY29uZmlnCiAgICAjIGdldCBzZXR0aW5nIGNvbnRlbnQgZm9yIEltcG9ydEV4cG9ydCBjb25maWcKICAgIG15ICRTZXR0aW5nID0gJENvbmZpZ09iamVjdC0+R2V0KCdGcm9udGVuZDo6TW9kdWxlJyk7CgogICAgIyB1cGRhdGUgbW9kdWxlIGxvY2F0aW9uCiAgICAkU2V0dGluZy0+eydBZG1pbkltcG9ydEV4cG9ydCd9LT57TmF2QmFyTW9kdWxlfS0+e01vZHVsZX0gPSAiS2VybmVsOjpPdXRwdXQ6OkhUTUw6Ok5hdkJhcjo6TW9kdWxlQWRtaW4iOwoKICAgICMgc2V0IG5ldyBzZXR0aW5nLAogICAgbXkgJFN1Y2Nlc3MgPSAkU3lzQ29uZmlnT2JqZWN0LT5Db25maWdJdGVtVXBkYXRlKAogICAgICAgIFZhbGlkID0+IDEsCiAgICAgICAgS2V5ICAgPT4gJ0Zyb250ZW5kOjpNb2R1bGUjIyNBZG1pbkltcG9ydEV4cG9ydCcsCiAgICAgICAgVmFsdWUgPT4gJFNldHRpbmctPnsnQWRtaW5JbXBvcnRFeHBvcnQnfSwKICAgICk7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cgo9ZW5kIEludGVybmFsOgoKPWJhY2sKCj1oZWFkMSBURVJNUyBBTkQgQ09ORElUSU9OUwoKVGhpcyBTb2Z0d2FyZSBpcyBwYXJ0IG9mIHRoZSBPVFJTIHByb2plY3QgKGh0dHA6Ly9vdHJzLm9yZy8pLgoKVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CmRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBMPGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC50eHQ+LgoKPWN1dAo=
LyoKQ29weXJpZ2h0IChDKSAyMDAxLTIwMTkgT1RSUyBBRywgaHR0cHM6Ly9vdHJzLmNvbS8KClRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCnRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQpkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wLnR4dC4KKi8KCi8qKgogKiBAcGFja2FnZSAgICAgU2tpbiAiRGVmYXVsdCIKICogQHNlY3Rpb24gICAgIEltcG9ydCBFeHBvcnQgU2NyZWVuCiAqLwoKQG1lZGlhIHNjcmVlbixwcm9qZWN0aW9uLHR2LGhhbmRoZWxkIHsKCi8qKgogKiBAc3Vic2VjdGlvbgogKi8KCgouTWFwSGVhZGVyUm93IGxhYmVsIHsKICAgIGNvbG9yOiAjOTI5MjkyOwp9CgouTWFwSGVhZGVyUm93IC5IZWFkZXIsCi5NYXBIZWFkZXJSb3cgLkZpZWxkIHsKICAgIGRpc3BsYXk6IGlubGluZTsKICAgIG1hcmdpbi1yaWdodDogMTVweDsKICAgIHBhZGRpbmctbGVmdDogMnB4Owp9CgpidXR0b24uQXJyb3dVcCwKYnV0dG9uLkFycm93RG93biB7CiAgICBoZWlnaHQ6IDE2cHg7CiAgICB3aWR0aDogMTZweDsKICAgIHBhZGRpbmc6IDFweDsKICAgIG1hcmdpbi10b3A6IDA7CiAgICBtYXJnaW4tYm90dG9tOiAxcHg7CiAgICBib3JkZXItc3R5bGU6IG5vbmU7CiAgICB0ZXh0LWluZGVudDogLTk5OTlweDsKICAgIGRpc3BsYXk6IGlubGluZS1ibG9jazsKICAgIHZlcnRpY2FsLWFsaWduOiBtaWRkbGU7CiAgICBjdXJzb3I6IHBvaW50ZXI7Cn0KCmJ1dHRvbi5BcnJvd1VwOmFjdGl2ZSwKYnV0dG9uLkFycm93RG93bjphY3RpdmUgewogICAgbWFyZ2luLXRvcDogMXB4OwogICAgbWFyZ2luLWJvdHRvbTogMDsKfQoKYnV0dG9uLkFycm93VXBbZGlzYWJsZWQ9ImRpc2FibGVkIl06YWN0aXZlLApidXR0b24uQXJyb3dEb3duW2Rpc2FibGVkPSJkaXNhYmxlZCJdOmFjdGl2ZSB7CiAgICBtYXJnaW4tdG9wOiAwcHg7CiAgICBtYXJnaW4tYm90dG9tOiAxcHg7Cn0KCmJ1dHRvbi5BcnJvd1VwIHsKICAgIGJhY2tncm91bmQ6IHVybCguLi9pbWcvaWNvbnMvaW1wb3J0ZXhwb3J0X2Fycm93X3VwLnBuZyk7Cn0KCmJ1dHRvbi5BcnJvd1VwW2Rpc2FibGVkPSJkaXNhYmxlZCJdIHsKICAgIGJhY2tncm91bmQ6IHVybCguLi9pbWcvaWNvbnMvaW1wb3J0ZXhwb3J0X2Fycm93X3VwX2Rpc2FibGVkLnBuZyk7CiAgICBjdXJzb3I6IGRlZmF1bHQ7Cn0KCmJ1dHRvbi5BcnJvd0Rvd24gewogICAgYmFja2dyb3VuZDogdXJsKC4uL2ltZy9pY29ucy9pbXBvcnRleHBvcnRfYXJyb3dfZG93bi5wbmcpOwp9CgpidXR0b24uQXJyb3dEb3duW2Rpc2FibGVkPSJkaXNhYmxlZCJdewogICAgYmFja2dyb3VuZDogdXJsKC4uL2ltZy9pY29ucy9pbXBvcnRleHBvcnRfYXJyb3dfZG93bl9kaXNhYmxlZC5wbmcpOwogICAgY3Vyc29yOiBkZWZhdWx0Owp9Cgp9IC8qIGVuZCBAbWVkaWEgKi8K
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==