123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413 |
- <?php
- namespace OSInet\Finite_State_Machine;
- abstract class Machine {
- const VERSION = '12D22';
- const IDLE_EVENT = 'idle';
- const INIT_STATE = 'init';
- const FINAL_STATE = 'final';
- const EVENT_NORMAL = 1;
- const EVENT_QUEUE = 2;
- const EVENT_SINK = 3;
- public $idleWork = TRUE;
- public $allowActions = TRUE;
- protected $fEventMode = self::EVENT_NORMAL;
- protected $fQueue = array();
-
- protected $fState;
-
- public $fTransitions = null;
-
- public function __construct() {
- $this->checkTransitions();
- reset($this->fTransitions);
- $x = each($this->fTransitions);
- $x = $x[0];
- $this->fState = $x;
- }
-
- protected function checkTransitions() {
- if (!isset($this->fTransitions))
- throw new Exception('No FSM processing is allowed without a transitions table\n');
- }
-
- public function getState() {
- return $this->fState;
- }
-
- public function getAcceptedEvents() {
- $this->checkTransitions();
- try {
- $events = array_keys($this->fTransitions[$this->fState]);
-
- }
- catch (Exception $e) {
- echo "Exception in getAcceptedEvents" . print_r($e);
- print_r(debug_backtrace());
- $events = array();
- }
- return $events;
- }
-
- public function getAcceptedOutcomes($eventName) {
-
- $this->checkTransitions();
-
- $outcomes = array_keys($this->fTransitions[$this->fState][$eventName]);
-
-
- return $outcomes;
- }
-
- public function isEventAllowed($eventName) {
-
- $this->checkTransitions();
- $ret = in_array($eventName, $this->getAcceptedEvents());
-
- return $ret;
- }
-
- public function isOutcomeAllowed($eventName, $outcome) {
- $this->checkTransitions();
- if (!$this->isEventAllowed($eventName)) {
- return false;
- }
- $ret = array_key_exists($outcome, $this->getAcceptedOutcomes($eventName));
- return $ret;
- }
-
- public function applyEvent($eventName) {
-
- do {
- $result = $this->applySimpleEvent($eventName);
- if ($this->allowActions) {
- $eventName = $result->fsmAction;
- } else {
- $eventName = NULL;
- }
- } while ($eventName);
-
- return $result;
- }
-
- private function applySimpleEvent($eventName) {
-
- $currentState = $this->fState;
- if (($eventName == self::IDLE_EVENT) && !$this->idleWork) {
- return new Result();
- }
- if (!$this->isEventAllowed($eventName)) {
- die("Event $eventName not allowed in current state $currentState.\n");
-
- }
- $outcomes = $this->getAcceptedOutcomes($eventName);
- $methodName = "event$eventName";
- if (!method_exists($this, $methodName)) {
- die (func_name() . ": missing method "
- . get_class($this) . "::$methodName. Aborting.\n");
- }
- $outcome = $this->$methodName();
- if (!in_array($outcome, $outcomes)) {
- throw new \Exception(func_name()
- . ": event guard. Transition on \"$eventName\" return invalid outcome: "
- . print_r($outcome, true)
- . " for state $this->fState\n");
- }
- $transition = &$this->fTransitions[$currentState][$eventName][$outcome];
- $result = new Result (
- $outcome,
- $transition[0],
- $transition[1]
- );
- if (isset($result->fsmState)) {
- $this->fState = $result->fsmState;
- }
-
- if (!isset($outcome)) {
- $outcome = 'NULL';
- }
-
- return $result;
- }
-
- public function fIdle() {
- return TRUE;
- }
-
- public function idle() {
- return $this->applyEvent(self::IDLE_EVENT);
- }
-
- public function getEventMode() {
- return $this->fEventMode;
- }
-
- public function setEventMode($mode) {
- switch ($mode) {
- case self::EVENT_NORMAL :
- if (count($this->fQueue) > 0) {
- while (($event = array_shift($this->fQueue)) !== NULL) {
- $this->applyEvent($event);
- }
- }
- break;
-
- case self::EVENT_QUEUE:
- break;
-
- case self::EVENT_SINK :
- if (count($this->fQueue) > 0) {
- $this->fQueue = array();
- }
- break;
- default:
- throw new Exception("Trying to set unknown FSM mode $mode");
- }
- $this->fEventMode = $mode;
- return $this->fEventMode;
- }
-
- public function loadFsm($url = NULL) {
- $osd = FALSE;
- if ($osd) {
- echo "Loading FSM from $url\n";
- }
- if (!isset($url)) {
- $url = get_class($this) . ".xml";
- }
- $fsm = simplexml_load_file($url);
- $fsmVersion = (string) $fsm['fsm_version'];
- if ($fsmVersion !== '1.3') {
- die("Revision $fsmVersion of schema is not supported.\n");
- }
- $this->idleWork = ($fsm['idle_work'] == 1);
- $this->allowActions = ($fsm['allow_actions'] == 1);
- $this->fTransitions = array();
- $t = &$this->fTransitions;
-
- foreach ($fsm->state as $state) {
- $id = (string) $state['id'];
- if ($osd) {
- echo "State $id :\n";
- }
- $t[$id] = array();
- switch ($id) {
- case self::INIT_STATE :
- if ($osd) {
- echo " Initial state\n";
- }
- break;
- case self::FINAL_STATE :
- if ($osd) {
- echo " Final state\n";
- }
- break;
- }
- foreach ($state->event as $event) {
- $name = (string) $event['name'];
- if ($osd) {
- echo " Event $name";
- }
- if (!isset($event['type'])) {
- $event['type'] = 'void';
- }
- $eventType = (string) $event['type'];
- if ($osd) {
- echo ", type $eventType\n";
- }
- foreach ($event as $next) {
- if ($event['type'] == 'void') {
- $next['result'] = 'always';
- $result = null;
- }
- else {
- $result = (string) $next['result'];
- }
- if (!isset($next['state'])) {
- $next['state'] = (string) $state['id'];
- }
- if ($osd) {
- echo " Next(" . $next['result'] . ') = ' . $next['state'];
- if (isset($next['action'])) {
- echo " + event " . $next['action'];
- }
- echo PHP_EOL;
- }
- $t[$id][$name][$result] = array(
- (string) $next['state'],
- (string) $next['action']);
- }
- }
- }
- }
- }
|