GeneralCatalog
1.2.1
OTRS AG
http://otrs.org/
GNU GENERAL PUBLIC LICENSE Version 2, June 1991
Build for OTRS::ITSM 1.2.1.
Build for OTRS::ITSM 1.2.0 rc1.
Build for OTRS::ITSM 1.2.0 beta3.
Build for OTRS::ITSM 1.2.0 beta2.
Build for OTRS::ITSM 1.2.0 beta1.
The general catalog package.
Das General Catalog Paket.
2.3.x
<br>
<b>ATTENTION</b>
<br>
<br>
If you uninstall this package, all database tables that were created during installation will be deleted.
All data from this table will be irrevocably lost.
<br>
<br>
((enjoy))<br>
<br>
<br>
<b>ACHTUNG</b>
<br>
<br>
Bei der Deinstallation werden die von diesem Paket angelegten Datenbank-Tabellen gelöscht.
Alle darin enthaltenen Daten gehen unwiderruflich verloren!
<br>
<br>
((enjoy))<br>
<br>
2008-08-18 09:25:13
opms.otrs.com
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSIgPz4KPG90cnNfY29uZmlnIHZlcnNpb249IjEuMCIgaW5pdD0iRnJhbWV3b3JrIj4KICAgIDxDVlM+JElkOiBHZW5lcmFsQ2F0YWxvZy54bWwsdiAxLjMgMjAwOC8wMS8yMyAwODo1ODoyNiBtaCBFeHAgJDwvQ1ZTPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iRnJvbnRlbmQ6Ok1vZHVsZSMjI0FkbWluR2VuZXJhbENhdGFsb2ciIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5Gcm9udGVuZCBtb2R1bGUgcmVnaXN0cmF0aW9uIGZvciB0aGUgQWRtaW5HZW5lcmFsQ2F0YWxvZyBjb25maWd1cmF0aW9uIGluIHRoZSBhZG1pbiBhcmVhLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5Gcm9udGVuZG1vZHVsLVJlZ2lzdHJhdGlvbiBkZXIgQWRtaW5HZW5lcmFsQ2F0YWxvZyBLb25maWd1cmF0aW9uIGltIEFkbWluLUJlcmVpY2guPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+RnJhbWV3b3JrPC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+RnJvbnRlbmQ6OkFkbWluOjpNb2R1bGVSZWdpc3RyYXRpb248L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8RnJvbnRlbmRNb2R1bGVSZWc+CiAgICAgICAgICAgICAgICA8R3JvdXA+YWRtaW48L0dyb3VwPgogICAgICAgICAgICAgICAgPERlc2NyaXB0aW9uPkFkbWluPC9EZXNjcmlwdGlvbj4KICAgICAgICAgICAgICAgIDxUaXRsZT5HZW5lcmFsIENhdGFsb2c8L1RpdGxlPgogICAgICAgICAgICAgICAgPE5hdkJhck5hbWU+QWRtaW48L05hdkJhck5hbWU+CiAgICAgICAgICAgICAgICA8TmF2QmFyTW9kdWxlPgogICAgICAgICAgICAgICAgICAgIDxNb2R1bGU+S2VybmVsOjpPdXRwdXQ6OkhUTUw6Ok5hdkJhck1vZHVsZUFkbWluPC9Nb2R1bGU+CiAgICAgICAgICAgICAgICAgICAgPE5hbWU+R2VuZXJhbCBDYXRhbG9nPC9OYW1lPgogICAgICAgICAgICAgICAgICAgIDxCbG9jaz5CbG9jazM8L0Jsb2NrPgogICAgICAgICAgICAgICAgICAgIDxQcmlvPjQxMDwvUHJpbz4KICAgICAgICAgICAgICAgIDwvTmF2QmFyTW9kdWxlPgogICAgICAgICAgICA8L0Zyb250ZW5kTW9kdWxlUmVnPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KPC9vdHJzX2NvbmZpZz4=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9iZ19HZW5lcmFsQ2F0YWxvZy5wbSAtIHRoZSBidWxnYXJpYW4gdHJhbnNsYXRpb24gb2YgR2VuZXJhbENhdGFsb2cKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgQ29weXJpZ2h0IChDKSAyMDA3LTIwMDggTWlsZW4gS291dGV2CiMgLS0KIyAkSWQ6IGJnX0dlbmVyYWxDYXRhbG9nLnBtLHYgMS4xMCAyMDA4LzA4LzE0IDExOjAwOjIzIG1oIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTIuMC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6YmdfR2VuZXJhbENhdGFsb2c7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjEwICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAkTGFuZy0+eydHZW5lcmFsIENhdGFsb2cnfSAgICAgICAgICAgID0gJ87x7e7i5e0g6uDy4Ovu4yc7CiAgICAkTGFuZy0+eydHZW5lcmFsIENhdGFsb2cgTWFuYWdlbWVudCd9ID0gJ9Pv8ODi6+Xt6OUg7eAg7vHt7uLt6P8g6uDy4Ovu4yc7CiAgICAkTGFuZy0+eydDYXRhbG9nIENsYXNzJ30gICAgICAgICAgICAgID0gJ8rr4PHu4uUg4iDq4PLg6+7j4Cc7CiAgICAkTGFuZy0+eydBZGQgYSBuZXcgQ2F0YWxvZyBDbGFzcy4nfSAgID0gJ8Tu4eDi6CDt7uIg6uvg8SDiIOrg8uDr7uPgLic7CiAgICAkTGFuZy0+eydBZGQgQ2F0YWxvZyBJdGVtJ30gICAgICAgICAgID0gJ8Tu4eDi6CDl6+Xs5e3yIOr67CDq4PLg6+7j4Cc7CiAgICAkTGFuZy0+eydBZGQgQ2F0YWxvZyBDbGFzcyd9ICAgICAgICAgID0gJ8Tu4eDi6CDq6+DxIOIg6uDy4Ovu4+AnOwogICAgJExhbmctPnsnRnVuY3Rpb25hbGl0eSd9ICAgICAgICAgICAgICA9ICfU8+3q9uju7eDr7e7x8ic7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9jdF9HZW5lcmFsQ2F0YWxvZy5wbSAtIHRoZSBjYXRhbGFuIHRyYW5zbGF0aW9uIG9mIEdlbmVyYWxDYXRhbG9nCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDggT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIENvcHlyaWdodCAoQykgMjAwOCBTaXN0ZW1lcyBPVElDIChpYnNhbHV0KSAtIEFudG9uaW8gTGluZGUKIyAtLQojICRJZDogY3RfR2VuZXJhbENhdGFsb2cucG0sdiAxLjIgMjAwOC8wOC8xNCAxMTowMDo0NyBtaCBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0yLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmN0X0dlbmVyYWxDYXRhbG9nOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS4yICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAkTGFuZy0+eydHZW5lcmFsIENhdGFsb2cnfSAgICAgICAgICAgID0gJ0NhdOBsZWcgZ2VuZXJhbCc7CiAgICAkTGFuZy0+eydHZW5lcmFsIENhdGFsb2cgTWFuYWdlbWVudCd9ID0gJ0dlc3Rp8yBkZWwgY2F04GxlZyBnZW5lcmFsJzsKICAgICRMYW5nLT57J0NhdGFsb2cgQ2xhc3MnfSAgICAgICAgICAgICAgPSAnQ2xhc3NlIGRlIGNhdOFsZWcnOwogICAgJExhbmctPnsnQWRkIGEgbmV3IENhdGFsb2cgQ2xhc3MuJ30gICA9ICdBZmVnaXIgdW5hIG5vdmEgY2xhc3NlIGRlIGNhdOBsZWcnOwogICAgJExhbmctPnsnQWRkIENhdGFsb2cgSXRlbSd9ICAgICAgICAgICA9ICdBZmVnaXIgYXJ0aWNsZSBkZSBjYXTgbGVnJzsKICAgICRMYW5nLT57J0FkZCBDYXRhbG9nIENsYXNzJ30gICAgICAgICAgPSAnQWZlZ2lyIGNsYXNzZSBkZSBjYXTgbGVnJzsKICAgICRMYW5nLT57J0Z1bmN0aW9uYWxpdHknfSAgICAgICAgICAgICAgPSAnRnVuY2lvbmFsaXRhdCc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9jel9HZW5lcmFsQ2F0YWxvZy5wbSAtIHRoZSBjemVjaCB0cmFuc2xhdGlvbiBvZiBHZW5lcmFsQ2F0YWxvZwojIENvcHlyaWdodCAoQykgMjAwMS0yMDA4IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLm9yZy8KIyBDb3B5cmlnaHQgKEMpIDIwMDctMjAwOCBNaWxlbiBLb3V0ZXYKIyAtLQojICRJZDogY3pfR2VuZXJhbENhdGFsb2cucG0sdiAxLjggMjAwOC8wOC8xNCAxMTowMDo0NyBtaCBFeHAgJAojIC0tCiMgVGhpcyBzb2Z0d2FyZSBjb21lcyB3aXRoIEFCU09MVVRFTFkgTk8gV0FSUkFOVFkuIEZvciBkZXRhaWxzLCBzZWUKIyB0aGUgZW5jbG9zZWQgZmlsZSBDT1BZSU5HIGZvciBsaWNlbnNlIGluZm9ybWF0aW9uIChHUEwpLiBJZiB5b3UKIyBkaWQgbm90IHJlY2VpdmUgdGhpcyBmaWxlLCBzZWUgaHR0cDovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0yLjAudHh0LgojIC0tCgpwYWNrYWdlIEtlcm5lbDo6TGFuZ3VhZ2U6OmN6X0dlbmVyYWxDYXRhbG9nOwoKdXNlIHN0cmljdDsKdXNlIHdhcm5pbmdzOwoKdXNlIHZhcnMgcXcoJFZFUlNJT04pOwokVkVSU0lPTiA9IHF3KCRSZXZpc2lvbjogMS44ICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAkTGFuZy0+eydHZW5lcmFsIENhdGFsb2cnfSAgICAgICAgICAgID0gJ1bFoWVvYmVjbsO9IGthdGFsb2cnOwogICAgJExhbmctPnsnR2VuZXJhbCBDYXRhbG9nIE1hbmFnZW1lbnQnfSA9ICfFmMOtemVuw60gdsWhZW9iZWNuw6lobyBrYXRhbG9ndSc7CiAgICAkTGFuZy0+eydDYXRhbG9nIENsYXNzJ30gICAgICAgICAgICAgID0gJ1TFmcOtZHkgdiBrYXRhbG9ndSc7CiAgICAkTGFuZy0+eydBZGQgYSBuZXcgQ2F0YWxvZyBDbGFzcy4nfSAgID0gJ1DFmWlkYXQgbm92b3UgdMWZw61kdSBkbyBrYXRhbG9ndS4nOwogICAgJExhbmctPnsnQWRkIENhdGFsb2cgSXRlbSd9ICAgICAgICAgICA9ICdQxZlpZGF0IHBydmVrIGRvIGthdGFsb2d1JzsKICAgICRMYW5nLT57J0FkZCBDYXRhbG9nIENsYXNzJ30gICAgICAgICAgPSAnUMWZaWRhdCB0xZnDrWR1IGRvIGthdGFsb2d1JzsKICAgICRMYW5nLT57J0Z1bmN0aW9uYWxpdHknfSAgICAgICAgICAgICAgPSAnRnVua2Npb25hbGl0YSc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9kZV9HZW5lcmFsQ2F0YWxvZy5wbSAtIHRoZSBnZXJtYW4gdHJhbnNsYXRpb24gb2YgR2VuZXJhbENhdGFsb2cKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgLS0KIyAkSWQ6IGRlX0dlbmVyYWxDYXRhbG9nLnBtLHYgMS4xMyAyMDA4LzA1LzIxIDA4OjM2OjM3IG1oIEV4cCAkCiMgLS0KIyBUaGlzIHNvZnR3YXJlIGNvbWVzIHdpdGggQUJTT0xVVEVMWSBOTyBXQVJSQU5UWS4gRm9yIGRldGFpbHMsIHNlZQojIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTIuMC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpMYW5ndWFnZTo6ZGVfR2VuZXJhbENhdGFsb2c7Cgp1c2Ugc3RyaWN0Owp1c2Ugd2FybmluZ3M7Cgp1c2UgdmFycyBxdygkVkVSU0lPTik7CiRWRVJTSU9OID0gcXcoJFJldmlzaW9uOiAxLjEzICQpIFsxXTsKCnN1YiBEYXRhIHsKICAgIG15ICRTZWxmID0gc2hpZnQ7CgogICAgbXkgJExhbmcgPSAkU2VsZi0+e1RyYW5zbGF0aW9ufTsKCiAgICByZXR1cm4gaWYgcmVmICRMYW5nIG5lICdIQVNIJzsKCiAgICAkTGFuZy0+eydHZW5lcmFsIENhdGFsb2cnfSAgICAgICAgICAgID0gJ0dlbmVyYWwgQ2F0YWxvZyc7CiAgICAkTGFuZy0+eydHZW5lcmFsIENhdGFsb2cgTWFuYWdlbWVudCd9ID0gJ0dlbmVyYWwgQ2F0YWxvZyBWZXJ3YWx0dW5nJzsKICAgICRMYW5nLT57J0NhdGFsb2cgQ2xhc3MnfSAgICAgICAgICAgICAgPSAnS2F0YWxvZyBLbGFzc2UnOwogICAgJExhbmctPnsnQWRkIGEgbmV3IENhdGFsb2cgQ2xhc3MuJ30gICA9ICdFaW5lIG5ldWUgS2F0YWxvZyBLbGFzc2UgaGluenVm/Gdlbi4nOwogICAgJExhbmctPnsnQWRkIENhdGFsb2cgSXRlbSd9ICAgICAgICAgICA9ICdLYXRhbG9nIEVpbnRyYWcgaGluenVm/Gdlbic7CiAgICAkTGFuZy0+eydBZGQgQ2F0YWxvZyBDbGFzcyd9ICAgICAgICAgID0gJ0thdGFsb2cgS2xhc3NlIGhpbnp1ZvxnZW4nOwogICAgJExhbmctPnsnRnVuY3Rpb25hbGl0eSd9ICAgICAgICAgICAgICA9ICdGdW5rdGlvbmFsaXTkdCc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9lc19HZW5lcmFsQ2F0YWxvZy5wbSAtIHRoZSBzcGFuaXNoIHRyYW5zbGF0aW9uIG9mIEdlbmVyYWxDYXRhbG9nCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDggT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIENvcHlyaWdodCAoQykgMjAwOCBBcXVpbGVzIENvaGVuCiMgLS0KIyAkSWQ6IGVzX0dlbmVyYWxDYXRhbG9nLnBtLHYgMS4zIDIwMDgvMDgvMTQgMTE6MDA6NDcgbWggRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMi4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjplc19HZW5lcmFsQ2F0YWxvZzsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuMyAkKSBbMV07CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgIG15ICRMYW5nID0gJFNlbGYtPntUcmFuc2xhdGlvbn07CgogICAgcmV0dXJuIGlmIHJlZiAkTGFuZyBuZSAnSEFTSCc7CgogICAgJExhbmctPnsnR2VuZXJhbCBDYXRhbG9nJ30gICAgICAgICAgICA9ICdDYXRhbG9nbyBHZW5lcmFsJzsKICAgICRMYW5nLT57J0dlbmVyYWwgQ2F0YWxvZyBNYW5hZ2VtZW50J30gPSAnQWRtaW5pc3RyYWNp824gZGUgQ2F0YWxvZ28gR2VuZXJhbCc7CiAgICAkTGFuZy0+eydDYXRhbG9nIENsYXNzJ30gICAgICAgICAgICAgID0gJ0NsYXNlIGRlIENhdGFsb2dvJzsKICAgICRMYW5nLT57J0FkZCBhIG5ldyBDYXRhbG9nIENsYXNzLid9ICAgPSAnQfFhZGlyIHVuYSBudWV2YSBDbGFzZSBhbCBDYXRhbG9nbyc7CiAgICAkTGFuZy0+eydBZGQgQ2F0YWxvZyBJdGVtJ30gICAgICAgICAgID0gJ0HxYWRpciBFbGVtZW50byBkZSBDYXRhbG9nbyc7CiAgICAkTGFuZy0+eydBZGQgQ2F0YWxvZyBDbGFzcyd9ICAgICAgICAgID0gJ0HxYWRpciBDbGFzZSBkZSBDYXRhbG9nbyc7CiAgICAkTGFuZy0+eydGdW5jdGlvbmFsaXR5J30gICAgICAgICAgICAgID0gJ0Z1bmNpb25hbGlkYWQnOwoKICAgIHJldHVybiAxOwp9CgoxOwo=
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9wbF9HZW5lcmFsQ2F0YWxvZy5wbSAtIHRoZSBwb2xpc2ggdHJhbnNsYXRpb24gb2YgR2VuZXJhbENhdGFsb2cKIyBDb3B5cmlnaHQgKEMpIDIwMDEtMjAwOCBPVFJTIEFHLCBodHRwOi8vb3Rycy5vcmcvCiMgQ29weXJpZ2h0IChDKSAyMDA4IE1hY2llaiBMb3N6YWpjCiMgLS0KIyAkSWQ6IHBsX0dlbmVyYWxDYXRhbG9nLnBtLHYgMS4zIDIwMDgvMDgvMTQgMTE6MDA6NDcgbWggRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMi4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpwbF9HZW5lcmFsQ2F0YWxvZzsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuMyAkKSBbMV07CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgIG15ICRMYW5nID0gJFNlbGYtPntUcmFuc2xhdGlvbn07CgogICAgcmV0dXJuIGlmIHJlZiAkTGFuZyBuZSAnSEFTSCc7CgogICAgJExhbmctPnsnR2VuZXJhbCBDYXRhbG9nJ30gICAgICAgICAgICA9ICdLYXRhbG9nIGez83dueSc7CiAgICAkTGFuZy0+eydHZW5lcmFsIENhdGFsb2cgTWFuYWdlbWVudCd9ID0gJ1phcnqxZHphbmllIGthdGFsb2dpZW0gZ7Pzd255bSc7CiAgICAkTGFuZy0+eydDYXRhbG9nIENsYXNzJ30gICAgICAgICAgICAgID0gJyc7CiAgICAkTGFuZy0+eydBZGQgYSBuZXcgQ2F0YWxvZyBDbGFzcy4nfSAgID0gJyc7CiAgICAkTGFuZy0+eydBZGQgQ2F0YWxvZyBJdGVtJ30gICAgICAgICAgID0gJ0RvZGFqIGVsZW1lbnQga2F0YWxvZ3UnOwogICAgJExhbmctPnsnQWRkIENhdGFsb2cgQ2xhc3MnfSAgICAgICAgICA9ICcnOwogICAgJExhbmctPnsnRnVuY3Rpb25hbGl0eSd9ICAgICAgICAgICAgICA9ICdGdW5rY2pvbmFsbm+25ic7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
IyAtLQojIEtlcm5lbC9MYW5ndWFnZS9ydV9HZW5lcmFsQ2F0YWxvZy5wbSAtIHRoZSBydXNzaWFuIHRyYW5zbGF0aW9uIG9mIEdlbmVyYWxDYXRhbG9nCiMgQ29weXJpZ2h0IChDKSAyMDAxLTIwMDggT1RSUyBBRywgaHR0cDovL290cnMub3JnLwojIENvcHlyaWdodCAoQykgMjAwOCBFZ29yIFRzaWxlbmtvIDxiZzhzIGF0IHN5bWxpbmsucnU+CiMgLS0KIyAkSWQ6IHJ1X0dlbmVyYWxDYXRhbG9nLnBtLHYgMS4xIDIwMDgvMDgvMTUgMTQ6NDY6MzEgbWggRXhwICQKIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoR1BMKS4gSWYgeW91CiMgZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMi4wLnR4dC4KIyAtLQoKcGFja2FnZSBLZXJuZWw6Okxhbmd1YWdlOjpydV9HZW5lcmFsQ2F0YWxvZzsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnVzZSB2YXJzIHF3KCRWRVJTSU9OKTsKJFZFUlNJT04gPSBxdygkUmV2aXNpb246IDEuMSAkKSBbMV07CgpzdWIgRGF0YSB7CiAgICBteSAkU2VsZiA9IHNoaWZ0OwoKICAgIG15ICRMYW5nID0gJFNlbGYtPntUcmFuc2xhdGlvbn07CgogICAgcmV0dXJuIGlmIHJlZiAkTGFuZyBuZSAnSEFTSCc7CgogICAgJExhbmctPnsnR2VuZXJhbCBDYXRhbG9nJ30gICAgICAgICAgICA9ICfO4fno6SDq4PLg6+7jJzsKICAgICRMYW5nLT57J0dlbmVyYWwgQ2F0YWxvZyBNYW5hZ2VtZW50J30gPSAn0+/w4OLr5e3o5SDu4fno7CDq4PLg6+7j7uwnOwogICAgJExhbmctPnsnQ2F0YWxvZyBDbGFzcyd9ICAgICAgICAgICAgICA9ICfK6+Dx8SDq4PLg6+7j4Cc7CiAgICAkTGFuZy0+eydBZGQgYSBuZXcgQ2F0YWxvZyBDbGFzcy4nfSAgID0gJ8Tu4eDi6PL8IO3u4vvpIOrr4PHxIOrg8uDr7uPgJzsKICAgICRMYW5nLT57J0FkZCBDYXRhbG9nIEl0ZW0nfSAgICAgICAgICAgPSAnxO7h4OLr5e3o5SD96+Xs5e3y4CDq4PLg6+7j4Cc7CiAgICAkTGFuZy0+eydBZGQgQ2F0YWxvZyBDbGFzcyd9ICAgICAgICAgID0gJ8Tu4eDi6+Xt6OUg6uvg8fHgIOrg8uDr7uPgJzsKICAgICRMYW5nLT57J0Z1bmN0aW9uYWxpdHknfSAgICAgICAgICAgICAgPSAn1PPt6vbo7u3g6/zt7vHy/Cc7CgogICAgcmV0dXJuIDE7Cn0KCjE7Cg==
# --
# Kernel/Modules/AdminGeneralCatalog.pm - admin frontend of general catalog management
# Copyright (C) 2001-2008 OTRS AG, http://otrs.org/
# --
# $Id: AdminGeneralCatalog.pm,v 1.21 2008/03/06 14:28:04 mh Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl-2.0.txt.
# --

