>> $all_types * * @return array>> * * @psalm-pure */ public static function negateTypes(array $all_types): array { $negated_types = []; foreach ($all_types as $key => $anded_types) { if (count($anded_types) > 1) { $new_anded_types = []; foreach ($anded_types as $orred_types) { if (count($orred_types) === 1) { $new_anded_types[] = $orred_types[0]->getNegation(); } else { continue 2; } } assert($new_anded_types !== []); $negated_types[$key] = [$new_anded_types]; continue; } $new_orred_types = []; foreach ($anded_types[0] as $orred_type) { $new_orred_types[] = [$orred_type->getNegation()]; } $negated_types[$key] = $new_orred_types; } return $negated_types; } /** * This is a very simple simplification heuristic * for CNF formulae. * * It simplifies formulae: * ($a) && ($a || $b) => $a * (!$a) && (!$b) && ($a || $b || $c) => $c * * @param list $clauses * * @return list * * @psalm-pure */ public static function simplifyCNF(array $clauses): array { $clause_count = count($clauses); //65536 seems to be a significant threshold, when put at 65537, the code https://psalm.dev/r/216f362ea6 goes //from seconds in analysis to many minutes if ($clause_count > 65_536) { return []; } if ($clause_count > 50) { $all_has_unknown = true; foreach ($clauses as $clause) { $clause_has_unknown = false; foreach ($clause->possibilities as $key => $_) { if ($key[0] === '*') { $clause_has_unknown = true; break; } } if (!$clause_has_unknown) { $all_has_unknown = false; break; } } if ($all_has_unknown) { return $clauses; } } $cloned_clauses = []; // avoid strict duplicates foreach ($clauses as $clause) { $cloned_clauses[$clause->hash] = $clause; } // remove impossible types foreach ($cloned_clauses as $clause_a_hash => $clause_a) { if (!$clause_a->reconcilable || $clause_a->wedge) { continue; } if (count($clause_a->possibilities) !== 1 || count(array_values($clause_a->possibilities)[0]) !== 1) { foreach ($cloned_clauses as $clause_b) { if ($clause_a === $clause_b || !$clause_b->reconcilable || $clause_b->wedge) { continue; } if (array_keys($clause_a->possibilities) === array_keys($clause_b->possibilities)) { $opposing_keys = []; foreach ($clause_a->possibilities as $key => $a_possibilities) { $b_possibilities = $clause_b->possibilities[$key]; if (array_keys($clause_a->possibilities[$key]) === array_keys($clause_b->possibilities[$key]) ) { continue; } if (count($a_possibilities) === 1 && count($b_possibilities) === 1) { if (reset($a_possibilities)->isNegationOf(reset($b_possibilities))) { $opposing_keys[] = $key; continue; } } continue 2; } if (count($opposing_keys) === 1) { unset($cloned_clauses[$clause_a_hash]); $clause_a = $clause_a->removePossibilities($opposing_keys[0]); if (!$clause_a) { continue 2; } $cloned_clauses[$clause_a->hash] = $clause_a; } } } continue; } $clause_var = array_keys($clause_a->possibilities)[0]; $only_type = array_pop(array_values($clause_a->possibilities)[0]); $negated_clause_type = $only_type->getNegation(); $negated_clause_type_string = (string)$negated_clause_type; foreach ($cloned_clauses as $clause_b_hash => $clause_b) { if ($clause_a === $clause_b || !$clause_b->reconcilable || $clause_b->wedge) { continue; } if (isset($clause_b->possibilities[$clause_var])) { $unmatched = []; $matched = []; foreach ($clause_b->possibilities[$clause_var] as $k => $possible_type) { if ((string)$possible_type === $negated_clause_type_string) { $matched[] = $possible_type; } else { $unmatched[$k] = $possible_type; } } if ($matched) { $clause_var_possibilities = $unmatched; unset($cloned_clauses[$clause_b_hash]); if (!$clause_var_possibilities) { $updated_clause = $clause_b->removePossibilities($clause_var); if ($updated_clause) { $cloned_clauses[$updated_clause->hash] = $updated_clause; } } else { $updated_clause = $clause_b->addPossibilities( $clause_var, $clause_var_possibilities ); $cloned_clauses[$updated_clause->hash] = $updated_clause; } } } } } $simplified_clauses = []; foreach ($cloned_clauses as $clause_a) { $is_redundant = false; foreach ($cloned_clauses as $clause_b) { if ($clause_a === $clause_b || !$clause_b->reconcilable || $clause_b->wedge || $clause_a->wedge ) { continue; } if ($clause_a->contains($clause_b)) { $is_redundant = true; break; } } if (!$is_redundant) { $simplified_clauses[$clause_a->hash] = $clause_a; } } $clause_count = count($simplified_clauses); // simplify (A || X) && (!A || Y) && (X || Y) // to // simplify (A || X) && (!A || Y) // where X and Y are sets of orred terms if ($clause_count > 2 && $clause_count < 256) { $clauses = array_values($simplified_clauses); for ($i = 0; $i < $clause_count; $i++) { $clause_a = $clauses[$i]; for ($k = $i + 1; $k < $clause_count; $k++) { $clause_b = $clauses[$k]; $common_keys = array_keys( array_intersect_key($clause_a->possibilities, $clause_b->possibilities) ); if ($common_keys) { $common_negated_keys = []; foreach ($common_keys as $common_key) { if (count($clause_a->possibilities[$common_key]) === 1 && count($clause_b->possibilities[$common_key]) === 1 && reset($clause_a->possibilities[$common_key])->isNegationOf( reset($clause_b->possibilities[$common_key]) ) ) { $common_negated_keys[] = $common_key; } } if ($common_negated_keys) { $new_possibilities = []; foreach ($clause_a->possibilities as $var_id => $possibilities) { if (in_array($var_id, $common_negated_keys, true)) { continue; } if (!isset($new_possibilities[$var_id])) { $new_possibilities[$var_id] = $possibilities; } else { $new_possibilities[$var_id] = array_merge( $new_possibilities[$var_id], $possibilities, ); } } foreach ($clause_b->possibilities as $var_id => $possibilities) { if (in_array($var_id, $common_negated_keys, true)) { continue; } if (!isset($new_possibilities[$var_id])) { $new_possibilities[$var_id] = $possibilities; } else { $new_possibilities[$var_id] = array_merge( $new_possibilities[$var_id], $possibilities, ); } } /** @psalm-suppress MixedArgumentTypeCoercion due I think to Psalm bug */ $conflict_clause = (new Clause( $new_possibilities, $clause_a->creating_conditional_id, $clause_a->creating_conditional_id, false, true, true, [] )); unset($simplified_clauses[$conflict_clause->hash]); } } } } } return array_values($simplified_clauses); } /** * Look for clauses with only one possible value * * @param list $clauses * @param array $cond_referenced_var_ids * @param array>> $active_truths * * @return array>> */ public static function getTruthsFromFormula( array $clauses, ?int $creating_conditional_id = null, array &$cond_referenced_var_ids = [], array &$active_truths = [] ): array { $truths = []; $active_truths = []; if ($clauses === []) { return []; } foreach ($clauses as $clause) { if (!$clause->reconcilable || count($clause->possibilities) !== 1) { continue; } foreach ($clause->possibilities as $var => $possible_types) { if ($var[0] === '*') { continue; } // if there's only one possible type, return it if (count($possible_types) === 1) { $possible_type = array_pop($possible_types); if (isset($truths[$var]) && !isset($clause->redefined_vars[$var])) { $truths[$var][] = [$possible_type]; } else { $truths[$var] = [[$possible_type]]; } if ($creating_conditional_id && $creating_conditional_id === $clause->creating_conditional_id) { if (!isset($active_truths[$var])) { $active_truths[$var] = []; } $active_truths[$var][count($truths[$var]) - 1] = [$possible_type]; } } else { // if there's only one active clause, return all the non-negation clause members ORed together $things_that_can_be_said = []; foreach ($possible_types as $assertion) { if ($assertion instanceof Falsy || !$assertion->isNegation()) { $things_that_can_be_said[(string)$assertion] = $assertion; } } if ($things_that_can_be_said && count($things_that_can_be_said) === count($possible_types)) { if ($clause->generated && count($possible_types) > 1) { unset($cond_referenced_var_ids[$var]); } $truths[$var] = [array_values($things_that_can_be_said)]; if ($creating_conditional_id && $creating_conditional_id === $clause->creating_conditional_id) { $active_truths[$var] = [array_values($things_that_can_be_said)]; } } } } } return $truths; } /** * @param non-empty-list $clauses * * @return list * * @psalm-pure */ public static function groupImpossibilities(array $clauses): array { $complexity = 1; $seed_clauses = []; $clause = array_pop($clauses); if (!$clause->wedge) { if ($clause->impossibilities === null) { throw new UnexpectedValueException('$clause->impossibilities should not be null'); } foreach ($clause->impossibilities as $var => $impossible_types) { foreach ($impossible_types as $impossible_type) { $seed_clause = new Clause( [$var => [(string)$impossible_type => $impossible_type]], $clause->creating_conditional_id, $clause->creating_object_id ); $seed_clauses[] = $seed_clause; ++$complexity; } } } if (!$clauses || !$seed_clauses) { return $seed_clauses; } $complexity_upper_bound = count($seed_clauses); foreach ($clauses as $clause) { $i = 0; foreach ($clause->possibilities as $p) { $i += count($p); } $complexity_upper_bound *= $i; if ($complexity_upper_bound > 20_000) { throw new ComplicatedExpressionException(); } } while ($clauses) { $clause = array_pop($clauses); $new_clauses = []; foreach ($seed_clauses as $grouped_clause) { if ($clause->impossibilities === null) { throw new UnexpectedValueException('$clause->impossibilities should not be null'); } foreach ($clause->impossibilities as $var => $impossible_types) { foreach ($impossible_types as $impossible_type) { $new_clause_possibilities = $grouped_clause->possibilities; if (isset($new_clause_possibilities[$var])) { $impossible_type_string = (string)$impossible_type; $new_clause_possibilities[$var][$impossible_type_string] = $impossible_type; foreach ($new_clause_possibilities[$var] as $ak => $av) { foreach ($new_clause_possibilities[$var] as $bk => $bv) { if ($ak == $bk) { break; } if ($ak !== $impossible_type_string && $bk !== $impossible_type_string) { continue; } if ($av->isNegationOf($bv)) { break 3; } } } } else { $new_clause_possibilities[$var] = [(string)$impossible_type => $impossible_type]; } $new_clause = new Clause( $new_clause_possibilities, $grouped_clause->creating_conditional_id, $clause->creating_object_id, false, true, true, [] ); $new_clauses[] = $new_clause; ++$complexity; if ($complexity > 20_000) { throw new ComplicatedExpressionException(); } } } } $seed_clauses = $new_clauses; } return $seed_clauses; } /** * @param list $left_clauses * @param list $right_clauses * * @return list * * @psalm-pure */ public static function combineOredClauses( array $left_clauses, array $right_clauses, int $conditional_object_id ): array { if (count($left_clauses) > 60_000 || count($right_clauses) > 60_000) { return []; } $clauses = []; $all_wedges = true; $has_wedge = false; foreach ($left_clauses as $left_clause) { foreach ($right_clauses as $right_clause) { $all_wedges = $all_wedges && ($left_clause->wedge && $right_clause->wedge); $has_wedge = $has_wedge || ($left_clause->wedge && $right_clause->wedge); } } if ($all_wedges) { return [new Clause([], $conditional_object_id, $conditional_object_id, true)]; } foreach ($left_clauses as $left_clause) { foreach ($right_clauses as $right_clause) { if ($left_clause->wedge && $right_clause->wedge) { // handled below continue; } /** @var array> */ $possibilities = []; $can_reconcile = true; if ($left_clause->wedge || $right_clause->wedge || !$left_clause->reconcilable || !$right_clause->reconcilable ) { $can_reconcile = false; } foreach ($left_clause->possibilities as $var => $possible_types) { if (isset($right_clause->redefined_vars[$var])) { continue; } if (isset($possibilities[$var])) { $possibilities[$var] = array_merge($possibilities[$var], $possible_types); } else { $possibilities[$var] = $possible_types; } } foreach ($right_clause->possibilities as $var => $possible_types) { if (isset($possibilities[$var])) { $possibilities[$var] = array_merge($possibilities[$var], $possible_types); } else { $possibilities[$var] = $possible_types; } } foreach ($possibilities as $var_possibilities) { if (count($var_possibilities) === 2) { $vals = array_values($var_possibilities); /** @psalm-suppress PossiblyUndefinedIntArrayOffset */ if ($vals[0]->isNegationOf($vals[1])) { continue 2; } } } $creating_conditional_id = $right_clause->creating_conditional_id === $left_clause->creating_conditional_id ? $right_clause->creating_conditional_id : $conditional_object_id; $clauses[] = new Clause( $possibilities, $creating_conditional_id, $creating_conditional_id, false, $can_reconcile, $right_clause->generated || $left_clause->generated || count($left_clauses) > 1 || count($right_clauses) > 1, [] ); } } if ($has_wedge) { $clauses[] = new Clause([], $conditional_object_id, $conditional_object_id, true); } return $clauses; } /** * Negates a set of clauses * negateClauses([$a || $b]) => !$a && !$b * negateClauses([$a, $b]) => !$a || !$b * negateClauses([$a, $b || $c]) => * (!$a || !$b) && * (!$a || !$c) * negateClauses([$a, $b || $c, $d || $e || $f]) => * (!$a || !$b || !$d) && * (!$a || !$b || !$e) && * (!$a || !$b || !$f) && * (!$a || !$c || !$d) && * (!$a || !$c || !$e) && * (!$a || !$c || !$f) * * @param list $clauses * * @return non-empty-list */ public static function negateFormula(array $clauses): array { $clauses = array_filter( $clauses, static fn(Clause $clause): bool => $clause->reconcilable ); if (!$clauses) { $cond_id = mt_rand(0, 100_000_000); return [new Clause([], $cond_id, $cond_id, true)]; } $clauses_with_impossibilities = []; foreach ($clauses as $clause) { $clauses_with_impossibilities[] = $clause->calculateNegation(); } unset($clauses); $impossible_clauses = self::groupImpossibilities($clauses_with_impossibilities); if (!$impossible_clauses) { $cond_id = mt_rand(0, 100_000_000); return [new Clause([], $cond_id, $cond_id, true)]; } $negated = self::simplifyCNF($impossible_clauses); if (!$negated) { $cond_id = mt_rand(0, 100_000_000); return [new Clause([], $cond_id, $cond_id, true)]; } return $negated; } }