2011-12-22 15:17:41 +13:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A Directed Acyclic Graph - used for doing topological sorts on dependencies, such as the before/after conditions
|
|
|
|
* in config yaml fragments
|
|
|
|
*/
|
2012-08-27 16:08:10 +12:00
|
|
|
class SS_DAG implements IteratorAggregate {
|
2011-12-22 15:17:41 +13:00
|
|
|
/** @var array|null - The nodes/vertices in the graph. Should be a numeric sequence of items (no string keys, no gaps). */
|
|
|
|
protected $data;
|
|
|
|
|
|
|
|
/** @var array - The edges in the graph, in $to_idx => [$from_idx1, $from_idx2, ...] format */
|
|
|
|
protected $dag;
|
|
|
|
|
2012-09-19 12:07:39 +02:00
|
|
|
public function __construct($data = null) {
|
2011-12-22 15:17:41 +13:00
|
|
|
$data = $data ? array_values($data) : array();
|
|
|
|
|
|
|
|
$this->data = $data;
|
|
|
|
$this->dag = array_fill_keys(array_keys($data), array());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add another node/vertex
|
|
|
|
* @param $item anything - The item to add to the graph
|
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public function additem($item) {
|
2011-12-22 15:17:41 +13:00
|
|
|
$this->data[] = $item;
|
|
|
|
$this->dag[] = array();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add an edge from one vertex to another
|
|
|
|
* @param $from integer|any - The index in $data of the node/vertex, or the node/vertex itself, that the edge goes from
|
|
|
|
* @param $to integer|any - The index in $data of the node/vertex, or the node/vertex itself, that the edge goes to
|
|
|
|
*
|
|
|
|
* When passing actual nodes (as opposed to indexes), uses array_search with strict = true to find
|
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public function addedge($from, $to) {
|
2011-12-22 15:17:41 +13:00
|
|
|
$i = is_numeric($from) ? $from : array_search($from, $this->data, true);
|
|
|
|
$j = is_numeric($to) ? $to : array_search($to, $this->data, true);
|
|
|
|
|
|
|
|
if ($i === false) throw new Exception("Couldnt find 'from' item in data when adding edge to DAG");
|
|
|
|
if ($j === false) throw new Exception("Couldnt find 'to' item in data when adding edge to DAG");
|
|
|
|
|
|
|
|
if (!isset($this->dag[$j])) $this->dag[$j] = array();
|
|
|
|
$this->dag[$j][] = $i;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sort graph so that each node (a) comes before any nodes (b) where an edge exists from a to b
|
|
|
|
* @return array - The nodes
|
|
|
|
* @throws Exception - If the graph is cyclic (and so can't be sorted)
|
|
|
|
*/
|
2012-09-19 12:07:39 +02:00
|
|
|
public function sort() {
|
2011-12-22 15:17:41 +13:00
|
|
|
$data = $this->data; $dag = $this->dag; $sorted = array();
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
$withedges = array_filter($dag, 'count');
|
|
|
|
$starts = array_diff_key($dag, $withedges);
|
|
|
|
|
|
|
|
if (!count($starts)) break;
|
|
|
|
|
|
|
|
foreach ($starts as $i => $foo) $sorted[] = $data[$i];
|
|
|
|
|
|
|
|
foreach ($withedges as $j => $deps) {
|
|
|
|
$withedges[$j] = array_diff($withedges[$j], array_keys($starts));
|
|
|
|
}
|
|
|
|
|
|
|
|
$dag = $withedges;
|
|
|
|
}
|
|
|
|
|
2012-08-27 16:08:10 +12:00
|
|
|
if ($dag) {
|
|
|
|
$remainder = new SS_DAG($data); $remainder->dag = $dag;
|
|
|
|
throw new SS_DAG_CyclicException("DAG has cyclic requirements", $remainder);
|
|
|
|
}
|
2011-12-22 15:17:41 +13:00
|
|
|
return $sorted;
|
|
|
|
}
|
2012-08-27 16:08:10 +12:00
|
|
|
|
2012-09-19 12:07:39 +02:00
|
|
|
public function getIterator() {
|
2012-08-27 16:08:10 +12:00
|
|
|
return new SS_DAG_Iterator($this->data, $this->dag);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class SS_DAG_CyclicException extends Exception {
|
|
|
|
|
|
|
|
public $dag;
|
|
|
|
|
2012-09-19 12:07:39 +02:00
|
|
|
public function __construct($message, $dag) {
|
2012-08-27 16:08:10 +12:00
|
|
|
$this->dag = $dag;
|
|
|
|
parent::__construct($message);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
class SS_DAG_Iterator implements Iterator {
|
|
|
|
|
|
|
|
protected $data;
|
|
|
|
protected $dag;
|
|
|
|
|
|
|
|
protected $dagkeys;
|
|
|
|
protected $i;
|
|
|
|
|
2012-09-19 12:07:39 +02:00
|
|
|
public function __construct($data, $dag) {
|
2012-08-27 16:08:10 +12:00
|
|
|
$this->data = $data;
|
|
|
|
$this->dag = $dag;
|
|
|
|
$this->rewind();
|
|
|
|
}
|
|
|
|
|
2012-09-19 12:07:39 +02:00
|
|
|
public function key() {
|
2012-08-27 16:08:10 +12:00
|
|
|
return $this->i;
|
|
|
|
}
|
|
|
|
|
2012-09-19 12:07:39 +02:00
|
|
|
public function current() {
|
2012-08-27 16:08:10 +12:00
|
|
|
$res = array();
|
|
|
|
|
|
|
|
$res['from'] = $this->data[$this->i];
|
|
|
|
|
|
|
|
$res['to'] = array();
|
|
|
|
foreach ($this->dag[$this->i] as $to) $res['to'][] = $this->data[$to];
|
|
|
|
|
|
|
|
return $res;
|
|
|
|
}
|
|
|
|
|
2012-09-19 12:07:39 +02:00
|
|
|
public function next() {
|
2012-08-27 16:08:10 +12:00
|
|
|
$this->i = array_shift($this->dagkeys);
|
|
|
|
}
|
|
|
|
|
2012-09-19 12:07:39 +02:00
|
|
|
public function rewind() {
|
2012-08-27 16:08:10 +12:00
|
|
|
$this->dagkeys = array_keys($this->dag);
|
|
|
|
$this->next();
|
|
|
|
}
|
|
|
|
|
2012-09-19 12:07:39 +02:00
|
|
|
public function valid() {
|
2012-08-27 16:08:10 +12:00
|
|
|
return $this->i !== null;
|
|
|
|
}
|
2012-03-24 16:04:52 +13:00
|
|
|
}
|