package Kernel::Modules::AdminGeneralCatalog;

use strict;
use warnings;

use Kernel::System::GeneralCatalog;
use Kernel::System::Valid;

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

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

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

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

    return $Self;
}

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

    # ------------------------------------------------------------ #
    # catalog item list
    # ------------------------------------------------------------ #
    if ( $Self->{Subaction} eq 'ItemList' ) {
        my $Class = $Self->{ParamObject}->GetParam( Param => "Class" ) || '';

        # check needed class
        return $Self->{LayoutObject}->Redirect( OP => "Action=$Self->{Action}" ) if !$Class;

        # get catalog class list
        my $ClassList       = $Self->{GeneralCatalogObject}->ClassList();
        my $ClassOptionStrg = $Self->{LayoutObject}->BuildSelection(
            Name         => 'Class',
            Data         => $ClassList,
            SelectedID   => $Class,
            PossibleNone => 1,
            Translation  => 0,
        );

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

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

        # get catalog item list
        my $ItemIDList = $Self->{GeneralCatalogObject}->ItemList(
            Class => $Class,
            Valid => 0,
        );

        # check item list
        return $Self->{LayoutObject}->ErrorScreen()
            if !$ItemIDList || !%{$ItemIDList};

        my $CssClass = '';
        for my $ItemID ( sort { $ItemIDList->{$a} cmp $ItemIDList->{$b} } keys %{$ItemIDList} ) {

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

            # get item data
            my $ItemData = $Self->{GeneralCatalogObject}->ItemGet(
                ItemID => $ItemID,
            );

            # output overview item list
            $Self->{LayoutObject}->Block(
                Name => 'OverviewItemList',
                Data => {
                    %{$ItemData},
                    CssClass      => $CssClass,
                    Functionality => $ItemData->{Functionality} || '-',
                    Valid         => $ValidList{ $ItemData->{ValidID} },
                },
            );
        }

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

        # create output string
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AdminGeneralCatalog',
            Data         => \%Param,
        );

        # add footer
        $Output .= $Self->{LayoutObject}->Footer();

        return $Output;
    }

    # ------------------------------------------------------------ #
    # catalog item edit
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'ItemEdit' ) {
        my %ItemData;

        # get params
        $ItemData{ItemID} = $Self->{ParamObject}->GetParam( Param => "ItemID" );
        if ( $ItemData{ItemID} eq 'NEW' ) {

            # get class
            $ItemData{Class} = $Self->{ParamObject}->GetParam( Param => "Class" );

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

            # get item data
            my $ItemDataRef = $Self->{GeneralCatalogObject}->ItemGet(
                ItemID => $ItemData{ItemID},
            );

            # check item data
            return $Self->{LayoutObject}->ErrorScreen()
                if !$ItemDataRef;

            %ItemData = %{$ItemDataRef};
        }

        # generate ClassOptionStrg
        my $ClassList       = $Self->{GeneralCatalogObject}->ClassList();
        my $ClassOptionStrg = $Self->{LayoutObject}->BuildSelection(
            Name         => 'Class',
            Data         => $ClassList,
            SelectedID   => $ItemData{Class},
            PossibleNone => 1,
            Translation  => 0,
        );

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

        # get functionality list
        my $FunctionalityRef = $Self->{GeneralCatalogObject}->FunctionalityList(
            Class => $ItemData{Class},
        );

        # prepare functionality list
        my %FunctionalityList;
        for my $Functionality ( @{$FunctionalityRef} ) {
            $FunctionalityList{$Functionality} = $Functionality || '-';
        }
        if ( !%FunctionalityList ) {
            %FunctionalityList = (
                '' => '-',
            );
        }

        # generate FunctionalityOptionStrg
        my $FunctionalityOptionStrg = $Self->{LayoutObject}->BuildSelection(
            Name        => 'Functionality',
            Data        => \%FunctionalityList,
            SelectedID  => $ItemData{Functionality} || '',
            Translation => 0,
        );

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

        # output ItemEdit
        $Self->{LayoutObject}->Block(
            Name => 'ItemEdit',
            Data => {
                %ItemData,
                FunctionalityOptionStrg => $FunctionalityOptionStrg,
                ValidOptionStrg         => $ValidOptionStrg,
            },
        );

        if ( $ItemData{Class} eq 'NEW' ) {

            # output ItemEditClassAdd
            $Self->{LayoutObject}->Block(
                Name => 'ItemEditClassAdd',
                Data => {
                    Class => $ItemData{Class},
                },
            );
        }
        else {

            # output ItemEditClassExist
            $Self->{LayoutObject}->Block(
                Name => 'ItemEditClassExist',
                Data => {
                    Class => $ItemData{Class},
                },
            );
        }

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

        # create output string
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AdminGeneralCatalog',
            Data         => \%Param,
        );

        # add footer
        $Output .= $Self->{LayoutObject}->Footer();

        return $Output;
    }

    # ------------------------------------------------------------ #
    # catalog item save
    # ------------------------------------------------------------ #
    elsif ( $Self->{Subaction} eq 'ItemSave' ) {
        my %ItemData;

        # get params
        for my $Param (qw(Class ItemID Name Functionality ValidID Comment)) {
            $ItemData{$Param} = $Self->{ParamObject}->GetParam( Param => $Param ) || '';
        }

        # check class
        return $Self->{LayoutObject}->Redirect( OP => "Action=$Self->{Action}" )
            if !$ItemData{Class} || $ItemData{Class} eq 'NEW';

        # save to database
        my $Success;
        if ( $ItemData{ItemID} eq 'NEW' ) {
            $Success = $Self->{GeneralCatalogObject}->ItemAdd(
                %ItemData,
                UserID => $Self->{UserID},
            );
        }
        else {
            $Success = $Self->{GeneralCatalogObject}->ItemUpdate(
                %ItemData,
                UserID => $Self->{UserID},
            );
        }

        return $Self->{LayoutObject}->ErrorScreen() if !$Success;

        # redirect to overview class list
        return $Self->{LayoutObject}->Redirect(
            OP => "Action=$Self->{Action}&Subaction=ItemList&Class=$ItemData{Class}"
        );
    }

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

        # get catalog class list
        my $ClassList       = $Self->{GeneralCatalogObject}->ClassList();
        my $ClassOptionStrg = $Self->{LayoutObject}->BuildSelection(
            Name         => 'Class',
            Data         => $ClassList,
            PossibleNone => 1,
            Translation  => 0,
        );

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

        my $CssClass = '';
        for my $Class ( @{$ClassList} ) {

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

            # output overview class list
            $Self->{LayoutObject}->Block(
                Name => 'OverviewClassList',
                Data => {
                    Class    => $Class,
                    CssClass => $CssClass,
                },
            );
        }

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

        # create output string
        $Output .= $Self->{LayoutObject}->Output(
            TemplateFile => 'AdminGeneralCatalog',
            Data         => \%Param,
        );

        # add footer
        $Output .= $Self->{LayoutObject}->Footer();

        return $Output;
    }
}

