MasterSlave
1.0.3
OTRS AG
http://otrs.org/
GNU AFFERO GENERAL PUBLIC LICENSE Version 3, November 2007
Build for MasterSlave 1.0.3.
Fixed bug #5241 - MasterSlave Update also updates non-slave child tickets.
Init release.
Includes "Ticket Master/Slave" feature.
Enthält "Ticket Master/Slave" Funktionalität.
2.4.x
<b>ATTENTION</b>
<br/>
<br/>
You're upgrading from an earlier version of the MasterSlave module.
The behavior of the MasterSlave module changed slightly since version
1.0.1.<br/>
<br/>
If you're upgrading from that version, please check
<a href="http://faq.otrs.org/otrs/public.pl?Action=PublicFAQ;CategoryID=33;ItemID=365" target="_blank">
this FAQ article</a> for details.
<br/>
<br/>
((enjoy))<br/>
<br/>
2017-10-09 14:32:27
opms.otrs.com
PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iaXNvLTg4NTktMSIgPz4KPG90cnNfY29uZmlnIHZlcnNpb249IjEuMCIgaW5pdD0iQ2hhbmdlcyI+CiAgICA8Q29uZmlnSXRlbSBOYW1lPSJNYXN0ZXJUaWNrZXRGcmVlVGV4dEZpZWxkIiBSZXF1aXJlZD0iMCIgVmFsaWQ9IjEiPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJlbiI+RGVmaW5lIGZyZWUgdGV4dCBmaWVsZCBmb3IgbWFzdGVyIHRpY2tldCBmZWF0dXJlLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5EZWZpbml0aW9uIGRlcyBGcmVpU2NobPxzc2VsRmVsZGVzIGb8ciBkaWUgTWFzdGVyLVRpY2tldCBGdW5rdGlvbmFsaXTkdC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaWNrZXQ8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5NYXN0ZXJTbGF2ZTwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxPcHRpb24gU2VsZWN0ZWRJRD0iMTIiPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSIxIj4xPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSIyIj4yPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSIzIj4zPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSI0Ij40PC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSI1Ij41PC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSI2Ij42PC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSI3Ij43PC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSI4Ij44PC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSI5Ij45PC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSIxMCI+MTA8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjExIj4xMTwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMTIiPjEyPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSIxMyI+MTM8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IjE0Ij4xNDwvSXRlbT4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iMTUiPjE1PC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSIxNiI+MTY8L0l0ZW0+CiAgICAgICAgICAgIDwvT3B0aW9uPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpY2tldDo6RXZlbnRNb2R1bGVQb3N0IyMjTWFzdGVyU2xhdmUiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5SZWdpc3RyYXRpb24gb2YgdGhlIHRpY2tldCBldmVudCBtb2R1bC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxEZXNjcmlwdGlvbiBMYW5nPSJkZSI+UmVnaXN0cmllcnVuZyBkZXMgVGlja2V0LUV2ZW50LU1vZHVsZXMuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGlja2V0PC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+TWFzdGVyU2xhdmU8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iTW9kdWxlIj5LZXJuZWw6OlN5c3RlbTo6VGlja2V0OjpFdmVudDo6TWFzdGVyU2xhdmU8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9IkV2ZW50Ij4oQXJ0aWNsZUNyZWF0ZXxBcnRpY2xlU2VuZHxUaWNrZXRTdGF0ZVVwZGF0ZXxUaWNrZXRQcmlvcml0eVVwZGF0ZXxUaWNrZXRQZW5kaW5nVGltZVVwZGF0ZXxUaWNrZXRMb2NrVXBkYXRlfFRpY2tldE93bmVyVXBkYXRlfFRpY2tldFJlc3BvbnNpYmxlVXBkYXRlfFRpY2tldEZyZWVUZXh0VXBkYXRlKTwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlByZUFwcGxpY2F0aW9uTW9kdWxlIyMjQWdlbnRNYXN0ZXJTbGF2ZVRpY2tldFByZXBhcmUiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5UaGlzIG1vZHVsZSBpcyBwcmVwYXJpbmcgbWFzdGVyL3NsYXZlIHB1bGxkb3duIGluIGVtYWlsIGFuZCBwaG9uZSB0aWNrZXQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRpZXNlcyBNb2R1bCBkaWVudCB6dXIgZ2VuZXJpZXJ1bmcgZGVzIE1hc3Rlci9TbGF2ZSBQdWxsZG93bnMgaW4gRW1haWwtIHVuZCBUZWxlZm9uIFRpY2tldC48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaWNrZXQ8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5NYXN0ZXJTbGF2ZTwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxTdHJpbmcgUmVnZXg9IiI+S2VybmVsOjpNb2R1bGVzOjpBZ2VudE1hc3RlclNsYXZlUHJlcGFyZVRpY2tldDwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpY2tldEZyZWVLZXkxMiIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIxIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPkRlZmluZSB0aGUgZnJlZSBrZXkgZmllbGQgMTIgZm9yIHRpY2tldHMuIFdpdGggdGhpcyBzZXR0aW5nIHlvdSBjYW4gZGVmaW5lIGEgbmV3IHRpY2tldCBhdHRyaWJ1dGUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaXRpb24gZGVzIEZyZWlTY2hs/HNzZWxGZWxkZXMgMTIgZvxyIFRpY2tldHMuIEhpZXL8YmVyIGv2bm5lbiB6dXPkdHpsaWNoZSBUaWNrZXRhdHRyaWJ1dGUgZGVmaW5pZXJ0IHdlcmRlbi48L0Rlc2NyaXB0aW9uPgogICAgICAgIDxHcm91cD5UaWNrZXQ8L0dyb3VwPgogICAgICAgIDxTdWJHcm91cD5Db3JlOjpUaWNrZXRGcmVlVGV4dDwvU3ViR3JvdXA+CiAgICAgICAgPFNldHRpbmc+CiAgICAgICAgICAgIDxIYXNoPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJNYXN0ZXJUaWNrZXQiPk1hc3RlclRpY2tldDwvSXRlbT4KICAgICAgICAgICAgPC9IYXNoPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpY2tldEZyZWVLZXkxMjo6RGVmYXVsdFNlbGVjdGlvbiIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIwIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlRoZSBkZWZhdWx0IHNlbGVjdGlvbiBvZiBmcmVlIGtleSBmaWVsZCAxMiBpZiBtb3JlIHRoYW4gb25lIG9wdGlvbiBpcyBkZWZpbmVkLjwvRGVzY3JpcHRpb24+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImRlIj5EaWUgU3RhbmRhcmQtQXVzd2FobCBkZXMgRnJlaVNjaGz8c3NlbEZlbGRlcyAxMiB3ZW5uIG1laHJlcmUgT3B0aW9uZW4gZGVmaW5pZXJ0IHNpbmQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGlja2V0PC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6VGlja2V0RnJlZVRleHQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPjwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KICAgIDxDb25maWdJdGVtIE5hbWU9IlRpY2tldEZyZWVUZXh0MTIiIFJlcXVpcmVkPSIwIiBWYWxpZD0iMSI+CiAgICAgICAgPERlc2NyaXB0aW9uIExhbmc9ImVuIj5EZWZpbmUgdGhlIEZyZWVUZXh0RmllbGQgMTIgZm9yIHRpY2tldHMuIFdpdGggdGhpcyBzZXR0aW5nIHlvdSBjYW4gZGVmaW5lIGEgbmV3IHRpY2tldCBhdHRyaWJ1dGUuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRlZmluaXRpb24gZGVzIEZyZWlUZXh0RmVsZGVzIDEyIGb8ciBUaWNrZXRzLiBIaWVy/GJlciBr9m5uZW4genVz5HR6bGljaGUgVGlja2V0YXR0cmlidXRlIGRlZmluaWVydCB3ZXJkZW4uPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGlja2V0PC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6VGlja2V0RnJlZVRleHQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8SGFzaD4KICAgICAgICAgICAgICAgIDxJdGVtIEtleT0iIj4tPC9JdGVtPgogICAgICAgICAgICAgICAgPEl0ZW0gS2V5PSJTbGF2ZSI+U2xhdmU8L0l0ZW0+CiAgICAgICAgICAgICAgICA8SXRlbSBLZXk9Ik1hc3RlciI+TWFzdGVyPC9JdGVtPgogICAgICAgICAgICA8L0hhc2g+CiAgICAgICAgPC9TZXR0aW5nPgogICAgPC9Db25maWdJdGVtPgogICAgPENvbmZpZ0l0ZW0gTmFtZT0iVGlja2V0RnJlZVRleHQxMjo6RGVmYXVsdFNlbGVjdGlvbiIgUmVxdWlyZWQ9IjAiIFZhbGlkPSIwIj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZW4iPlRoZSBkZWZhdWx0IHNlbGVjdGlvbiBvZiBGcmVlVGV4dEZpZWxkIDEyIGlmIG1vcmUgdGhhbiBvbmUgb3B0aW9uIGlzIGRlZmluZWQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8RGVzY3JpcHRpb24gTGFuZz0iZGUiPkRpZSBTdGFuZGFyZC1BdXN3YWhsIGRlcyBGcmVpVGV4dEZlbGRlcyAxMiB3ZW5uIG1laHJlcmUgT3B0aW9uZW4gZGVmaW5pZXJ0IHNpbmQuPC9EZXNjcmlwdGlvbj4KICAgICAgICA8R3JvdXA+VGlja2V0PC9Hcm91cD4KICAgICAgICA8U3ViR3JvdXA+Q29yZTo6VGlja2V0RnJlZVRleHQ8L1N1Ykdyb3VwPgogICAgICAgIDxTZXR0aW5nPgogICAgICAgICAgICA8U3RyaW5nIFJlZ2V4PSIiPjwvU3RyaW5nPgogICAgICAgIDwvU2V0dGluZz4KICAgIDwvQ29uZmlnSXRlbT4KPC9vdHJzX2NvbmZpZz4K
IyAtLQojIENvcHlyaWdodCAoQykgMjAwMS0yMDE3IE9UUlMgQUcsIGh0dHA6Ly9vdHJzLmNvbS8KIyAtLQojIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCiMgdGhlIGVuY2xvc2VkIGZpbGUgQ09QWUlORyBmb3IgbGljZW5zZSBpbmZvcm1hdGlvbiAoQUdQTCkuIElmIHlvdQojIGRpZCBub3QgcmVjZWl2ZSB0aGlzIGZpbGUsIHNlZSBodHRwOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvYWdwbC50eHQuCiMgLS0KCnBhY2thZ2UgS2VybmVsOjpNb2R1bGVzOjpBZ2VudE1hc3RlclNsYXZlUHJlcGFyZVRpY2tldDsKCnVzZSBzdHJpY3Q7CnVzZSB3YXJuaW5nczsKCnN1YiBuZXcgewogICAgbXkgKCAkVHlwZSwgJVBhcmFtICkgPSBAXzsKCiAgICAjIGFsbG9jYXRlIG5ldyBoYXNoIGZvciBvYmplY3QKICAgIG15ICRTZWxmID0geyVQYXJhbX07CiAgICBibGVzcyggJFNlbGYsICRUeXBlICk7CgogICAgIyBjaGVjayBuZWVkZWQgT2JqZWN0cwogICAgZm9yIChxdyhQYXJhbU9iamVjdCBEQk9iamVjdCBMYXlvdXRPYmplY3QgTG9nT2JqZWN0IENvbmZpZ09iamVjdCBUaWNrZXRPYmplY3QpKSB7CiAgICAgICAgaWYgKCAhJFNlbGYtPnskX30gKSB7CiAgICAgICAgICAgICRTZWxmLT57TGF5b3V0T2JqZWN0fS0+RmF0YWxFcnJvciggTWVzc2FnZSA9PiAiR290IG5vICRfISIgKTsKICAgICAgICB9CiAgICB9CgogICAgcmV0dXJuICRTZWxmOwp9CgpzdWIgUHJlUnVuIHsKICAgIG15ICggJFNlbGYsICVQYXJhbSApID0gQF87CgogICAgIyBkbyBvbmx5IHVzZSB0aGlzIGluIHBob25lIGFuZCBlbWFpbCB0aWNrZXQKICAgIHJldHVybiBpZiAoICRTZWxmLT57QWN0aW9ufSAhfiAvXkFnZW50VGlja2V0KEVtYWlsfFBob25lKSQvICk7CgogICAgIyBnZXQgbWFzdGVyL3NsYXZlIHRpY2tldCBmcmVlIGZpZWxkCiAgICBteSAkQ291bnQgPSAkU2VsZi0+e0NvbmZpZ09iamVjdH0tPkdldCgnTWFzdGVyVGlja2V0RnJlZVRleHRGaWVsZCcpOwoKICAgICMgcmV0dXJuIGlmIG5vIGNvbmZpZyBvcHRpb24gaXMgdXNlZAogICAgcmV0dXJuIGlmICEkQ291bnQ7CgogICAgIyBkZWZpbmUgVGlja2V0RnJlZVRleHQgZmllbGQKICAgIG15ICRUaWNrZXRGcmVlVGV4dCA9ICdUaWNrZXRGcmVlVGV4dCcgLiAkQ291bnQ7CgogICAgIyBmaW5kIGFsbCBjdXJyZW50IG9wZW4gbWFzdGVyIHNsYXZlIHRpY2tldHMKICAgIG15IEBUaWNrZXRJRHMgPSAkU2VsZi0+e1RpY2tldE9iamVjdH0tPlRpY2tldFNlYXJjaCgKCiAgICAgICAgIyByZXN1bHQgKHJlcXVpcmVkKQogICAgICAgIFJlc3VsdCAgICAgICAgICA9PiAnQVJSQVknLAogICAgICAgICRUaWNrZXRGcmVlVGV4dCA9PiAnTWFzdGVyJywKICAgICAgICBTdGF0ZVR5cGUgICAgICAgPT4gJ09wZW4nLAoKICAgICAgICAjIHJlc3VsdCBsaW1pdAogICAgICAgIExpbWl0ICAgICAgPT4gNjAsCiAgICAgICAgVXNlcklEICAgICA9PiAkU2VsZi0+e1VzZXJJRH0sCiAgICAgICAgUGVybWlzc2lvbiA9PiAncm8nLAogICAgKTsKCiAgICAjIHNldCBmcmVlIGZpZWxkIGFzIHNob3duCiAgICAkU2VsZi0+e0NvbmZpZ09iamVjdH0tPnsiVGlja2V0OjpGcm9udGVuZDo6JFNlbGYtPntBY3Rpb259In0tPntUaWNrZXRGcmVlVGV4dH0tPnskQ291bnR9ID0gMTsKCiAgICAjIHNldCBmcmVlIGZpZWxkcwogICAgJFNlbGYtPntDb25maWdPYmplY3R9LT57JFRpY2tldEZyZWVUZXh0fSAgICAgICAgICAgPSB1bmRlZjsKICAgICRTZWxmLT57Q29uZmlnT2JqZWN0fS0+eyRUaWNrZXRGcmVlVGV4dH0tPnsnJ30gICAgID0gJy0nOwogICAgJFNlbGYtPntDb25maWdPYmplY3R9LT57JFRpY2tldEZyZWVUZXh0fS0+e01hc3Rlcn0gPSAnTmV3IE1hc3RlciBUaWNrZXQnOwogICAgZm9yIG15ICRUaWNrZXRJRCAoQFRpY2tldElEcykgewogICAgICAgIG15ICVUaWNrZXQgPSAkU2VsZi0+e1RpY2tldE9iamVjdH0tPlRpY2tldEdldCggVGlja2V0SUQgPT4gJFRpY2tldElEICk7CiAgICAgICAgbmV4dCBpZiAhJVRpY2tldDsKICAgICAgICAkU2VsZi0+e0NvbmZpZ09iamVjdH0tPnskVGlja2V0RnJlZVRleHR9LT57IlNsYXZlT2Y6JFRpY2tldHtUaWNrZXROdW1iZXJ9In0KICAgICAgICAgICAgPSAiU2xhdmUgb2YgVGlja2V0IyRUaWNrZXR7VGlja2V0TnVtYmVyfTogJFRpY2tldHtUaXRsZX0iOwogICAgfQoKICAgIHJldHVybjsKfQoKMTsK
# --
# Copyright (C) 2001-2017 OTRS AG, http://otrs.com/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::System::Ticket::Event::MasterSlave;

