#[AllowDynamicProperties]
class Graph
{
/**
* Create a new graph of states to expore
*
* @param callable $states A function that returns the next states of the current one.
* Pass a set of fixed state relations to pick from it directly.
*/
private $states = null;
public function __construct($states, $target = "lowest")
{
// Either a function or a set of fixed states
if (is_callable($states)) {
$this->states = $states;
} else {
$this->states = function ($graph) use ($states) {
return $states[$graph->current];
};
}
$this->setTarget($target);
}
/**
* Get the next states to explore from the current one
*
* @param
* @return Set of states that should be explored
*/
public function getNextStates()
{
$states = call_user_func($this->states, $this);
// Prune the states, if needed
foreach ($this->pruning as $pruning) {
$states = $pruning($states, $this);
}
return $states;
}
/**
* Add a pruning function to the graph to filter the next states to explore.
*
* @return self
*/
public $pruning = [];
public function addPruning($pruning)
{
$this->pruning[] = $pruning;
return $this;
}
/**
* Define how a new state's value is calculated
*
* @param callback $get_new_value
* @return self
*/
public function defineNewValue($get_new_value)
{
$this->get_new_value = $get_new_value;
return $this;
}
/**
* Register a way to compare two values of a state
*
* @param callback $state_compare
* @return self
*/
public function defineBetterValue($is_better_value)
{
$this->is_better_value = $is_better_value;
return $this;
}
/**
* Set a priority heuristic for the given state
* Used for the A* algorithm
*
* @param callback $priority_heuristic
* @return self
*/
public function definePriority($get_priority)
{
$this->get_priority = $get_priority;
return $this;
}
/**
* Define how the end state is determined.
*
* @param callback $end_definition
* @return self
*/
public function defineEnd($is_end)
{
$this->is_end = $is_end;
return $this;
}
/**
* Set the type/priority of the queue to use for exploration.
* Either lowest or highest value first
*
* @param string $priority "lowest" or "highest" / "FIFO" or "LIFO"
* @return self
*/
public function setQueue($priority)
{
$this->queueType = $priority;
$this->queue = in_array($priority, ["lowest", "highest"])
? new PriorityQueue($priority)
: new Queue($priority);
return $this;
}
/**
* Set what should be returned when the end is reached
*
* @param callback $completion
* @return self
*/
public function onEnd($on_end)
{
$this->on_end = $on_end;
return $this;
}
/**
* Set what should be returned when all path have been explored
*
* @param callback $completion
* @return self
*/
public function onCompletion($on_completion)
{
$this->on_completion = $on_completion;
return $this;
}
// ==================================================
// > PRE-DEFINED PARAMETERS
// ==================================================
/**
* Set the target : lowest or highest outcome
*
* @param string $target
* @return self
*/
public function setTarget($target)
{
$this->target = $target;
switch ($this->target) {
// Lowest value possible : BFS or lowest priority queue and stop when end is reached
case "lowest":
// End when end node is reached
$this->defineEnd(function ($graph) {
return $graph->current == $graph->end;
});
// New value is the sum of the previous value and the relation
$this->defineNewValue(function ($relation, $state, $graph) {
return ($graph->values[$graph->current] ?? 0) + $relation;
});
// On end : return the path to the end
$this->onEnd(function ($graph) {
return $graph->getPathTo($graph->current);
});
// Explore by lowest value first
$this->setQueue("lowest")->definePriority(function ($graph, $state, $value) {
return $value;
});
// Keep the lowest values for each state
$this->defineBetterValue(function ($before, $after, $state, $graph) {
return $before > $after;
});
// If completion, no path found
$this->onCompletion(function ($graph) {
return false;
});
break;
// Highest value possible : DFS or highest priority queue and stop when all paths have been explored
case "highest":
// No end, expore everything
$this->defineEnd(function ($graph) {
return false;
});
// No priority, just FIFO
$this->setQueue("FIFO")->definePriority(fn () => false);
// New value is the sum of the previous value and the relation
$this->defineNewValue(function ($relation, $state, $graph) {
return ($graph->values[$graph->current] ?? 0) + $relation;
});
// Keep the highest values for each state
$this->defineBetterValue(function ($before, $after) {
return $before < $after;
});
// Return the highest value at completion
$this->onCompletion(function ($graph) {
return $graph->getPathTo(array_search(max($graph->values), $graph->values));
});
break;
}
return $this;
}
// =============================================================================
// > COMPUTATIONS
// =============================================================================
/**
* Explore the graph, starting from a given state until completion or the given end is reached.
* Return the path to the end and the value of the end state by default, or any value that was asked.
* @param string $from The state to start from
* @param string $to The state to end to, if any
* @param string $target "lowest" or "highest" outcome
* @return mixed
*/
public function explore($start = false, $end = false)
{
// Initialize with start state.
// Can be bypassed to continue exploring after changing the end goal.
if ($start) {
$this->start = $start;
$this->end = $end;
$this->previous = [];
$this->values = [];
$this->setQueue($this->queueType);
// Queue of nodes to visit
if (is_callable($this->start)) {
call_user_func($this->start, $this);
} else {
$this->queue->insert($start, 0);
$this->values[$start] = 0;
}
}
while ($this->queue->count()) {
$this->current = $this->queue->extract();
// Reached the end, stop searching
if (call_user_func($this->is_end, $this)) {
return call_user_func($this->on_end, $this);
}
// Add the current state to the visited list
foreach ($this->getNextStates() as $state => $relation) {
// Compute the new value
$new_value = call_user_func($this->get_new_value, $relation, $state, $this);
if (!isset($this->values[$state]) || call_user_func($this->is_better_value, $this->values[$state], $new_value, $state, $this)) {
$this->values[$state] = $new_value;
$this->previous[$state] = $this->current;
$this->queue->insert($state, call_user_func($this->get_priority, $this, $state, $new_value, $relation));
}
}
}
return call_user_func($this->on_completion, $this);
}
/**
* Get the final path to the current state
*
* @return array of states visited and the value at the end
*/
public function getPathTo($state)
{
$path = [];
$value = $this->values[$state];
while (!is_null($state)) {
$path[] = $state;
$state = $this->previous[$state] ?? null;
}
return [array_reverse($path), $value];
}
/**
* Get direct relations betewwen specific states (or all of them).
* Warning : should only be used for graph that are not too big
* @param array $states to get relations from
* @return Set
*/
public function getDirectRelations($states, $retrace = false)
{
$relations = set();
$states = (array) $states;
// Switch to flood-fill exploration
$graph = (clone $this)->defineEnd(function ($graph) use ($states, $relations, $retrace) {
if (!in_array($graph->current, $states)) return false;
if ($graph->current == $graph->start) return false;
// When a new state is reached, add it to the relations
$relations[$graph->start][$graph->current] = $retrace ? $graph->getPathTo($graph->current) : $graph->values[$graph->current];
// When all states have been found, stop
return count($relations[$graph->start]) >= count($states) - 1;
});
foreach ($states as $from) {
$graph->start = $from;
$graph->explore($from);
}
return $relations->instanciate("Set");
}
}