1;

# --
# AdminGeneralCatalog.dtl - provides HTML form for AdminGeneralCatalog
# Copyright (C) 2001-2008 OTRS AG, http://otrs.org/
# --
# $Id: AdminGeneralCatalog.dtl,v 1.6 2008/01/28 14:58:45 mh Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl-2.0.txt.
# --

<!-- dtl:block:Overview -->
<table border="0" width="100%" cellspacing="0" cellpadding="3">
  <tr>
    <td colspan="2" class="mainhead">
      $Env{"Box0"}$Text{"General Catalog Management"}$Env{"Box1"}
    </td>
  </tr>
  <tr>
    <td width="30%" class="mainbody">
      <form action="$Env{"CGIHandle"}" method="get">
        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="ItemEdit">
        <input type="hidden" name="ItemID" value="NEW">
        <table width="100%" cellspacing="0" cellpadding="4">
          <tr>
            <td class="contenthead">$Text{"Add Catalog Item"}:</td>
          </tr>
          <tr>
            <td class="contentbody">
              <table>
                <tr>
                  <td class="contentkey">$Text{"Catalog Class"}:</td>
                  <td class="contentvalue">$Data{"ClassOptionStrg"}</td>
                </tr>
              </table>
            </td>
          </tr>
          <tr>
            <td class="contentfooter">
              <input class="button" type="submit" value="$Text{"Add"}">
            </td>
          </tr>
        </table>
      </form>
      <br>
      <form action="$Env{"CGIHandle"}" method="get">
        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="ItemEdit">
        <input type="hidden" name="ItemID" value="NEW">
        <input type="hidden" name="Class" value="NEW">
        <table width="100%" cellspacing="0" cellpadding="4">
          <tr>
            <td class="contenthead">$Text{"Add Catalog Class"}:</td>
          </tr>
          <tr>
            <td class="contentbody">$Text{"Add a new Catalog Class."}</td>
          </tr>
          <tr>
            <td class="contentfooter">
              <input class="button" type="submit" value="$Text{"Add"}">
            </td>
          </tr>
        </table>
      </form>
    </td>
    <td width="70%" class="mainbody">
