2020-05-19 00:57:02 +02:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Internal\Analyzer\Statements\Expression\BinaryOp;
|
|
|
|
|
|
|
|
use PhpParser;
|
|
|
|
use Psalm\Internal\Analyzer\Statements\ExpressionAnalyzer;
|
|
|
|
use Psalm\Internal\Analyzer\StatementsAnalyzer;
|
|
|
|
use Psalm\Internal\Analyzer\Statements\Block\IfAnalyzer;
|
|
|
|
use Psalm\CodeLocation;
|
|
|
|
use Psalm\Context;
|
|
|
|
use Psalm\Type\Algebra;
|
|
|
|
use Psalm\Type\Reconciler;
|
|
|
|
use function array_merge;
|
|
|
|
use function array_diff_key;
|
|
|
|
use function array_filter;
|
|
|
|
use function array_values;
|
|
|
|
use function array_map;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
class AndAnalyzer
|
|
|
|
{
|
|
|
|
public static function analyze(
|
|
|
|
StatementsAnalyzer $statements_analyzer,
|
|
|
|
PhpParser\Node\Expr\BinaryOp $stmt,
|
|
|
|
Context $context,
|
|
|
|
bool $from_stmt = false
|
|
|
|
) : bool {
|
|
|
|
if ($from_stmt) {
|
|
|
|
$fake_if_stmt = new PhpParser\Node\Stmt\If_(
|
|
|
|
$stmt->left,
|
|
|
|
[
|
|
|
|
'stmts' => [
|
|
|
|
new PhpParser\Node\Stmt\Expression(
|
|
|
|
$stmt->right
|
|
|
|
)
|
|
|
|
]
|
|
|
|
],
|
|
|
|
$stmt->getAttributes()
|
|
|
|
);
|
|
|
|
|
|
|
|
return IfAnalyzer::analyze($statements_analyzer, $fake_if_stmt, $context) !== false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$pre_referenced_var_ids = $context->referenced_var_ids;
|
|
|
|
|
|
|
|
$pre_assigned_var_ids = $context->assigned_var_ids;
|
|
|
|
|
|
|
|
$left_context = clone $context;
|
|
|
|
|
|
|
|
$left_context->referenced_var_ids = [];
|
|
|
|
$left_context->assigned_var_ids = [];
|
|
|
|
|
|
|
|
/** @var list<string> */
|
|
|
|
$left_context->reconciled_expression_clauses = [];
|
|
|
|
|
|
|
|
if (ExpressionAnalyzer::analyze($statements_analyzer, $stmt->left, $left_context) === false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$codebase = $statements_analyzer->getCodebase();
|
|
|
|
|
2020-08-26 21:35:29 +02:00
|
|
|
$left_cond_id = \spl_object_id($stmt->left);
|
|
|
|
|
2020-05-19 00:57:02 +02:00
|
|
|
$left_clauses = Algebra::getFormula(
|
2020-08-26 21:35:29 +02:00
|
|
|
$left_cond_id,
|
|
|
|
$left_cond_id,
|
2020-05-19 00:57:02 +02:00
|
|
|
$stmt->left,
|
|
|
|
$context->self,
|
|
|
|
$statements_analyzer,
|
|
|
|
$codebase
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach ($left_context->vars_in_scope as $var_id => $type) {
|
|
|
|
if (isset($left_context->assigned_var_ids[$var_id])) {
|
|
|
|
$context->vars_in_scope[$var_id] = $type;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @var array<string, bool> */
|
|
|
|
$left_referenced_var_ids = $left_context->referenced_var_ids;
|
|
|
|
$context->referenced_var_ids = array_merge($pre_referenced_var_ids, $left_referenced_var_ids);
|
|
|
|
|
|
|
|
$left_assigned_var_ids = array_diff_key($left_context->assigned_var_ids, $pre_assigned_var_ids);
|
|
|
|
|
|
|
|
$left_referenced_var_ids = array_diff_key($left_referenced_var_ids, $left_assigned_var_ids);
|
|
|
|
|
|
|
|
$context_clauses = array_merge($left_context->clauses, $left_clauses);
|
|
|
|
|
|
|
|
if ($left_context->reconciled_expression_clauses) {
|
|
|
|
$reconciled_expression_clauses = $left_context->reconciled_expression_clauses;
|
|
|
|
|
|
|
|
$context_clauses = array_values(
|
|
|
|
array_filter(
|
|
|
|
$context_clauses,
|
2020-09-12 17:24:05 +02:00
|
|
|
function ($c) use ($reconciled_expression_clauses): bool {
|
2020-08-26 16:41:47 +02:00
|
|
|
return !\in_array($c->hash, $reconciled_expression_clauses);
|
2020-05-19 00:57:02 +02:00
|
|
|
}
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
if (\count($context_clauses) === 1
|
|
|
|
&& $context_clauses[0]->wedge
|
|
|
|
&& !$context_clauses[0]->possibilities
|
|
|
|
) {
|
|
|
|
$context_clauses = [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$simplified_clauses = Algebra::simplifyCNF($context_clauses);
|
|
|
|
|
|
|
|
$active_left_assertions = [];
|
|
|
|
|
|
|
|
$left_type_assertions = Algebra::getTruthsFromFormula(
|
|
|
|
$simplified_clauses,
|
2020-08-26 21:35:29 +02:00
|
|
|
$left_cond_id,
|
2020-05-19 00:57:02 +02:00
|
|
|
$left_referenced_var_ids,
|
|
|
|
$active_left_assertions
|
|
|
|
);
|
|
|
|
|
|
|
|
$changed_var_ids = [];
|
|
|
|
|
|
|
|
$right_context = clone $left_context;
|
|
|
|
|
|
|
|
if ($left_type_assertions) {
|
|
|
|
// while in an and, we allow scope to boil over to support
|
|
|
|
// statements of the form if ($x && $x->foo())
|
|
|
|
$right_vars_in_scope = Reconciler::reconcileKeyedTypes(
|
|
|
|
$left_type_assertions,
|
|
|
|
$active_left_assertions,
|
|
|
|
$context->vars_in_scope,
|
|
|
|
$changed_var_ids,
|
|
|
|
$left_referenced_var_ids,
|
|
|
|
$statements_analyzer,
|
|
|
|
[],
|
|
|
|
$context->inside_loop,
|
2020-10-07 23:26:37 +02:00
|
|
|
new CodeLocation($statements_analyzer->getSource(), $stmt->left),
|
|
|
|
$context->inside_negation
|
2020-05-19 00:57:02 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
$right_context->vars_in_scope = $right_vars_in_scope;
|
|
|
|
|
|
|
|
if ($context->if_scope) {
|
|
|
|
$context->if_scope->if_cond_changed_var_ids += $changed_var_ids;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$partitioned_clauses = Context::removeReconciledClauses($left_clauses, $changed_var_ids);
|
|
|
|
|
|
|
|
$right_context->clauses = $partitioned_clauses[0];
|
|
|
|
|
|
|
|
if (ExpressionAnalyzer::analyze($statements_analyzer, $stmt->right, $right_context) === false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$context->referenced_var_ids = array_merge(
|
|
|
|
$right_context->referenced_var_ids,
|
|
|
|
$left_context->referenced_var_ids
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($context->inside_conditional) {
|
|
|
|
$context->updateChecks($right_context);
|
|
|
|
|
|
|
|
$context->vars_possibly_in_scope = array_merge(
|
|
|
|
$right_context->vars_possibly_in_scope,
|
|
|
|
$left_context->vars_possibly_in_scope
|
|
|
|
);
|
|
|
|
|
|
|
|
$context->assigned_var_ids = array_merge(
|
|
|
|
$left_context->assigned_var_ids,
|
|
|
|
$right_context->assigned_var_ids
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($context->if_context && !$context->inside_negation) {
|
|
|
|
$context->vars_in_scope = $right_context->vars_in_scope;
|
|
|
|
$if_context = $context->if_context;
|
|
|
|
|
|
|
|
foreach ($right_context->vars_in_scope as $var_id => $type) {
|
|
|
|
if (!isset($if_context->vars_in_scope[$var_id])) {
|
|
|
|
$if_context->vars_in_scope[$var_id] = $type;
|
|
|
|
} elseif (isset($context->vars_in_scope[$var_id])) {
|
|
|
|
$if_context->vars_in_scope[$var_id] = $context->vars_in_scope[$var_id];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$if_context->referenced_var_ids = array_merge(
|
|
|
|
$context->referenced_var_ids,
|
|
|
|
$if_context->referenced_var_ids
|
|
|
|
);
|
|
|
|
|
|
|
|
$if_context->assigned_var_ids = array_merge(
|
|
|
|
$context->assigned_var_ids,
|
|
|
|
$if_context->assigned_var_ids
|
|
|
|
);
|
|
|
|
|
|
|
|
$if_context->reconciled_expression_clauses = array_merge(
|
|
|
|
$if_context->reconciled_expression_clauses,
|
|
|
|
array_map(
|
|
|
|
function ($c) {
|
2020-08-26 16:41:47 +02:00
|
|
|
return $c->hash;
|
2020-05-19 00:57:02 +02:00
|
|
|
},
|
|
|
|
$partitioned_clauses[1]
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
$if_context->vars_possibly_in_scope = array_merge(
|
|
|
|
$context->vars_possibly_in_scope,
|
|
|
|
$if_context->vars_possibly_in_scope
|
|
|
|
);
|
|
|
|
|
|
|
|
$if_context->updateChecks($context);
|
|
|
|
} else {
|
|
|
|
$context->vars_in_scope = $left_context->vars_in_scope;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|