Skip to main content
Drupal API
User account menu
  • Log in

Breadcrumb

  1. Drupal Core 11.1.x
  2. RuleWatchGraph.php

class RuleWatchGraph

The RuleWatchGraph efficiently propagates decisions to other rules

All rules generated for solving a SAT problem should be inserted into the graph. When a decision on a literal is made, the graph can be used to propagate the decision to all other rules involving the literal, leading to other trivial decisions resulting from unit clauses.

@author Nils Adermann <naderman@naderman.de>

Hierarchy

  • class \Composer\DependencyResolver\RuleWatchGraph

Expanded class hierarchy of RuleWatchGraph

File

vendor/composer/composer/src/Composer/DependencyResolver/RuleWatchGraph.php, line 25

Namespace

Composer\DependencyResolver
View source
class RuleWatchGraph {
    
    /** @var array<int, RuleWatchChain> */
    protected $watchChains = [];
    
    /**
     * Inserts a rule node into the appropriate chains within the graph
     *
     * The node is prepended to the watch chains for each of the two literals it
     * watches.
     *
     * Assertions are skipped because they only depend on a single package and
     * have no alternative literal that could be true, so there is no need to
     * watch changes in any literals.
     *
     * @param RuleWatchNode $node The rule node to be inserted into the graph
     */
    public function insert(RuleWatchNode $node) : void {
        if ($node->getRule()
            ->isAssertion()) {
            return;
        }
        if (!$node->getRule() instanceof MultiConflictRule) {
            foreach ([
                $node->watch1,
                $node->watch2,
            ] as $literal) {
                if (!isset($this->watchChains[$literal])) {
                    $this->watchChains[$literal] = new RuleWatchChain();
                }
                $this->watchChains[$literal]
                    ->unshift($node);
            }
        }
        else {
            foreach ($node->getRule()
                ->getLiterals() as $literal) {
                if (!isset($this->watchChains[$literal])) {
                    $this->watchChains[$literal] = new RuleWatchChain();
                }
                $this->watchChains[$literal]
                    ->unshift($node);
            }
        }
    }
    
    /**
     * Propagates a decision on a literal to all rules watching the literal
     *
     * If a decision, e.g. +A has been made, then all rules containing -A, e.g.
     * (-A|+B|+C) now need to satisfy at least one of the other literals, so
     * that the rule as a whole becomes true, since with +A applied the rule
     * is now (false|+B|+C) so essentially (+B|+C).
     *
     * This means that all rules watching the literal -A need to be updated to
     * watch 2 other literals which can still be satisfied instead. So literals
     * that conflict with previously made decisions are not an option.
     *
     * Alternatively it can occur that a unit clause results: e.g. if in the
     * above example the rule was (-A|+B), then A turning true means that
     * B must now be decided true as well.
     *
     * @param  int       $decidedLiteral The literal which was decided (A in our example)
     * @param  int       $level          The level at which the decision took place and at which
     *                                   all resulting decisions should be made.
     * @param  Decisions $decisions      Used to check previous decisions and to
     *                                   register decisions resulting from propagation
     * @return Rule|null If a conflict is found the conflicting rule is returned
     */
    public function propagateLiteral(int $decidedLiteral, int $level, Decisions $decisions) : ?Rule {
        // we invert the decided literal here, example:
        // A was decided => (-A|B) now requires B to be true, so we look for
        // rules which are fulfilled by -A, rather than A.
        $literal = -$decidedLiteral;
        if (!isset($this->watchChains[$literal])) {
            return null;
        }
        $chain = $this->watchChains[$literal];
        $chain->rewind();
        while ($chain->valid()) {
            $node = $chain->current();
            if (!$node->getRule() instanceof MultiConflictRule) {
                $otherWatch = $node->getOtherWatch($literal);
                if (!$node->getRule()
                    ->isDisabled() && !$decisions->satisfy($otherWatch)) {
                    $ruleLiterals = $node->getRule()
                        ->getLiterals();
                    $alternativeLiterals = array_filter($ruleLiterals, static function ($ruleLiteral) use ($literal, $otherWatch, $decisions) : bool {
                        return $literal !== $ruleLiteral && $otherWatch !== $ruleLiteral && !$decisions->conflict($ruleLiteral);
                    });
                    if (\count($alternativeLiterals) > 0) {
                        reset($alternativeLiterals);
                        $this->moveWatch($literal, current($alternativeLiterals), $node);
                        continue;
                    }
                    if ($decisions->conflict($otherWatch)) {
                        return $node->getRule();
                    }
                    $decisions->decide($otherWatch, $level, $node->getRule());
                }
            }
            else {
                foreach ($node->getRule()
                    ->getLiterals() as $otherLiteral) {
                    if ($literal !== $otherLiteral && !$decisions->satisfy($otherLiteral)) {
                        if ($decisions->conflict($otherLiteral)) {
                            return $node->getRule();
                        }
                        $decisions->decide($otherLiteral, $level, $node->getRule());
                    }
                }
            }
            $chain->next();
        }
        return null;
    }
    
    /**
     * Moves a rule node from one watch chain to another
     *
     * The rule node's watched literals are updated accordingly.
     *
     * @param int           $fromLiteral A literal the node used to watch
     * @param int           $toLiteral   A literal the node should watch now
     * @param RuleWatchNode $node        The rule node to be moved
     */
    protected function moveWatch(int $fromLiteral, int $toLiteral, RuleWatchNode $node) : void {
        if (!isset($this->watchChains[$toLiteral])) {
            $this->watchChains[$toLiteral] = new RuleWatchChain();
        }
        $node->moveWatch($fromLiteral, $toLiteral);
        $this->watchChains[$fromLiteral]
            ->remove();
        $this->watchChains[$toLiteral]
            ->unshift($node);
    }

}

Members

Title Sort descending Modifiers Object type Summary
RuleWatchGraph::$watchChains protected property @var array&lt;int, RuleWatchChain&gt;
RuleWatchGraph::insert public function Inserts a rule node into the appropriate chains within the graph
RuleWatchGraph::moveWatch protected function Moves a rule node from one watch chain to another
RuleWatchGraph::propagateLiteral public function Propagates a decision on a literal to all rules watching the literal

API Navigation

  • Drupal Core 11.1.x
  • Topics
  • Classes
  • Functions
  • Constants
  • Globals
  • Files
  • Namespaces
  • Deprecated
  • Services
RSS feed
Powered by Drupal