model.itop-tickets.php 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584
  1. <?php
  2. // Copyright (C) 2010 Combodo SARL
  3. //
  4. // This program is free software; you can redistribute it and/or modify
  5. // it under the terms of the GNU General Public License as published by
  6. // the Free Software Foundation; version 3 of the License.
  7. //
  8. // This program is distributed in the hope that it will be useful,
  9. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. // GNU General Public License for more details.
  12. //
  13. // You should have received a copy of the GNU General Public License
  14. // along with this program; if not, write to the Free Software
  15. // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  16. /**
  17. * Persistent classes for a CMDB
  18. *
  19. * @author Erwan Taloc <erwan.taloc@combodo.com>
  20. * @author Romain Quetiez <romain.quetiez@combodo.com>
  21. * @author Denis Flaven <denis.flaven@combodo.com>
  22. * @license http://www.opensource.org/licenses/gpl-3.0.html LGPL
  23. */
  24. abstract class Ticket extends cmdbAbstractObject
  25. {
  26. public static function Init()
  27. {
  28. $aParams = array
  29. (
  30. "category" => "bizmodel,searchable,ticketing",
  31. "key_type" => "autoincrement",
  32. "name_attcode" => "ref",
  33. "state_attcode" => "",
  34. "reconc_keys" => array("ref"),
  35. "db_table" => "ticket",
  36. "db_key_field" => "id",
  37. "db_finalclass_field" => "",
  38. "display_template" => "",
  39. );
  40. MetaModel::Init_Params($aParams);
  41. MetaModel::Init_InheritAttributes();
  42. MetaModel::Init_AddAttribute(new AttributeString("ref", array("allowed_values"=>null, "sql"=>"ref", "default_value"=>"", "is_null_allowed"=>true, "depends_on"=>array())));
  43. MetaModel::Init_AddAttribute(new AttributeString("title", array("allowed_values"=>null, "sql"=>"title", "default_value"=>"", "is_null_allowed"=>false, "depends_on"=>array())));
  44. MetaModel::Init_AddAttribute(new AttributeText("description", array("allowed_values"=>null, "sql"=>"description", "default_value"=>"", "is_null_allowed"=>false, "depends_on"=>array())));
  45. MetaModel::Init_AddAttribute(new AttributeText("ticket_log", array("allowed_values"=>null, "sql"=>"ticket_log", "default_value"=>"", "is_null_allowed"=>true, "depends_on"=>array())));
  46. MetaModel::Init_AddAttribute(new AttributeDateTime("start_date", array("allowed_values"=>null, "sql"=>"start_date", "default_value"=>null, "is_null_allowed"=>false, "depends_on"=>array())));
  47. MetaModel::Init_AddAttribute(new AttributeLinkedSetIndirect("document_list", array("linked_class"=>"lnkTicketToDoc", "ext_key_to_me"=>"ticket_id", "ext_key_to_remote"=>"document_id", "allowed_values"=>null, "count_min"=>0, "count_max"=>0, "depends_on"=>array())));
  48. MetaModel::Init_AddAttribute(new AttributeLinkedSetIndirect("ci_list", array("linked_class"=>"lnkTicketToCI", "ext_key_to_me"=>"ticket_id", "ext_key_to_remote"=>"ci_id", "allowed_values"=>null, "count_min"=>0, "count_max"=>0, "depends_on"=>array())));
  49. MetaModel::Init_AddAttribute(new AttributeLinkedSetIndirect("contact_list", array("linked_class"=>"lnkTicketToContact", "ext_key_to_me"=>"ticket_id", "ext_key_to_remote"=>"contact_id", "allowed_values"=>null, "count_min"=>0, "count_max"=>0, "depends_on"=>array())));
  50. MetaModel::Init_SetZListItems('details', array('ref', 'title', 'description', 'ticket_log', 'start_date', 'document_list', 'ci_list', 'contact_list'));
  51. MetaModel::Init_SetZListItems('advanced_search', array('finalclass', 'ref', 'title', 'ticket_log', 'start_date'));
  52. MetaModel::Init_SetZListItems('standard_search', array('finalclass', 'ref', 'title', 'ticket_log', 'start_date'));
  53. MetaModel::Init_SetZListItems('list', array('finalclass', 'ref', 'title', 'ticket_log', 'start_date'));
  54. }
  55. }
  56. class lnkTicketToDoc extends cmdbAbstractObject
  57. {
  58. public static function Init()
  59. {
  60. $aParams = array
  61. (
  62. "category" => "bizmodel,searchable,ticketing",
  63. "key_type" => "autoincrement",
  64. "name_attcode" => "ticket_id",
  65. "state_attcode" => "",
  66. "reconc_keys" => array(),
  67. "db_table" => "lnktickettodoc",
  68. "db_key_field" => "id",
  69. "db_finalclass_field" => "",
  70. "display_template" => "",
  71. );
  72. MetaModel::Init_Params($aParams);
  73. MetaModel::Init_InheritAttributes();
  74. MetaModel::Init_AddAttribute(new AttributeExternalKey("ticket_id", array("targetclass"=>"Ticket", "jointype"=>null, "allowed_values"=>null, "sql"=>"ticket_id", "is_null_allowed"=>false, "on_target_delete"=>DEL_AUTO, "depends_on"=>array())));
  75. MetaModel::Init_AddAttribute(new AttributeExternalField("ticket_ref", array("allowed_values"=>null, "extkey_attcode"=>"ticket_id", "target_attcode"=>"ref", "is_null_allowed"=>true, "depends_on"=>array())));
  76. MetaModel::Init_AddAttribute(new AttributeExternalKey("document_id", array("targetclass"=>"Document", "jointype"=>null, "allowed_values"=>null, "sql"=>"document_id", "is_null_allowed"=>false, "on_target_delete"=>DEL_AUTO, "depends_on"=>array())));
  77. MetaModel::Init_AddAttribute(new AttributeExternalField("document_name", array("allowed_values"=>null, "extkey_attcode"=>"document_id", "target_attcode"=>"name", "is_null_allowed"=>true, "depends_on"=>array())));
  78. MetaModel::Init_SetZListItems('details', array('ticket_id', 'document_id'));
  79. MetaModel::Init_SetZListItems('advanced_search', array('ticket_id', 'document_id'));
  80. MetaModel::Init_SetZListItems('standard_search', array('ticket_id', 'document_id'));
  81. MetaModel::Init_SetZListItems('list', array('ticket_id', 'document_id'));
  82. }
  83. }
  84. class lnkTicketToContact extends cmdbAbstractObject
  85. {
  86. public static function Init()
  87. {
  88. $aParams = array
  89. (
  90. "category" => "bizmodel,searchable,ticketing",
  91. "key_type" => "autoincrement",
  92. "name_attcode" => "ticket_id",
  93. "state_attcode" => "",
  94. "reconc_keys" => array(),
  95. "db_table" => "lnktickettocontact",
  96. "db_key_field" => "id",
  97. "db_finalclass_field" => "",
  98. "display_template" => "",
  99. );
  100. MetaModel::Init_Params($aParams);
  101. MetaModel::Init_InheritAttributes();
  102. MetaModel::Init_AddAttribute(new AttributeExternalKey("ticket_id", array("targetclass"=>"Ticket", "jointype"=>null, "allowed_values"=>null, "sql"=>"ticket_id", "is_null_allowed"=>false, "on_target_delete"=>DEL_AUTO, "depends_on"=>array())));
  103. MetaModel::Init_AddAttribute(new AttributeExternalField("ticket_ref", array("allowed_values"=>null, "extkey_attcode"=>"ticket_id", "target_attcode"=>"ref", "is_null_allowed"=>true, "depends_on"=>array())));
  104. MetaModel::Init_AddAttribute(new AttributeExternalKey("contact_id", array("targetclass"=>"Contact", "jointype"=>null, "allowed_values"=>null, "sql"=>"contact_id", "is_null_allowed"=>false, "on_target_delete"=>DEL_AUTO, "depends_on"=>array())));
  105. MetaModel::Init_AddAttribute(new AttributeExternalField("contact_name", array("allowed_values"=>null, "extkey_attcode"=>"contact_id", "target_attcode"=>"name", "is_null_allowed"=>true, "depends_on"=>array())));
  106. MetaModel::Init_AddAttribute(new AttributeExternalField("contact_email", array("allowed_values"=>null, "extkey_attcode"=>"contact_id", "target_attcode"=>"email", "is_null_allowed"=>true, "depends_on"=>array())));
  107. MetaModel::Init_AddAttribute(new AttributeString("role", array("allowed_values"=>null, "sql"=>"role", "default_value"=>"", "is_null_allowed"=>true, "depends_on"=>array())));
  108. MetaModel::Init_SetZListItems('details', array('ticket_id', 'contact_id', 'contact_email', 'role'));
  109. MetaModel::Init_SetZListItems('advanced_search', array('ticket_id', 'contact_id', 'contact_email', 'role'));
  110. MetaModel::Init_SetZListItems('standard_search', array('ticket_id', 'contact_id', 'contact_email', 'role'));
  111. MetaModel::Init_SetZListItems('list', array('ticket_id', 'contact_id', 'contact_email', 'role'));
  112. }
  113. }
  114. class lnkTicketToCI extends cmdbAbstractObject
  115. {
  116. public static function Init()
  117. {
  118. $aParams = array
  119. (
  120. "category" => "bizmodel,searchable,ticketing",
  121. "key_type" => "autoincrement",
  122. "name_attcode" => "ticket_id",
  123. "state_attcode" => "",
  124. "reconc_keys" => array(),
  125. "db_table" => "lnktickettoci",
  126. "db_key_field" => "id",
  127. "db_finalclass_field" => "",
  128. "display_template" => "",
  129. );
  130. MetaModel::Init_Params($aParams);
  131. MetaModel::Init_InheritAttributes();
  132. MetaModel::Init_AddAttribute(new AttributeExternalKey("ticket_id", array("targetclass"=>"Ticket", "jointype"=>null, "allowed_values"=>null, "sql"=>"ticket_id", "is_null_allowed"=>false, "on_target_delete"=>DEL_AUTO, "depends_on"=>array())));
  133. MetaModel::Init_AddAttribute(new AttributeExternalField("ticket_ref", array("allowed_values"=>null, "extkey_attcode"=>"ticket_id", "target_attcode"=>"ref", "is_null_allowed"=>true, "depends_on"=>array())));
  134. MetaModel::Init_AddAttribute(new AttributeExternalKey("ci_id", array("targetclass"=>"FunctionalCI", "jointype"=>null, "allowed_values"=>null, "sql"=>"ci_id", "is_null_allowed"=>false, "on_target_delete"=>DEL_AUTO, "depends_on"=>array())));
  135. MetaModel::Init_AddAttribute(new AttributeExternalField("ci_name", array("allowed_values"=>null, "extkey_attcode"=>"ci_id", "target_attcode"=>"name", "is_null_allowed"=>true, "depends_on"=>array())));
  136. MetaModel::Init_AddAttribute(new AttributeExternalField("ci_status", array("allowed_values"=>null, "extkey_attcode"=>"ci_id", "target_attcode"=>"status", "is_null_allowed"=>true, "depends_on"=>array())));
  137. MetaModel::Init_SetZListItems('details', array('ticket_id', 'ci_id', 'ci_status'));
  138. MetaModel::Init_SetZListItems('advanced_search', array('ticket_id', 'ci_id', 'ci_status'));
  139. MetaModel::Init_SetZListItems('standard_search', array('ticket_id', 'ci_id', 'ci_status'));
  140. MetaModel::Init_SetZListItems('list', array('ticket_id', 'ci_id', 'ci_status'));
  141. }
  142. }
  143. abstract class ResponseTicket extends Ticket
  144. {
  145. public static function Init()
  146. {
  147. $aParams = array
  148. (
  149. "category" => "bizmodel",
  150. "key_type" => "autoincrement",
  151. "name_attcode" => "ref",
  152. "state_attcode" => "status",
  153. "reconc_keys" => array("ref"),
  154. "db_table" => "ticket_response",
  155. "db_key_field" => "id",
  156. "db_finalclass_field" => "",
  157. "display_template" => "",
  158. );
  159. MetaModel::Init_Params($aParams);
  160. MetaModel::Init_InheritAttributes();
  161. MetaModel::Init_AddAttribute(new AttributeEnum("status", array("allowed_values"=>new ValueSetEnum('new,assigned,frozen,escalated_tto,escalated_ttr,resolved,closed'), "sql"=>"status", "default_value"=>"new", "is_null_allowed"=>false, "depends_on"=>array())));
  162. MetaModel::Init_AddAttribute(new AttributeExternalKey("caller_id", array("targetclass"=>"Person", "jointype"=>null, "allowed_values"=>new ValueSetObjects('SELECT Person WHERE org_id = :this->org_id'), "sql"=>"caller_id", "is_null_allowed"=>true, "on_target_delete"=>DEL_MANUAL, "depends_on"=>array("org_id"))));
  163. MetaModel::Init_AddAttribute(new AttributeExternalField("caller_email", array("allowed_values"=>null, "extkey_attcode"=>"caller_id", "target_attcode"=>"email", "is_null_allowed"=>true, "depends_on"=>array())));
  164. MetaModel::Init_AddAttribute(new AttributeExternalKey("org_id", array("targetclass"=>"Organization", "jointype"=>null, "allowed_values"=>null, "sql"=>"org_id", "is_null_allowed"=>false, "on_target_delete"=>DEL_AUTO, "depends_on"=>array())));
  165. MetaModel::Init_AddAttribute(new AttributeExternalField("org_name", array("allowed_values"=>null, "extkey_attcode"=>"org_id", "target_attcode"=>"name", "is_null_allowed"=>true, "depends_on"=>array())));
  166. MetaModel::Init_AddAttribute(new AttributeExternalKey("service_id", array("targetclass"=>"Service", "jointype"=>null, "allowed_values"=>new ValueSetObjects('SELECT Service AS s JOIN SLA AS sla ON sla.service_id=s.id JOIN lnkContractToSLA AS ln ON ln.sla_id=sla.id JOIN CustomerContract AS cc ON ln.contract_id=cc.id WHERE cc.org_id =:this->org_id'), "sql"=>"service_id", "is_null_allowed"=>false, "on_target_delete"=>DEL_MANUAL, "depends_on"=>array("org_id"))));
  167. MetaModel::Init_AddAttribute(new AttributeExternalField("service_name", array("allowed_values"=>null, "extkey_attcode"=>"service_id", "target_attcode"=>"name", "is_null_allowed"=>true, "depends_on"=>array())));
  168. MetaModel::Init_AddAttribute(new AttributeExternalKey("servicesubcategory_id", array("targetclass"=>"ServiceSubcategory", "jointype"=>null, "allowed_values"=>new ValueSetObjects('SELECT ServiceSubcategory WHERE service_id = :this->service_id'), "sql"=>"servicesubcategory_id", "is_null_allowed"=>false, "on_target_delete"=>DEL_MANUAL, "depends_on"=>array("service_id"))));
  169. MetaModel::Init_AddAttribute(new AttributeExternalField("servicesubcategory_name", array("allowed_values"=>null, "extkey_attcode"=>"servicesubcategory_id", "target_attcode"=>"name", "is_null_allowed"=>true, "depends_on"=>array())));
  170. MetaModel::Init_AddAttribute(new AttributeString("product", array("allowed_values"=>null, "sql"=>"product", "default_value"=>"", "is_null_allowed"=>true, "depends_on"=>array())));
  171. MetaModel::Init_AddAttribute(new AttributeEnum("impact", array("allowed_values"=>new ValueSetEnum('1,2,3'), "sql"=>"impact", "default_value"=>"1", "is_null_allowed"=>false, "depends_on"=>array())));
  172. MetaModel::Init_AddAttribute(new AttributeEnum("urgency", array("allowed_values"=>new ValueSetEnum('1,2,3'), "sql"=>"urgency", "default_value"=>"1", "is_null_allowed"=>false, "depends_on"=>array())));
  173. MetaModel::Init_AddAttribute(new AttributeEnum("priority", array("allowed_values"=>new ValueSetEnum('1,2,3'), "sql"=>"priority", "default_value"=>"1", "is_null_allowed"=>false, "depends_on"=>array())));
  174. MetaModel::Init_AddAttribute(new AttributeExternalKey("workgroup_id", array("targetclass"=>"Team", "jointype"=>null, "allowed_values"=>new ValueSetObjects('SELECT Team AS t JOIN CustomerContract AS cc ON cc.support_team_id=t.id JOIN lnkContractToSLA AS ln ON ln.contract_id=cc.id JOIN SLA AS sla ON ln.sla_id=sla.id WHERE sla.service_id = :this->service_id AND cc.org_id = :this->org_id'), "sql"=>"workgroup_id", "is_null_allowed"=>false, "on_target_delete"=>DEL_MANUAL, "depends_on"=>array("org_id","service_id"))));
  175. MetaModel::Init_AddAttribute(new AttributeExternalField("workgroup_name", array("allowed_values"=>null, "extkey_attcode"=>"workgroup_id", "target_attcode"=>"name", "is_null_allowed"=>true, "depends_on"=>array())));
  176. MetaModel::Init_AddAttribute(new AttributeExternalKey("agent_id", array("targetclass"=>"Person", "jointype"=>null, "allowed_values"=>new ValueSetObjects('SELECT Person AS p JOIN lnkTeamToContact AS l ON l.contact_id=p.id JOIN Team AS t ON l.team_id=t.id WHERE t.id = :this->workgroup_id'), "sql"=>"agent_id", "is_null_allowed"=>true, "on_target_delete"=>DEL_MANUAL, "depends_on"=>array("workgroup_id"))));
  177. MetaModel::Init_AddAttribute(new AttributeExternalField("agent_name", array("allowed_values"=>null, "extkey_attcode"=>"agent_id", "target_attcode"=>"name", "is_null_allowed"=>true, "depends_on"=>array())));
  178. MetaModel::Init_AddAttribute(new AttributeExternalField("agent_email", array("allowed_values"=>null, "extkey_attcode"=>"agent_id", "target_attcode"=>"email", "is_null_allowed"=>true, "depends_on"=>array())));
  179. MetaModel::Init_AddAttribute(new AttributeExternalKey("related_problem_id", array("targetclass"=>"Problem", "jointype"=>null, "allowed_values"=>null, "sql"=>"related_problem_id", "is_null_allowed"=>true, "on_target_delete"=>DEL_MANUAL, "depends_on"=>array())));
  180. MetaModel::Init_AddAttribute(new AttributeExternalField("related_problem_ref", array("allowed_values"=>null, "extkey_attcode"=>"related_problem_id", "target_attcode"=>"ref", "is_null_allowed"=>true, "depends_on"=>array())));
  181. MetaModel::Init_AddAttribute(new AttributeExternalKey("related_change_id", array("targetclass"=>"Change", "jointype"=>null, "allowed_values"=>null, "sql"=>"related_change_id", "is_null_allowed"=>true, "on_target_delete"=>DEL_MANUAL, "depends_on"=>array())));
  182. MetaModel::Init_AddAttribute(new AttributeExternalField("related_change_ref", array("allowed_values"=>null, "extkey_attcode"=>"related_change_id", "target_attcode"=>"ref", "is_null_allowed"=>true, "depends_on"=>array())));
  183. MetaModel::Init_AddAttribute(new AttributeDateTime("close_date", array("allowed_values"=>null, "sql"=>"close_date", "default_value"=>null, "is_null_allowed"=>true, "depends_on"=>array())));
  184. MetaModel::Init_AddAttribute(new AttributeDateTime("last_update", array("allowed_values"=>null, "sql"=>"last_update", "default_value"=>null, "is_null_allowed"=>true, "depends_on"=>array())));
  185. MetaModel::Init_AddAttribute(new AttributeDateTime("assignment_date", array("allowed_values"=>null, "sql"=>"assignment_date", "default_value"=>null, "is_null_allowed"=>true, "depends_on"=>array())));
  186. MetaModel::Init_AddAttribute(new AttributeDeadline("escalation_deadline", array("allowed_values"=>null, "sql"=>"escalation_deadline", "default_value"=>null, "is_null_allowed"=>true, "depends_on"=>array())));
  187. MetaModel::Init_AddAttribute(new AttributeDeadline("closure_deadline", array("allowed_values"=>null, "sql"=>"closure_deadline", "default_value"=>null, "is_null_allowed"=>true, "depends_on"=>array())));
  188. MetaModel::Init_AddAttribute(new AttributeEnum("resolution_code", array("allowed_values"=>new ValueSetEnum('fixed,duplicate,couldnotreproduce,irrelevant'), "sql"=>"resolution_code", "default_value"=>"fixed", "is_null_allowed"=>true, "depends_on"=>array())));
  189. MetaModel::Init_AddAttribute(new AttributeText("solution", array("allowed_values"=>null, "sql"=>"solution", "default_value"=>"", "is_null_allowed"=>true, "depends_on"=>array())));
  190. MetaModel::Init_AddAttribute(new AttributeEnum("user_satisfaction", array("allowed_values"=>new ValueSetEnum('1,2,3,4'), "sql"=>"user_satisfaction", "default_value"=>"1", "is_null_allowed"=>true, "depends_on"=>array())));
  191. MetaModel::Init_AddAttribute(new AttributeText("user_commment", array("allowed_values"=>null, "sql"=>"user_commment", "default_value"=>"", "is_null_allowed"=>true, "depends_on"=>array())));
  192. MetaModel::Init_SetZListItems('details', array('ref', 'title', 'org_id', 'ticket_log', 'start_date', 'escalation_deadline', 'closure_deadline', 'document_list', 'ci_list', 'contact_list', 'status', 'caller_id', 'service_id', 'servicesubcategory_id', 'product', 'impact', 'urgency', 'priority', 'workgroup_id', 'agent_id', 'agent_email', 'related_problem_id', 'related_change_id', 'close_date', 'last_update', 'assignment_date', 'resolution_code', 'solution', 'user_satisfaction', 'user_commment'));
  193. MetaModel::Init_SetZListItems('advanced_search', array('finalclass', 'ref', 'title', 'org_id', 'start_date', 'status', 'caller_id', 'service_id', 'servicesubcategory_id', 'product', 'impact', 'urgency', 'priority', 'workgroup_id', 'agent_id', 'agent_email', 'related_problem_id', 'related_change_id', 'close_date', 'last_update', 'assignment_date', 'escalation_deadline', 'closure_deadline', 'resolution_code', 'solution', 'user_satisfaction', 'user_commment'));
  194. MetaModel::Init_SetZListItems('standard_search', array('finalclass', 'ref', 'title', 'org_id', 'start_date', 'status', 'caller_id', 'service_id', 'servicesubcategory_id', 'product', 'impact', 'urgency', 'priority', 'workgroup_id', 'agent_id', 'agent_email', 'related_problem_id', 'related_change_id', 'close_date', 'last_update', 'assignment_date', 'escalation_deadline', 'closure_deadline', 'resolution_code', 'solution', 'user_satisfaction', 'user_commment'));
  195. MetaModel::Init_SetZListItems('list', array('finalclass', 'ref', 'title', 'org_id', 'start_date', 'status', 'caller_id', 'service_id', 'servicesubcategory_id', 'product', 'impact', 'urgency', 'priority', 'workgroup_id', 'agent_id', 'agent_email', 'related_problem_id', 'related_change_id', 'close_date', 'last_update', 'assignment_date', 'escalation_deadline', 'closure_deadline', 'resolution_code', 'solution', 'user_satisfaction', 'user_commment'));
  196. // Lifecycle
  197. MetaModel::Init_DefineState(
  198. "new",
  199. array(
  200. "attribute_inherit" => null,
  201. "attribute_list" => array(
  202. 'ref' => OPT_ATT_READONLY,
  203. 'ticket_log' => OPT_ATT_HIDDEN,
  204. 'caller_id' => OPT_ATT_MANDATORY,
  205. 'related_change_id' => OPT_ATT_HIDDEN,
  206. 'description' => OPT_ATT_MUSTCHANGE,
  207. 'contact_list' => OPT_ATT_READONLY,
  208. 'start_date' => OPT_ATT_READONLY,
  209. 'last_update' => OPT_ATT_READONLY,
  210. 'assignment_date' => OPT_ATT_HIDDEN,
  211. 'escalation_deadline' => OPT_ATT_READONLY,
  212. 'closure_deadline' => OPT_ATT_HIDDEN,
  213. 'close_date' => OPT_ATT_HIDDEN,
  214. 'org_id' => OPT_ATT_MUSTCHANGE,
  215. 'service_id' => OPT_ATT_MUSTCHANGE,
  216. 'servicesubcategory_id' => OPT_ATT_MUSTCHANGE,
  217. 'product' => OPT_ATT_MUSTPROMPT,
  218. 'impact' => OPT_ATT_MUSTCHANGE,
  219. 'urgency' => OPT_ATT_MUSTCHANGE,
  220. 'priority' => OPT_ATT_READONLY,
  221. 'workgroup_id' => OPT_ATT_MUSTCHANGE,
  222. 'agent_id' => OPT_ATT_HIDDEN,
  223. 'agent_email' => OPT_ATT_HIDDEN,
  224. 'resolution_code' => OPT_ATT_HIDDEN,
  225. 'solution' => OPT_ATT_HIDDEN,
  226. 'user_satisfaction' => OPT_ATT_HIDDEN,
  227. 'user_commment' => OPT_ATT_HIDDEN,
  228. ),
  229. )
  230. );
  231. MetaModel::Init_DefineState(
  232. "escalated_tto",
  233. array(
  234. "attribute_inherit" => 'new',
  235. "attribute_list" => array(
  236. ),
  237. )
  238. );
  239. MetaModel::Init_DefineState(
  240. "assigned",
  241. array(
  242. "attribute_inherit" => 'new',
  243. "attribute_list" => array(
  244. 'title' => OPT_ATT_READONLY,
  245. 'caller_id' => OPT_ATT_READONLY,
  246. 'org_id' => OPT_ATT_READONLY,
  247. 'ticket_log' => OPT_ATT_NORMAL,
  248. 'description' => OPT_ATT_READONLY,
  249. 'agent_id' => OPT_ATT_MUSTPROMPT | OPT_ATT_MANDATORY,
  250. 'agent_email' => OPT_ATT_READONLY,
  251. 'workgroup_id' => OPT_ATT_MUSTPROMPT | OPT_ATT_MANDATORY,
  252. 'closure_deadline' => OPT_ATT_HIDDEN,
  253. 'escalation_deadline' => OPT_ATT_READONLY,
  254. ),
  255. )
  256. );
  257. MetaModel::Init_DefineState(
  258. "escalated_ttr",
  259. array(
  260. "attribute_inherit" => 'assigned',
  261. "attribute_list" => array(
  262. ),
  263. )
  264. );
  265. MetaModel::Init_DefineState(
  266. "frozen",
  267. array(
  268. "attribute_inherit" => 'assigned',
  269. "attribute_list" => array(
  270. ),
  271. )
  272. );
  273. MetaModel::Init_DefineState(
  274. "resolved",
  275. array(
  276. "attribute_inherit" => 'assigned',
  277. "attribute_list" => array(
  278. 'service_id' => OPT_ATT_READONLY,
  279. 'servicesubcategory_id' => OPT_ATT_READONLY,
  280. 'product' => OPT_ATT_READONLY,
  281. 'impact' => OPT_ATT_READONLY,
  282. 'workgroup_id' => OPT_ATT_READONLY,
  283. 'agent_id' => OPT_ATT_READONLY,
  284. 'urgency' => OPT_ATT_READONLY,
  285. 'resolution_code' => OPT_ATT_MUSTPROMPT,
  286. 'solution' => OPT_ATT_MUSTPROMPT,
  287. 'closure_deadline' => OPT_ATT_READONLY,
  288. 'escalation_deadline' => OPT_ATT_HIDDEN,
  289. ),
  290. )
  291. );
  292. MetaModel::Init_DefineState(
  293. "closed",
  294. array(
  295. "attribute_inherit" => 'resolved',
  296. "attribute_list" => array(
  297. 'ticket_log' => OPT_ATT_READONLY,
  298. 'user_satisfaction' => OPT_ATT_MUSTPROMPT,
  299. 'user_commment' => OPT_ATT_MUSTPROMPT,
  300. 'resolution_code' => OPT_ATT_READONLY,
  301. 'solution' => OPT_ATT_READONLY,
  302. 'close_date' => OPT_ATT_READONLY,
  303. 'closure_deadline' => OPT_ATT_HIDDEN,
  304. ),
  305. )
  306. );
  307. MetaModel::Init_DefineStimulus(new StimulusUserAction("ev_assign", array()));
  308. MetaModel::Init_DefineStimulus(new StimulusUserAction("ev_reassign", array()));
  309. MetaModel::Init_DefineStimulus(new StimulusInternal("ev_timeout", array()));
  310. MetaModel::Init_DefineStimulus(new StimulusUserAction("ev_resolve", array()));
  311. MetaModel::Init_DefineStimulus(new StimulusUserAction("ev_close", array()));
  312. MetaModel::Init_DefineTransition("new", "ev_assign", array("target_state"=>"assigned", "actions"=>array(), "user_restriction"=>null));
  313. MetaModel::Init_DefineTransition("new", "ev_timeout", array("target_state"=>"escalated_tto", "actions"=>array(), "user_restriction"=>null));
  314. MetaModel::Init_DefineTransition("escalated_tto", "ev_assign", array("target_state"=>"assigned", "actions"=>array(), "user_restriction"=>null));
  315. MetaModel::Init_DefineTransition("assigned", "ev_reassign", array("target_state"=>"assigned", "actions"=>array(), "user_restriction"=>null));
  316. MetaModel::Init_DefineTransition("assigned", "ev_timeout", array("target_state"=>"escalated_ttr", "actions"=>array(), "user_restriction"=>null));
  317. MetaModel::Init_DefineTransition("assigned", "ev_resolve", array("target_state"=>"resolved", "actions"=>array(), "user_restriction"=>null));
  318. MetaModel::Init_DefineTransition("escalated_ttr", "ev_reassign", array("target_state"=>"escalated_ttr", "actions"=>array(), "user_restriction"=>null));
  319. MetaModel::Init_DefineTransition("escalated_ttr", "ev_resolve", array("target_state"=>"resolved", "actions"=>array(), "user_restriction"=>null));
  320. MetaModel::Init_DefineTransition("resolved", "ev_reassign", array("target_state"=>"assigned", "actions"=>array(), "user_restriction"=>null));
  321. MetaModel::Init_DefineTransition("resolved", "ev_close", array("target_state"=>"closed", "actions"=>array('SetClosureDate'), "user_restriction"=>null));
  322. }
  323. // Lifecycle actions
  324. //
  325. public function SetClosureDate($sStimulusCode)
  326. {
  327. $this->Set('close_date', time());
  328. return true;
  329. }
  330. /**
  331. * Determines the shortest SLT, for this ticket, for the given metric. Returns null is no SLT was found
  332. * @param string $sMetric Type of metric 'TTO', 'TTR', etc as defined in the SLT class
  333. * @return hash Array with 'SLT' => name of the SLT selected, 'value' => duration in seconds of the SLT metric, null if no SLT applies to this ticket
  334. */
  335. public function ComputeSLT($sMetric = 'TTO')
  336. {
  337. $aResult = null;
  338. if (MetaModel::IsValidClass('SLT'))
  339. {
  340. $sOQL = "SELECT SLT JOIN lnkSLTToSLA AS L1 ON L1.slt_id=SLT.id JOIN SLA ON L1.sla_id = SLA.id JOIN lnkContractToSLA AS L2 ON L2.sla_id = SLA.id JOIN CustomerContract ON L2.contract_id = CustomerContract.id
  341. WHERE SLT.ticket_priority = :priority AND SLA.service_id = :service_id AND SLT.metric = :metric AND CustomerContract.org_id = :org_id";
  342. $oSLTSet = new DBObjectSet(DBObjectSearch::FromOQL($sOQL),
  343. array(),
  344. array(
  345. 'priority' => $this->Get('priority'),
  346. 'service_id' => $this->Get('service_id'),
  347. 'metric' => $sMetric,
  348. 'org_id' => $this->Get('org_id'),
  349. )
  350. );
  351. $iMinDuration = PHP_INT_MAX;
  352. $sSLTName = '';
  353. while($oSLT = $oSLTSet->Fetch())
  354. {
  355. $iDuration = (int)$oSLT->Get('value');
  356. $sUnit = $oSLT->Get('value_unit');
  357. //echo "<p>Found SLT: ".$oSLT->GetName()." - $iDuration ($sUnit)</p>\n";
  358. switch($sUnit)
  359. {
  360. case 'days':
  361. $iDuration = $iDuration * 24; // 24 hours in 1 days
  362. // Fall though
  363. case 'hours':
  364. $iDuration = $iDuration * 60; // 60 minutes in 1 hour
  365. // Fall though
  366. case 'minutes':
  367. $iDuration = $iDuration * 60;
  368. }
  369. if ($iDuration < $iMinDuration)
  370. {
  371. $iMinDuration = $iDuration;
  372. $sSLTName = $oSLT->GetName();
  373. }
  374. }
  375. if ($iMinDuration == PHP_INT_MAX)
  376. {
  377. $aResult = null;
  378. }
  379. else
  380. {
  381. $aResult = array('SLT' => $sSLTName, 'value' => $iMinDuration);
  382. }
  383. }
  384. return $aResult;
  385. }
  386. /**
  387. * Compute the priority of the ticket based on its impact and urgency
  388. * @return integer The priority of the ticket 1(high) .. 3(low)
  389. */
  390. public function ComputePriority()
  391. {
  392. // priority[impact][urgency]
  393. $aPriorities = array(
  394. // single person
  395. 1 => array(
  396. 1 => 1,
  397. 2 => 1,
  398. 3 => 2,
  399. ),
  400. // a group
  401. 2 => array(
  402. 1 => 1,
  403. 2 => 2,
  404. 3 => 3,
  405. ),
  406. // a departement!
  407. 3 => array(
  408. 1 => 2,
  409. 2 => 3,
  410. 3 => 3,
  411. ),
  412. );
  413. $iPriority = $aPriorities[(int)$this->Get('impact')][(int)$this->Get('urgency')];
  414. return $iPriority;
  415. }
  416. public function ComputeValues()
  417. {
  418. // Compute the priority of the ticket
  419. $this->Set('priority', $this->ComputePriority());
  420. // Compute the SLA deadlines, if any is applicable to this ticket
  421. $aSLT = $this->ComputeSLT('TTO');
  422. if ($aSLT != null)
  423. {
  424. //echo "<p>TTO: SLT found: {$aSLT['SLT']}, value: {$aSLT['value']}</p>\n";
  425. $iStartDate = AttributeDateTime::GetAsUnixSeconds($this->Get('start_date'));
  426. $this->Set('escalation_deadline', $iStartDate + $aSLT['value']);
  427. }
  428. else
  429. {
  430. $this->Set('escalation_deadline', null);
  431. }
  432. $aSLT = $this->ComputeSLT('TTR');
  433. if ($aSLT != null)
  434. {
  435. //echo "<p>TTR: SLT found: {$aSLT['SLT']}, value: {$aSLT['value']}</p>\n";
  436. $iStartDate = AttributeDateTime::GetAsUnixSeconds($this->Get('start_date'));
  437. $this->Set('closure_deadline', $iStartDate + $aSLT['value']);
  438. }
  439. else
  440. {
  441. $this->Set('closure_deadline', null);
  442. }
  443. }
  444. /**
  445. * Determines if the ticket must be hilighted in the list, if we're about to miss a SLA for instance
  446. */
  447. public function GetHilightClass()
  448. {
  449. $sHilightClass = '';
  450. switch($this->GetState())
  451. {
  452. case 'new':
  453. $oEscalationDeadline = $this->Get('escalation_deadline');
  454. if ($oEscalationDeadline != null)
  455. {
  456. // A SLA is running
  457. $iStartDate = AttributeDateTime::GetAsUnixSeconds($this->Get('start_date'));
  458. $iEscalationDeadline = AttributeDateTime::GetAsUnixSeconds($oEscalationDeadline);
  459. $ratio = ($iEscalationDeadline - time())/($iEscalationDeadline - $iStartDate);
  460. if ($ratio <= 0)
  461. {
  462. $sHilightClass = HILIGHT_CLASS_CRITICAL;
  463. }
  464. else if ($ratio <= 0.25)
  465. {
  466. $sHilightClass = HILIGHT_CLASS_WARNING;
  467. }
  468. }
  469. break;
  470. case 'assigned':
  471. $oClosureDeadline = $this->Get('closure_deadline');
  472. if ($oClosureDeadline != null)
  473. {
  474. // A SLA is running
  475. $iStartDate = AttributeDateTime::GetAsUnixSeconds($this->Get('start_date'));
  476. $iClosureDeadline = AttributeDateTime::GetAsUnixSeconds($oClosureDeadline);
  477. $ratio = ($iClosureDeadline - time())/($iClosureDeadline - $iStartDate);
  478. if ($ratio <= 0)
  479. {
  480. $sHilightClass = HILIGHT_CLASS_CRITICAL;
  481. }
  482. else if ($ratio <= 0.25)
  483. {
  484. $sHilightClass = HILIGHT_CLASS_WARNING;
  485. }
  486. }
  487. break;
  488. case 'escalated_tto':
  489. case 'escalated_ttr':
  490. $sHilightClass = HILIGHT_CLASS_CRITICAL;
  491. break;
  492. }
  493. return $sHilightClass;
  494. }
  495. protected function OnInsert()
  496. {
  497. $this->Set('last_update', time());
  498. }
  499. protected function OnUpdate()
  500. {
  501. $this->Set('last_update', time());
  502. }
  503. /*
  504. EXAMPLE: OnInsert....
  505. protected function OnInsert()
  506. {
  507. // Romain: ajouter cette ligne
  508. $oToNotify = $this->Get('contacts_a_notifier');
  509. // Romain: ca c'etait pour verifier que ca fonctionne bien
  510. // $oFirstContact = MetaModel::GetObject('bizPerson', 6);
  511. // $oNewLink = new lnkContactTicket();
  512. // $oNewLink->Set('contact_id', 6);
  513. // $oNewLink->Set('role', 'created before');
  514. // $oToNotify->AddObject($oNewLink);
  515. $oImpactedInfras = DBObjectSet::FromLinkSet($this, 'impacted_infra_manual', 'infra_id');
  516. $aComputed = $oImpactedInfras->GetRelatedObjects('impacts', 10);
  517. if (array_key_exists('logRealObject', $aComputed))
  518. {
  519. foreach($aComputed['logRealObject'] as $iKey => $oObject)
  520. {
  521. if (MetaModel::IsParentClass('bizContact', get_class($oObject)))
  522. {
  523. $oNewLink = new lnkContactTicket();
  524. $oNewLink->Set('contact_id', $iKey);
  525. //$oNewLink->Set('ticket_id', $this->GetKey()); // unkown at that time!
  526. $oNewLink->Set('role', 'contact automatically computed');
  527. // Romain: transformer cette ligne
  528. $oToNotify->AddObject($oNewLink);
  529. }
  530. }
  531. // Romain: supprimer cette ligne
  532. // $this->Set('contacts_a_notifier', $oToNotify);
  533. }
  534. }
  535. */
  536. }
  537. ?>