var, $statements_analyzer->getFQCLN(), $statements_analyzer, $nesting ); self::updateArrayType( $statements_analyzer, $stmt, $assign_value, $assignment_value_type, $context ); if (!$statements_analyzer->node_data->getType($stmt->var) && $var_id) { $context->vars_in_scope[$var_id] = Type::getMixed(); } } /** * * @return false|null */ public static function updateArrayType( StatementsAnalyzer $statements_analyzer, PhpParser\Node\Expr\ArrayDimFetch $stmt, ?PhpParser\Node\Expr $assign_value, 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; if (ExpressionAnalyzer::analyze( $statements_analyzer, $root_array_expr, $context, true ) === false) { // fall through } $codebase = $statements_analyzer->getCodebase(); $root_type = $statements_analyzer->node_data->getType($root_array_expr) ?: Type::getMixed(); if ($root_type->hasMixed()) { 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 } } 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 $root_var_id = ExpressionAnalyzer::getRootVarId( $root_array_expr, $statements_analyzer->getFQCLN(), $statements_analyzer ); $var_id_additions = []; $parent_var_id = null; $offset_already_existed = false; $full_var_id = true; $child_stmt = null; $taint_sources = []; $taint_type = 0; if ($codebase->taint && $assign_value && ($assign_value_type = $statements_analyzer->node_data->getType($assign_value)) ) { $taint_sources = $assign_value_type->sources; $taint_type = $assign_value_type->tainted ?: 0; } // 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); } $child_stmt_dim_type = null; if ($child_stmt->dim) { if (ExpressionAnalyzer::analyze( $statements_analyzer, $child_stmt->dim, $context ) === false) { return false; } if (!($child_stmt_dim_type = $statements_analyzer->node_data->getType($child_stmt->dim))) { return null; } if ($child_stmt->dim instanceof PhpParser\Node\Scalar\String_ || ($child_stmt->dim instanceof PhpParser\Node\Expr\ConstFetch && $child_stmt_dim_type->isSingleStringLiteral()) ) { if ($child_stmt->dim instanceof PhpParser\Node\Scalar\String_) { $value = $child_stmt->dim->value; } else { $value = $child_stmt_dim_type->getSingleStringLiteral()->value; } 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_type->isSingleIntLiteral()) ) { if ($child_stmt->dim instanceof PhpParser\Node\Scalar\LNumber) { $value = $child_stmt->dim->value; } else { $value = $child_stmt_dim_type->getSingleIntLiteral()->value; } $var_id_additions[] = '[' . $value . ']'; } elseif ($child_stmt->dim instanceof PhpParser\Node\Expr\Variable && is_string($child_stmt->dim->name) ) { $var_id_additions[] = '[$' . $child_stmt->dim->name . ']'; } 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 . ']'; } } elseif ($child_stmt->dim instanceof PhpParser\Node\Expr\ClassConstFetch && $child_stmt->dim->name instanceof PhpParser\Node\Identifier && $child_stmt->dim->class instanceof PhpParser\Node\Name ) { $object_name = ClassLikeAnalyzer::getFQCLNFromNameObject( $child_stmt->dim->class, $statements_analyzer->getAliases() ); $var_id_additions[] = '[' . $object_name . '::' . $child_stmt->dim->name->name . ']'; } else { $var_id_additions[] = '[' . $child_stmt_dim_type . ']'; $full_var_id = false; } } else { $var_id_additions[] = ''; $full_var_id = false; } if (!($child_stmt_var_type = $statements_analyzer->node_data->getType($child_stmt->var))) { return null; } if ($child_stmt_var_type->isEmpty()) { $child_stmt_var_type = Type::getEmptyArray(); $statements_analyzer->node_data->setType($child_stmt->var, $child_stmt_var_type); } $array_var_id = $root_var_id . implode('', $var_id_additions); if ($parent_var_id && isset($context->vars_in_scope[$parent_var_id])) { $child_stmt_var_type = clone $context->vars_in_scope[$parent_var_id]; $statements_analyzer->node_data->setType($child_stmt->var, $child_stmt_var_type); } $array_type = clone $child_stmt_var_type; $child_stmt_type = ArrayFetchAnalyzer::getArrayAccessTypeGivenOffset( $statements_analyzer, $child_stmt, $array_type, $child_stmt_dim_type ?: Type::getInt(), true, $array_var_id, $context, $assign_value, $child_stmts ? null : $assignment_type ); $statements_analyzer->node_data->setType( $child_stmt, $child_stmt_type ); $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; } if (!$child_stmts) { // 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; } $child_stmt_type = $assignment_type; $statements_analyzer->node_data->setType($child_stmt, $assignment_type); } $current_type = $child_stmt_type; $current_dim = $child_stmt->dim; $parent_var_id = $array_var_id; } if ($root_var_id && $full_var_id && $child_stmt && ($child_stmt_var_type = $statements_analyzer->node_data->getType($child_stmt->var)) && !$child_stmt_var_type->hasObjectType() ) { $array_var_id = $root_var_id . implode('', $var_id_additions); $parent_var_id = $root_var_id . implode('', \array_slice($var_id_additions, 0, -1)); if (isset($context->vars_in_scope[$array_var_id]) && !$context->vars_in_scope[$array_var_id]->possibly_undefined ) { $offset_already_existed = true; } $context->vars_in_scope[$array_var_id] = clone $assignment_type; $context->possibly_assigned_var_ids[$array_var_id] = true; } // only update as many child stmts are we were able to process above foreach ($reversed_child_stmts as $child_stmt) { $child_stmt_type = $statements_analyzer->node_data->getType($child_stmt); if (!$child_stmt_type) { throw new \InvalidArgumentException('Should never get here'); } $key_value = null; if ($current_dim instanceof PhpParser\Node\Scalar\String_ || $current_dim instanceof PhpParser\Node\Scalar\LNumber ) { $key_value = $current_dim->value; } elseif ($current_dim instanceof PhpParser\Node\Expr\ConstFetch && ($current_dim_type = $statements_analyzer->node_data->getType($current_dim)) ) { $is_single_string_literal = $current_dim_type->isSingleStringLiteral(); if ($is_single_string_literal || $current_dim_type->isSingleIntLiteral()) { if ($is_single_string_literal) { $key_value = $current_dim_type->getSingleStringLiteral()->value; } else { $key_value = $current_dim_type->getSingleIntLiteral()->value; } } } if ($key_value !== null) { $has_matching_objectlike_property = false; $has_matching_string = false; foreach ($child_stmt_type->getAtomicTypes() as $type) { if ($type instanceof ObjectLike) { if (isset($type->properties[$key_value])) { $has_matching_objectlike_property = true; $type->properties[$key_value] = clone $current_type; } } if ($type instanceof Type\Atomic\TString && \is_int($key_value)) { $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; } } } } if (!$has_matching_objectlike_property && !$has_matching_string) { $array_assignment_type = new Type\Union([ new ObjectLike([$key_value => $current_type]), ]); $new_child_type = Type::combineUnionTypes( $child_stmt_type, $array_assignment_type, $codebase, true, true ); } else { $new_child_type = $child_stmt_type; // noop } } else { if (!$current_dim) { $array_assignment_type = new Type\Union([ new TList($current_type), ]); } else { $current_dim_type = $statements_analyzer->node_data->getType($current_dim); $array_assignment_type = new Type\Union([ new TArray([ $current_dim_type && !$current_dim_type->hasMixed() ? $current_dim_type : Type::getArrayKey(), $current_type, ]), ]); } $new_child_type = Type::combineUnionTypes( $child_stmt_type, $array_assignment_type, $codebase, true, true ); } $new_child_type->removeType('null'); $new_child_type->possibly_undefined = false; if (!$child_stmt_type->hasObjectType()) { $child_stmt_type = $new_child_type; $statements_analyzer->node_data->setType($child_stmt, $new_child_type); } $current_type = $child_stmt_type; $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_type; $context->possibly_assigned_var_ids[$array_var_id] = true; } } $root_is_string = $root_type->isString(); $key_value = null; if ($current_dim instanceof PhpParser\Node\Scalar\String_ || ($current_dim instanceof PhpParser\Node\Scalar\LNumber && !$root_is_string) ) { $key_value = $current_dim->value; } elseif ($current_dim instanceof PhpParser\Node\Expr\ConstFetch && ($current_dim_type = $statements_analyzer->node_data->getType($current_dim)) && !$root_is_string ) { $is_single_string_literal = $current_dim_type->isSingleStringLiteral(); if ($is_single_string_literal || $current_dim_type->isSingleIntLiteral()) { if ($is_single_string_literal) { $key_value = $current_dim_type->getSingleStringLiteral()->value; } else { $key_value = $current_dim_type->getSingleIntLiteral()->value; } } } if ($key_value !== null) { $has_matching_objectlike_property = false; foreach ($root_type->getAtomicTypes() as $type) { if ($type instanceof ObjectLike) { if (isset($type->properties[$key_value])) { $has_matching_objectlike_property = true; $type->properties[$key_value] = clone $current_type; } } 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 ); } } if (!$has_matching_objectlike_property) { $object_like = new ObjectLike([$key_value => $current_type]); $object_like->sealed = true; $array_assignment_type = new Type\Union([ $object_like, ]); $new_child_type = Type::combineUnionTypes( $root_type, $array_assignment_type, $codebase, true, false ); } else { $new_child_type = $root_type; // noop } } elseif (!$root_is_string) { if ($current_dim) { if ($current_dim_type = $statements_analyzer->node_data->getType($current_dim)) { if ($current_dim_type->hasMixed()) { $current_dim_type = Type::getArrayKey(); } $array_atomic_key_type = ArrayFetchAnalyzer::replaceOffsetTypeWithInts( $current_dim_type ); } else { $array_atomic_key_type = Type::getArrayKey(); } if ($offset_already_existed && $child_stmt && $parent_var_id && ($parent_type = $context->vars_in_scope[$parent_var_id] ?? null) ) { if ($parent_type->hasList()) { $array_atomic_type = new TNonEmptyList( $current_type ); } elseif ($parent_type->hasClassStringMap() && $current_dim_type && $current_dim_type->isTemplatedClassString() ) { /** * @var Type\Atomic\TClassStringMap * @psalm-suppress PossiblyUndefinedStringArrayOffset */ $class_string_map = $parent_type->getAtomicTypes()['array']; /** * @var Type\Atomic\TTemplateParamClass */ $offset_type_part = \array_values($current_dim_type->getAtomicTypes())[0]; $template_result = new \Psalm\Internal\Type\TemplateResult( [], [ $offset_type_part->param_name => [ $offset_type_part->defining_class => [ new Type\Union([ new Type\Atomic\TTemplateParam( $class_string_map->param_name, $offset_type_part->as_type ? new Type\Union([$offset_type_part->as_type]) : Type::getObject(), 'class-string-map' ) ]) ] ] ] ); $current_type->replaceTemplateTypesWithArgTypes( $template_result, $codebase ); $array_atomic_type = new Type\Atomic\TClassStringMap( $class_string_map->param_name, $class_string_map->as_type, $current_type ); } else { $array_atomic_type = new TNonEmptyArray([ $array_atomic_key_type, $current_type, ]); } } else { $array_atomic_type = new TNonEmptyArray([ $array_atomic_key_type, $current_type, ]); } } else { $array_atomic_type = new TNonEmptyList($current_type); } $from_countable_object_like = false; $new_child_type = null; if (!$current_dim && !$context->inside_loop) { $atomic_root_types = $root_type->getAtomicTypes(); if (isset($atomic_root_types['array'])) { if ($array_atomic_type instanceof Type\Atomic\TClassStringMap) { $array_atomic_type = new TNonEmptyArray([ $array_atomic_type->getStandinKeyParam(), $array_atomic_type->value_param ]); } elseif ($atomic_root_types['array'] instanceof TNonEmptyArray || $atomic_root_types['array'] instanceof TNonEmptyList ) { $array_atomic_type->count = $atomic_root_types['array']->count; } elseif ($atomic_root_types['array'] instanceof ObjectLike && $atomic_root_types['array']->sealed ) { $array_atomic_type->count = count($atomic_root_types['array']->properties); $from_countable_object_like = true; 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 ); } else { $array_atomic_type = new TNonEmptyArray( $array_atomic_type->type_params ); } } } $array_assignment_type = new Type\Union([ $array_atomic_type, ]); if (!$new_child_type) { $new_child_type = Type::combineUnionTypes( $root_type, $array_assignment_type, $codebase, true, true ); } if ($from_countable_object_like) { $atomic_root_types = $new_child_type->getAtomicTypes(); if (isset($atomic_root_types['array']) && ($atomic_root_types['array'] instanceof TNonEmptyArray || $atomic_root_types['array'] instanceof TNonEmptyList) && $atomic_root_types['array']->count !== null ) { $atomic_root_types['array']->count++; } } } else { $new_child_type = $root_type; } $new_child_type->removeType('null'); if (!$root_type->hasObjectType()) { $root_type = $new_child_type; } if ($codebase->taint && $taint_sources) { $root_type->sources = \array_merge($taint_sources, $root_type->sources ?: []); $root_type->tainted = $taint_type | $root_type->tainted; } $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) { PropertyAssignmentAnalyzer::analyzeInstance( $statements_analyzer, $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) { return false; } if (ExpressionAnalyzer::analyze($statements_analyzer, $root_array_expr->var, $context) === false) { return false; } } } elseif ($root_array_expr instanceof PhpParser\Node\Expr\StaticPropertyFetch && $root_array_expr->name instanceof PhpParser\Node\Identifier ) { PropertyAssignmentAnalyzer::analyzeStatic( $statements_analyzer, $root_array_expr, null, $root_type, $context ); } elseif ($root_var_id) { $context->vars_in_scope[$root_var_id] = $root_type; } 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', new \Psalm\CodeLocation($statements_analyzer->getSource(), $root_array_expr) ), $statements_analyzer->getSuppressedIssues() ) ) { // do nothing } } } return null; } }