*/ public static function getFormula( PhpParser\Node\Expr $conditional, $this_class_name, StatementsSource $source ) { if ($conditional instanceof PhpParser\Node\Expr\BinaryOp\BooleanAnd || $conditional instanceof PhpParser\Node\Expr\BinaryOp\LogicalAnd ) { $left_assertions = self::getFormula( $conditional->left, $this_class_name, $source ); $right_assertions = self::getFormula( $conditional->right, $this_class_name, $source ); return array_merge( $left_assertions, $right_assertions ); } if ($conditional instanceof PhpParser\Node\Expr\BinaryOp\BooleanOr || $conditional instanceof PhpParser\Node\Expr\BinaryOp\LogicalOr ) { // at the moment we only support formulae in CNF if (!$conditional->left instanceof PhpParser\Node\Expr\BinaryOp\BooleanAnd && !$conditional->left instanceof PhpParser\Node\Expr\BinaryOp\LogicalAnd ) { $left_clauses = self::getFormula( $conditional->left, $this_class_name, $source ); } else { $left_clauses = [new Clause([], true)]; } if (!$conditional->right instanceof PhpParser\Node\Expr\BinaryOp\BooleanAnd && !$conditional->right instanceof PhpParser\Node\Expr\BinaryOp\LogicalAnd ) { $right_clauses = self::getFormula( $conditional->right, $this_class_name, $source ); } else { $right_clauses = [new Clause([], true)]; } return self::combineOredClauses($left_clauses, $right_clauses); } $assertions = AssertionFinder::getAssertions( $conditional, $this_class_name, $source ); if ($assertions) { $clauses = []; foreach ($assertions as $var => $type) { if ($type === 'isset') { $key_parts = preg_split( '/(\]|\[)/', $var, -1, PREG_SPLIT_NO_EMPTY ); $base = array_shift($key_parts); $clauses[] = new Clause([$base => ['isset']]); if (count($key_parts)) { $clauses[] = new Clause([$base => ['!false']]); $clauses[] = new Clause([$base => ['!int']]); } foreach ($key_parts as $i => $key_part_dim) { $base .= '[' . $key_part_dim . ']'; $clauses[] = new Clause([$base => ['isset']]); if ($i < count($key_parts) - 1) { $clauses[] = new Clause([$base => ['!false']]); $clauses[] = new Clause([$base => ['!int']]); } } } else { $clauses[] = new Clause([$var => [$type]]); } } return $clauses; } return [new Clause([], true)]; } /** * 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 array $clauses * * @return array */ public static function negateFormula(array $clauses) { foreach ($clauses as $clause) { self::calculateNegation($clause); } return self::groupImpossibilities($clauses); } /** * @param Clause $clause * * @return void */ public static function calculateNegation(Clause $clause) { if ($clause->impossibilities !== null) { return; } $impossibilities = []; foreach ($clause->possibilities as $var_id => $possiblity) { $impossibility = []; foreach ($possiblity as $type) { if ($type[0] !== '^' && (!isset($type[1]) || $type[1] !== '^')) { $impossibility[] = TypeChecker::negateType($type); } } if ($impossibility) { $impossibilities[$var_id] = $impossibility; } } $clause->impossibilities = $impossibilities; } /** * This looks to see if there are any clauses in one formula that contradict * clauses in another formula, or clauses that duplicate previous clauses * * e.g. * if ($a) { } * elseif ($a) { } * * @param array $formula1 * @param array $formula2 * @param StatementsChecker $statements_checker, * @param PhpParser\Node $stmt * * @return void */ public static function checkForParadox( array $formula1, array $formula2, StatementsChecker $statements_checker, PhpParser\Node $stmt ) { $negated_formula2 = self::negateFormula($formula2); // remove impossible types foreach ($negated_formula2 as $clause_a) { foreach ($clause_a->possibilities as $key => $values) { if (count($values) > 1 && count(array_unique($values)) < count($values)) { if (IssueBuffer::accepts( new RedundantCondition( 'Found a redundant condition when evaluating ' . $key, new CodeLocation($statements_checker, $stmt) ), $statements_checker->getSuppressedIssues() )) { // fall through } } } if (!$clause_a->reconcilable || $clause_a->wedge) { continue; } foreach ($formula1 as $clause_b) { if ($clause_a === $clause_b || !$clause_b->reconcilable || $clause_b->wedge) { continue; } $clause_a_contains_b_possibilities = true; foreach ($clause_b->possibilities as $key => $keyed_possibilities) { if (!isset($clause_a->possibilities[$key])) { $clause_a_contains_b_possibilities = false; break; } if ($clause_a->possibilities[$key] != $keyed_possibilities) { $clause_a_contains_b_possibilities = false; break; } } if ($clause_a_contains_b_possibilities) { if (IssueBuffer::accepts( new ParadoxicalCondition( 'Encountered a paradox when evaluating the conditional', new CodeLocation($statements_checker, $stmt) ), $statements_checker->getSuppressedIssues() )) { // fall through } return; } } } } /** * This is a very simple simplification heuristic * for CNF formulae. * * It simplifies formulae: * ($a) && ($a || $b) => $a * (!$a) && (!$b) && ($a || $b || $c) => $c * * @param array $clauses * * @return array */ public static function simplifyCNF(array $clauses) { $cloned_clauses = []; // avoid strict duplicates foreach ($clauses as $clause) { $cloned_clauses[$clause->getHash()] = clone $clause; } // remove impossible types foreach ($cloned_clauses as $clause_a) { if (count($clause_a->possibilities) !== 1 || count(array_values($clause_a->possibilities)[0]) !== 1) { continue; } if (!$clause_a->reconcilable || $clause_a->wedge) { continue; } $clause_var = array_keys($clause_a->possibilities)[0]; $only_type = array_pop(array_values($clause_a->possibilities)[0]); $negated_clause_type = TypeChecker::negateType($only_type); foreach ($cloned_clauses as $clause_b) { if ($clause_a === $clause_b || !$clause_b->reconcilable || $clause_b->wedge) { continue; } if (isset($clause_b->possibilities[$clause_var]) && in_array($negated_clause_type, $clause_b->possibilities[$clause_var], true) ) { $clause_b->possibilities[$clause_var] = array_filter( $clause_b->possibilities[$clause_var], /** * @param string $possible_type * * @return bool */ function ($possible_type) use ($negated_clause_type) { return $possible_type !== $negated_clause_type; } ); if (count($clause_b->possibilities[$clause_var]) === 0) { unset($clause_b->possibilities[$clause_var]); } } } } $cloned_clauses = array_filter( $cloned_clauses, /** * @return bool */ function (Clause $clause) { return (bool)count($clause->possibilities); } ); $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) { continue; } if ($clause_a->contains($clause_b)) { $is_redundant = true; break; } } if (!$is_redundant) { $simplified_clauses[] = $clause_a; } } return $simplified_clauses; } /** * Look for clauses with only one possible value * * @param array $clauses * * @return array */ public static function getTruthsFromFormula(array $clauses) { $truths = []; if (empty($clauses)) { return []; } foreach ($clauses as $clause) { if (!$clause->reconcilable) { continue; } foreach ($clause->possibilities as $var => $possible_types) { // if there's only one possible type, return it if (count($clause->possibilities) === 1 && count($possible_types) === 1) { if (isset($truths[$var])) { $truths[$var] .= '&' . array_pop($possible_types); } else { $truths[$var] = array_pop($possible_types); } } elseif (count($clause->possibilities) === 1) { // if there's only one active clause, return all the non-negation clause members ORed together $things_that_can_be_said = array_filter( $possible_types, /** * @param string $possible_type * * @return bool */ function ($possible_type) { return $possible_type[0] !== '!'; } ); if ($things_that_can_be_said && count($things_that_can_be_said) === count($possible_types)) { $things_that_can_be_said = array_unique($things_that_can_be_said); $truths[$var] = implode('|', array_unique($things_that_can_be_said)); } } } } return $truths; } /** * @param array $clauses * * @return array */ protected static function groupImpossibilities(array $clauses) { $clause = array_pop($clauses); $new_clauses = []; if (count($clauses)) { $grouped_clauses = self::groupImpossibilities($clauses); foreach ($grouped_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($grouped_clause->possibilities[$var])) { $new_clause_possibilities[$var][] = $impossible_type; } else { $new_clause_possibilities[$var] = [$impossible_type]; } $new_clauses[] = new Clause($new_clause_possibilities); } } } } elseif ($clause && !$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) { $new_clauses[] = new Clause([$var => [$impossible_type]]); } } } return $new_clauses; } /** * @param array $left_clauses * @param array $right_clauses * * @return array */ public static function combineOredClauses(array $left_clauses, array $right_clauses) { // we cannot deal, at the moment, with orring non-CNF clauses if (count($left_clauses) !== 1 || count($right_clauses) !== 1) { return [new Clause([], true)]; } /** @var array> */ $possibilities = []; if ($left_clauses[0]->wedge && $right_clauses[0]->wedge) { return [new Clause([], true)]; } $can_reconcile = true; if ($left_clauses[0]->wedge || $right_clauses[0]->wedge || !$left_clauses[0]->reconcilable || !$right_clauses[0]->reconcilable ) { $can_reconcile = false; } foreach ($left_clauses[0]->possibilities as $var => $possible_types) { if (isset($possibilities[$var])) { $possibilities[$var] = array_merge($possibilities[$var], $possible_types); } else { $possibilities[$var] = $possible_types; } } foreach ($right_clauses[0]->possibilities as $var => $possible_types) { if (isset($possibilities[$var])) { $possibilities[$var] = array_merge($possibilities[$var], $possible_types); } else { $possibilities[$var] = $possible_types; } } return [new Clause($possibilities, false, $can_reconcile)]; } }