|
@@ -0,0 +1,223 @@
|
|
|
+<?php
|
|
|
+
|
|
|
+ * Input converter for FSM from Dia
|
|
|
+ *
|
|
|
+ * This class converts an UML diagram from Dia 0.9.6
|
|
|
+ * into an abstract FSM graph which can then be output by another class
|
|
|
+ * to the FSM XML format used by the OSInet FSM 1.6.
|
|
|
+ *
|
|
|
+ * @copyright 2007 Ouest Systèmes Informatiques
|
|
|
+ * @author Frederic G. MARAND
|
|
|
+ * @license CeCILL 2.0
|
|
|
+ * @version CVS: $Id: Fsm_From_Dia.php,v 1.1 2007-06-10 19:37:09 marand Exp $
|
|
|
+ * @link http:
|
|
|
+ * @since FSM 1.6
|
|
|
+ * @package fsm
|
|
|
+ * @subpackage fsm.ui
|
|
|
+*/
|
|
|
+
|
|
|
+$erFsmFromDia = error_reporting(E_ALL|E_STRICT);
|
|
|
+
|
|
|
+
|
|
|
+ * This class converts an UML diagram from Dia 0.9.6
|
|
|
+ * into an abstract FSM graph.
|
|
|
+ *
|
|
|
+ * @todo Validate the diagram: currently it will just choke on non-accepted diagrams
|
|
|
+ */
|
|
|
+class Fsm_From_Dia
|
|
|
+ {
|
|
|
+ const DIA_NAMESPACE = "http://www.lysator.liu.se/~alla/dia/";
|
|
|
+
|
|
|
+
|
|
|
+ * The DOM for the source Dia diagram
|
|
|
+ *
|
|
|
+ * @var DOMDocument
|
|
|
+ */
|
|
|
+ protected $dom;
|
|
|
+
|
|
|
+
|
|
|
+ * Extract the initial/final status from a Dia "UML - State Term"
|
|
|
+ *
|
|
|
+ * @param DOMElement $element
|
|
|
+ * @return array
|
|
|
+ */
|
|
|
+ protected function getStateTermInfo($element)
|
|
|
+ {
|
|
|
+ $query = 'dia:attribute[@name="is_final"]/dia:boolean';
|
|
|
+
|
|
|
+ $xpath = new DOMXPath($this->dom);
|
|
|
+ $xpath->registerNamespace("dia", Fsm_From_Dia::DIA_NAMESPACE);
|
|
|
+
|
|
|
+ $id = $element->getAttribute('id');
|
|
|
+ $dia_attributes = $xpath->query($query, $element);
|
|
|
+ if ($dia_attributes->length == 1)
|
|
|
+ {
|
|
|
+ $dia_boolean = $dia_attributes->item(0);
|
|
|
+ $val = $dia_boolean->getAttribute('val');
|
|
|
+ switch ($val)
|
|
|
+ {
|
|
|
+ case 'true': $ret = 'final'; break;
|
|
|
+ case 'false': $ret = 'initial'; break;
|
|
|
+ default: $ret = "anomalous($val)"; break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ echo "Initial/final state #$id does not bear the is_final attribute: anomaly.\n";
|
|
|
+ }
|
|
|
+
|
|
|
+ return array('type' => $ret, 'name' => $ret);
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ * Extract the name from a Dia "UML - State"
|
|
|
+ *
|
|
|
+ * @param DOMElement $element
|
|
|
+ * @return array
|
|
|
+ */
|
|
|
+ protected function getStateInfo($element)
|
|
|
+ {
|
|
|
+ $query = 'dia:attribute[@name="text"]/dia:composite/dia:attribute[@name="string"]/dia:string';
|
|
|
+ $xpath = new DOMXPath($this->dom);
|
|
|
+ $xpath->registerNamespace("dia", Fsm_From_Dia::DIA_NAMESPACE);
|
|
|
+
|
|
|
+ $id = $element->getAttribute('id');
|
|
|
+ $dia_string = $xpath->query($query, $element);
|
|
|
+ if ($dia_string->length == 1)
|
|
|
+ {
|
|
|
+ $dia_text = $dia_string->item(0);
|
|
|
+ $ret = trim($dia_text->textContent, '#');
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ echo "Standard state #$id does not contain the expected content: anomaly.\n";
|
|
|
+ }
|
|
|
+ return array('type' => 'standard', 'name' => $ret);
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+
|
|
|
+ * Extract the actual text content from a query for a Dia string element.
|
|
|
+ * This is to be used over XPath query results
|
|
|
+ *
|
|
|
+ * @param DOMNodeList $nodes
|
|
|
+ * @return string
|
|
|
+ */
|
|
|
+ private function getTextFromDiaString ($nodes)
|
|
|
+ {
|
|
|
+ if ($nodes->length == 1)
|
|
|
+ {
|
|
|
+ $ret = $nodes->item(0);
|
|
|
+ $ret = trim($ret->textContent, '#');
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ $ret = NULL;
|
|
|
+ }
|
|
|
+ return $ret;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ * Extract the various information fields from a Dia "UML - Connection"
|
|
|
+ *
|
|
|
+ * @param DOMElement $element
|
|
|
+ * @return array
|
|
|
+ */
|
|
|
+ protected function getTransitionInfo($element)
|
|
|
+ {
|
|
|
+ $xpath = new DOMXPath($this->dom);
|
|
|
+ $xpath->registerNamespace("dia", Fsm_From_Dia::DIA_NAMESPACE );
|
|
|
+ $id = $element->getAttribute('id');
|
|
|
+
|
|
|
+ $baseQuery = 'dia:attribute[@name="%s"]/dia:string';
|
|
|
+ foreach (array('trigger', 'action', 'guard') as $parameter)
|
|
|
+ {
|
|
|
+ $query = sprintf($baseQuery, $parameter);
|
|
|
+ $nodes = $xpath->query($query, $element);
|
|
|
+ $ret[$parameter] = $this->getTextFromDiaString($nodes);
|
|
|
+ if ($parameter <> 'guard')
|
|
|
+ {
|
|
|
+ $ret[$parameter] = ucfirst($ret[$parameter]);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ $query = 'dia:connections/dia:connection';
|
|
|
+ $dia_connections = $xpath->query($query, $element);
|
|
|
+
|
|
|
+ if ($dia_connections->length == 2)
|
|
|
+ {
|
|
|
+
|
|
|
+ foreach ($dia_connections as $end)
|
|
|
+ {
|
|
|
+ $handle = $end->getAttribute('handle');
|
|
|
+ $link = $end->getAttribute('to');
|
|
|
+ switch ($handle)
|
|
|
+ {
|
|
|
+ case '0' : $kind = 'from'; $ret['from'] = $link ; break;
|
|
|
+ case '1' : $kind = 'to'; $ret['to'] = $link ; break;
|
|
|
+ default: $kind = "anomaly($handle)"; break;
|
|
|
+ }
|
|
|
+
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ echo "Anomaly detected on the connection properties of transiition #$id\n";
|
|
|
+ }
|
|
|
+
|
|
|
+ return $ret;
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ * Load a Dia file into the DOM
|
|
|
+ *
|
|
|
+ * @param string $filePath
|
|
|
+ * @return void
|
|
|
+ */
|
|
|
+ public function __construct($filePath)
|
|
|
+ {
|
|
|
+ $this->dom = new DOMDocument();
|
|
|
+ $this->dom->load($filePath);
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+ * Parse the DOM to extract the various UML elements to an abstract FSM array
|
|
|
+ * @return array
|
|
|
+ */
|
|
|
+ public function parse()
|
|
|
+ {
|
|
|
+ $query = '//dia:object';
|
|
|
+ $xpath = new DOMXPath($this->dom);
|
|
|
+ $xpath->registerNamespace("dia", Fsm_From_Dia::DIA_NAMESPACE);
|
|
|
+ $result = $xpath->query($query);
|
|
|
+
|
|
|
+ foreach ($result as $object)
|
|
|
+ {
|
|
|
+ $type = $object->getAttribute('type');
|
|
|
+ $id = $object->getAttribute('id');
|
|
|
+ switch ($type)
|
|
|
+ {
|
|
|
+ case 'UML - State Term':
|
|
|
+ $states[$id] = $this->getStateTermInfo($object);
|
|
|
+ break;
|
|
|
+
|
|
|
+ case 'UML - State':
|
|
|
+ $states[$id] = $this->getStateInfo($object);
|
|
|
+ break;
|
|
|
+
|
|
|
+ case 'UML - Transition':
|
|
|
+ $transitions[$id] = $this->getTransitionInfo($object);
|
|
|
+ break;
|
|
|
+
|
|
|
+ default:
|
|
|
+ echo "Object #$id is of unknown type $type: ignored.\n";
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return array('states' => $states, 'transitions' => $transitions);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+
|
|
|
+error_reporting($erFsmFromDia);
|
|
|
+unset ($erFsmFromDia);
|