<!-- dtl:block:OverviewClass -->
      <table width="100%" cellspacing="0" cellpadding="4">
        <tr>
          <td class="contenthead">$Text{"List"}:</td>
        </tr>
        <tr>
          <td class="contentbody">
            <table width="100%" border="0" cellspacing="0" cellpadding="3">
              <tr>
                <td class="contentkey" width="60%">$Text{"Catalog Class"}</td>
              </tr>
<!-- dtl:block:OverviewClassList -->
              <tr>
                <td class="$QData{"CssClass"}">
                  <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=ItemList&Class=$LQData{"Class"}">
                  $QData{"Class"}
                  </a>
                </td>
              </tr>
<!-- dtl:block:OverviewClassList -->
            </table>
          </td>
        </tr>
        <tr>
          <td class="contentfooter">
            &nbsp;
          </td>
        </tr>
      </table>
<!-- dtl:block:OverviewClass -->
<!-- dtl:block:OverviewItem -->
      <table width="100%" cellspacing="0" cellpadding="4">
        <tr>
          <td class="contenthead">$Text{"List"}:</td>
        </tr>
        <tr>
          <td class="contentbody">
            <table width="100%" border="0" cellspacing="0" cellpadding="3">
              <tr>
                <td class="contentkey" width="60%">$QData{"Class"}</td>
                <td class="contentkey" align="center">$Text{"Functionality"}</td>
                <td class="contentkey" align="center">$Text{"Valid"}</td>
              </tr>
<!-- dtl:block:OverviewItemList -->
              <tr>
                <td class="$QData{"CssClass"}">
                  <a href="$Env{"Baselink"}Action=$Env{"Action"}&Subaction=ItemEdit&ItemID=$QData{"ItemID"}">
                  $QData{"Name"}
                  </a>
                </td>
                <td class="$QData{"CssClass"}" align="center">
                  $QData{"Functionality"}
                </td>
                <td class="$QData{"CssClass"}" align="center">
                  $Text{"$QData{"Valid"}"}
                </td>
              </tr>
<!-- dtl:block:OverviewItemList -->
            </table>
          </td>
        </tr>
        <tr>
          <td class="contentfooter">
            &nbsp;
          </td>
        </tr>
      </table>
<!-- dtl:block:OverviewItem -->
<!-- dtl:block:ItemEdit -->
      <form action="$Env{"CGIHandle"}" method="get">
        <input type="hidden" name="Action" value="$Env{"Action"}">
        <input type="hidden" name="Subaction" value="ItemSave">
        <input type="hidden" name="ItemID" value="$QData{"ItemID"}">
        <table width="100%" cellspacing="0" cellpadding="4">
          <tr>
            <td class="contenthead">$Text{"Item"} $Text{"Edit"}:</td>
          </tr>
          <tr>
            <td class="contentbody">
              <table border="0" cellspacing="0" cellpadding="3">
<!-- dtl:block:ItemEditClassAdd -->
                <tr>
                  <td class="contentkey">$Text{"Catalog Class"}: </td>
                  <td class="contentvalue">
                    <input type="text" name="Class" value="$QData{"Name"}" size="45" maxlength="100">
                  </td>
                </tr>
<!-- dtl:block:ItemEditClassAdd -->
<!-- dtl:block:ItemEditClassExist -->
                <tr>
                  <td class="contentkey">$Text{"Catalog Class"}: </td>
                  <td class="contentvalue">
                    $QData{"Class"}
                    <input type="hidden" name="Class" value="$QData{"Class"}">
                  </td>
                </tr>
<!-- dtl:block:ItemEditClassExist -->
                <tr>
                  <td class="contentkey">$Text{"Name"}: </td>
                  <td class="contentvalue">
                    <input type="text" name="Name" value="$QData{"Name"}" size="45" maxlength="100">
                  </td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Functionality"}: </td>
                  <td class="contentvalue">$Data{"FunctionalityOptionStrg"}</td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Valid"}: </td>
                  <td class="contentvalue">$Data{"ValidOptionStrg"}</td>
                </tr>
                <tr>
                  <td class="contentkey">$Text{"Comment"}: </td>
                  <td class="contentvalue">
                    <input type="text" name="Comment" value="$QData{"Comment"}" size="50" maxlength="200">
                  </td>
                </tr>
              </table>
            </td>
          </tr>
          <tr>
            <td class="contentfooter">
              <input class="button" type="submit" value="$Text{"Submit"}">
            </td>
          </tr>
        </table>
      </form>
