Utility classes

2019
#[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");
    }
}