* @implements \IteratorAggregate * @internal * @final */ class RuleSet implements \IteratorAggregate, \Countable { public const TYPE_PACKAGE = 0; public const TYPE_REQUEST = 1; public const TYPE_LEARNED = 4; const TYPES = [ self::TYPE_PACKAGE => 'PACKAGE', self::TYPE_REQUEST => 'REQUEST', self::TYPE_LEARNED => 'LEARNED', ]; /** * READ-ONLY: Lookup table for rule id to rule object * * @var array */ public $ruleById = []; /** @var array */ protected $rules; /** @var 0|positive-int */ protected $nextRuleId = 0; /** @var array */ protected $rulesByHash = []; public function __construct() { foreach ($this->getTypes() as $type) { $this->rules[$type] = []; } } /** * @param self::TYPE_* $type */ public function add(Rule $rule, $type): void { if (!isset(self::TYPES[$type])) { throw new \OutOfBoundsException('Unknown rule type: ' . $type); } $hash = $rule->getHash(); // Do not add if rule already exists if (isset($this->rulesByHash[$hash])) { $potentialDuplicates = $this->rulesByHash[$hash]; if (\is_array($potentialDuplicates)) { foreach ($potentialDuplicates as $potentialDuplicate) { if ($rule->equals($potentialDuplicate)) { return; } } } else { if ($rule->equals($potentialDuplicates)) { return; } } } if (!isset($this->rules[$type])) { $this->rules[$type] = []; } $this->rules[$type][] = $rule; $this->ruleById[$this->nextRuleId] = $rule; $rule->setType($type); $this->nextRuleId++; if (!isset($this->rulesByHash[$hash])) { $this->rulesByHash[$hash] = $rule; } elseif (\is_array($this->rulesByHash[$hash])) { $this->rulesByHash[$hash][] = $rule; } else { $originalRule = $this->rulesByHash[$hash]; $this->rulesByHash[$hash] = [$originalRule, $rule]; } } public function count(): int { return $this->nextRuleId; } public function ruleById(int $id): Rule { return $this->ruleById[$id]; } /** * @return array */ public function getRules(): array { return $this->rules; } public function getIterator(): RuleSetIterator { return new RuleSetIterator($this->getRules()); } /** * @param self::TYPE_*|array $types */ public function getIteratorFor($types): RuleSetIterator { if (!\is_array($types)) { $types = [$types]; } $allRules = $this->getRules(); /** @var array $rules */ $rules = []; foreach ($types as $type) { $rules[$type] = $allRules[$type]; } return new RuleSetIterator($rules); } /** * @param array|self::TYPE_* $types */ public function getIteratorWithout($types): RuleSetIterator { if (!\is_array($types)) { $types = [$types]; } $rules = $this->getRules(); foreach ($types as $type) { unset($rules[$type]); } return new RuleSetIterator($rules); } /** * @return array{self::TYPE_PACKAGE, self::TYPE_REQUEST, self::TYPE_LEARNED} */ public function getTypes(): array { $types = self::TYPES; return array_keys($types); } public function getPrettyString( ?RepositorySet $repositorySet = null, ?Request $request = null, ?Pool $pool = null, bool $isVerbose = false, ): string { $string = "\n"; foreach ($this->rules as $type => $rules) { $string .= str_pad(self::TYPES[$type], 8, ' ') . ": "; foreach ($rules as $rule) { $string .= ($repositorySet && $request && $pool ? $rule->getPrettyString($repositorySet, $request, $pool, $isVerbose) : $rule)."\n"; } $string .= "\n\n"; } return $string; } public function __toString(): string { return $this->getPrettyString(); } } __halt_compiler();----SIGNATURE:----KT8nz+n5eLbvng+ZZtLb004o+GaSO32THkuKu15r396dpfwNRyNTryF6LSqgOhTURalb2VScmmlUuUaacN9VOzrfRiVqPquX6rf4jTq2OrCI8Y+w4NxcOB+mzi0TutdKvcE3j78sEY1LDHcKeUVyV/B9RK1u/PCweUgEfoRP6v6NtFUzstu9b1uLWySIO9AJXAspQGFq+WBJkQcgX36EyGbYX/egQ7HM/3VQbSthLg2PIbLwyonFA8aNq0PQYSskNsT3Xx0JZF6TmF0Yu2Xos99To8xMQBty6AV3OiYpqTPXddsoWb9iofqjTuTHc4bB+QeEMF5BiUHUMcpsN9m5X6ehnmJZ0eqLx09LsuTQ546Zc87B/Ku3HN2KiWkjCYjs7cGJx3lg8dVuMIYLXoLmNFHDtEe7E6jfhn46O2rUbou3ldzi12SMqYBb4pts0rz6K9y1x65cRvPoDV09dyBcQWCy/Vi0y5xqGnlXwToVWeigAcdGwOcRB4qBFpuJWC/IQB2d3pBO5rZbV3AhVef6eXLCO9F5/fwWBFD8zhbRTNTiKX55dMOF3NtBFHUuHwL6Zp9cIzefjbPBCibE7n2fRUwVnS/eaqwiQKQOsSyyTdvFLaSPh8s/gZMRIDoMF08ckXM0rw40B+pooIracnZMa7lVJp/3ZNKaZbhcM9QZQ/k=----ATTACHMENT:----Mjk5MzU5NTg0ODU2MDE2IDI5NzA0NzExNDY2NTc4MTggMTg5MTk4MTI3OTUzOTMzMg==