2018-01-14 18:09:40 +01:00
|
|
|
<?php
|
2021-12-15 04:58:32 +01:00
|
|
|
|
2018-11-06 03:57:36 +01:00
|
|
|
namespace Psalm\Internal\Analyzer\Statements\Expression\Assignment;
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2021-12-03 21:40:18 +01:00
|
|
|
use InvalidArgumentException;
|
2018-01-14 18:09:40 +01:00
|
|
|
use PhpParser;
|
2022-01-07 23:16:36 +01:00
|
|
|
use PhpParser\Node\Expr\Variable;
|
2021-12-03 20:11:20 +01:00
|
|
|
use Psalm\CodeLocation;
|
|
|
|
use Psalm\Codebase;
|
2021-06-08 04:55:21 +02:00
|
|
|
use Psalm\Context;
|
2019-12-13 23:17:14 +01:00
|
|
|
use Psalm\Internal\Analyzer\ClassLikeAnalyzer;
|
2020-05-18 21:13:27 +02:00
|
|
|
use Psalm\Internal\Analyzer\Statements\Expression\ExpressionIdentifier;
|
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;
|
2020-09-30 18:28:13 +02:00
|
|
|
use Psalm\Internal\Codebase\VariableUseGraph;
|
2021-12-03 20:11:20 +01:00
|
|
|
use Psalm\Internal\DataFlow\DataFlowNode;
|
2020-11-29 22:16:16 +01:00
|
|
|
use Psalm\Internal\Type\TemplateInferredTypeReplacer;
|
2021-12-03 20:11:20 +01:00
|
|
|
use Psalm\Internal\Type\TemplateResult;
|
2019-11-08 17:56:33 +01:00
|
|
|
use Psalm\Issue\InvalidArrayAssignment;
|
2021-06-08 04:55:21 +02:00
|
|
|
use Psalm\IssueBuffer;
|
2018-01-14 18:09:40 +01:00
|
|
|
use Psalm\Type;
|
|
|
|
use Psalm\Type\Atomic\TArray;
|
2021-12-13 04:45:57 +01:00
|
|
|
use Psalm\Type\Atomic\TClassStringMap;
|
|
|
|
use Psalm\Type\Atomic\TDependentListKey;
|
2022-12-13 21:40:19 +01:00
|
|
|
use Psalm\Type\Atomic\TIntRange;
|
2021-06-08 04:55:21 +02:00
|
|
|
use Psalm\Type\Atomic\TKeyedArray;
|
2019-10-09 00:44:46 +02:00
|
|
|
use Psalm\Type\Atomic\TList;
|
2021-12-13 04:45:57 +01:00
|
|
|
use Psalm\Type\Atomic\TLiteralClassString;
|
|
|
|
use Psalm\Type\Atomic\TLiteralInt;
|
|
|
|
use Psalm\Type\Atomic\TLiteralString;
|
2018-11-09 16:56:27 +01:00
|
|
|
use Psalm\Type\Atomic\TNonEmptyArray;
|
2021-12-13 04:45:57 +01:00
|
|
|
use Psalm\Type\Atomic\TString;
|
|
|
|
use Psalm\Type\Atomic\TTemplateIndexedAccess;
|
|
|
|
use Psalm\Type\Atomic\TTemplateKeyOf;
|
|
|
|
use Psalm\Type\Atomic\TTemplateParam;
|
|
|
|
use Psalm\Type\Atomic\TTemplateParamClass;
|
2021-12-13 16:28:14 +01:00
|
|
|
use Psalm\Type\Union;
|
2021-06-08 04:55:21 +02:00
|
|
|
|
2022-12-13 21:40:19 +01:00
|
|
|
use function array_fill;
|
2021-06-08 04:55:21 +02:00
|
|
|
use function array_pop;
|
2019-06-26 22:52:29 +02:00
|
|
|
use function array_reverse;
|
|
|
|
use function array_shift;
|
2021-12-03 21:07:25 +01:00
|
|
|
use function array_slice;
|
2022-11-04 19:04:23 +01:00
|
|
|
use function assert;
|
2021-06-08 04:55:21 +02:00
|
|
|
use function count;
|
2022-03-03 00:49:57 +01:00
|
|
|
use function end;
|
2019-06-26 22:52:29 +02:00
|
|
|
use function implode;
|
2021-12-03 21:07:25 +01:00
|
|
|
use function in_array;
|
2021-06-08 04:55:21 +02:00
|
|
|
use function is_string;
|
|
|
|
use function preg_match;
|
2021-12-03 21:07:25 +01:00
|
|
|
use function strlen;
|
2022-12-13 21:40:19 +01:00
|
|
|
use function strpos;
|
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
|
|
|
{
|
|
|
|
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,
|
2020-07-22 01:40:35 +02:00
|
|
|
?PhpParser\Node\Expr $assign_value,
|
2021-12-13 16:28:14 +01:00
|
|
|
Union $assignment_value_type
|
2020-09-12 17:24:05 +02:00
|
|
|
): void {
|
2020-11-19 01:19:07 +01:00
|
|
|
$nesting = 0;
|
2020-05-18 21:13:27 +02:00
|
|
|
$var_id = ExpressionIdentifier::getVarId(
|
2018-01-14 18:09:40 +01:00
|
|
|
$stmt->var,
|
2018-11-11 18:01:14 +01:00
|
|
|
$statements_analyzer->getFQCLN(),
|
|
|
|
$statements_analyzer,
|
2020-11-19 01:19:07 +01:00
|
|
|
$nesting
|
2018-01-14 18:09:40 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
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
|
|
|
|
);
|
|
|
|
|
2019-11-25 17:44:54 +01:00
|
|
|
if (!$statements_analyzer->node_data->getType($stmt->var) && $var_id) {
|
2018-01-14 18:09:40 +01:00
|
|
|
$context->vars_in_scope[$var_id] = Type::getMixed();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return false|null
|
2021-10-11 17:23:21 +02:00
|
|
|
* @psalm-suppress PossiblyUnusedReturnValue not used but seems important
|
2018-01-14 18:09:40 +01:00
|
|
|
*/
|
|
|
|
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,
|
2021-12-13 16:28:14 +01:00
|
|
|
Union $assignment_type,
|
2018-01-14 18:09:40 +01:00
|
|
|
Context $context
|
2020-09-04 22:26:33 +02:00
|
|
|
): ?bool {
|
2018-01-14 18:09:40 +01:00
|
|
|
$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();
|
|
|
|
|
2021-10-13 18:35:16 +02:00
|
|
|
$root_type = $statements_analyzer->node_data->getType($root_array_expr) ?? Type::getMixed();
|
2018-01-14 18:09:40 +01:00
|
|
|
|
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
|
|
|
}
|
|
|
|
|
|
|
|
$current_type = $root_type;
|
|
|
|
|
|
|
|
$current_dim = $stmt->dim;
|
|
|
|
|
|
|
|
// gets a variable id that *may* contain array keys
|
2022-02-04 18:49:12 +01:00
|
|
|
$root_var_id = ExpressionIdentifier::getExtendedVarId(
|
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
|
|
|
);
|
|
|
|
|
2018-08-21 06:28:39 +02:00
|
|
|
$parent_var_id = null;
|
|
|
|
|
2019-12-19 00:48:25 +01:00
|
|
|
$offset_already_existed = false;
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
self::analyzeNestedArrayAssignment(
|
|
|
|
$statements_analyzer,
|
|
|
|
$codebase,
|
|
|
|
$context,
|
|
|
|
$assign_value,
|
|
|
|
$assignment_type,
|
2022-03-03 00:49:57 +01:00
|
|
|
$child_stmts,
|
2020-11-30 19:08:42 +01:00
|
|
|
$root_var_id,
|
|
|
|
$parent_var_id,
|
|
|
|
$root_type,
|
|
|
|
$current_type,
|
|
|
|
$current_dim,
|
|
|
|
$offset_already_existed
|
|
|
|
);
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
$root_is_string = $root_type->isString();
|
|
|
|
$key_values = [];
|
|
|
|
|
|
|
|
if ($current_dim instanceof PhpParser\Node\Scalar\String_) {
|
2021-12-13 04:45:57 +01:00
|
|
|
$key_values[] = new TLiteralString($current_dim->value);
|
2020-11-30 19:08:42 +01:00
|
|
|
} elseif ($current_dim instanceof PhpParser\Node\Scalar\LNumber && !$root_is_string) {
|
2021-12-13 04:45:57 +01:00
|
|
|
$key_values[] = new TLiteralInt($current_dim->value);
|
2020-11-30 19:08:42 +01:00
|
|
|
} elseif ($current_dim
|
2021-06-09 18:55:37 +02:00
|
|
|
&& ($key_type = $statements_analyzer->node_data->getType($current_dim))
|
2020-11-30 19:08:42 +01:00
|
|
|
&& !$root_is_string
|
|
|
|
) {
|
2021-06-09 18:55:37 +02:00
|
|
|
$string_literals = $key_type->getLiteralStrings();
|
|
|
|
$int_literals = $key_type->getLiteralInts();
|
2020-11-30 19:08:42 +01:00
|
|
|
|
2021-06-09 18:55:37 +02:00
|
|
|
$all_atomic_types = $key_type->getAtomicTypes();
|
2020-11-30 19:08:42 +01:00
|
|
|
|
|
|
|
if (count($string_literals) + count($int_literals) === count($all_atomic_types)) {
|
|
|
|
foreach ($string_literals as $string_literal) {
|
2022-11-04 19:04:23 +01:00
|
|
|
$key_values[] = $string_literal;
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($int_literals as $int_literal) {
|
2022-11-04 19:04:23 +01:00
|
|
|
$key_values[] = $int_literal;
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
if ($key_values) {
|
|
|
|
$new_child_type = self::updateTypeWithKeyValues(
|
|
|
|
$codebase,
|
|
|
|
$root_type,
|
|
|
|
$current_type,
|
|
|
|
$key_values
|
|
|
|
);
|
|
|
|
} elseif (!$root_is_string) {
|
|
|
|
$new_child_type = self::updateArrayAssignmentChildType(
|
|
|
|
$statements_analyzer,
|
|
|
|
$codebase,
|
|
|
|
$current_dim,
|
|
|
|
$context,
|
|
|
|
$current_type,
|
|
|
|
$root_type,
|
|
|
|
$offset_already_existed,
|
|
|
|
$parent_var_id
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
$new_child_type = $root_type;
|
|
|
|
}
|
2019-11-25 17:44:54 +01:00
|
|
|
|
2022-10-03 10:45:36 +02:00
|
|
|
$new_child_type = $new_child_type->getBuilder();
|
2020-11-30 19:08:42 +01:00
|
|
|
$new_child_type->removeType('null');
|
2022-10-03 10:45:36 +02:00
|
|
|
$new_child_type = $new_child_type->freeze();
|
2020-06-19 00:48:19 +02:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
if (!$root_type->hasObjectType()) {
|
|
|
|
$root_type = $new_child_type;
|
|
|
|
}
|
2020-09-30 18:28:13 +02:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
$statements_analyzer->node_data->setType($root_array_expr, $root_type);
|
|
|
|
|
|
|
|
if ($root_array_expr instanceof PhpParser\Node\Expr\PropertyFetch) {
|
|
|
|
if ($root_array_expr->name instanceof PhpParser\Node\Identifier) {
|
|
|
|
InstancePropertyAssignmentAnalyzer::analyze(
|
2018-11-11 18:01:14 +01:00
|
|
|
$statements_analyzer,
|
2020-11-30 19:08:42 +01:00
|
|
|
$root_array_expr,
|
|
|
|
$root_array_expr->name->name,
|
|
|
|
null,
|
|
|
|
$root_type,
|
|
|
|
$context,
|
|
|
|
false
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
if (ExpressionAnalyzer::analyze($statements_analyzer, $root_array_expr->name, $context) === false) {
|
2018-01-14 18:09:40 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
if (ExpressionAnalyzer::analyze($statements_analyzer, $root_array_expr->var, $context) === false) {
|
|
|
|
return false;
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
|
|
|
} elseif ($root_array_expr instanceof PhpParser\Node\Expr\StaticPropertyFetch
|
|
|
|
&& $root_array_expr->name instanceof PhpParser\Node\Identifier
|
|
|
|
) {
|
2021-10-11 17:23:21 +02:00
|
|
|
if (StaticPropertyAssignmentAnalyzer::analyze(
|
2020-11-30 19:08:42 +01:00
|
|
|
$statements_analyzer,
|
|
|
|
$root_array_expr,
|
|
|
|
null,
|
|
|
|
$root_type,
|
|
|
|
$context
|
2021-10-11 17:23:21 +02:00
|
|
|
) === false) {
|
|
|
|
return false;
|
|
|
|
}
|
2020-11-30 19:08:42 +01:00
|
|
|
} elseif ($root_var_id) {
|
|
|
|
$context->vars_in_scope[$root_var_id] = $root_type;
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
if ($root_array_expr instanceof PhpParser\Node\Expr\MethodCall
|
|
|
|
|| $root_array_expr instanceof PhpParser\Node\Expr\StaticCall
|
|
|
|
|| $root_array_expr instanceof PhpParser\Node\Expr\FuncCall
|
|
|
|
) {
|
|
|
|
if ($root_type->hasArray()) {
|
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new InvalidArrayAssignment(
|
|
|
|
'Assigning to the output of a function has no effect',
|
2021-12-03 20:11:20 +01:00
|
|
|
new CodeLocation($statements_analyzer->getSource(), $root_array_expr)
|
2020-11-30 19:08:42 +01:00
|
|
|
),
|
|
|
|
$statements_analyzer->getSuppressedIssues()
|
|
|
|
)
|
2018-05-31 22:49:01 +02:00
|
|
|
) {
|
2020-11-30 19:08:42 +01:00
|
|
|
// do nothing
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-05-31 22:49:01 +02:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-12-13 04:45:57 +01:00
|
|
|
* @param non-empty-list<TLiteralInt|TLiteralString> $key_values
|
2020-11-30 19:08:42 +01:00
|
|
|
*/
|
|
|
|
private static function updateTypeWithKeyValues(
|
2021-12-03 20:11:20 +01:00
|
|
|
Codebase $codebase,
|
2021-12-13 16:28:14 +01:00
|
|
|
Union $child_stmt_type,
|
|
|
|
Union $current_type,
|
2020-11-30 19:08:42 +01:00
|
|
|
array $key_values
|
2021-12-13 16:28:14 +01:00
|
|
|
): Union {
|
2020-11-30 19:08:42 +01:00
|
|
|
$has_matching_objectlike_property = false;
|
|
|
|
$has_matching_string = false;
|
|
|
|
|
2022-10-03 15:13:47 +02:00
|
|
|
$changed = false;
|
|
|
|
$types = [];
|
2020-11-30 19:08:42 +01:00
|
|
|
foreach ($child_stmt_type->getAtomicTypes() as $type) {
|
2022-12-13 21:40:19 +01:00
|
|
|
if ($type instanceof TList) {
|
|
|
|
$type = $type->getKeyedArray();
|
|
|
|
}
|
2022-10-03 15:13:47 +02:00
|
|
|
$old_type = $type;
|
2021-12-13 04:45:57 +01:00
|
|
|
if ($type instanceof TTemplateParam) {
|
2022-10-03 15:13:47 +02:00
|
|
|
$type = $type->replaceAs(self::updateTypeWithKeyValues(
|
2021-05-05 06:46:43 +02:00
|
|
|
$codebase,
|
|
|
|
$type->as,
|
|
|
|
$current_type,
|
|
|
|
$key_values
|
2022-10-03 15:13:47 +02:00
|
|
|
));
|
2021-05-05 06:46:43 +02:00
|
|
|
$has_matching_objectlike_property = true;
|
2022-10-03 15:13:47 +02:00
|
|
|
} elseif ($type instanceof TKeyedArray) {
|
|
|
|
$properties = $type->properties;
|
|
|
|
foreach ($key_values as $key_value) {
|
|
|
|
if (isset($properties[$key_value->value])) {
|
2020-11-30 19:08:42 +01:00
|
|
|
$has_matching_objectlike_property = true;
|
|
|
|
|
2022-11-04 19:04:23 +01:00
|
|
|
$properties[$key_value->value] = $current_type;
|
2018-05-31 22:49:01 +02:00
|
|
|
}
|
2022-10-03 15:13:47 +02:00
|
|
|
}
|
|
|
|
$type = $type->setProperties($properties);
|
|
|
|
} elseif ($type instanceof TString) {
|
|
|
|
foreach ($key_values as $key_value) {
|
|
|
|
if ($key_value instanceof TLiteralInt) {
|
|
|
|
$has_matching_string = true;
|
|
|
|
|
|
|
|
if ($type instanceof TLiteralString
|
|
|
|
&& $current_type->isSingleStringLiteral()
|
|
|
|
) {
|
|
|
|
$new_char = $current_type->getSingleStringLiteral()->value;
|
|
|
|
|
|
|
|
if (strlen($new_char) === 1 && $type->value[0] !== $new_char) {
|
|
|
|
$v = $type->value;
|
|
|
|
$v[0] = $new_char;
|
|
|
|
$changed = true;
|
|
|
|
$type = new TLiteralString($v);
|
|
|
|
break;
|
|
|
|
}
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
2019-10-03 21:27:50 +02:00
|
|
|
}
|
2022-10-03 15:13:47 +02:00
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
2022-10-03 15:13:47 +02:00
|
|
|
$types[$type->getKey()] = $type;
|
|
|
|
$changed = $changed || $old_type !== $type;
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2022-10-03 15:13:47 +02:00
|
|
|
if ($changed) {
|
|
|
|
$child_stmt_type = $child_stmt_type->getBuilder()->setTypes($types)->freeze();
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
if (!$has_matching_objectlike_property && !$has_matching_string) {
|
|
|
|
if (count($key_values) === 1) {
|
|
|
|
$key_value = $key_values[0];
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
$object_like = new TKeyedArray(
|
2022-11-04 19:04:23 +01:00
|
|
|
[$key_value->value => $current_type],
|
2021-12-13 04:45:57 +01:00
|
|
|
$key_value instanceof TLiteralClassString
|
2020-11-30 19:08:42 +01:00
|
|
|
? [$key_value->value => true]
|
2022-11-10 15:18:27 +01:00
|
|
|
: null
|
2020-11-30 19:08:42 +01:00
|
|
|
);
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2021-12-13 16:28:14 +01:00
|
|
|
$array_assignment_type = new Union([
|
2020-11-30 19:08:42 +01:00
|
|
|
$object_like,
|
|
|
|
]);
|
|
|
|
} else {
|
|
|
|
$array_assignment_literals = $key_values;
|
2018-08-21 06:28:39 +02:00
|
|
|
|
2021-12-13 16:28:14 +01:00
|
|
|
$array_assignment_type = new Union([
|
2021-12-13 04:45:57 +01:00
|
|
|
new TNonEmptyArray([
|
2021-12-13 16:28:14 +01:00
|
|
|
new Union($array_assignment_literals),
|
2022-11-04 19:04:23 +01:00
|
|
|
$current_type
|
2020-11-30 19:08:42 +01:00
|
|
|
])
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Type::combineUnionTypes(
|
|
|
|
$child_stmt_type,
|
|
|
|
$array_assignment_type,
|
|
|
|
$codebase,
|
2018-01-14 18:09:40 +01:00
|
|
|
true,
|
2020-11-30 19:08:42 +01:00
|
|
|
false
|
2018-01-14 18:09:40 +01:00
|
|
|
);
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
return $child_stmt_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-12-13 04:45:57 +01:00
|
|
|
* @param list<TLiteralInt|TLiteralString> $key_values $key_values
|
2020-11-30 19:08:42 +01:00
|
|
|
*/
|
|
|
|
private static function taintArrayAssignment(
|
|
|
|
StatementsAnalyzer $statements_analyzer,
|
|
|
|
PhpParser\Node\Expr\ArrayDimFetch $expr,
|
2022-11-04 19:04:23 +01:00
|
|
|
Union &$stmt_type,
|
2021-12-13 16:28:14 +01:00
|
|
|
Union $child_stmt_type,
|
2020-11-30 19:08:42 +01:00
|
|
|
?string $var_var_id,
|
|
|
|
array $key_values
|
2021-12-05 18:51:26 +01:00
|
|
|
): void {
|
2020-11-30 19:08:42 +01:00
|
|
|
if ($statements_analyzer->data_flow_graph
|
|
|
|
&& ($statements_analyzer->data_flow_graph instanceof VariableUseGraph
|
2021-12-03 21:07:25 +01:00
|
|
|
|| !in_array('TaintedInput', $statements_analyzer->getSuppressedIssues()))
|
2020-11-30 19:08:42 +01:00
|
|
|
) {
|
2021-12-03 20:11:20 +01:00
|
|
|
$var_location = new CodeLocation($statements_analyzer->getSource(), $expr->var);
|
2020-11-30 19:08:42 +01:00
|
|
|
|
2021-12-03 20:11:20 +01:00
|
|
|
$parent_node = DataFlowNode::getForAssignment(
|
2020-11-30 19:08:42 +01:00
|
|
|
$var_var_id ?: 'assignment',
|
|
|
|
$var_location
|
2019-11-25 17:44:54 +01:00
|
|
|
);
|
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
$statements_analyzer->data_flow_graph->addNode($parent_node);
|
2019-08-27 23:00:00 +02:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
$old_parent_nodes = $stmt_type->parent_nodes;
|
|
|
|
|
2022-11-04 19:04:23 +01:00
|
|
|
$stmt_type = $stmt_type->setParentNodes([$parent_node->id => $parent_node]);
|
2020-11-30 19:08:42 +01:00
|
|
|
|
|
|
|
foreach ($old_parent_nodes as $old_parent_node) {
|
|
|
|
$statements_analyzer->data_flow_graph->addPath(
|
|
|
|
$old_parent_node,
|
|
|
|
$parent_node,
|
|
|
|
'='
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($stmt_type->by_ref) {
|
|
|
|
$statements_analyzer->data_flow_graph->addPath(
|
|
|
|
$parent_node,
|
|
|
|
$old_parent_node,
|
|
|
|
'='
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($stmt_type->parent_nodes as $parent_node) {
|
|
|
|
foreach ($child_stmt_type->parent_nodes as $child_parent_node) {
|
|
|
|
if ($key_values) {
|
|
|
|
foreach ($key_values as $key_value) {
|
|
|
|
$statements_analyzer->data_flow_graph->addPath(
|
|
|
|
$child_parent_node,
|
|
|
|
$parent_node,
|
2021-03-24 20:23:56 +01:00
|
|
|
'arrayvalue-assignment-\'' . $key_value->value . '\''
|
2020-11-30 19:08:42 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$statements_analyzer->data_flow_graph->addPath(
|
|
|
|
$child_parent_node,
|
|
|
|
$parent_node,
|
2021-03-24 20:23:56 +01:00
|
|
|
'arrayvalue-assignment'
|
2020-11-30 19:08:42 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static function updateArrayAssignmentChildType(
|
|
|
|
StatementsAnalyzer $statements_analyzer,
|
2021-12-03 20:11:20 +01:00
|
|
|
Codebase $codebase,
|
2020-11-30 19:08:42 +01:00
|
|
|
?PhpParser\Node\Expr $current_dim,
|
|
|
|
Context $context,
|
2021-12-13 16:28:14 +01:00
|
|
|
Union $value_type,
|
|
|
|
Union $root_type,
|
2020-11-30 19:08:42 +01:00
|
|
|
bool $offset_already_existed,
|
|
|
|
?string $parent_var_id
|
2021-12-13 16:28:14 +01:00
|
|
|
): Union {
|
2021-06-09 18:55:37 +02:00
|
|
|
$templated_assignment = false;
|
|
|
|
|
2022-12-13 21:40:19 +01:00
|
|
|
$array_atomic_type_class_string = null;
|
|
|
|
$array_atomic_type_array = null;
|
|
|
|
$array_atomic_type_list = null;
|
2020-11-30 19:08:42 +01:00
|
|
|
if ($current_dim) {
|
2021-06-09 18:55:37 +02:00
|
|
|
$key_type = $statements_analyzer->node_data->getType($current_dim);
|
|
|
|
|
|
|
|
if ($key_type) {
|
|
|
|
if ($key_type->hasMixed()) {
|
|
|
|
$key_type = Type::getArrayKey();
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
|
|
|
|
2021-06-09 18:55:37 +02:00
|
|
|
if ($key_type->isSingle()) {
|
2021-12-11 20:44:34 +01:00
|
|
|
$key_type_type = $key_type->getSingleAtomic();
|
2020-12-07 01:14:52 +01:00
|
|
|
|
2022-12-13 21:40:19 +01:00
|
|
|
if (($key_type_type instanceof TIntRange
|
|
|
|
&& $key_type_type->dependent_list_key === $parent_var_id
|
|
|
|
) || ($key_type_type instanceof TDependentListKey
|
|
|
|
&& $key_type_type->var_id === $parent_var_id
|
|
|
|
)) {
|
2020-12-07 01:14:52 +01:00
|
|
|
$offset_already_existed = true;
|
|
|
|
}
|
2021-06-09 18:55:37 +02:00
|
|
|
|
2021-12-13 04:45:57 +01:00
|
|
|
if ($key_type_type instanceof TTemplateParam
|
2021-06-09 18:55:37 +02:00
|
|
|
&& $key_type_type->as->isSingle()
|
|
|
|
&& $root_type->isSingle()
|
|
|
|
&& $value_type->isSingle()
|
|
|
|
) {
|
2021-12-11 20:44:34 +01:00
|
|
|
$key_type_as_type = $key_type_type->as->getSingleAtomic();
|
|
|
|
$value_atomic_type = $value_type->getSingleAtomic();
|
|
|
|
$root_atomic_type = $root_type->getSingleAtomic();
|
2021-06-09 18:55:37 +02:00
|
|
|
|
2021-12-13 04:45:57 +01:00
|
|
|
if ($key_type_as_type instanceof TTemplateKeyOf
|
|
|
|
&& $root_atomic_type instanceof TTemplateParam
|
|
|
|
&& $value_atomic_type instanceof TTemplateIndexedAccess
|
2021-06-09 18:55:37 +02:00
|
|
|
&& $key_type_as_type->param_name === $root_atomic_type->param_name
|
|
|
|
&& $key_type_as_type->defining_class === $root_atomic_type->defining_class
|
|
|
|
&& $value_atomic_type->array_param_name === $root_atomic_type->param_name
|
|
|
|
&& $value_atomic_type->offset_param_name === $key_type_type->param_name
|
|
|
|
&& $value_atomic_type->defining_class === $root_atomic_type->defining_class
|
|
|
|
) {
|
|
|
|
$templated_assignment = true;
|
|
|
|
$offset_already_existed = true;
|
|
|
|
}
|
|
|
|
}
|
2020-12-07 01:14:52 +01:00
|
|
|
}
|
|
|
|
|
2022-10-03 10:45:36 +02:00
|
|
|
$array_atomic_key_type = ArrayFetchAnalyzer::replaceOffsetTypeWithInts($key_type);
|
2020-11-30 19:08:42 +01:00
|
|
|
} else {
|
|
|
|
$array_atomic_key_type = Type::getArrayKey();
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($offset_already_existed
|
|
|
|
&& $parent_var_id
|
|
|
|
&& ($parent_type = $context->vars_in_scope[$parent_var_id] ?? null)
|
|
|
|
) {
|
2022-12-13 21:40:19 +01:00
|
|
|
if ($parent_type->hasList() && strpos($parent_var_id, '[') === false) {
|
|
|
|
$array_atomic_type_list = $value_type;
|
2020-11-30 19:08:42 +01:00
|
|
|
} elseif ($parent_type->hasClassStringMap()
|
2021-06-09 18:55:37 +02:00
|
|
|
&& $key_type
|
|
|
|
&& $key_type->isTemplatedClassString()
|
2020-11-30 19:08:42 +01:00
|
|
|
) {
|
|
|
|
/**
|
2021-12-13 04:45:57 +01:00
|
|
|
* @var TClassStringMap
|
2020-11-30 19:08:42 +01:00
|
|
|
*/
|
2022-12-13 21:40:19 +01:00
|
|
|
$class_string_map = $parent_type->getArray();
|
2020-11-30 19:08:42 +01:00
|
|
|
/**
|
2021-12-13 04:45:57 +01:00
|
|
|
* @var TTemplateParamClass
|
2020-11-30 19:08:42 +01:00
|
|
|
*/
|
2021-12-11 20:44:34 +01:00
|
|
|
$offset_type_part = $key_type->getSingleAtomic();
|
2020-11-30 19:08:42 +01:00
|
|
|
|
2021-12-03 20:11:20 +01:00
|
|
|
$template_result = new TemplateResult(
|
2020-11-30 19:08:42 +01:00
|
|
|
[],
|
|
|
|
[
|
|
|
|
$offset_type_part->param_name => [
|
2021-12-13 16:28:14 +01:00
|
|
|
$offset_type_part->defining_class => new Union([
|
2021-12-13 04:45:57 +01:00
|
|
|
new TTemplateParam(
|
2020-11-30 19:08:42 +01:00
|
|
|
$class_string_map->param_name,
|
|
|
|
$offset_type_part->as_type
|
2021-12-13 16:28:14 +01:00
|
|
|
? new Union([$offset_type_part->as_type])
|
2020-11-30 19:08:42 +01:00
|
|
|
: Type::getObject(),
|
|
|
|
'class-string-map'
|
|
|
|
)
|
|
|
|
])
|
|
|
|
]
|
|
|
|
]
|
|
|
|
);
|
|
|
|
|
2022-10-03 10:45:36 +02:00
|
|
|
$value_type = TemplateInferredTypeReplacer::replace(
|
2021-06-09 18:55:37 +02:00
|
|
|
$value_type,
|
2020-11-30 19:08:42 +01:00
|
|
|
$template_result,
|
|
|
|
$codebase
|
|
|
|
);
|
|
|
|
|
2022-12-13 21:40:19 +01:00
|
|
|
$array_atomic_type_class_string = new TClassStringMap(
|
2020-11-30 19:08:42 +01:00
|
|
|
$class_string_map->param_name,
|
|
|
|
$class_string_map->as_type,
|
2021-06-09 18:55:37 +02:00
|
|
|
$value_type
|
2020-11-30 19:08:42 +01:00
|
|
|
);
|
|
|
|
} else {
|
2022-12-13 21:40:19 +01:00
|
|
|
$array_atomic_type_array = [
|
2020-11-30 19:08:42 +01:00
|
|
|
$array_atomic_key_type,
|
2021-06-09 18:55:37 +02:00
|
|
|
$value_type,
|
2022-12-13 21:40:19 +01:00
|
|
|
];
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
|
|
|
} else {
|
2022-12-13 21:40:19 +01:00
|
|
|
$array_atomic_type_array = [
|
2020-11-30 19:08:42 +01:00
|
|
|
$array_atomic_key_type,
|
2021-06-09 18:55:37 +02:00
|
|
|
$value_type,
|
2022-12-13 21:40:19 +01:00
|
|
|
];
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
|
|
|
} else {
|
2022-12-13 21:40:19 +01:00
|
|
|
$array_atomic_type_list = $value_type;
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
$from_countable_object_like = false;
|
|
|
|
|
|
|
|
$new_child_type = null;
|
|
|
|
|
2022-12-13 21:40:19 +01:00
|
|
|
$array_atomic_type = null;
|
2020-11-30 19:08:42 +01:00
|
|
|
if (!$current_dim && !$context->inside_loop) {
|
|
|
|
$atomic_root_types = $root_type->getAtomicTypes();
|
|
|
|
|
|
|
|
if (isset($atomic_root_types['array'])) {
|
2022-12-13 21:40:19 +01:00
|
|
|
$atomic_root_type_array = $atomic_root_types['array'];
|
|
|
|
if ($atomic_root_type_array instanceof TList) {
|
|
|
|
$atomic_root_type_array = $atomic_root_type_array->getKeyedArray();
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($array_atomic_type_class_string) {
|
2020-11-30 19:08:42 +01:00
|
|
|
$array_atomic_type = new TNonEmptyArray([
|
2022-12-13 21:40:19 +01:00
|
|
|
$array_atomic_type_class_string->getStandinKeyParam(),
|
|
|
|
$array_atomic_type_class_string->value_param
|
2020-11-30 19:08:42 +01:00
|
|
|
]);
|
2022-12-13 21:40:19 +01:00
|
|
|
} elseif ($atomic_root_type_array instanceof TKeyedArray
|
|
|
|
&& $atomic_root_type_array->is_list
|
|
|
|
&& $atomic_root_type_array->fallback_params === null
|
2020-11-30 19:08:42 +01:00
|
|
|
) {
|
2022-12-13 21:40:19 +01:00
|
|
|
$array_atomic_type = $atomic_root_type_array;
|
|
|
|
} elseif ($atomic_root_type_array instanceof TNonEmptyArray
|
|
|
|
|| ($atomic_root_type_array instanceof TKeyedArray
|
|
|
|
&& $atomic_root_type_array->is_list
|
|
|
|
&& $atomic_root_type_array->isNonEmpty()
|
|
|
|
)
|
2020-11-30 19:08:42 +01:00
|
|
|
) {
|
2022-12-13 21:40:19 +01:00
|
|
|
$prop_count = null;
|
|
|
|
if ($atomic_root_type_array instanceof TNonEmptyArray) {
|
|
|
|
$prop_count = $atomic_root_type_array->count;
|
|
|
|
} else {
|
|
|
|
$min_count = $atomic_root_type_array->getMinCount();
|
|
|
|
if ($min_count === $atomic_root_type_array->getMaxCount()) {
|
|
|
|
$prop_count = $min_count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ($array_atomic_type_array) {
|
|
|
|
$array_atomic_type = new TNonEmptyArray(
|
|
|
|
$array_atomic_type_array,
|
|
|
|
$prop_count
|
|
|
|
);
|
|
|
|
} elseif ($prop_count !== null) {
|
|
|
|
assert($array_atomic_type_list !== null);
|
|
|
|
$array_atomic_type = new TKeyedArray(
|
|
|
|
array_fill(
|
|
|
|
0,
|
|
|
|
$prop_count,
|
|
|
|
$array_atomic_type_list
|
|
|
|
),
|
|
|
|
null,
|
|
|
|
[
|
|
|
|
Type::getListKey(),
|
|
|
|
$array_atomic_type_list
|
|
|
|
],
|
|
|
|
true
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} elseif ($atomic_root_type_array instanceof TKeyedArray
|
|
|
|
&& $atomic_root_type_array->fallback_params === null
|
|
|
|
) {
|
|
|
|
if ($array_atomic_type_array) {
|
|
|
|
$array_atomic_type = new TNonEmptyArray(
|
|
|
|
$array_atomic_type_array,
|
|
|
|
count($atomic_root_type_array->properties)
|
|
|
|
);
|
|
|
|
} elseif ($atomic_root_type_array->is_list) {
|
|
|
|
$array_atomic_type = $atomic_root_type_array;
|
2022-11-04 19:04:23 +01:00
|
|
|
$new_child_type = new Union([$array_atomic_type], [
|
|
|
|
'parent_nodes' => $root_type->parent_nodes
|
|
|
|
]);
|
2022-12-13 21:40:19 +01:00
|
|
|
} else {
|
|
|
|
assert($array_atomic_type_list !== null);
|
|
|
|
$array_atomic_type = new TKeyedArray(
|
|
|
|
array_fill(
|
|
|
|
0,
|
|
|
|
count($atomic_root_type_array->properties),
|
|
|
|
$array_atomic_type_list
|
|
|
|
),
|
|
|
|
null,
|
|
|
|
[
|
|
|
|
Type::getListKey(),
|
|
|
|
$array_atomic_type_list
|
|
|
|
],
|
|
|
|
true
|
|
|
|
);
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
2022-12-13 21:40:19 +01:00
|
|
|
$from_countable_object_like = true;
|
|
|
|
} elseif ($array_atomic_type_list) {
|
|
|
|
$array_atomic_type = Type::getNonEmptyListAtomic(
|
|
|
|
$array_atomic_type_list
|
2020-11-30 19:08:42 +01:00
|
|
|
);
|
|
|
|
} else {
|
2022-12-13 21:40:19 +01:00
|
|
|
assert($array_atomic_type_array !== null);
|
2020-11-30 19:08:42 +01:00
|
|
|
$array_atomic_type = new TNonEmptyArray(
|
2022-12-13 21:40:19 +01:00
|
|
|
$array_atomic_type_array
|
2020-11-30 19:08:42 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-13 21:40:19 +01:00
|
|
|
$array_atomic_type ??= $array_atomic_type_class_string
|
|
|
|
?? ($array_atomic_type_list !== null
|
|
|
|
? Type::getNonEmptyListAtomic($array_atomic_type_list)
|
|
|
|
: null
|
|
|
|
) ?? ($array_atomic_type_array !== null
|
|
|
|
? new TNonEmptyArray($array_atomic_type_array)
|
|
|
|
: null
|
|
|
|
)
|
|
|
|
;
|
|
|
|
assert($array_atomic_type !== null);
|
|
|
|
|
|
|
|
$array_assignment_type = new Union([$array_atomic_type]);
|
2020-11-30 19:08:42 +01:00
|
|
|
|
|
|
|
if (!$new_child_type) {
|
2021-06-09 18:55:37 +02:00
|
|
|
if ($templated_assignment) {
|
|
|
|
$new_child_type = $root_type;
|
|
|
|
} else {
|
|
|
|
$new_child_type = Type::combineUnionTypes(
|
|
|
|
$root_type,
|
|
|
|
$array_assignment_type,
|
|
|
|
$codebase,
|
|
|
|
true,
|
|
|
|
true
|
|
|
|
);
|
|
|
|
}
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($from_countable_object_like) {
|
|
|
|
$atomic_root_types = $new_child_type->getAtomicTypes();
|
|
|
|
|
2022-12-13 21:40:19 +01:00
|
|
|
if (isset($atomic_root_types['array'])) {
|
|
|
|
$atomic_root_type_array = $atomic_root_types['array'];
|
|
|
|
if ($atomic_root_type_array instanceof TList) {
|
|
|
|
$atomic_root_type_array = $atomic_root_type_array->getKeyedArray();
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($atomic_root_type_array instanceof TNonEmptyArray
|
|
|
|
&& $atomic_root_type_array->count !== null
|
|
|
|
) {
|
|
|
|
$atomic_root_types['array'] =
|
|
|
|
$atomic_root_type_array->setCount($atomic_root_type_array->count+1);
|
|
|
|
$new_child_type = new Union($atomic_root_types);
|
|
|
|
} elseif ($atomic_root_type_array instanceof TKeyedArray
|
|
|
|
&& $atomic_root_type_array->is_list) {
|
|
|
|
$properties = $atomic_root_type_array->properties;
|
|
|
|
$had_undefined = false;
|
|
|
|
foreach ($properties as &$property) {
|
|
|
|
if ($property->possibly_undefined) {
|
|
|
|
$property = $property->setPossiblyUndefined(true);
|
|
|
|
$had_undefined = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$had_undefined && $atomic_root_type_array->fallback_params) {
|
|
|
|
$properties []= $atomic_root_type_array->fallback_params[1];
|
|
|
|
}
|
|
|
|
|
|
|
|
$atomic_root_types['array'] =
|
|
|
|
$atomic_root_type_array->setProperties($properties);
|
|
|
|
|
|
|
|
$new_child_type = new Union($atomic_root_types);
|
|
|
|
}
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
|
|
|
}
|
2020-12-07 01:14:52 +01:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
return $new_child_type;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2022-03-02 23:27:58 +01:00
|
|
|
* @param non-empty-list<PhpParser\Node\Expr\ArrayDimFetch> $child_stmts
|
|
|
|
* @param-out PhpParser\Node\Expr $child_stmt
|
2020-11-30 19:08:42 +01:00
|
|
|
*/
|
|
|
|
private static function analyzeNestedArrayAssignment(
|
|
|
|
StatementsAnalyzer $statements_analyzer,
|
2021-12-03 20:11:20 +01:00
|
|
|
Codebase $codebase,
|
2020-11-30 19:08:42 +01:00
|
|
|
Context $context,
|
|
|
|
?PhpParser\Node\Expr $assign_value,
|
2021-12-13 16:28:14 +01:00
|
|
|
Union $assignment_type,
|
2020-11-30 19:08:42 +01:00
|
|
|
array $child_stmts,
|
|
|
|
?string $root_var_id,
|
|
|
|
?string &$parent_var_id,
|
2021-12-13 16:28:14 +01:00
|
|
|
Union &$root_type,
|
|
|
|
Union &$current_type,
|
2020-11-30 19:08:42 +01:00
|
|
|
?PhpParser\Node\Expr &$current_dim,
|
|
|
|
bool &$offset_already_existed
|
|
|
|
): void {
|
|
|
|
$var_id_additions = [];
|
2020-12-07 01:14:52 +01:00
|
|
|
|
2022-03-03 00:49:57 +01:00
|
|
|
$root_var = end($child_stmts)->var;
|
2022-01-07 23:16:36 +01:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
// First go from the root element up, and go as far as we can to figure out what
|
|
|
|
// array types there are
|
2022-03-03 00:49:57 +01:00
|
|
|
foreach (array_reverse($child_stmts) as $i => $child_stmt) {
|
2020-11-30 19:08:42 +01:00
|
|
|
$child_stmt_dim_type = null;
|
|
|
|
|
|
|
|
$offset_type = null;
|
|
|
|
|
|
|
|
if ($child_stmt->dim) {
|
2021-06-25 15:54:39 +02:00
|
|
|
$was_inside_general_use = $context->inside_general_use;
|
|
|
|
$context->inside_general_use = true;
|
2020-11-30 19:08:42 +01:00
|
|
|
|
|
|
|
if (ExpressionAnalyzer::analyze(
|
|
|
|
$statements_analyzer,
|
|
|
|
$child_stmt->dim,
|
|
|
|
$context
|
|
|
|
) === false) {
|
2021-12-20 23:58:49 +01:00
|
|
|
$context->inside_general_use = $was_inside_general_use;
|
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-06-25 15:54:39 +02:00
|
|
|
$context->inside_general_use = $was_inside_general_use;
|
2020-11-30 19:08:42 +01:00
|
|
|
|
|
|
|
if (!($child_stmt_dim_type = $statements_analyzer->node_data->getType($child_stmt->dim))) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
[$offset_type, $var_id_addition, $full_var_id] = self::getArrayAssignmentOffsetType(
|
|
|
|
$statements_analyzer,
|
|
|
|
$child_stmt,
|
|
|
|
$child_stmt_dim_type
|
|
|
|
);
|
|
|
|
|
|
|
|
$var_id_additions[] = $var_id_addition;
|
|
|
|
} else {
|
|
|
|
$var_id_additions[] = '';
|
|
|
|
$full_var_id = false;
|
|
|
|
}
|
|
|
|
|
2022-11-04 19:04:23 +01:00
|
|
|
if (!($array_type = $statements_analyzer->node_data->getType($child_stmt->var))) {
|
2020-11-30 19:08:42 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-11-04 19:04:23 +01:00
|
|
|
if ($array_type->isNever()) {
|
|
|
|
$array_type = Type::getEmptyArray();
|
|
|
|
$statements_analyzer->node_data->setType($child_stmt->var, $array_type);
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
|
|
|
|
2022-02-04 18:49:12 +01:00
|
|
|
$extended_var_id = $root_var_id . implode('', $var_id_additions);
|
2020-11-30 19:08:42 +01:00
|
|
|
|
|
|
|
if ($parent_var_id && isset($context->vars_in_scope[$parent_var_id])) {
|
2022-11-04 19:04:23 +01:00
|
|
|
$array_type = $context->vars_in_scope[$parent_var_id];
|
|
|
|
$statements_analyzer->node_data->setType($child_stmt->var, $array_type);
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
|
|
|
|
2022-03-03 00:49:57 +01:00
|
|
|
$is_last = $i === count($child_stmts) - 1;
|
|
|
|
|
2022-10-03 10:45:36 +02:00
|
|
|
$child_stmt_dim_type_or_int = $child_stmt_dim_type ?? Type::getInt();
|
2020-11-30 19:08:42 +01:00
|
|
|
$child_stmt_type = ArrayFetchAnalyzer::getArrayAccessTypeGivenOffset(
|
|
|
|
$statements_analyzer,
|
|
|
|
$child_stmt,
|
|
|
|
$array_type,
|
2022-10-03 10:45:36 +02:00
|
|
|
$child_stmt_dim_type_or_int,
|
2020-11-30 19:08:42 +01:00
|
|
|
true,
|
2022-02-04 18:49:12 +01:00
|
|
|
$extended_var_id,
|
2020-11-30 19:08:42 +01:00
|
|
|
$context,
|
|
|
|
$assign_value,
|
2022-03-03 00:49:57 +01:00
|
|
|
!$is_last ? null : $assignment_type
|
2020-11-30 19:08:42 +01:00
|
|
|
);
|
2022-10-03 10:45:36 +02:00
|
|
|
if ($child_stmt->dim) {
|
2022-11-04 19:04:23 +01:00
|
|
|
$statements_analyzer->node_data->setType(
|
|
|
|
$child_stmt->dim,
|
|
|
|
$child_stmt_dim_type_or_int
|
|
|
|
);
|
2022-10-03 10:45:36 +02:00
|
|
|
}
|
2020-11-30 19:08:42 +01:00
|
|
|
|
|
|
|
$statements_analyzer->node_data->setType(
|
|
|
|
$child_stmt,
|
|
|
|
$child_stmt_type
|
|
|
|
);
|
|
|
|
|
2022-03-03 00:49:57 +01:00
|
|
|
if ($is_last) {
|
2019-12-27 18:49:28 +01:00
|
|
|
// we need this slight hack as the type we're putting it has to be
|
|
|
|
// different from the type we're getting out
|
|
|
|
if ($array_type->isSingle() && $array_type->hasClassStringMap()) {
|
|
|
|
$assignment_type = $child_stmt_type;
|
|
|
|
}
|
|
|
|
|
2019-11-25 17:44:54 +01:00
|
|
|
$child_stmt_type = $assignment_type;
|
|
|
|
$statements_analyzer->node_data->setType($child_stmt, $assignment_type);
|
2020-06-19 00:48:19 +02:00
|
|
|
|
2020-10-13 23:28:12 +02:00
|
|
|
if ($statements_analyzer->data_flow_graph) {
|
2020-09-28 06:45:02 +02:00
|
|
|
self::taintArrayAssignment(
|
|
|
|
$statements_analyzer,
|
|
|
|
$child_stmt,
|
|
|
|
$array_type,
|
|
|
|
$assignment_type,
|
2022-02-04 18:49:12 +01:00
|
|
|
ExpressionIdentifier::getExtendedVarId(
|
2020-09-28 06:45:02 +02:00
|
|
|
$child_stmt->var,
|
|
|
|
$statements_analyzer->getFQCLN(),
|
|
|
|
$statements_analyzer
|
|
|
|
),
|
2020-11-30 19:08:42 +01:00
|
|
|
$offset_type !== null ? [$offset_type] : []
|
2020-09-28 06:45:02 +02:00
|
|
|
);
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
|
2022-11-04 19:04:23 +01:00
|
|
|
$statements_analyzer->node_data->setType($child_stmt->var, $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;
|
|
|
|
$context->possibly_assigned_var_ids[$rooted_parent_id] = true;
|
|
|
|
}
|
|
|
|
|
2019-11-25 17:44:54 +01:00
|
|
|
$current_type = $child_stmt_type;
|
2018-01-14 18:09:40 +01:00
|
|
|
$current_dim = $child_stmt->dim;
|
|
|
|
|
2022-02-04 18:49:12 +01:00
|
|
|
$parent_var_id = $extended_var_id;
|
2022-03-03 00:49:57 +01:00
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2022-01-07 23:16:36 +01:00
|
|
|
if ($statements_analyzer->data_flow_graph instanceof VariableUseGraph
|
|
|
|
&& $root_var_id !== null
|
|
|
|
&& isset($context->references_to_external_scope[$root_var_id])
|
2022-03-02 23:27:58 +01:00
|
|
|
&& $root_var instanceof Variable && is_string($root_var->name)
|
2022-01-07 23:16:36 +01:00
|
|
|
&& $root_var_id === '$' . $root_var->name
|
|
|
|
) {
|
|
|
|
// Array is a reference to an external scope, mark it as used
|
|
|
|
$statements_analyzer->data_flow_graph->addPath(
|
|
|
|
DataFlowNode::getForAssignment(
|
|
|
|
$root_var_id,
|
|
|
|
new CodeLocation($statements_analyzer->getSource(), $root_var)
|
|
|
|
),
|
|
|
|
new DataFlowNode('variable-use', 'variable use', null),
|
|
|
|
'variable-use'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
if ($root_var_id
|
2018-02-17 18:32:19 +01:00
|
|
|
&& $full_var_id
|
2019-11-25 17:44:54 +01:00
|
|
|
&& ($child_stmt_var_type = $statements_analyzer->node_data->getType($child_stmt->var))
|
|
|
|
&& !$child_stmt_var_type->hasObjectType()
|
2018-01-14 18:09:40 +01:00
|
|
|
) {
|
2022-02-04 18:49:12 +01:00
|
|
|
$extended_var_id = $root_var_id . implode('', $var_id_additions);
|
2021-12-03 21:07:25 +01:00
|
|
|
$parent_var_id = $root_var_id . implode('', array_slice($var_id_additions, 0, -1));
|
2019-12-19 00:48:25 +01:00
|
|
|
|
2022-02-04 18:49:12 +01:00
|
|
|
if (isset($context->vars_in_scope[$extended_var_id])
|
|
|
|
&& !$context->vars_in_scope[$extended_var_id]->possibly_undefined
|
2019-12-19 00:48:25 +01:00
|
|
|
) {
|
|
|
|
$offset_already_existed = true;
|
|
|
|
}
|
|
|
|
|
2022-11-04 19:04:23 +01:00
|
|
|
$context->vars_in_scope[$extended_var_id] = $assignment_type;
|
2022-02-04 18:49:12 +01:00
|
|
|
$context->possibly_assigned_var_ids[$extended_var_id] = true;
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
|
2022-03-03 00:49:57 +01:00
|
|
|
array_shift($child_stmts);
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
// only update as many child stmts are we were able to process above
|
2022-03-03 00:49:57 +01:00
|
|
|
foreach ($child_stmts as $child_stmt) {
|
2019-11-25 17:44:54 +01:00
|
|
|
$child_stmt_type = $statements_analyzer->node_data->getType($child_stmt);
|
|
|
|
|
|
|
|
if (!$child_stmt_type) {
|
2021-12-03 21:40:18 +01:00
|
|
|
throw new InvalidArgumentException('Should never get here');
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
|
2022-06-21 18:42:26 +02:00
|
|
|
$key_values = $current_dim ? self::getDimKeyValues($statements_analyzer, $current_dim) : [];
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2020-07-02 00:57:11 +02:00
|
|
|
if ($key_values) {
|
|
|
|
$new_child_type = self::updateTypeWithKeyValues(
|
|
|
|
$codebase,
|
|
|
|
$child_stmt_type,
|
|
|
|
$current_type,
|
|
|
|
$key_values
|
|
|
|
);
|
2018-01-14 18:09:40 +01:00
|
|
|
} else {
|
2019-10-09 00:44:46 +02:00
|
|
|
if (!$current_dim) {
|
2022-12-13 21:40:19 +01:00
|
|
|
$array_assignment_type = Type::getList($current_type);
|
2019-10-09 00:44:46 +02:00
|
|
|
} else {
|
2021-06-09 18:55:37 +02:00
|
|
|
$key_type = $statements_analyzer->node_data->getType($current_dim);
|
2020-02-23 02:52:39 +01:00
|
|
|
|
2021-12-13 16:28:14 +01:00
|
|
|
$array_assignment_type = new Union([
|
2019-10-09 00:44:46 +02:00
|
|
|
new TArray([
|
2021-06-09 18:55:37 +02:00
|
|
|
$key_type && !$key_type->hasMixed()
|
|
|
|
? $key_type
|
2020-02-23 02:52:39 +01:00
|
|
|
: Type::getArrayKey(),
|
2019-10-09 00:44:46 +02:00
|
|
|
$current_type,
|
|
|
|
]),
|
|
|
|
]);
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
|
|
|
|
$new_child_type = Type::combineUnionTypes(
|
2019-11-25 17:44:54 +01:00
|
|
|
$child_stmt_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,
|
2020-04-09 16:42:54 +02:00
|
|
|
true
|
2018-01-14 18:09:40 +01:00
|
|
|
);
|
|
|
|
}
|
2022-11-04 19:04:23 +01:00
|
|
|
if ($new_child_type->hasNull() || $new_child_type->possibly_undefined) {
|
|
|
|
$new_child_type = $new_child_type->getBuilder();
|
|
|
|
$new_child_type->removeType('null');
|
|
|
|
$new_child_type->possibly_undefined = false;
|
|
|
|
$new_child_type = $new_child_type->freeze();
|
|
|
|
}
|
2019-11-25 17:44:54 +01:00
|
|
|
if (!$child_stmt_type->hasObjectType()) {
|
|
|
|
$child_stmt_type = $new_child_type;
|
|
|
|
$statements_analyzer->node_data->setType($child_stmt, $new_child_type);
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
|
2019-11-25 17:44:54 +01:00
|
|
|
$current_type = $child_stmt_type;
|
2018-01-14 18:09:40 +01:00
|
|
|
$current_dim = $child_stmt->dim;
|
|
|
|
|
|
|
|
array_pop($var_id_additions);
|
|
|
|
|
2020-09-28 06:45:02 +02:00
|
|
|
$parent_array_var_id = null;
|
2020-06-19 00:48:19 +02:00
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
if ($root_var_id) {
|
2022-02-04 18:49:12 +01:00
|
|
|
$extended_var_id = $root_var_id . implode('', $var_id_additions);
|
2021-12-03 21:07:25 +01:00
|
|
|
$parent_array_var_id = $root_var_id . implode('', array_slice($var_id_additions, 0, -1));
|
2022-11-04 19:04:23 +01:00
|
|
|
$context->vars_in_scope[$extended_var_id] = $child_stmt_type;
|
2022-02-04 18:49:12 +01:00
|
|
|
$context->possibly_assigned_var_ids[$extended_var_id] = true;
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
2020-06-19 00:48:19 +02:00
|
|
|
|
2020-10-13 23:28:12 +02:00
|
|
|
if ($statements_analyzer->data_flow_graph) {
|
2022-11-04 19:04:23 +01:00
|
|
|
$t_orig = $statements_analyzer->node_data->getType($child_stmt->var);
|
|
|
|
$array_type = $t_orig ?? Type::getMixed();
|
2020-06-19 00:48:19 +02:00
|
|
|
self::taintArrayAssignment(
|
|
|
|
$statements_analyzer,
|
2020-09-28 06:45:02 +02:00
|
|
|
$child_stmt,
|
2022-11-04 19:04:23 +01:00
|
|
|
$array_type,
|
2020-06-19 00:48:19 +02:00
|
|
|
$new_child_type,
|
2020-09-28 06:45:02 +02:00
|
|
|
$parent_array_var_id,
|
2022-06-21 18:42:26 +02:00
|
|
|
$child_stmt->dim ? self::getDimKeyValues($statements_analyzer, $child_stmt->dim) : [],
|
2020-06-19 00:48:19 +02:00
|
|
|
);
|
2022-11-04 19:04:23 +01:00
|
|
|
if ($t_orig) {
|
|
|
|
$statements_analyzer->node_data->setType($child_stmt->var, $array_type);
|
|
|
|
}
|
|
|
|
if ($root_var_id) {
|
|
|
|
if ($parent_array_var_id === $root_var_id) {
|
|
|
|
$rooted_parent_id = $root_var_id;
|
|
|
|
$root_type = $array_type;
|
|
|
|
} else {
|
|
|
|
assert($parent_array_var_id !== null);
|
|
|
|
$rooted_parent_id = $parent_array_var_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
$context->vars_in_scope[$rooted_parent_id] = $array_type;
|
|
|
|
}
|
2020-06-19 00:48:19 +02:00
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
2020-11-30 19:08:42 +01:00
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2022-06-21 18:42:26 +02:00
|
|
|
/**
|
|
|
|
* @return list<TLiteralInt|TLiteralString>
|
|
|
|
*/
|
|
|
|
private static function getDimKeyValues(
|
|
|
|
StatementsAnalyzer $statements_analyzer,
|
2022-06-21 20:59:01 +02:00
|
|
|
PhpParser\Node\Expr $dim
|
2022-06-21 18:42:26 +02:00
|
|
|
): array {
|
|
|
|
$key_values = [];
|
|
|
|
|
|
|
|
if ($dim instanceof PhpParser\Node\Scalar\String_) {
|
|
|
|
$key_values[] = new TLiteralString($dim->value);
|
|
|
|
} elseif ($dim instanceof PhpParser\Node\Scalar\LNumber) {
|
|
|
|
$key_values[] = new TLiteralInt($dim->value);
|
|
|
|
} else {
|
|
|
|
$key_type = $statements_analyzer->node_data->getType($dim);
|
|
|
|
|
|
|
|
if ($key_type) {
|
|
|
|
$string_literals = $key_type->getLiteralStrings();
|
|
|
|
$int_literals = $key_type->getLiteralInts();
|
|
|
|
|
|
|
|
$all_atomic_types = $key_type->getAtomicTypes();
|
|
|
|
|
|
|
|
if (count($string_literals) + count($int_literals) === count($all_atomic_types)) {
|
|
|
|
foreach ($string_literals as $string_literal) {
|
2022-11-04 19:04:23 +01:00
|
|
|
$key_values[] = $string_literal;
|
2022-06-21 18:42:26 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($int_literals as $int_literal) {
|
2022-11-04 19:04:23 +01:00
|
|
|
$key_values[] = $int_literal;
|
2022-06-21 18:42:26 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $key_values;
|
|
|
|
}
|
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
/**
|
2022-11-12 02:14:21 +01:00
|
|
|
* @return array{TLiteralInt|TLiteralString|null, string, bool}
|
2020-11-30 19:08:42 +01:00
|
|
|
*/
|
|
|
|
private static function getArrayAssignmentOffsetType(
|
|
|
|
StatementsAnalyzer $statements_analyzer,
|
|
|
|
PhpParser\Node\Expr\ArrayDimFetch $child_stmt,
|
2021-12-13 16:28:14 +01:00
|
|
|
Union $child_stmt_dim_type
|
2020-11-30 19:08:42 +01:00
|
|
|
): array {
|
|
|
|
if ($child_stmt->dim instanceof PhpParser\Node\Scalar\String_
|
|
|
|
|| (($child_stmt->dim instanceof PhpParser\Node\Expr\ConstFetch
|
|
|
|
|| $child_stmt->dim instanceof PhpParser\Node\Expr\ClassConstFetch)
|
|
|
|
&& $child_stmt_dim_type->isSingleStringLiteral())
|
2018-12-19 22:10:09 +01:00
|
|
|
) {
|
2020-11-30 19:08:42 +01:00
|
|
|
if ($child_stmt->dim instanceof PhpParser\Node\Scalar\String_) {
|
2021-12-13 04:45:57 +01:00
|
|
|
$offset_type = new TLiteralString($child_stmt->dim->value);
|
2020-11-30 19:08:42 +01:00
|
|
|
} else {
|
|
|
|
$offset_type = $child_stmt_dim_type->getSingleStringLiteral();
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
2019-10-09 00:44:46 +02:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
if (preg_match('/^(0|[1-9][0-9]*)$/', $offset_type->value)) {
|
|
|
|
$var_id_addition = '[' . $offset_type->value . ']';
|
2018-05-03 19:56:30 +02:00
|
|
|
} else {
|
2020-11-30 19:08:42 +01:00
|
|
|
$var_id_addition = '[\'' . $offset_type->value . '\']';
|
2018-05-03 19:56:30 +02:00
|
|
|
}
|
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
return [$offset_type, $var_id_addition, true];
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
if ($child_stmt->dim instanceof PhpParser\Node\Scalar\LNumber
|
|
|
|
|| (($child_stmt->dim instanceof PhpParser\Node\Expr\ConstFetch
|
|
|
|
|| $child_stmt->dim instanceof PhpParser\Node\Expr\ClassConstFetch)
|
|
|
|
&& $child_stmt_dim_type->isSingleIntLiteral())
|
|
|
|
) {
|
|
|
|
if ($child_stmt->dim instanceof PhpParser\Node\Scalar\LNumber) {
|
2021-12-13 04:45:57 +01:00
|
|
|
$offset_type = new TLiteralInt($child_stmt->dim->value);
|
2020-11-30 19:08:42 +01:00
|
|
|
} else {
|
|
|
|
$offset_type = $child_stmt_dim_type->getSingleIntLiteral();
|
2019-10-09 00:44:46 +02:00
|
|
|
}
|
2018-05-03 19:56:30 +02:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
$var_id_addition = '[' . $offset_type->value . ']';
|
2018-05-03 19:56:30 +02:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
return [$offset_type, $var_id_addition, true];
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
if ($child_stmt->dim instanceof PhpParser\Node\Expr\Variable
|
|
|
|
&& is_string($child_stmt->dim->name)
|
|
|
|
) {
|
|
|
|
$var_id_addition = '[$' . $child_stmt->dim->name . ']';
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
return [null, $var_id_addition, true];
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
if ($child_stmt->dim instanceof PhpParser\Node\Expr\PropertyFetch
|
|
|
|
&& $child_stmt->dim->name instanceof PhpParser\Node\Identifier
|
2019-12-27 16:34:51 +01:00
|
|
|
) {
|
2022-02-04 18:49:12 +01:00
|
|
|
$object_id = ExpressionIdentifier::getExtendedVarId(
|
2020-11-30 19:08:42 +01:00
|
|
|
$child_stmt->dim->var,
|
|
|
|
$statements_analyzer->getFQCLN(),
|
|
|
|
$statements_analyzer
|
2019-12-27 16:34:51 +01:00
|
|
|
);
|
2020-07-02 00:57:11 +02:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
if ($object_id) {
|
|
|
|
$var_id_addition = '[' . $object_id . '->' . $child_stmt->dim->name->name . ']';
|
2020-07-02 00:57:11 +02:00
|
|
|
} else {
|
2020-11-30 19:08:42 +01:00
|
|
|
$var_id_addition = '[' . $child_stmt_dim_type . ']';
|
2020-07-02 00:57:11 +02:00
|
|
|
}
|
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
return [null, $var_id_addition, true];
|
2020-07-02 00:57:11 +02:00
|
|
|
}
|
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
if ($child_stmt->dim instanceof PhpParser\Node\Expr\ClassConstFetch
|
|
|
|
&& $child_stmt->dim->name instanceof PhpParser\Node\Identifier
|
|
|
|
&& $child_stmt->dim->class instanceof PhpParser\Node\Name
|
2020-06-19 00:48:19 +02:00
|
|
|
) {
|
2020-11-30 19:08:42 +01:00
|
|
|
$object_name = ClassLikeAnalyzer::getFQCLNFromNameObject(
|
|
|
|
$child_stmt->dim->class,
|
|
|
|
$statements_analyzer->getAliases()
|
2020-11-10 04:44:36 +01:00
|
|
|
);
|
2020-11-30 19:08:42 +01:00
|
|
|
$var_id_addition = '[' . $object_name . '::' . $child_stmt->dim->name->name . ']';
|
2020-11-10 04:44:36 +01:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
return [null, $var_id_addition, true];
|
|
|
|
}
|
2020-11-10 04:44:36 +01:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
$var_id_addition = '[' . $child_stmt_dim_type . ']';
|
2020-09-28 06:45:02 +02:00
|
|
|
|
2020-11-30 19:08:42 +01:00
|
|
|
return [null, $var_id_addition, false];
|
2020-06-19 00:48:19 +02:00
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|