<!-- dtl:block:ItemEdit -->
    </td>
  </tr>
</table>
<!-- dtl:block:Overview -->

# --
# Kernel/System/GeneralCatalog.pm - all general catalog functions
# Copyright (C) 2001-2008 OTRS AG, http://otrs.org/
# --
# $Id: GeneralCatalog.pm,v 1.43 2008/07/23 09:02:02 ub Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl-2.0.txt.
# --

package Kernel::System::GeneralCatalog;

use strict;
use warnings;

use Kernel::System::Valid;
use Kernel::System::CheckItem;

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

=head1 NAME

Kernel::System::GeneralCatalog - general catalog lib

=head1 SYNOPSIS

All general catalog functions.

=head1 PUBLIC INTERFACE

=over 4

=cut

=item new()

create an object

    use Kernel::Config;
    use Kernel::System::DB;
    use Kernel::System::GeneralCatalog;
    use Kernel::System::Log;
    use Kernel::System::Main;

    my $ConfigObject = Kernel::Config->new();
    my $LogObject = Kernel::System::Log->new(
        ConfigObject => $ConfigObject,
    );
    my $MainObject = Kernel::System::Main->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
    );
    my $DBObject = Kernel::System::DB->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
        MainObject   => $MainObject,
    );
    my $GeneralCatalogObject = Kernel::System::GeneralCatalog->new(
        ConfigObject => $ConfigObject,
        LogObject    => $LogObject,
        DBObject     => $DBObject,
        MainObject   => $MainObject,
    );

=cut

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

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

    # check needed objects
    for my $Object (qw(DBObject ConfigObject LogObject MainObject)) {
        $Self->{$Object} = $Param{$Object} || die "Got no $Object!";
    }
    $Self->{ValidObject}     = Kernel::System::Valid->new( %{$Self} );
    $Self->{CheckItemObject} = Kernel::System::CheckItem->new( %{$Self} );

    return $Self;
}

=item ClassList()

return an array reference of all general catalog classes

    my $ArrayRef = $GeneralCatalogObject->ClassList();

=cut

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

    # ask database
    $Self->{DBObject}->Prepare(
        SQL => 'SELECT DISTINCT(general_catalog_class) '
            . 'FROM general_catalog ORDER BY general_catalog_class',
    );

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

    return \@ClassList;
}

=item ClassRename()

rename a general catalog class

    my $True = $GeneralCatalogObject->ClassRename(
        ClassOld => 'ITSM::ConfigItem::State',
        ClassNew => 'ITSM::ConfigItem::DeploymentState',
    );

=cut

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

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

    # cleanup given params
    for my $Argument (qw(ClassOld ClassNew)) {
        $Self->{CheckItemObject}->StringClean(
            StringRef         => \$Param{$Argument},
            RemoveAllNewlines => 1,
            RemoveAllTabs     => 1,
            RemoveAllSpaces   => 1,
        );
    }

    return 1 if $Param{ClassNew} eq $Param{ClassOld};

    # check if new class name already exists
    $Self->{DBObject}->Prepare(
        SQL   => 'SELECT id FROM general_catalog WHERE general_catalog_class = ?',
        Bind  => [ \$Param{ClassNew} ],
        Limit => 1,
    );

    # fetch the result
    my $AlreadyExists = 0;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        $AlreadyExists = 1;
    }

    if ($AlreadyExists) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't rename class $Param{ClassOld}! New classname already exists."
        );
        return;
    }

    # reset cache
    delete $Self->{Cache}->{ItemGet}->{Class}->{ $Param{ClassOld} };
    delete $Self->{Cache}->{ItemGet}->{Class}->{ $Param{ClassNew} };
    delete $Self->{Cache}->{ItemGet}->{ItemID};
    delete $Self->{Cache}->{ItemList};

    # rename general catalog class
    return $Self->{DBObject}->Do(
        SQL => 'UPDATE general_catalog SET general_catalog_class = ? '
            . 'WHERE general_catalog_class = ?',
        Bind => [ \$Param{ClassNew}, \$Param{ClassOld} ],
    );
}

=item ItemList()

return a list as hash reference of one general catalog class

    my $HashRef = $GeneralCatalogObject->ItemList(
        Class         => 'ITSM::Service::Type',
        Functionality => 'active',               # (optional) string or array reference
        Valid         => 0,                      # (optional) default 1
    );

=cut

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

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

    # set default value
    if ( !defined $Param{Valid} ) {
        $Param{Valid} = 1;
    }

    # create sql string
    my $SQL  = 'SELECT id, name FROM general_catalog WHERE general_catalog_class = ? ';
    my @BIND = ( \$Param{Class} );

    # add valid string to sql string
    if ( $Param{Valid} ) {
        $SQL .= 'AND valid_id = 1 ';
    }

    # create cache key
    my $CacheKey = $Param{Class} . '####' . $Param{Valid} . '####';

    # add functionality to sql string
    if ( $Param{Functionality} ) {

        # create array reference, if functionality is give as sting
        if ( ref $Param{Functionality} ne 'ARRAY' ) {
            $Param{Functionality} = [ $Param{Functionality} ];
        }

        # create functionality string
        my $FunctionalityString = join q{, }, map {'?'} @{ $Param{Functionality} };

        # add functionality list to cache key
        $CacheKey .= join q{####}, map {$_} @{ $Param{Functionality} };

        # create and add bind parameters
        my @BindParams = map { \$_ } @{ $Param{Functionality} };
        push @BIND, @BindParams;

        # add functionality string to sql string
        $SQL .= "AND functionality IN ($FunctionalityString)";
    }

    # check if result is already cached
    return $Self->{Cache}->{ItemList}->{$CacheKey}
        if $Self->{Cache}->{ItemList}->{$CacheKey};

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

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

    # check item
    if ( !%Data ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'Class not found in database!',
        );
        return;
    }

    # cache the result
    $Self->{Cache}->{ItemList}->{$CacheKey} = \%Data;

    return \%Data;
}

=item FunctionalityList()

return an array reference of all functionalities of a general catalog class

    my $ArrayRef = $GeneralCatalogObject->FunctionalityList(
        Class => 'ITSM::Service::Type',
    );

=cut

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

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

    # ask database
    $Self->{DBObject}->Prepare(
        SQL => 'SELECT DISTINCT(functionality) FROM general_catalog '
            . 'WHERE general_catalog_class = ? ORDER BY functionality',
        Bind => [ \$Param{Class} ],
    );

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

    return \@FunctionalityList;
}

=item ItemGet()

get a general catalog item

Return
    $ItemData{ItemID}
    $ItemData{Class}
    $ItemData{Name}
    $ItemData{Functionality}
    $ItemData{ValidID}
    $ItemData{Comment}
    $ItemData{CreateTime}
    $ItemData{CreateBy}
    $ItemData{ChangeTime}
    $ItemData{ChangeBy}

    my $ItemDataRef = $GeneralCatalogObject->ItemGet(
        ItemID  => 3,
    );

    or

    my $ItemDataRef = $GeneralCatalogObject->ItemGet(
        Class => 'ITSM::Service::Type',
        Name  => 'Item Name',
    );

=cut

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

    # check needed stuff
    if ( !$Param{ItemID} && ( !$Param{Class} || !$Param{Name} ) ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'Need ItemID OR Class and Name!'
        );
        return;
    }

    # create sql string
    my $SQL = 'SELECT id, general_catalog_class, name, functionality, valid_id, comments, '
        . 'create_time, create_by, change_time, change_by FROM general_catalog WHERE ';
    my @BIND;

    # add options to sql string
    if ( $Param{Class} && $Param{Name} ) {

        # check if result is already cached
        return $Self->{Cache}->{ItemGet}->{Class}->{ $Param{Class} }->{ $Param{Name} }
            if $Self->{Cache}->{ItemGet}->{Class}->{ $Param{Class} }->{ $Param{Name} };

        # add class and name to sql string
        $SQL .= 'general_catalog_class = ? AND name = ?';
        push @BIND, ( \$Param{Class}, \$Param{Name} );
    }
    else {

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

        # add item id to sql string
        $SQL .= 'id = ?';
        push @BIND, \$Param{ItemID};
    }

    # ask database
    $Self->{DBObject}->Prepare(
        SQL   => $SQL,
        Bind  => \@BIND,
        Limit => 1,
    );

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

    # check item
    if ( !$ItemData{ItemID} ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => 'Item not found in database!',
        );
        return;
    }

    # cache the result
    $Self->{Cache}->{ItemGet}->{Class}->{ $ItemData{Class} }->{ $ItemData{Name} } = \%ItemData;
    $Self->{Cache}->{ItemGet}->{ItemID}->{ $ItemData{ItemID} } = \%ItemData;

    return \%ItemData;
}

