u_fsm.php 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. <?php
  2. /**
  3. * Finite state machine skeleton
  4. * embodying a Moore FSM
  5. * (c) 2006 Ouest Systèmes Informatiques (OSI)
  6. * Licensed under the CeCILL 2.0 license
  7. *
  8. * $Id: u_fsm.php,v 1.1 2006-12-03 23:20:08 marand Exp $
  9. */
  10. require_once('misc.php'); // for func_name()
  11. abstract class fsm
  12. {
  13. /**
  14. * the current state of the object
  15. * @var string
  16. */
  17. protected $f_state;
  18. /**
  19. * Transitions holds the transitions table
  20. * state1
  21. * event1
  22. * result1
  23. * action1
  24. * event2
  25. * ...
  26. * ..
  27. * ..
  28. * @var array
  29. */
  30. protected $f_transitions = null;
  31. /**
  32. * constructor initializes the FSM to the first
  33. * state in the transitions table
  34. * @return void
  35. */
  36. public function __construct()
  37. {
  38. $this->_check_transitions();
  39. reset($this->f_transitions);
  40. $x = each($this->f_transitions);
  41. $x = $x[0];
  42. $this->f_state = $x;
  43. }
  44. /**
  45. * make sure a transitions graph has been defined
  46. * @return void
  47. */
  48. private function _check_transitions()
  49. {
  50. if (!isset($this->f_transitions))
  51. throw new Exception('No FSM processing is allowed without a transitions table');
  52. }
  53. /**
  54. * getter for f_state
  55. */
  56. public function get_state()
  57. {
  58. return $this->f_state;
  59. }
  60. /**
  61. * return the list of events accepted in the current state
  62. * @return array
  63. */
  64. public function get_accepted_events()
  65. {
  66. $this->_check_transitions();
  67. $events = array_keys($this->f_transitions[$this->f_state]);
  68. // echo func_name() . ": " . print_r($events, true). "\n";
  69. return $events;
  70. }
  71. /**
  72. * return the list of outcome accepted in the current state for the give event
  73. * @param string $event_name
  74. * @param mixed $outcome
  75. * @return array
  76. */
  77. public function get_accepted_outcomes($event_name)
  78. {
  79. // echo func_name() . "\n";
  80. $this->_check_transitions();
  81. if (!$this->is_event_allowed($event_name))
  82. throw new Exception(func_name() . ": event \"$event_name\" not allowed in state \"$this->f_state\"\n");
  83. $outcomes = array_keys($this->f_transitions[$this->f_state][$event_name]);
  84. //echo "outcomes for event $event_name: " . print_r($outcomes, true) . "\n";
  85. return $outcomes;
  86. }
  87. /**
  88. * is this event accepted in the current state
  89. * the FSM is in ?
  90. *
  91. * @param string $event_name
  92. * @return boolean
  93. */
  94. public function is_event_allowed($event_name)
  95. {
  96. // echo func_name() . "($event_name)";
  97. $this->_check_transitions();
  98. $ret = in_array($event_name, $this->get_accepted_events());
  99. // echo ", result = <$ret>\n";
  100. return $ret;
  101. }
  102. /**
  103. * is a given outcome available for a given event,
  104. * considering the current context ?
  105. *
  106. * @param string $event_name
  107. * @param mixed $outcome
  108. * @return boolean
  109. */
  110. public function is_outcome_allowed($event_name, $outcome)
  111. {
  112. $this->_check_transitions();
  113. if (!$this->is_event_allowed($event_name))
  114. return false;
  115. $ret = array_key_exists($outcome, $this->get_accepted_outcomes($event_name));
  116. return $ret;
  117. }
  118. /**
  119. * apply an event
  120. *
  121. * @param string $event_name
  122. * @param array $params the
  123. * @return string resulting state
  124. */
  125. public function apply_event($event_name)
  126. {
  127. //echo func_name() . "\n";
  128. if (! $this->is_event_allowed($event_name))
  129. throw new Exception(func_name()
  130. . ": Event \"$event_name\" not accepted in current state \"$this->f_state\"");
  131. $method_name = "f_$event_name";
  132. $outcomes = $this->get_accepted_outcomes($event_name);
  133. $result = $this->$method_name();
  134. if (!in_array($result, $outcomes))
  135. throw new Exception(func_name()
  136. . ": event guard. Transition on \"$event_name\" return invalid result: "
  137. . var_dump($result)
  138. . "\n");
  139. $this->f_state = $this->f_transitions[$this->f_state][$event_name][$result];
  140. // echo "new state: $this->f_state\n";
  141. return $result;
  142. }
  143. }