items)) { $statements_analyzer->node_data->setType($stmt, Type::getEmptyArray()); return null; } $item_key_atomic_types = []; $item_value_atomic_types = []; $property_types = []; $class_strings = []; $can_create_objectlike = true; $array_keys = []; $int_offset_diff = 0; $codebase = $statements_analyzer->getCodebase(); $all_list = true; $taint_sources = []; $either_tainted = 0; foreach ($stmt->items as $int_offset => $item) { if ($item === null) { continue; } if (ExpressionAnalyzer::analyze($statements_analyzer, $item->value, $context) === false) { return false; } if ($item->unpack) { $unpacked_array_type = $statements_analyzer->node_data->getType($item->value); if (!$unpacked_array_type) { continue; } foreach ($unpacked_array_type->getTypes() as $unpacked_atomic_type) { if ($unpacked_atomic_type instanceof Type\Atomic\ObjectLike) { $unpacked_array_offset = 0; foreach ($unpacked_atomic_type->properties as $key => $property_value) { if (is_string($key)) { if (IssueBuffer::accepts( new DuplicateArrayKey( 'String keys are not supported in unpacked arrays', new CodeLocation($statements_analyzer->getSource(), $item->value) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } continue; } $item_key_atomic_types[] = new Type\Atomic\TLiteralInt($key); $item_value_atomic_types = array_merge( $item_value_atomic_types, array_values($property_value->getTypes()) ); $array_keys[$int_offset + $int_offset_diff + $unpacked_array_offset] = true; $property_types[$int_offset + $int_offset_diff + $unpacked_array_offset] = $property_value; $unpacked_array_offset++; } $int_offset_diff += $unpacked_array_offset - 1; } else { $can_create_objectlike = false; if ($unpacked_atomic_type instanceof Type\Atomic\TArray) { if ($unpacked_atomic_type->type_params[0]->hasString()) { if (IssueBuffer::accepts( new DuplicateArrayKey( 'String keys are not supported in unpacked arrays', new CodeLocation($statements_analyzer->getSource(), $item->value) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } elseif ($unpacked_atomic_type->type_params[0]->hasInt()) { $item_key_atomic_types[] = new Type\Atomic\TInt(); } $item_value_atomic_types = array_merge( $item_value_atomic_types, array_values($unpacked_atomic_type->type_params[1]->getTypes()) ); } elseif ($unpacked_atomic_type instanceof Type\Atomic\TList) { $item_key_atomic_types[] = new Type\Atomic\TInt(); $item_value_atomic_types = array_merge( $item_value_atomic_types, array_values($unpacked_atomic_type->type_param->getTypes()) ); } } } continue; } $item_key_value = null; if ($item->key) { $all_list = false; if (ExpressionAnalyzer::analyze($statements_analyzer, $item->key, $context) === false) { return false; } if ($item_key_type = $statements_analyzer->node_data->getType($item->key)) { $key_type = $item_key_type; if ($key_type->isNull()) { $key_type = Type::getString(''); } if ($item->key instanceof PhpParser\Node\Scalar\String_ && preg_match('/^(0|[1-9][0-9]*)$/', $item->key->value) ) { $key_type = Type::getInt(false, (int) $item->key->value); } $item_key_atomic_types = array_merge($item_key_atomic_types, array_values($key_type->getTypes())); if ($key_type->isSingleStringLiteral()) { $item_key_literal_type = $key_type->getSingleStringLiteral(); $item_key_value = $item_key_literal_type->value; if ($item_key_literal_type instanceof Type\Atomic\TLiteralClassString) { $class_strings[$item_key_value] = true; } } elseif ($key_type->isSingleIntLiteral()) { $item_key_value = $key_type->getSingleIntLiteral()->value; if ($item_key_value > $int_offset + $int_offset_diff) { $int_offset_diff = $item_key_value - $int_offset; } } } } else { $item_key_value = $int_offset + $int_offset_diff; $item_key_atomic_types[] = new Type\Atomic\TInt(); } if ($item_key_value !== null) { if (isset($array_keys[$item_key_value])) { if (IssueBuffer::accepts( new DuplicateArrayKey( 'Key \'' . $item_key_value . '\' already exists on array', new CodeLocation($statements_analyzer->getSource(), $item) ), $statements_analyzer->getSuppressedIssues() )) { // fall through } } $array_keys[$item_key_value] = true; } if ($codebase->taint) { if ($item_value_type = $statements_analyzer->node_data->getType($item->value)) { $taint_sources = array_merge($taint_sources, $item_value_type->sources ?: []); $either_tainted = $either_tainted | $item_value_type->tainted; } if ($item->key && ($item_key_type = $statements_analyzer->node_data->getType($item->key))) { $taint_sources = array_merge($taint_sources, $item_key_type->sources ?: []); $either_tainted = $either_tainted | $item_key_type->tainted; } } if ($item->byRef) { $var_id = ExpressionAnalyzer::getArrayVarId( $item->value, $statements_analyzer->getFQCLN(), $statements_analyzer ); if ($var_id) { $context->removeDescendents( $var_id, $context->vars_in_scope[$var_id] ?? null, null, $statements_analyzer ); $context->vars_in_scope[$var_id] = Type::getMixed(); } } if ($item_value_atomic_types && !$can_create_objectlike) { continue; } if ($item_value_type = $statements_analyzer->node_data->getType($item->value)) { if ($item_key_value !== null && count($property_types) <= 100) { $property_types[$item_key_value] = $item_value_type; } else { $can_create_objectlike = false; } $item_value_atomic_types = array_merge( $item_value_atomic_types, array_values($item_value_type->getTypes()) ); } else { $item_value_atomic_types[] = new Type\Atomic\TMixed(); if ($item_key_value !== null && count($property_types) <= 100) { $property_types[$item_key_value] = Type::getMixed(); } else { $can_create_objectlike = false; } } } if ($item_key_atomic_types) { $item_key_type = TypeCombination::combineTypes( $item_key_atomic_types, $codebase, false, true, 30 ); } else { $item_key_type = null; } if ($item_value_atomic_types) { $item_value_type = TypeCombination::combineTypes( $item_value_atomic_types, $codebase, false, true, 30 ); } else { $item_value_type = null; } // if this array looks like an object-like array, let's return that instead if ($item_value_type && $item_key_type && ($item_key_type->hasString() || $item_key_type->hasInt()) && $can_create_objectlike ) { $object_like = new Type\Atomic\ObjectLike($property_types, $class_strings); $object_like->sealed = true; $object_like->is_list = $all_list; $stmt_type = new Type\Union([$object_like]); if ($taint_sources) { $stmt_type->sources = $taint_sources; } if ($either_tainted) { $stmt_type->tainted = $either_tainted; } $statements_analyzer->node_data->setType($stmt, $stmt_type); return null; } if ($all_list) { $array_type = new Type\Atomic\TNonEmptyList($item_value_type ?: Type::getMixed()); $array_type->count = count($stmt->items); $stmt_type = new Type\Union([ $array_type, ]); if ($taint_sources) { $stmt_type->sources = $taint_sources; } if ($either_tainted) { $stmt_type->tainted = $either_tainted; } $statements_analyzer->node_data->setType($stmt, $stmt_type); return null; } $array_type = new Type\Atomic\TNonEmptyArray([ $item_key_type ?: new Type\Union([new TInt, new TString]), $item_value_type ?: Type::getMixed(), ]); $array_type->count = count($stmt->items); $stmt_type = new Type\Union([ $array_type, ]); if ($taint_sources) { $stmt_type->sources = $taint_sources; } if ($either_tainted) { $stmt_type->tainted = $either_tainted; } $statements_analyzer->node_data->setType($stmt, $stmt_type); return null; } }