=item ItemAdd()

add a new general catalog item

    my $ItemID = $GeneralCatalogObject->ItemAdd(
        Class         => 'ITSM::Service::Type',
        Name          => 'Item Name',
        Functionality => 'Func3',                # (optional)
        ValidID       => 1,
        Comment       => 'Comment',              # (optional)
        UserID        => 1,
    );

=cut

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

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

    # set default values
    for my $Argument (qw(Functionality Comment)) {
        $Param{$Argument} ||= '';
    }

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

    # find exiting item with same name
    $Self->{DBObject}->Prepare(
        SQL => 'SELECT id FROM general_catalog '
            . 'WHERE general_catalog_class = ? AND name = ?',
        Bind => [ \$Param{Class}, \$Param{Name} ],
        Limit => 1,
    );

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

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

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

    # insert new item
    return if !$Self->{DBObject}->Do(
        SQL => 'INSERT INTO general_catalog '
            . '(general_catalog_class, name, functionality, valid_id, comments, '
            . 'create_time, create_by, change_time, change_by) VALUES '
            . '(?, ?, ?, ?, ?, current_timestamp, ?, current_timestamp, ?)',
        Bind => [
            \$Param{Class},         \$Param{Name},
            \$Param{Functionality}, \$Param{ValidID},
            \$Param{Comment},       \$Param{UserID},
            \$Param{UserID},
        ],
    );

    # find id of new item
    $Self->{DBObject}->Prepare(
        SQL => 'SELECT id FROM general_catalog '
            . 'WHERE general_catalog_class = ? AND name = ?',
        Bind => [ \$Param{Class}, \$Param{Name} ],
        Limit => 1,
    );

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

    return $ItemID;
}

=item ItemUpdate()

update a existing general catalog item

    my $True = $GeneralCatalogObject->ItemUpdate(
        ItemID        => 123,
        Name          => 'Item Name',
        Functionality => 'Func3',      # (optional)
        ValidID       => 1,
        Comment       => 'Comment',    # (optional)
        UserID        => 1,
    );

=cut

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

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

    # set default values
    for my $Argument (qw(Functionality Comment)) {
        $Param{$Argument} ||= '';
    }

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

    # get class of item
    $Self->{DBObject}->Prepare(
        SQL   => 'SELECT general_catalog_class, functionality FROM general_catalog WHERE id = ?',
        Bind  => [ \$Param{ItemID} ],
        Limit => 1,
    );

    # fetch the result
    my $Class;
    my $OldFunctionality;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        $Class = $Row[0];
        $OldFunctionality = $Row[1] || '';
    }

    if ( !$Class ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "Can't update item! General catalog item not found in this class.",
        );
        return;
    }

    # find exiting item with same name
    $Self->{DBObject}->Prepare(
        SQL   => 'SELECT id FROM general_catalog WHERE general_catalog_class = ? AND name = ?',
        Bind  => [ \$Class, \$Param{Name} ],
        Limit => 1,
    );

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

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

    # count the functionality
    $Self->{DBObject}->Prepare(
        SQL => 'SELECT COUNT(functionality) FROM general_catalog '
            . 'WHERE general_catalog_class = ? AND functionality = ?',
        Bind => [ \$Class, \$OldFunctionality ],
        Limit => 1,
    );

    # fetch the result
    my $LastFunctionality = 1;
    while ( my @Row = $Self->{DBObject}->FetchrowArray() ) {
        if ( $Row[0] > 1 || $Param{Functionality} eq $OldFunctionality ) {
            $LastFunctionality = 0;
        }
    }

    # abort update, if functionality is the last one
    if ($LastFunctionality) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message =>
                "Can't update item! The functionality of this item is the last in this class.",
        );
        return;
    }

    # reset cache
    delete $Self->{Cache}->{ItemGet}->{Class}->{$Class}->{ $Param{Name} };
    delete $Self->{Cache}->{ItemGet}->{ItemID}->{ $Param{ItemID} };
    delete $Self->{Cache}->{ItemList};

    return $Self->{DBObject}->Do(
        SQL => 'UPDATE general_catalog SET '
            . 'name = ?, functionality = ?,'
            . 'valid_id = ?, comments = ?, '
            . 'change_time = current_timestamp, change_by = ? '
            . 'WHERE id = ?',
        Bind => [
            \$Param{Name},    \$Param{Functionality},
            \$Param{ValidID}, \$Param{Comment},
            \$Param{UserID},  \$Param{ItemID},
        ],
    );
}

1;

=back

=head1 TERMS AND CONDITIONS

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

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

=cut

=head1 VERSION

$Revision: 1.43 $ $Date: 2008/07/23 09:02:02 $

=cut

# --
# GeneralCatalog.t - general catalog tests
# Copyright (C) 2001-2008 OTRS AG, http://otrs.org/
# --
# $Id: GeneralCatalog.t,v 1.17 2008/03/19 15:06:14 mh Exp $
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see http://www.gnu.org/licenses/gpl-2.0.txt.
# --

use strict;
use warnings;
use utf8;

use vars qw($Self);

use Kernel::System::GeneralCatalog;
use Kernel::System::User;

$Self->{GeneralCatalogObject} = Kernel::System::GeneralCatalog->new( %{$Self} );
$Self->{UserObject}           = Kernel::System::User->new( %{$Self} );

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

# create needed users
my @UserIDs;
{

    # disable email checks to create new user
    my $CheckEmailAddressesOrg = $Self->{ConfigObject}->Get('CheckEmailAddresses') || 1;
    $Self->{ConfigObject}->Set(
        Key   => 'CheckEmailAddresses',
        Value => 0,
    );

    for my $Counter ( 1 .. 2 ) {

        # create new users for the tests
        my $UserID = $Self->{UserObject}->UserAdd(
            UserFirstname => 'GeneralCatalog' . $Counter,
            UserLastname  => 'UnitTest',
            UserLogin     => 'UnitTest-GeneralCatalog-' . $Counter . int rand 1_000_000,
            UserEmail     => 'UnitTest-GeneralCatalog-' . $Counter . '@localhost',
            ValidID       => 1,
            ChangeUserID  => 1,
        );

        push @UserIDs, $UserID;
    }

    # restore original email check param
    $Self->{ConfigObject}->Set(
        Key   => 'CheckEmailAddresses',
        Value => $CheckEmailAddressesOrg,
    );
}

# create needed random classes
my @ClassRand;

for my $Counter ( 1 .. 3 ) {

    push @ClassRand, int rand 1_000_000;
}

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

