2018-01-14 18:09:40 +01:00
|
|
|
<?php
|
2018-11-06 03:57:36 +01:00
|
|
|
namespace Psalm\Internal\Analyzer\Statements\Expression\Assignment;
|
2018-01-14 18:09:40 +01:00
|
|
|
|
|
|
|
use PhpParser;
|
2018-11-06 03:57:36 +01:00
|
|
|
use Psalm\Internal\Analyzer\Statements\Expression\Fetch\ArrayFetchAnalyzer;
|
|
|
|
use Psalm\Internal\Analyzer\Statements\ExpressionAnalyzer;
|
|
|
|
use Psalm\Internal\Analyzer\StatementsAnalyzer;
|
2018-01-14 18:09:40 +01:00
|
|
|
use Psalm\Context;
|
|
|
|
use Psalm\Type;
|
|
|
|
use Psalm\Type\Atomic\ObjectLike;
|
|
|
|
use Psalm\Type\Atomic\TArray;
|
2019-10-09 00:44:46 +02:00
|
|
|
use Psalm\Type\Atomic\TList;
|
2018-11-09 16:56:27 +01:00
|
|
|
use Psalm\Type\Atomic\TNonEmptyArray;
|
2019-10-09 00:44:46 +02:00
|
|
|
use Psalm\Type\Atomic\TNonEmptyList;
|
2019-06-26 22:52:29 +02:00
|
|
|
use function array_reverse;
|
|
|
|
use function array_shift;
|
|
|
|
use function count;
|
|
|
|
use function array_unshift;
|
|
|
|
use function preg_match;
|
|
|
|
use function is_string;
|
|
|
|
use function implode;
|
|
|
|
use function array_pop;
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2018-12-02 00:37:49 +01:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2018-11-06 03:57:36 +01:00
|
|
|
class ArrayAssignmentAnalyzer
|
2018-01-14 18:09:40 +01:00
|
|
|
{
|
|
|
|
/**
|
2019-01-31 18:45:47 +01:00
|
|
|
* @param StatementsAnalyzer $statements_analyzer
|
2018-01-14 18:09:40 +01:00
|
|
|
* @param PhpParser\Node\Expr\ArrayDimFetch $stmt
|
|
|
|
* @param Context $context
|
2019-01-31 18:45:47 +01:00
|
|
|
* @param PhpParser\Node\Expr|null $assign_value
|
2018-01-14 18:09:40 +01:00
|
|
|
* @param Type\Union $assignment_value_type
|
|
|
|
*
|
2018-03-07 17:16:56 +01:00
|
|
|
* @return void
|
2018-01-14 18:09:40 +01:00
|
|
|
*/
|
|
|
|
public static function analyze(
|
2018-11-11 18:01:14 +01:00
|
|
|
StatementsAnalyzer $statements_analyzer,
|
2018-01-14 18:09:40 +01:00
|
|
|
PhpParser\Node\Expr\ArrayDimFetch $stmt,
|
|
|
|
Context $context,
|
2019-01-31 18:45:47 +01:00
|
|
|
$assign_value,
|
2018-01-14 18:09:40 +01:00
|
|
|
Type\Union $assignment_value_type
|
|
|
|
) {
|
|
|
|
$nesting = 0;
|
2018-11-06 03:57:36 +01:00
|
|
|
$var_id = ExpressionAnalyzer::getVarId(
|
2018-01-14 18:09:40 +01:00
|
|
|
$stmt->var,
|
2018-11-11 18:01:14 +01:00
|
|
|
$statements_analyzer->getFQCLN(),
|
|
|
|
$statements_analyzer,
|
2018-01-14 18:09:40 +01:00
|
|
|
$nesting
|
|
|
|
);
|
|
|
|
|
|
|
|
self::updateArrayType(
|
2018-11-11 18:01:14 +01:00
|
|
|
$statements_analyzer,
|
2018-01-14 18:09:40 +01:00
|
|
|
$stmt,
|
2019-01-31 18:45:47 +01:00
|
|
|
$assign_value,
|
2018-01-14 18:09:40 +01:00
|
|
|
$assignment_value_type,
|
|
|
|
$context
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!isset($stmt->var->inferredType) && $var_id) {
|
|
|
|
$context->vars_in_scope[$var_id] = Type::getMixed();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @return false|null
|
|
|
|
*/
|
|
|
|
public static function updateArrayType(
|
2018-11-11 18:01:14 +01:00
|
|
|
StatementsAnalyzer $statements_analyzer,
|
2018-01-14 18:09:40 +01:00
|
|
|
PhpParser\Node\Expr\ArrayDimFetch $stmt,
|
2019-10-12 18:23:40 +02:00
|
|
|
?PhpParser\Node\Expr $assign_value,
|
2018-01-14 18:09:40 +01:00
|
|
|
Type\Union $assignment_type,
|
|
|
|
Context $context
|
|
|
|
) {
|
|
|
|
$root_array_expr = $stmt;
|
|
|
|
|
|
|
|
$child_stmts = [];
|
|
|
|
|
|
|
|
while ($root_array_expr->var instanceof PhpParser\Node\Expr\ArrayDimFetch) {
|
|
|
|
$child_stmts[] = $root_array_expr;
|
|
|
|
$root_array_expr = $root_array_expr->var;
|
|
|
|
}
|
|
|
|
|
|
|
|
$child_stmts[] = $root_array_expr;
|
|
|
|
$root_array_expr = $root_array_expr->var;
|
|
|
|
|
2018-11-06 03:57:36 +01:00
|
|
|
if (ExpressionAnalyzer::analyze(
|
2018-11-11 18:01:14 +01:00
|
|
|
$statements_analyzer,
|
2018-01-14 18:09:40 +01:00
|
|
|
$root_array_expr,
|
|
|
|
$context,
|
|
|
|
true
|
|
|
|
) === false) {
|
|
|
|
// fall through
|
|
|
|
}
|
|
|
|
|
2019-01-05 06:15:53 +01:00
|
|
|
$codebase = $statements_analyzer->getCodebase();
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
$root_type = isset($root_array_expr->inferredType) ? $root_array_expr->inferredType : Type::getMixed();
|
|
|
|
|
2018-12-08 19:18:55 +01:00
|
|
|
if ($root_type->hasMixed()) {
|
2019-03-31 21:27:52 +02:00
|
|
|
if (ExpressionAnalyzer::analyze(
|
|
|
|
$statements_analyzer,
|
|
|
|
$stmt->var,
|
|
|
|
$context,
|
|
|
|
true
|
|
|
|
) === false) {
|
|
|
|
// fall through
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($stmt->dim) {
|
|
|
|
if (ExpressionAnalyzer::analyze(
|
|
|
|
$statements_analyzer,
|
|
|
|
$stmt->dim,
|
|
|
|
$context
|
|
|
|
) === false) {
|
|
|
|
// fall through
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$child_stmts = array_reverse($child_stmts);
|
|
|
|
|
|
|
|
$current_type = $root_type;
|
|
|
|
|
|
|
|
$current_dim = $stmt->dim;
|
|
|
|
|
|
|
|
$reversed_child_stmts = [];
|
|
|
|
|
|
|
|
// gets a variable id that *may* contain array keys
|
2018-11-06 03:57:36 +01:00
|
|
|
$root_var_id = ExpressionAnalyzer::getRootVarId(
|
2018-01-14 18:09:40 +01:00
|
|
|
$root_array_expr,
|
2018-11-11 18:01:14 +01:00
|
|
|
$statements_analyzer->getFQCLN(),
|
|
|
|
$statements_analyzer
|
2018-01-14 18:09:40 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
$var_id_additions = [];
|
|
|
|
|
2018-08-21 06:28:39 +02:00
|
|
|
$parent_var_id = null;
|
|
|
|
|
2018-02-17 18:32:19 +01:00
|
|
|
$full_var_id = true;
|
2018-01-14 18:09:40 +01:00
|
|
|
|
|
|
|
$child_stmt = null;
|
|
|
|
|
2019-10-12 18:23:40 +02:00
|
|
|
$taint_sources = [];
|
|
|
|
$taint_type = 0;
|
|
|
|
|
|
|
|
if ($codebase->taint && isset($assign_value->inferredType)) {
|
|
|
|
$taint_sources = $assign_value->inferredType->sources;
|
|
|
|
$taint_type = $assign_value->inferredType->tainted ?: 0;
|
|
|
|
}
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
// First go from the root element up, and go as far as we can to figure out what
|
|
|
|
// array types there are
|
|
|
|
while ($child_stmts) {
|
|
|
|
$child_stmt = array_shift($child_stmts);
|
|
|
|
|
|
|
|
if (count($child_stmts)) {
|
|
|
|
array_unshift($reversed_child_stmts, $child_stmt);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($child_stmt->dim) {
|
2018-11-06 03:57:36 +01:00
|
|
|
if (ExpressionAnalyzer::analyze(
|
2018-11-11 18:01:14 +01:00
|
|
|
$statements_analyzer,
|
2018-01-14 18:09:40 +01:00
|
|
|
$child_stmt->dim,
|
|
|
|
$context
|
|
|
|
) === false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isset($child_stmt->dim->inferredType)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2018-05-31 22:49:01 +02:00
|
|
|
if ($child_stmt->dim instanceof PhpParser\Node\Scalar\String_
|
|
|
|
|| ($child_stmt->dim instanceof PhpParser\Node\Expr\ConstFetch
|
|
|
|
&& $child_stmt->dim->inferredType->isSingleStringLiteral())
|
|
|
|
) {
|
|
|
|
if ($child_stmt->dim instanceof PhpParser\Node\Scalar\String_) {
|
|
|
|
$value = $child_stmt->dim->value;
|
|
|
|
} else {
|
2018-08-09 03:31:13 +02:00
|
|
|
$value = $child_stmt->dim->inferredType->getSingleStringLiteral()->value;
|
2018-05-31 22:49:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (preg_match('/^(0|[1-9][0-9]*)$/', $value)) {
|
|
|
|
$var_id_additions[] = '[' . $value . ']';
|
|
|
|
}
|
|
|
|
$var_id_additions[] = '[\'' . $value . '\']';
|
|
|
|
} elseif ($child_stmt->dim instanceof PhpParser\Node\Scalar\LNumber
|
|
|
|
|| ($child_stmt->dim instanceof PhpParser\Node\Expr\ConstFetch
|
|
|
|
&& $child_stmt->dim->inferredType->isSingleIntLiteral())
|
|
|
|
) {
|
|
|
|
if ($child_stmt->dim instanceof PhpParser\Node\Scalar\LNumber) {
|
|
|
|
$value = $child_stmt->dim->value;
|
|
|
|
} else {
|
2018-08-09 03:31:13 +02:00
|
|
|
$value = $child_stmt->dim->inferredType->getSingleIntLiteral()->value;
|
2018-05-03 19:56:30 +02:00
|
|
|
}
|
2018-05-31 22:49:01 +02:00
|
|
|
|
|
|
|
$var_id_additions[] = '[' . $value . ']';
|
2018-02-17 17:24:08 +01:00
|
|
|
} elseif ($child_stmt->dim instanceof PhpParser\Node\Expr\Variable
|
|
|
|
&& is_string($child_stmt->dim->name)
|
|
|
|
) {
|
|
|
|
$var_id_additions[] = '[$' . $child_stmt->dim->name . ']';
|
2019-10-03 21:27:50 +02:00
|
|
|
} elseif ($child_stmt->dim instanceof PhpParser\Node\Expr\PropertyFetch
|
|
|
|
&& $child_stmt->dim->name instanceof PhpParser\Node\Identifier
|
|
|
|
) {
|
|
|
|
$object_id = ExpressionAnalyzer::getArrayVarId(
|
|
|
|
$child_stmt->dim->var,
|
|
|
|
$statements_analyzer->getFQCLN(),
|
|
|
|
$statements_analyzer
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($object_id) {
|
|
|
|
$var_id_additions[] = '[' . $object_id . '->' . $child_stmt->dim->name->name . ']';
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
} else {
|
|
|
|
$var_id_additions[] = '[' . $child_stmt->dim->inferredType . ']';
|
2018-02-17 18:32:19 +01:00
|
|
|
$full_var_id = false;
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$var_id_additions[] = '';
|
2018-02-17 18:32:19 +01:00
|
|
|
$full_var_id = false;
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!isset($child_stmt->var->inferredType)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($child_stmt->var->inferredType->isEmpty()) {
|
|
|
|
$child_stmt->var->inferredType = Type::getEmptyArray();
|
|
|
|
}
|
|
|
|
|
|
|
|
$array_var_id = $root_var_id . implode('', $var_id_additions);
|
|
|
|
|
2018-08-21 06:28:39 +02:00
|
|
|
if ($parent_var_id && isset($context->vars_in_scope[$parent_var_id])) {
|
|
|
|
$child_stmt->var->inferredType = clone $context->vars_in_scope[$parent_var_id];
|
|
|
|
}
|
|
|
|
|
2019-08-27 23:00:00 +02:00
|
|
|
$array_type = clone $child_stmt->var->inferredType;
|
2018-08-21 06:28:39 +02:00
|
|
|
|
2018-11-06 03:57:36 +01:00
|
|
|
$child_stmt->inferredType = ArrayFetchAnalyzer::getArrayAccessTypeGivenOffset(
|
2018-11-11 18:01:14 +01:00
|
|
|
$statements_analyzer,
|
2018-01-14 18:09:40 +01:00
|
|
|
$child_stmt,
|
2019-08-27 23:00:00 +02:00
|
|
|
$array_type,
|
2018-01-14 18:09:40 +01:00
|
|
|
isset($child_stmt->dim->inferredType) ? $child_stmt->dim->inferredType : Type::getInt(),
|
|
|
|
true,
|
|
|
|
$array_var_id,
|
2019-01-30 21:40:37 +01:00
|
|
|
$context,
|
2019-04-10 23:25:25 +02:00
|
|
|
$assign_value,
|
2018-01-14 18:09:40 +01:00
|
|
|
$child_stmts ? null : $assignment_type
|
|
|
|
);
|
|
|
|
|
2019-08-27 23:00:00 +02:00
|
|
|
$child_stmt->var->inferredType = $array_type;
|
|
|
|
|
|
|
|
if ($root_var_id) {
|
|
|
|
if (!$parent_var_id) {
|
|
|
|
$rooted_parent_id = $root_var_id;
|
|
|
|
$root_type = $array_type;
|
|
|
|
} else {
|
|
|
|
$rooted_parent_id = $parent_var_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
$context->vars_in_scope[$rooted_parent_id] = $array_type;
|
|
|
|
}
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
if (!$child_stmts) {
|
|
|
|
$child_stmt->inferredType = $assignment_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
$current_type = $child_stmt->inferredType;
|
|
|
|
$current_dim = $child_stmt->dim;
|
|
|
|
|
2019-08-27 23:00:00 +02:00
|
|
|
$parent_var_id = $array_var_id;
|
|
|
|
|
2018-12-08 19:18:55 +01:00
|
|
|
if ($child_stmt->var->inferredType->hasMixed()) {
|
2018-02-17 18:32:19 +01:00
|
|
|
$full_var_id = false;
|
2018-01-14 18:09:40 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($root_var_id
|
2018-02-17 18:32:19 +01:00
|
|
|
&& $full_var_id
|
2018-01-14 18:09:40 +01:00
|
|
|
&& isset($child_stmt->var->inferredType)
|
|
|
|
&& !$child_stmt->var->inferredType->hasObjectType()
|
|
|
|
) {
|
|
|
|
$array_var_id = $root_var_id . implode('', $var_id_additions);
|
|
|
|
$context->vars_in_scope[$array_var_id] = clone $assignment_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
// only update as many child stmts are we were able to process above
|
|
|
|
foreach ($reversed_child_stmts as $child_stmt) {
|
|
|
|
if (!isset($child_stmt->inferredType)) {
|
|
|
|
throw new \InvalidArgumentException('Should never get here');
|
|
|
|
}
|
|
|
|
|
2018-12-19 22:10:09 +01:00
|
|
|
$key_value = null;
|
2018-05-31 22:49:01 +02:00
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
if ($current_dim instanceof PhpParser\Node\Scalar\String_
|
|
|
|
|| $current_dim instanceof PhpParser\Node\Scalar\LNumber
|
|
|
|
) {
|
2018-12-19 22:10:09 +01:00
|
|
|
$key_value = $current_dim->value;
|
|
|
|
} elseif ($current_dim instanceof PhpParser\Node\Expr\ConstFetch
|
|
|
|
&& isset($current_dim->inferredType)
|
|
|
|
) {
|
|
|
|
$is_single_string_literal = $current_dim->inferredType->isSingleStringLiteral();
|
|
|
|
|
|
|
|
if ($is_single_string_literal || $current_dim->inferredType->isSingleIntLiteral()) {
|
|
|
|
if ($is_single_string_literal) {
|
|
|
|
$key_value = $current_dim->inferredType->getSingleStringLiteral()->value;
|
|
|
|
} else {
|
|
|
|
$key_value = $current_dim->inferredType->getSingleIntLiteral()->value;
|
|
|
|
}
|
2018-05-31 22:49:01 +02:00
|
|
|
}
|
2018-12-19 22:10:09 +01:00
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2018-12-19 22:10:09 +01:00
|
|
|
if ($key_value !== null) {
|
2018-01-14 18:09:40 +01:00
|
|
|
$has_matching_objectlike_property = false;
|
2019-11-06 01:03:59 +01:00
|
|
|
$has_matching_string = false;
|
2018-01-14 18:09:40 +01:00
|
|
|
|
|
|
|
foreach ($child_stmt->inferredType->getTypes() as $type) {
|
|
|
|
if ($type instanceof ObjectLike) {
|
|
|
|
if (isset($type->properties[$key_value])) {
|
|
|
|
$has_matching_objectlike_property = true;
|
|
|
|
|
|
|
|
$type->properties[$key_value] = clone $current_type;
|
|
|
|
}
|
|
|
|
}
|
2019-11-06 01:03:59 +01:00
|
|
|
|
|
|
|
if ($type instanceof Type\Atomic\TString) {
|
|
|
|
$has_matching_string = true;
|
|
|
|
|
|
|
|
if ($type instanceof Type\Atomic\TLiteralString
|
|
|
|
&& $current_type->isSingleStringLiteral()
|
|
|
|
) {
|
|
|
|
$new_char = $current_type->getSingleStringLiteral()->value;
|
|
|
|
|
|
|
|
if (strlen($new_char) === 1) {
|
|
|
|
$type->value[0] = $new_char;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
|
2019-11-06 01:03:59 +01:00
|
|
|
if (!$has_matching_objectlike_property && !$has_matching_string) {
|
2018-01-14 18:09:40 +01:00
|
|
|
$array_assignment_type = new Type\Union([
|
|
|
|
new ObjectLike([$key_value => $current_type]),
|
|
|
|
]);
|
|
|
|
|
|
|
|
$new_child_type = Type::combineUnionTypes(
|
|
|
|
$child_stmt->inferredType,
|
2018-11-28 16:41:49 +01:00
|
|
|
$array_assignment_type,
|
2019-01-05 06:15:53 +01:00
|
|
|
$codebase,
|
2018-12-08 19:18:55 +01:00
|
|
|
true,
|
|
|
|
false
|
2018-01-14 18:09:40 +01:00
|
|
|
);
|
|
|
|
} else {
|
|
|
|
$new_child_type = $child_stmt->inferredType; // noop
|
|
|
|
}
|
|
|
|
} else {
|
2019-10-09 00:44:46 +02:00
|
|
|
if (!$current_dim) {
|
|
|
|
$array_assignment_type = new Type\Union([
|
|
|
|
new TList($current_type),
|
|
|
|
]);
|
|
|
|
} else {
|
|
|
|
$array_assignment_type = new Type\Union([
|
|
|
|
new TArray([
|
|
|
|
$current_dim->inferredType ?? Type::getMixed(),
|
|
|
|
$current_type,
|
|
|
|
]),
|
|
|
|
]);
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
|
|
|
|
$new_child_type = Type::combineUnionTypes(
|
|
|
|
$child_stmt->inferredType,
|
2018-11-28 16:41:49 +01:00
|
|
|
$array_assignment_type,
|
2019-01-05 06:15:53 +01:00
|
|
|
$codebase,
|
2018-12-08 19:18:55 +01:00
|
|
|
true,
|
|
|
|
false
|
2018-01-14 18:09:40 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$new_child_type->removeType('null');
|
2018-04-10 00:02:45 +02:00
|
|
|
$new_child_type->possibly_undefined = false;
|
2018-01-14 18:09:40 +01:00
|
|
|
|
|
|
|
if (!$child_stmt->inferredType->hasObjectType()) {
|
|
|
|
$child_stmt->inferredType = $new_child_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
$current_type = $child_stmt->inferredType;
|
|
|
|
$current_dim = $child_stmt->dim;
|
|
|
|
|
|
|
|
array_pop($var_id_additions);
|
|
|
|
|
|
|
|
if ($root_var_id) {
|
|
|
|
$array_var_id = $root_var_id . implode('', $var_id_additions);
|
|
|
|
$context->vars_in_scope[$array_var_id] = clone $child_stmt->inferredType;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-18 17:02:50 +02:00
|
|
|
$root_is_string = $root_type->isString();
|
2018-12-19 22:10:09 +01:00
|
|
|
$key_value = null;
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2018-12-19 22:10:09 +01:00
|
|
|
if ($current_dim instanceof PhpParser\Node\Scalar\String_
|
|
|
|
|| ($current_dim instanceof PhpParser\Node\Scalar\LNumber && !$root_is_string)
|
2018-01-14 18:09:40 +01:00
|
|
|
) {
|
2018-12-19 22:10:09 +01:00
|
|
|
$key_value = $current_dim->value;
|
|
|
|
} elseif ($current_dim instanceof PhpParser\Node\Expr\ConstFetch
|
|
|
|
&& isset($current_dim->inferredType)
|
|
|
|
&& !$root_is_string
|
|
|
|
) {
|
|
|
|
$is_single_string_literal = $current_dim->inferredType->isSingleStringLiteral();
|
|
|
|
|
|
|
|
if ($is_single_string_literal || $current_dim->inferredType->isSingleIntLiteral()) {
|
|
|
|
if ($is_single_string_literal) {
|
|
|
|
$key_value = $current_dim->inferredType->getSingleStringLiteral()->value;
|
|
|
|
} else {
|
|
|
|
$key_value = $current_dim->inferredType->getSingleIntLiteral()->value;
|
|
|
|
}
|
2018-05-31 22:49:01 +02:00
|
|
|
}
|
2018-12-19 22:10:09 +01:00
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2018-12-19 22:10:09 +01:00
|
|
|
if ($key_value !== null) {
|
2018-01-14 18:09:40 +01:00
|
|
|
$has_matching_objectlike_property = false;
|
|
|
|
|
|
|
|
foreach ($root_type->getTypes() as $type) {
|
|
|
|
if ($type instanceof ObjectLike) {
|
|
|
|
if (isset($type->properties[$key_value])) {
|
|
|
|
$has_matching_objectlike_property = true;
|
|
|
|
|
|
|
|
$type->properties[$key_value] = clone $current_type;
|
|
|
|
}
|
2019-10-09 16:04:34 +02:00
|
|
|
} elseif ($type instanceof TNonEmptyList && $key_value === 0) {
|
|
|
|
$has_matching_objectlike_property = true;
|
|
|
|
|
|
|
|
$type->type_param = Type::combineUnionTypes(
|
|
|
|
clone $current_type,
|
|
|
|
$type->type_param,
|
|
|
|
$codebase,
|
|
|
|
true,
|
|
|
|
false
|
|
|
|
);
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$has_matching_objectlike_property) {
|
2018-12-19 22:15:19 +01:00
|
|
|
$object_like = new ObjectLike([$key_value => $current_type]);
|
|
|
|
$object_like->sealed = true;
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
$array_assignment_type = new Type\Union([
|
2018-12-19 22:15:19 +01:00
|
|
|
$object_like,
|
2018-01-14 18:09:40 +01:00
|
|
|
]);
|
|
|
|
|
|
|
|
$new_child_type = Type::combineUnionTypes(
|
|
|
|
$root_type,
|
2018-11-28 16:41:49 +01:00
|
|
|
$array_assignment_type,
|
2019-01-05 06:15:53 +01:00
|
|
|
$codebase,
|
2018-12-08 19:18:55 +01:00
|
|
|
true,
|
|
|
|
false
|
2018-01-14 18:09:40 +01:00
|
|
|
);
|
|
|
|
} else {
|
|
|
|
$new_child_type = $root_type; // noop
|
|
|
|
}
|
|
|
|
} elseif (!$root_is_string) {
|
2018-05-03 19:56:30 +02:00
|
|
|
if ($current_dim) {
|
|
|
|
if (isset($current_dim->inferredType)) {
|
2018-11-06 03:57:36 +01:00
|
|
|
$array_atomic_key_type = ArrayFetchAnalyzer::replaceOffsetTypeWithInts(
|
2018-05-03 19:56:30 +02:00
|
|
|
$current_dim->inferredType
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
$array_atomic_key_type = Type::getMixed();
|
|
|
|
}
|
2019-10-09 00:44:46 +02:00
|
|
|
|
|
|
|
$array_atomic_type = new TNonEmptyArray([
|
|
|
|
$array_atomic_key_type,
|
|
|
|
$current_type,
|
|
|
|
]);
|
2018-05-03 19:56:30 +02:00
|
|
|
} else {
|
2019-10-09 00:44:46 +02:00
|
|
|
$array_atomic_type = new TNonEmptyList($current_type);
|
2018-05-03 19:56:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$from_countable_object_like = false;
|
|
|
|
|
2019-10-09 00:44:46 +02:00
|
|
|
$new_child_type = null;
|
|
|
|
|
2018-05-03 19:56:30 +02:00
|
|
|
if (!$current_dim && !$context->inside_loop) {
|
|
|
|
$atomic_root_types = $root_type->getTypes();
|
|
|
|
|
|
|
|
if (isset($atomic_root_types['array'])) {
|
2019-10-09 00:44:46 +02:00
|
|
|
if ($atomic_root_types['array'] instanceof TNonEmptyArray
|
|
|
|
|| $atomic_root_types['array'] instanceof TNonEmptyList
|
|
|
|
) {
|
2018-05-03 19:56:30 +02:00
|
|
|
$array_atomic_type->count = $atomic_root_types['array']->count;
|
|
|
|
} elseif ($atomic_root_types['array'] instanceof ObjectLike
|
|
|
|
&& $atomic_root_types['array']->sealed
|
|
|
|
) {
|
2018-05-18 17:02:50 +02:00
|
|
|
$array_atomic_type->count = count($atomic_root_types['array']->properties);
|
2018-05-03 19:56:30 +02:00
|
|
|
$from_countable_object_like = true;
|
2019-10-09 00:44:46 +02:00
|
|
|
|
|
|
|
if ($atomic_root_types['array']->is_list
|
|
|
|
&& $array_atomic_type instanceof TList
|
|
|
|
) {
|
|
|
|
$array_atomic_type = clone $atomic_root_types['array'];
|
|
|
|
|
|
|
|
$new_child_type = new Type\Union([$array_atomic_type]);
|
|
|
|
}
|
|
|
|
} elseif ($array_atomic_type instanceof TList) {
|
|
|
|
$array_atomic_type = new TNonEmptyList(
|
|
|
|
$array_atomic_type->type_param
|
|
|
|
);
|
2018-12-19 22:15:19 +01:00
|
|
|
} else {
|
|
|
|
$array_atomic_type = new TNonEmptyArray(
|
|
|
|
$array_atomic_type->type_params
|
|
|
|
);
|
2018-05-03 19:56:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
$array_assignment_type = new Type\Union([
|
2018-05-03 19:56:30 +02:00
|
|
|
$array_atomic_type,
|
2018-01-14 18:09:40 +01:00
|
|
|
]);
|
|
|
|
|
2019-10-09 00:44:46 +02:00
|
|
|
if (!$new_child_type) {
|
|
|
|
$new_child_type = Type::combineUnionTypes(
|
|
|
|
$root_type,
|
|
|
|
$array_assignment_type,
|
|
|
|
$codebase,
|
|
|
|
true,
|
|
|
|
false
|
|
|
|
);
|
|
|
|
}
|
2018-05-03 19:56:30 +02:00
|
|
|
|
|
|
|
if ($from_countable_object_like) {
|
|
|
|
$atomic_root_types = $new_child_type->getTypes();
|
|
|
|
|
|
|
|
if (isset($atomic_root_types['array'])
|
2019-10-11 02:16:43 +02:00
|
|
|
&& ($atomic_root_types['array'] instanceof TNonEmptyArray
|
|
|
|
|| $atomic_root_types['array'] instanceof TNonEmptyList)
|
2018-05-18 17:02:50 +02:00
|
|
|
&& $atomic_root_types['array']->count !== null
|
2018-05-03 19:56:30 +02:00
|
|
|
) {
|
2018-05-18 17:02:50 +02:00
|
|
|
$atomic_root_types['array']->count++;
|
2018-05-03 19:56:30 +02:00
|
|
|
}
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
} else {
|
|
|
|
$new_child_type = $root_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
$new_child_type->removeType('null');
|
|
|
|
|
|
|
|
if (!$root_type->hasObjectType()) {
|
|
|
|
$root_type = $new_child_type;
|
|
|
|
}
|
|
|
|
|
2019-10-12 18:23:40 +02:00
|
|
|
if ($codebase->taint && $taint_sources) {
|
|
|
|
$root_type->sources = \array_merge($taint_sources, $root_type->sources ?: []);
|
|
|
|
$root_type->tainted = $taint_type | $root_type->tainted;
|
|
|
|
}
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
$root_array_expr->inferredType = $root_type;
|
|
|
|
|
|
|
|
if ($root_array_expr instanceof PhpParser\Node\Expr\PropertyFetch) {
|
2018-04-17 18:16:25 +02:00
|
|
|
if ($root_array_expr->name instanceof PhpParser\Node\Identifier) {
|
2018-11-06 03:57:36 +01:00
|
|
|
PropertyAssignmentAnalyzer::analyzeInstance(
|
2018-11-11 18:01:14 +01:00
|
|
|
$statements_analyzer,
|
2018-01-14 18:09:40 +01:00
|
|
|
$root_array_expr,
|
2018-04-17 18:16:25 +02:00
|
|
|
$root_array_expr->name->name,
|
2018-01-14 18:09:40 +01:00
|
|
|
null,
|
|
|
|
$root_type,
|
|
|
|
$context,
|
|
|
|
false
|
|
|
|
);
|
|
|
|
} else {
|
2018-11-11 18:01:14 +01:00
|
|
|
if (ExpressionAnalyzer::analyze($statements_analyzer, $root_array_expr->name, $context) === false) {
|
2018-01-14 18:09:40 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-11-11 18:01:14 +01:00
|
|
|
if (ExpressionAnalyzer::analyze($statements_analyzer, $root_array_expr->var, $context) === false) {
|
2018-01-14 18:09:40 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} elseif ($root_var_id) {
|
2019-07-04 22:38:31 +02:00
|
|
|
$context->vars_in_scope[$root_var_id] = $root_type;
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|