use strict;
use warnings;
use Kernel::System::LinkObject;

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

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

    # get needed objects
    for (
        qw(ConfigObject TicketObject LogObject UserObject CustomerUserObject SendmailObject TimeObject EncodeObject)
        )
    {
        $Self->{$_} = $Param{$_} || die "Got no $_!";
    }

    $Self->{LinkObject} = Kernel::System::LinkObject->new(%Param);

    return $Self;
}

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

    # check needed stuff
    for (qw(TicketID Event Config)) {
        if ( !$Param{$_} ) {
            $Self->{LogObject}->Log(
                Priority => 'error',
                Message  => "Need $_!"
            );
            return;
        }
    }

    # get ticket attributes
    my %Ticket         = $Self->{TicketObject}->TicketGet( TicketID => $Param{TicketID} );
    my $Count          = $Self->{ConfigObject}->Get('MasterTicketFreeTextField');
    my $TicketFreeText = 'TicketFreeText' . $Count;

    # link master/slave tickets
    if ( $Param{Event} eq 'TicketFreeTextUpdate' ) {
        if ( $Ticket{$TicketFreeText} && $Ticket{$TicketFreeText} =~ /^SlaveOf:(.+?)$/ ) {

            # lookup to find ticket id
            my $SourceKey = $Self->{TicketObject}->TicketIDLookup(
                TicketNumber => $1,
                UserID       => $Param{UserID},
            );

            # create link
            $Self->{LinkObject}->LinkAdd(
                SourceObject => 'Ticket',
                SourceKey    => $SourceKey,
                TargetObject => 'Ticket',
                TargetKey    => $Param{TicketID},
                Type         => 'ParentChild',
                State        => 'Valid',
                UserID       => $Param{UserID},
            );

            # reset free text field
            $Self->{TicketObject}->TicketFreeTextSet(
                Counter  => $Count,
                Value    => 'Slave',
                TicketID => $Param{TicketID},
                UserID   => $Param{UserID},
            );
        }
        return 1;
    }

    # check if it's a master/slave ticket
    return 1 if !$Ticket{$TicketFreeText};
    return 1 if $Ticket{$TicketFreeText} !~ /^(master|yes)$/i;

    # find slaves
    my %Links = $Self->{LinkObject}->LinkKeyList(
        Object1   => 'Ticket',
        Key1      => $Param{TicketID},
        Object2   => 'Ticket',
        State     => 'Valid',
        Type      => 'ParentChild',      # (optional)
        Direction => 'Target',           # (optional) default Both (Source|Target|Both)
        UserID    => $Param{UserID},
    );
    my @TicketIDs;
    for my $TicketID ( keys %Links ) {
        next if !$Links{$TicketID};

        # just take ticket with slave attributes for action
        my %Ticket = $Self->{TicketObject}->TicketGet( TicketID => $TicketID );
        next if !$Ticket{$TicketFreeText};
        next if $Ticket{$TicketFreeText} ne 'Slave';

        # remember ticket id
        push @TicketIDs, $TicketID;
    }

    # no slaves
    if ( !@TicketIDs ) {
        $Self->{LogObject}->Log(
            Priority => 'error',
            Message  => "No Slaves of ticket $Ticket{TicketID}!",
        );
        return 1;
    }

    # auto response action
    if ( $Param{Event} eq 'ArticleSend' ) {
        my @Index = $Self->{TicketObject}->ArticleIndex( TicketID => $Param{TicketID} );
        return 1 if !@Index;
        my %Article = $Self->{TicketObject}->ArticleGet( ArticleID => $Index[$#Index] );

        # mark ticket to prevent a loop
        $Self->{TicketObject}->HistoryAdd(
            TicketID     => $Param{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => 'MasterTicketAction: ArticleSend',
        );

        # perform action on linked tickets
        for my $TicketID (@TicketIDs) {
            next if !$Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: ArticleSend',
            );

            my %TicketSlave = $Self->{TicketObject}->TicketGet( TicketID => $TicketID );

            # no customer found, send no master message to customer
            if ( !$TicketSlave{CustomerUserID} ) {
                $Self->{TicketObject}->HistoryAdd(
                    TicketID     => $TicketID,
                    CreateUserID => $Param{UserID},
                    HistoryType  => 'Misc',
                    Name =>
                        "MasterTicket: no customer for this ticket found, send no master message to customer.",
                );
                next;
            }

            # no customer email found, send no master message to customer
            my %Customer = $Self->{CustomerUserObject}->CustomerUserDataGet(
                User => $TicketSlave{CustomerUserID},
            );
            if ( !$Customer{UserEmail} ) {
                $Self->{TicketObject}->HistoryAdd(
                    TicketID     => $TicketID,
                    CreateUserID => $Param{UserID},
                    HistoryType  => 'Misc',
                    Name =>
                        "MasterTicket: no customer email found, send no master message to customer.",
                );
                next;
            }

            # set recipient of slave ticket
            $Article{To} = $Customer{UserEmail};

            # rebuild subject
            $Self->{ConfigObject}->Set(
                Key   => 'Ticket::SubjectCleanAllNumbers',
                Value => 1,
            );
            my $Subject = $Self->{TicketObject}->TicketSubjectBuild(
                TicketNumber => $TicketSlave{TicketNumber},
                Subject      => $Article{Subject} || '',
            );

            # send article again
            $Self->{TicketObject}->ArticleSend(
                %Article,
                Subject        => $Subject,
                Cc             => '',
                Bcc            => '',
                HistoryType    => 'SendAnswer',
                HistoryComment => "Sent answer to '$Article{To}' based on master ticket.",
                TicketID       => $TicketID,
                UserID         => $Param{UserID},
            );
        }
        return 1;
    }

    # article create
    elsif ( $Param{Event} eq 'ArticleCreate' ) {
        my @Index = $Self->{TicketObject}->ArticleIndex( TicketID => $Param{TicketID} );
        return 1 if !@Index;
        my %Article = $Self->{TicketObject}->ArticleGet( ArticleID => $Index[$#Index] );

        # mark ticket to prevent a loop
        $Self->{TicketObject}->HistoryAdd(
            TicketID     => $Param{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => "MasterTicketAction: ArticleCreate",
        );

        # do not process email articles (already done in ArticleSend event!)
        if ( $Article{SenderType} eq 'agent' && $Article{ArticleType} eq 'email-external' ) {
            return 1;
        }

        # set the same state, but only for notes
        if ( $Article{ArticleType} !~ /^note-/i ) {
            return 1;
        }

        # perform action on linked tickets
        for my $TicketID (@TicketIDs) {
            next if !$Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: ArticleCreate',
            );

            # article create
            $Self->{TicketObject}->ArticleCreate(
                %Article,
                HistoryType    => 'AddNote',
                HistoryComment => 'Added article based on master ticket.',
                TicketID       => $TicketID,
                UserID         => $Param{UserID},
            );
        }
        return 1;
    }

    # state action
    elsif ( $Param{Event} eq 'TicketStateUpdate' ) {

        # mark ticket to prevent a loop
        $Self->{TicketObject}->HistoryAdd(
            TicketID     => $Param{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => "MasterTicketAction: TicketStateUpdate",
        );

        # perform action on linked tickets
        for my $TicketID (@TicketIDs) {
            next if !$Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: TicketStateUpdate',
            );

            # set the same state
            $Self->{TicketObject}->StateSet(
                StateID  => $Ticket{StateID},
                TicketID => $TicketID,
                UserID   => $Param{UserID},
            );
        }
        return 1;
    }

    # set pendig time
    elsif ( $Param{Event} eq 'TicketPendingTimeUpdate' ) {

        # mark ticket to prevent a loop
        $Self->{TicketObject}->HistoryAdd(
            TicketID     => $Param{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => "MasterTicketAction: TicketPendingTimeUpdate",
        );

        # perform action on linked tickets
        for my $TicketID (@TicketIDs) {
            next if !$Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: TicketPendingTimeUpdate',
            );

            # set the same pending time
            my ( $Sec, $Min, $Hour, $Day, $Month, $Year ) = $Self->{TimeObject}->SystemTime2Date(
                SystemTime => $Ticket{RealTillTimeNotUsed},
            );
            $Self->{TicketObject}->TicketPendingTimeSet(
                Year     => $Year,
                Month    => $Month,
                Day      => $Day,
                Hour     => $Hour,
                Minute   => $Min,
                TicketID => $TicketID,
                UserID   => $Param{UserID},
            );
        }
        return 1;
    }

    # priority action
    elsif ( $Param{Event} eq 'TicketPriorityUpdate' ) {

        # mark ticket to prevent a loop
        $Self->{TicketObject}->HistoryAdd(
            TicketID     => $Param{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => "MasterTicketAction: TicketPriorityUpdate",
        );

        # perform action on linked tickets
        for my $TicketID (@TicketIDs) {
            next if !$Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: TicketPriorityUpdate',
            );

            # set the same state
            $Self->{TicketObject}->PrioritySet(
                TicketID   => $TicketID,
                PriorityID => $Ticket{PriorityID},
                UserID     => $Param{UserID},
            );
        }
        return 1;
    }

    # owner action
    elsif ( $Param{Event} eq 'TicketOwnerUpdate' ) {

        # mark ticket to prevent a loop
        $Self->{TicketObject}->HistoryAdd(
            TicketID     => $Param{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => "MasterTicketAction: TicketOwnerUpdate",
        );

        # perform action on linked tickets
        for my $TicketID (@TicketIDs) {
            next if !$Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: TicketOwnerUpdate',
            );

            # set the same state
            $Self->{TicketObject}->OwnerSet(
                TicketID           => $TicketID,
                NewUserID          => $Ticket{OwnerID},
                SendNoNotification => 0,
                UserID             => $Param{UserID},
            );
        }
        return 1;
    }

    # responsible action
    elsif ( $Param{Event} eq 'TicketResponsibleUpdate' ) {

        # mark ticket to prevent a loop
        $Self->{TicketObject}->HistoryAdd(
            TicketID     => $Param{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => "MasterTicketAction: TicketResponsibleUpdate",
        );

        # perform action on linked tickets
        for my $TicketID (@TicketIDs) {
            next if !$Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: TicketResponsibleUpdate',
            );

            # set the same state
            $Self->{TicketObject}->ResponsibleSet(
                TicketID           => $TicketID,
                NewUserID          => $Ticket{ResponsibleID},
                SendNoNotification => 0,
                UserID             => $Param{UserID},
            );
        }
        return 1;
    }

    # unlock/lock action
    elsif ( $Param{Event} eq 'TicketLockUpdate' ) {

        # mark ticket to prevent a loop
        $Self->{TicketObject}->HistoryAdd(
            TicketID     => $Param{TicketID},
            CreateUserID => $Param{UserID},
            HistoryType  => 'Misc',
            Name         => "MasterTicketAction: TicketLockUpdate",
        );

        # perform action on linked tickets
        for my $TicketID (@TicketIDs) {
            next if !$Self->_LoopCheck(
                TicketID => $TicketID,
                UserID   => $Param{UserID},
                String   => 'MasterTicketAction: TicketLockUpdate',
            );

            # set the same state
            $Self->{TicketObject}->LockSet(
                Lock               => $Ticket{Lock},
                TicketID           => $TicketID,
                SendNoNotification => 1,
                UserID             => $Param{UserID},
            );
        }
        return 1;
    }
    return 1;
}

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

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

    my @Lines = $Self->{TicketObject}->HistoryGet(
        TicketID => $Param{TicketID},
        UserID   => $Param{UserID},
    );
    for my $Data ( reverse @Lines ) {
        if ( $Data->{HistoryType} eq 'Misc' && $Data->{Name} eq $Param{String} ) {
            my $TimeCreated = $Self->{TimeObject}->TimeStamp2SystemTime(
                String => $Data->{CreateTime}
            ) + 15;
            my $TimeCurrent = $Self->{TimeObject}->SystemTime();
            if ( $TimeCreated > $TimeCurrent ) {
                $Self->{TicketObject}->HistoryAdd(
                    TicketID     => $Param{TicketID},
                    CreateUserID => $Param{UserID},
                    HistoryType  => 'Misc',
                    Name         => "MasterTicketLoop: stopped",
                );
                return;
            }
        }
    }
    return 1;
}

1;