my $ItemData = [

    # this item is NOT complete and must not be added
    {
        Add => {
            Name    => 'TestItem1',
            ValidID => 1,
            UserID  => 1,
        },
    },

    # this item is NOT complete and must not be added
    {
        Add => {
            Class   => 'UnitTest::TestClass' . $ClassRand[0],
            ValidID => 1,
            UserID  => 1,
        },
    },

    # this item is NOT complete and must not be added
    {
        Add => {
            Class  => 'UnitTest::TestClass' . $ClassRand[0],
            Name   => 'TestItem2',
            UserID => 1,
        },
    },

    # this item is NOT complete and must not be added
    {
        Add => {
            Class   => 'UnitTest::TestClass' . $ClassRand[0],
            Name    => 'TestItem3',
            ValidID => 1,
        },
    },

    # this item must be inserted sucessfully
    {
        Add => {
            Class   => 'UnitTest::TestClass' . $ClassRand[0],
            Name    => 'TestItem4',
            ValidID => 1,
            UserID  => 1,
        },
        AddGet => {
            Class         => 'UnitTest::TestClass' . $ClassRand[0],
            Name          => 'TestItem4',
            Functionality => '',
            ValidID       => 1,
            Comment       => '',
            CreateBy      => 1,
            ChangeBy      => 1,
        },
    },

    # this item have the same name as one test before and must not be added
    {
        Add => {
            Class   => 'UnitTest::TestClass' . $ClassRand[0],
            Name    => 'TestItem4',
            ValidID => 1,
            UserID  => 1,
        },
    },

    # the item one add-test before must be NOT updated (Func '' is the last in this class)
    {
        Update => {
            Name          => 'TestItem4UPDATE1',
            Functionality => 'Test',
            ValidID       => $UserIDs[0],
            UserID        => $UserIDs[0],
        },
    },

    # this item must be inserted sucessfully
    {
        Add => {
            Class         => 'UnitTest::TestClass' . $ClassRand[0],
            Name          => 'TestItem5',
            Functionality => 'Test1',
            ValidID       => 1,
            UserID        => 1,
        },
        AddGet => {
            Class         => 'UnitTest::TestClass' . $ClassRand[0],
            Name          => 'TestItem5',
            Functionality => 'Test1',
            ValidID       => 1,
            Comment       => '',
            CreateBy      => 1,
            ChangeBy      => 1,
        },
    },

    # the item one add-test before must be NOT updated (item update arguments NOT complete)
    {
        Update => {
            ValidID => $UserIDs[0],
            UserID  => $UserIDs[0],
        },
    },

    # the item one add-test before must be NOT updated (item update arguments NOT complete)
    {
        Update => {
            Name   => 'TestItem5UPDATE1',
            UserID => $UserIDs[0],
        },
    },

    # the item one add-test before must be NOT updated (item update arguments NOT complete)
    {
        Update => {
            Functionality => 'Test1',
            ValidID       => $UserIDs[0],
        },
    },

    # the item one add-test before must be NOT updated (Func 'test1' is the last in this class)
    {
        Update => {
            Name    => 'TestItem5',
            ValidID => $UserIDs[0],
            UserID  => $UserIDs[0],
        },
    },

    # the item one add-test before must be NOT updated (Func 'test1' is the last in this class)
    {
        Update => {
            Name          => 'TestItem5',
            Functionality => 'Test2',
            ValidID       => $UserIDs[0],
            UserID        => $UserIDs[0],
        },
    },

    # the item one add-test before must be updated (item update arguments are complete)
    {
        Update => {
            Name          => 'TestItem5UPDATE2',
            Functionality => 'Test1',
            ValidID       => $UserIDs[0],
            UserID        => $UserIDs[0],
        },
        UpdateGet => {
            Name          => 'TestItem5UPDATE2',
            Functionality => 'Test1',
            ValidID       => $UserIDs[0],
            Comment       => '',
            CreateBy      => 1,
            ChangeBy      => $UserIDs[0],
        },
    },

    # the item one add-test before must be updated (item update arguments are complete)
    {
        Update => {
            Name          => 'TestItem5UPDATE3',
            Functionality => 'Test1',
            ValidID       => 1,
            UserID        => 1,
        },
        UpdateGet => {
            Name          => 'TestItem5UPDATE3',
            Functionality => 'Test1',
            ValidID       => 1,
            Comment       => '',
            CreateBy      => 1,
            ChangeBy      => 1,
        },
    },

    # this template must be inserted sucessfully (check string cleaner function)
    {
        Add => {
            Class         => " \t \n \r Unit Test :: Test Class \t \n \r " . $ClassRand[0],
            Name          => " \t \n \r Test Item \t \n \r ",
            Functionality => " \t \n \r Test Functionality \t \n \r ",
            ValidID       => 1,
            Comment       => " \t \n \r Test Comment \t \n \r ",
            UserID        => 1,
        },
        AddGet => {
            Class         => 'UnitTest::TestClass' . $ClassRand[0],
            Name          => 'Test Item',
            Functionality => 'TestFunctionality',
            ValidID       => 1,
            Comment       => 'Test Comment',
            CreateBy      => 1,
            ChangeBy      => 1,
        },
    },

    # the item one add-test before must be updated sucessfully (check string cleaner function)
    {
        Update => {
            Name          => " \t \n \r Test Item UPDATE1 \t \n \r ",
            Functionality => " \t \n \r Test Func tiona lity \t \n \r ",
            ValidID       => $UserIDs[0],
            Comment       => " \t \n \r Test Comment UPDATE1 \t \n \r ",
            UserID        => $UserIDs[0],
        },
        UpdateGet => {
            Name          => 'Test Item UPDATE1',
            Functionality => 'TestFunctionality',
            ValidID       => $UserIDs[0],
            Comment       => 'Test Comment UPDATE1',
            CreateBy      => 1,
            ChangeBy      => $UserIDs[0],
        },
    },

    # this item must be inserted sucessfully (unicode checks)
    {
        Add => {
            Class         => 'UnitTest::TestClass©' . $ClassRand[1],
            Name          => ' ϒ ϡ Test Item Ʃ Ϟ ',
            Functionality => ' Ѡ Ѥ TestFunctionality Ϡ Ω ',
            ValidID       => 1,
            Comment       => ' Ϡ Я Test Comment Ѭ Ѡ ',
            UserID        => 1,
        },
        AddGet => {
            Class         => 'UnitTest::TestClass©' . $ClassRand[1],
            Name          => 'ϒ ϡ Test Item Ʃ Ϟ',
            Functionality => 'ѠѤTestFunctionalityϠΩ',
            ValidID       => 1,
            Comment       => 'Ϡ Я Test Comment Ѭ Ѡ',
            CreateBy      => 1,
            ChangeBy      => 1,
        },
    },

    # the item one add-test before must be updated sucessfully (unicode checks)
    {
        Update => {
            Name          => 'Test Item Ʃ ɤ UPDATE1',
            Functionality => ' Ѡ Ѥ TestFunctionality Ϡ Ω ',
            ValidID       => $UserIDs[1],
            Comment       => ' Test Comment љ ђ UPDATE1 ',
            UserID        => $UserIDs[1],
        },
        UpdateGet => {
            Name          => 'Test Item Ʃ ɤ UPDATE1',
            Functionality => 'ѠѤTestFunctionalityϠΩ',
            ValidID       => $UserIDs[1],
            Comment       => 'Test Comment љ ђ UPDATE1',
            CreateBy      => 1,
            ChangeBy      => $UserIDs[1],
        },
    },

    # this item must be inserted sucessfully (a second item with Functionality 'test1')
    {
        Add => {
            Class         => 'UnitTest::TestClass' . $ClassRand[0],
            Name          => 'TestItem6',
            Functionality => 'Test1',
            ValidID       => 1,
            UserID        => 1,
        },
        AddGet => {
            Class         => 'UnitTest::TestClass' . $ClassRand[0],
            Name          => 'TestItem6',
            Functionality => 'Test1',
            ValidID       => 1,
            Comment       => '',
            CreateBy      => 1,
            ChangeBy      => 1,
        },
    },

    # the item one add-test before must be updated (set functionality to '')
    {
        Update => {
            Name          => 'TestItem6UPDATE1',
            Functionality => '',
            ValidID       => 1,
            UserID        => 1,
        },
        UpdateGet => {
            Name          => 'TestItem6UPDATE1',
            Functionality => '',
            ValidID       => 1,
            Comment       => '',
            CreateBy      => 1,
            ChangeBy      => 1,
        },
    },

    # this item must be inserted sucessfully (special character checks)
    {
        Add => {
            Class         => 'UnitTest::TestClass[test]%*\\' . $ClassRand[1],
            Name          => ' [test]%*\\ Test Item [test]%*\\ ',
            Functionality => ' [test]%*\\ TestFunctionality [test]%*\\ ',
            ValidID       => 1,
            Comment       => ' [test]%*\\ Test Comment [test]%*\\ ',
            UserID        => 1,
        },
        AddGet => {
            Class         => 'UnitTest::TestClass[test]%*\\' . $ClassRand[1],
            Name          => '[test]%*\\ Test Item [test]%*\\',
            Functionality => '[test]%*\\TestFunctionality[test]%*\\',
            ValidID       => 1,
            Comment       => '[test]%*\\ Test Comment [test]%*\\',
            CreateBy      => 1,
            ChangeBy      => 1,
        },
    },

    # the item one add-test before must be updated sucessfully (special character checks)
    {
        Update => {
            Name          => ' [test]%*\\ Test Item UPDATE1 [test]%*\\ ',
            Functionality => ' [test]%*\\ TestFunctionality [test]%*\\ ',
            ValidID       => $UserIDs[1],
            Comment       => ' [test]%*\\ Test Comment UPDATE1 [test]%*\\ ',
            UserID        => $UserIDs[1],
        },
        UpdateGet => {
            Name          => '[test]%*\\ Test Item UPDATE1 [test]%*\\',
            Functionality => '[test]%*\\TestFunctionality[test]%*\\',
            ValidID       => $UserIDs[1],
            Comment       => '[test]%*\\ Test Comment UPDATE1 [test]%*\\',
            CreateBy      => 1,
            ChangeBy      => $UserIDs[1],
        },
    },
];

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

my $TestCount = 1;
my $LastAddedItemID;
my %AddedItemCounter;

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

    if ( $Item->{Add} ) {

        # add new item
        my $ItemID = $Self->{GeneralCatalogObject}->ItemAdd(
            %{ $Item->{Add} },
        );

        # check if item was added successfully or not
        if ( $Item->{AddGet} ) {

            $Self->True(
                $ItemID,
                "Test $TestCount: ItemAdd() - ItemKey: $ItemID",
            );

            if ($ItemID) {

                # set last item id variable
                $LastAddedItemID = $ItemID;

                # increment the added item counter
                $AddedItemCounter{ $Item->{AddGet}->{Class} }++;
            }
        }
        else {
            $Self->False(
                $ItemID,
                "Test $TestCount: ItemAdd()",
            );
        }

        # get item data to check the values after creation of item
        my $ItemGet = $Self->{GeneralCatalogObject}->ItemGet(
            ItemID => $ItemID,
            UserID => $Item->{Add}->{UserID},
        );

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

    if ( $Item->{Update} ) {

        # check last item id varaible
        if ( !$LastAddedItemID ) {
            $Self->False(
                1,
                "Test $TestCount: NO LAST ITEM ID GIVEN",
            );
        }

        # update the item
        my $UpdateSucess = $Self->{GeneralCatalogObject}->ItemUpdate(
            %{ $Item->{Update} },
            ItemID => $LastAddedItemID,
        );

        # check if item was updated successfully or not
        if ( $Item->{UpdateGet} ) {
            $Self->True(
                $UpdateSucess,
                "Test $TestCount: ItemUpdate() - ItemKey: $LastAddedItemID",
            );
        }
        else {
            $Self->False(
                $UpdateSucess,
                "Test $TestCount: ItemUpdate()",
            );
        }

        # get item data to check the values after the update
        my $ItemGet2 = $Self->{GeneralCatalogObject}->ItemGet(
            ItemID => $LastAddedItemID,
            UserID => $Item->{Update}->{UserID},
        );

        # check item data after update
        for my $ItemAttribute ( keys %{ $Item->{UpdateGet} } ) {
            $Self->Is(
                $ItemGet2->{$ItemAttribute} || '',
                $Item->{UpdateGet}->{$ItemAttribute} || '',
                "Test $TestCount: ItemGet() - $ItemAttribute",
            );
        }
    }

    $TestCount++;
}

# ------------------------------------------------------------ #
# make preparations for later tests
# ------------------------------------------------------------ #

# create needed arrays
my %ExistingClassesTmp;
ITEM:
for my $Item ( @{$ItemData} ) {
    next ITEM if !$Item->{AddGet}->{Class};
    $ExistingClassesTmp{ $Item->{AddGet}->{Class} } = 1;
}
my @ExistingClasses = sort keys %ExistingClassesTmp;

my %ExistingFunctionalitiesTmp;
ITEM:
for my $Item ( @{$ItemData} ) {
    $Item->{UpdateGet}->{Functionality} ||= '';
    $ExistingFunctionalitiesTmp{ $Item->{UpdateGet}->{Functionality} } = 1;
}
my @ExistingFunctionalities = sort keys %ExistingFunctionalitiesTmp;

my @NonExistingClasses = ( 'UnitTest::NoExistingClass1', 'UnitTest::NoExistingClass2' );

# ------------------------------------------------------------ #
# ClassList test 1
# ------------------------------------------------------------ #

my $ClassList1 = $Self->{GeneralCatalogObject}->ClassList();

for my $Class (@ExistingClasses) {

    my $ClassCount = 0;
    if ( $ClassList1 && ref $ClassList1 eq 'ARRAY' ) {
        $ClassCount = grep { $_ eq $Class } @{$ClassList1};
    }

    $Self->Is(
        $ClassCount,
        1,
        "Test $TestCount: ClassList() - $Class listed",
    );

    $TestCount++;
}

# ------------------------------------------------------------ #
# ItemList test 1
# ------------------------------------------------------------ #

for my $Class (@NonExistingClasses) {

    my $ItemList = $Self->{GeneralCatalogObject}->ItemList(
        Class => $Class,
        Valid => 0,
    );

    $Self->False(
        $ItemList,
        "Test $TestCount: ItemList() - $Class not exists",
    );

    $TestCount++;
}

# ------------------------------------------------------------ #
# ItemList test 2
# ------------------------------------------------------------ #

for my $Class (@ExistingClasses) {

    my $ItemList = $Self->{GeneralCatalogObject}->ItemList(
        Class => $Class,
        Valid => 0,
    );

    my $ListCount = 'NULL';
    if ( defined $ItemList && ref $ItemList eq 'HASH' ) {
        $ListCount = keys %{$ItemList};
    }

    $Self->Is(
        $ListCount,
        $AddedItemCounter{$Class},
        "Test $TestCount: ItemList() - $Class correct number of items",
    );

    $TestCount++;
}

# ------------------------------------------------------------ #
# FunctionalityList test 1
# ------------------------------------------------------------ #

my %FunctionalityList1;
map { $FunctionalityList1{$_} = 1 } @ExistingFunctionalities;

for my $Class (@ExistingClasses) {

    my $FunctionalityList = $Self->{GeneralCatalogObject}->FunctionalityList(
        Class => $Class,
    );

    $Self->True(
        $FunctionalityList && ref $FunctionalityList eq 'ARRAY',
        "Test $TestCount: FunctionalityList() - return a array reference",
    );

    for my $Functionality ( @{$FunctionalityList} ) {
        delete $FunctionalityList1{$Functionality};
    }

    $TestCount++;
}

$Self->True(
    !keys %FunctionalityList1,
    "Test $TestCount: FunctionalityList()",
);

$TestCount++;

# ------------------------------------------------------------ #
# ClassRename test 1 (check normal rename)
# ------------------------------------------------------------ #

CLASS:
for my $Class (@ExistingClasses) {

    my $OldItemList = $Self->{GeneralCatalogObject}->ItemList(
        Class => $Class,
        Valid => 0,
    );

    my $Success = $Self->{GeneralCatalogObject}->ClassRename(
        ClassOld => $Class,
        ClassNew => $Class . 'RENAME1',
    );

    if ( !$Success ) {
        $Self->False(
            1,
            "Test $TestCount: ClassRename() - Rename failed",
        );
        next CLASS;
    }

    my $NewItemList = $Self->{GeneralCatalogObject}->ItemList(
        Class => $Class . 'RENAME1',
        Valid => 0,
    );

    if (
        !$OldItemList
        || !$NewItemList
        || ref $OldItemList ne 'HASH'
        || ref $NewItemList ne 'HASH'
        )
    {
        $Self->False(
            1,
            "Test $TestCount: ClassRename() - ItemList failed",
        );
        next CLASS;
    }

    OLDKEY:
    for my $OldKey ( keys %{$OldItemList} ) {

        if ( !exists $NewItemList->{$OldKey} ) {
            $NewItemList->{FailedDummy} = 1;
            next OLDKEY;
        }

        next OLDKEY if $OldItemList->{$OldKey} ne $NewItemList->{$OldKey};

        delete $NewItemList->{$OldKey};
    }

    $Self->True(
        !keys %{$NewItemList},
        "Test $TestCount: ClassRename()",
    );

    $TestCount++;
}

# ------------------------------------------------------------ #
# ClassRename test 2 (check string cleaner function)
# ------------------------------------------------------------ #

CLASS:
for my $Class (@ExistingClasses) {

    my $OldItemList = $Self->{GeneralCatalogObject}->ItemList(
        Class => $Class . 'RENAME1',
        Valid => 0,
    );

    my $Success = $Self->{GeneralCatalogObject}->ClassRename(
        ClassOld => $Class . 'RENAME1',
        ClassNew => ' ' . $Class . "RE NA ME 2 \n \r \t ",
    );

    if ( !$Success ) {
        $Self->False(
            1,
            "Test $TestCount: ClassRename() - Rename failed",
        );
        next CLASS;
    }

    my $NewItemList = $Self->{GeneralCatalogObject}->ItemList(
        Class => $Class . 'RENAME2',
        Valid => 0,
    );

    if (
        !$OldItemList
        || !$NewItemList
        || ref $OldItemList ne 'HASH'
        || ref $NewItemList ne 'HASH'
        )
    {
        $Self->False(
            1,
            "Test $TestCount: ClassRename() - ItemList failed",
        );
        next CLASS;
    }

    OLDKEY:
    for my $OldKey ( keys %{$OldItemList} ) {

        if ( !exists $NewItemList->{$OldKey} ) {
            $NewItemList->{FailedDummy} = 1;
            next OLDKEY;
        }

        next OLDKEY if $OldItemList->{$OldKey} ne $NewItemList->{$OldKey};

        delete $NewItemList->{$OldKey};
    }

    $Self->True(
        !keys %{$NewItemList},
        "Test $TestCount: ClassRename()",
    );

    $TestCount++;
}

# ------------------------------------------------------------ #
# ClassRename test 2 (identical name test)
# ------------------------------------------------------------ #

for my $Class (@ExistingClasses) {

    my $Success = $Self->{GeneralCatalogObject}->ClassRename(
        ClassOld => $Class . 'RENAME2',
        ClassNew => $Class . 'RENAME2',
    );

    $Self->True(
        $Success,
        "Test $TestCount: ClassRename() - oldname and newname identical",
    );

    $TestCount++;
}

# ------------------------------------------------------------ #
# ClassRename test 3 (new class name already exists)
# ------------------------------------------------------------ #

$Self->{GeneralCatalogObject}->ItemAdd(
    Class         => 'UnitTest::TestClass' . $ClassRand[2],
    Name          => 'Dummy',
    Functionality => '',
    ValidID       => 1,
    UserID        => 1,
);

for my $Class (@ExistingClasses) {

    my $Success = $Self->{GeneralCatalogObject}->ClassRename(
        ClassOld => $Class . 'RENAME2',
        ClassNew => 'UnitTest::TestClass' . $ClassRand[2],
    );

    $Self->False(
        $Success,
        "Test $TestCount: ClassRename() - new class name already exists",
    );

    $TestCount++;
}

1;
