2018-01-14 18:09:40 +01:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Checker\Statements\Expression\Fetch;
|
|
|
|
|
|
|
|
use PhpParser;
|
|
|
|
use Psalm\Checker\Statements\ExpressionChecker;
|
|
|
|
use Psalm\Checker\StatementsChecker;
|
|
|
|
use Psalm\Checker\TypeChecker;
|
|
|
|
use Psalm\CodeLocation;
|
|
|
|
use Psalm\Context;
|
|
|
|
use Psalm\Issue\EmptyArrayAccess;
|
|
|
|
use Psalm\Issue\InvalidArrayAccess;
|
|
|
|
use Psalm\Issue\InvalidArrayAssignment;
|
|
|
|
use Psalm\Issue\InvalidArrayOffset;
|
|
|
|
use Psalm\Issue\MixedArrayAccess;
|
|
|
|
use Psalm\Issue\MixedArrayAssignment;
|
|
|
|
use Psalm\Issue\MixedArrayOffset;
|
|
|
|
use Psalm\Issue\MixedStringOffsetAssignment;
|
|
|
|
use Psalm\Issue\NullArrayAccess;
|
|
|
|
use Psalm\Issue\NullArrayOffset;
|
|
|
|
use Psalm\Issue\PossiblyInvalidArrayAccess;
|
|
|
|
use Psalm\Issue\PossiblyInvalidArrayAssignment;
|
|
|
|
use Psalm\Issue\PossiblyInvalidArrayOffset;
|
|
|
|
use Psalm\Issue\PossiblyNullArrayAccess;
|
|
|
|
use Psalm\Issue\PossiblyNullArrayAssignment;
|
|
|
|
use Psalm\Issue\PossiblyNullArrayOffset;
|
2018-04-17 20:06:27 +02:00
|
|
|
use Psalm\Issue\PossiblyUndefinedArrayOffset;
|
2018-01-14 18:09:40 +01:00
|
|
|
use Psalm\IssueBuffer;
|
|
|
|
use Psalm\Type;
|
|
|
|
use Psalm\Type\Atomic\ObjectLike;
|
|
|
|
use Psalm\Type\Atomic\TArray;
|
|
|
|
use Psalm\Type\Atomic\TEmpty;
|
2018-05-05 23:30:18 +02:00
|
|
|
use Psalm\Type\Atomic\TLiteralFloat;
|
|
|
|
use Psalm\Type\Atomic\TLiteralInt;
|
|
|
|
use Psalm\Type\Atomic\TLiteralString;
|
|
|
|
use Psalm\Type\Atomic\TFloat;
|
2018-04-20 16:52:23 +02:00
|
|
|
use Psalm\Type\Atomic\TGenericParam;
|
2018-05-05 23:30:18 +02:00
|
|
|
use Psalm\Type\Atomic\TInt;
|
2018-01-14 18:09:40 +01:00
|
|
|
use Psalm\Type\Atomic\TMixed;
|
|
|
|
use Psalm\Type\Atomic\TNamedObject;
|
|
|
|
use Psalm\Type\Atomic\TNull;
|
2018-08-21 17:40:29 +02:00
|
|
|
use Psalm\Type\Atomic\TSingleLetter;
|
2018-01-14 18:09:40 +01:00
|
|
|
use Psalm\Type\Atomic\TString;
|
|
|
|
|
|
|
|
class ArrayFetchChecker
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @param StatementsChecker $statements_checker
|
|
|
|
* @param PhpParser\Node\Expr\ArrayDimFetch $stmt
|
|
|
|
* @param Context $context
|
|
|
|
*
|
|
|
|
* @return false|null
|
|
|
|
*/
|
|
|
|
public static function analyze(
|
|
|
|
StatementsChecker $statements_checker,
|
|
|
|
PhpParser\Node\Expr\ArrayDimFetch $stmt,
|
|
|
|
Context $context
|
|
|
|
) {
|
|
|
|
$array_var_id = ExpressionChecker::getArrayVarId(
|
|
|
|
$stmt->var,
|
|
|
|
$statements_checker->getFQCLN(),
|
|
|
|
$statements_checker
|
|
|
|
);
|
|
|
|
|
|
|
|
$keyed_array_var_id = ExpressionChecker::getArrayVarId(
|
|
|
|
$stmt,
|
|
|
|
$statements_checker->getFQCLN(),
|
|
|
|
$statements_checker
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($stmt->dim && ExpressionChecker::analyze($statements_checker, $stmt->dim, $context) === false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-05-12 05:14:44 +02:00
|
|
|
$dim_var_id = null;
|
|
|
|
$new_offset_type = null;
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
if ($stmt->dim) {
|
|
|
|
if (isset($stmt->dim->inferredType)) {
|
|
|
|
$used_key_type = $stmt->dim->inferredType;
|
|
|
|
} else {
|
|
|
|
$used_key_type = Type::getMixed();
|
|
|
|
}
|
2018-05-12 05:14:44 +02:00
|
|
|
|
|
|
|
$dim_var_id = ExpressionChecker::getArrayVarId(
|
|
|
|
$stmt->dim,
|
|
|
|
$statements_checker->getFQCLN(),
|
|
|
|
$statements_checker
|
|
|
|
);
|
2018-01-14 18:09:40 +01:00
|
|
|
} else {
|
|
|
|
$used_key_type = Type::getInt();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ExpressionChecker::analyze(
|
|
|
|
$statements_checker,
|
|
|
|
$stmt->var,
|
|
|
|
$context
|
|
|
|
) === false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-03-18 00:28:01 +01:00
|
|
|
if ($keyed_array_var_id
|
2018-04-10 07:27:26 +02:00
|
|
|
&& $context->hasVariable($keyed_array_var_id)
|
2018-03-18 00:28:01 +01:00
|
|
|
&& !$context->vars_in_scope[$keyed_array_var_id]->possibly_undefined
|
|
|
|
) {
|
2018-02-17 17:24:08 +01:00
|
|
|
$stmt->inferredType = clone $context->vars_in_scope[$keyed_array_var_id];
|
2018-02-17 17:36:20 +01:00
|
|
|
|
2018-02-17 17:24:08 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
if (isset($stmt->var->inferredType)) {
|
|
|
|
$var_type = $stmt->var->inferredType;
|
|
|
|
|
|
|
|
if ($var_type->isNull()) {
|
|
|
|
if (!$context->inside_isset) {
|
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new NullArrayAccess(
|
|
|
|
'Cannot access array value on null variable ' . $array_var_id,
|
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
|
|
|
// fall through
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($stmt->inferredType)) {
|
|
|
|
$stmt->inferredType = Type::combineUnionTypes($stmt->inferredType, Type::getNull());
|
|
|
|
} else {
|
|
|
|
$stmt->inferredType = Type::getNull();
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$stmt->inferredType = self::getArrayAccessTypeGivenOffset(
|
|
|
|
$statements_checker,
|
|
|
|
$stmt,
|
|
|
|
$stmt->var->inferredType,
|
|
|
|
$used_key_type,
|
|
|
|
false,
|
|
|
|
$array_var_id,
|
|
|
|
null,
|
|
|
|
$context->inside_isset
|
|
|
|
);
|
2018-05-12 05:14:44 +02:00
|
|
|
|
|
|
|
if ($context->inside_isset
|
|
|
|
&& $stmt->dim
|
|
|
|
&& isset($stmt->dim->inferredType)
|
|
|
|
&& $stmt->var->inferredType->hasArray()
|
|
|
|
&& ($stmt->var instanceof PhpParser\Node\Expr\ClassConstFetch
|
|
|
|
|| $stmt->var instanceof PhpParser\Node\Expr\ConstFetch)
|
|
|
|
) {
|
|
|
|
/** @var TArray|ObjectLike */
|
|
|
|
$array_type = $stmt->var->inferredType->getTypes()['array'];
|
|
|
|
|
|
|
|
if ($array_type instanceof TArray) {
|
|
|
|
$const_array_key_type = $array_type->type_params[0];
|
|
|
|
} else {
|
|
|
|
$const_array_key_type = $array_type->getGenericKeyType();
|
|
|
|
}
|
|
|
|
|
2018-05-12 17:17:41 +02:00
|
|
|
if ($dim_var_id && !$const_array_key_type->isMixed() && !$stmt->dim->inferredType->isMixed()) {
|
2018-05-12 05:14:44 +02:00
|
|
|
$new_offset_type = clone $stmt->dim->inferredType;
|
|
|
|
$const_array_key_atomic_types = $const_array_key_type->getTypes();
|
|
|
|
$project_checker = $statements_checker->getFileChecker()->project_checker;
|
|
|
|
|
|
|
|
foreach ($new_offset_type->getTypes() as $offset_key => $offset_atomic_type) {
|
|
|
|
if ($offset_atomic_type instanceof TString
|
|
|
|
|| $offset_atomic_type instanceof TInt
|
|
|
|
) {
|
2018-05-18 17:02:50 +02:00
|
|
|
if (!isset($const_array_key_atomic_types[$offset_key])
|
|
|
|
&& !TypeChecker::isContainedBy(
|
|
|
|
$project_checker->codebase,
|
|
|
|
new Type\Union([$offset_atomic_type]),
|
|
|
|
$const_array_key_type
|
|
|
|
)
|
|
|
|
) {
|
2018-05-12 05:14:44 +02:00
|
|
|
$new_offset_type->removeType($offset_key);
|
|
|
|
}
|
2018-05-12 17:17:41 +02:00
|
|
|
} elseif (!TypeChecker::isContainedBy(
|
|
|
|
$project_checker->codebase,
|
|
|
|
$const_array_key_type,
|
|
|
|
new Type\Union([$offset_atomic_type])
|
|
|
|
)) {
|
2018-05-12 05:14:44 +02:00
|
|
|
$new_offset_type->removeType($offset_key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
|
2018-01-28 23:28:34 +01:00
|
|
|
if ($keyed_array_var_id && $context->hasVariable($keyed_array_var_id, $statements_checker)) {
|
2018-01-14 18:09:40 +01:00
|
|
|
$stmt->inferredType = $context->vars_in_scope[$keyed_array_var_id];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isset($stmt->inferredType)) {
|
|
|
|
$stmt->inferredType = Type::getMixed();
|
2018-03-17 22:35:36 +01:00
|
|
|
} else {
|
|
|
|
if ($stmt->inferredType->possibly_undefined && !$context->inside_isset && !$context->inside_unset) {
|
2018-04-17 20:06:27 +02:00
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new PossiblyUndefinedArrayOffset(
|
|
|
|
'Possibly undefined array key ' . $keyed_array_var_id,
|
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
|
|
|
return false;
|
2018-03-17 22:35:36 +01:00
|
|
|
}
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
|
2018-05-12 17:19:31 +02:00
|
|
|
if ($context->inside_isset && $dim_var_id && $new_offset_type && $new_offset_type->getTypes()) {
|
2018-05-12 05:14:44 +02:00
|
|
|
$context->vars_in_scope[$dim_var_id] = $new_offset_type;
|
|
|
|
}
|
|
|
|
|
2018-04-11 20:19:42 +02:00
|
|
|
if ($keyed_array_var_id && !$context->inside_isset) {
|
2018-04-10 07:27:26 +02:00
|
|
|
$context->vars_in_scope[$keyed_array_var_id] = $stmt->inferredType;
|
|
|
|
$context->vars_possibly_in_scope[$keyed_array_var_id] = true;
|
|
|
|
|
|
|
|
// reference the variable too
|
|
|
|
$context->hasVariable($keyed_array_var_id, $statements_checker);
|
|
|
|
}
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param Type\Union $array_type
|
|
|
|
* @param Type\Union $offset_type
|
|
|
|
* @param bool $in_assignment
|
2018-02-18 23:55:11 +01:00
|
|
|
* @param null|string $array_var_id
|
2018-01-14 18:09:40 +01:00
|
|
|
* @param bool $inside_isset
|
|
|
|
*
|
|
|
|
* @return Type\Union
|
|
|
|
*/
|
|
|
|
public static function getArrayAccessTypeGivenOffset(
|
|
|
|
StatementsChecker $statements_checker,
|
|
|
|
PhpParser\Node\Expr\ArrayDimFetch $stmt,
|
|
|
|
Type\Union $array_type,
|
|
|
|
Type\Union $offset_type,
|
|
|
|
$in_assignment,
|
|
|
|
$array_var_id,
|
|
|
|
Type\Union $replacement_type = null,
|
|
|
|
$inside_isset = false
|
|
|
|
) {
|
|
|
|
$project_checker = $statements_checker->getFileChecker()->project_checker;
|
2018-02-01 06:50:01 +01:00
|
|
|
$codebase = $project_checker->codebase;
|
2018-01-14 18:09:40 +01:00
|
|
|
|
|
|
|
$has_array_access = false;
|
|
|
|
$non_array_types = [];
|
|
|
|
|
|
|
|
$has_valid_offset = false;
|
2018-02-06 17:27:01 +01:00
|
|
|
$expected_offset_types = [];
|
2018-01-14 18:09:40 +01:00
|
|
|
|
|
|
|
$key_value = null;
|
|
|
|
|
|
|
|
if ($stmt->dim instanceof PhpParser\Node\Scalar\String_
|
|
|
|
|| $stmt->dim instanceof PhpParser\Node\Scalar\LNumber
|
|
|
|
) {
|
|
|
|
$key_value = $stmt->dim->value;
|
2018-05-05 23:30:18 +02:00
|
|
|
} elseif (isset($stmt->dim->inferredType)) {
|
|
|
|
foreach ($stmt->dim->inferredType->getTypes() as $possible_value_type) {
|
|
|
|
if ($possible_value_type instanceof TLiteralString
|
|
|
|
|| $possible_value_type instanceof TLiteralInt
|
|
|
|
) {
|
2018-05-18 17:02:50 +02:00
|
|
|
if ($key_value !== null) {
|
2018-05-05 23:30:18 +02:00
|
|
|
$key_value = null;
|
|
|
|
break;
|
|
|
|
}
|
2018-05-18 17:02:50 +02:00
|
|
|
|
|
|
|
$key_value = $possible_value_type->value;
|
2018-05-05 23:30:18 +02:00
|
|
|
} elseif ($possible_value_type instanceof TString
|
|
|
|
|| $possible_value_type instanceof TInt
|
|
|
|
) {
|
|
|
|
$key_value = null;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
$array_access_type = null;
|
|
|
|
|
|
|
|
if ($offset_type->isNull()) {
|
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new NullArrayOffset(
|
|
|
|
'Cannot access value on variable ' . $array_var_id . ' using null offset',
|
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
|
|
|
// fall through
|
|
|
|
}
|
|
|
|
|
|
|
|
return Type::getMixed();
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($offset_type->isNullable() && !$offset_type->ignore_nullable_issues && !$inside_isset) {
|
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new PossiblyNullArrayOffset(
|
|
|
|
'Cannot access value on variable ' . $array_var_id
|
|
|
|
. ' using possibly null offset ' . $offset_type,
|
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt->var)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
|
|
|
// fall through
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($array_type->getTypes() as &$type) {
|
|
|
|
if ($type instanceof TNull) {
|
|
|
|
if ($array_type->ignore_nullable_issues) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($in_assignment) {
|
|
|
|
if ($replacement_type) {
|
|
|
|
if ($array_access_type) {
|
|
|
|
$array_access_type = Type::combineUnionTypes($array_access_type, $replacement_type);
|
|
|
|
} else {
|
|
|
|
$array_access_type = clone $replacement_type;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new PossiblyNullArrayAssignment(
|
|
|
|
'Cannot access array value on possibly null variable ' . $array_var_id .
|
|
|
|
' of type ' . $array_type,
|
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
|
|
|
// fall through
|
|
|
|
}
|
|
|
|
|
|
|
|
$array_access_type = new Type\Union([new TEmpty]);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!$inside_isset) {
|
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new PossiblyNullArrayAccess(
|
|
|
|
'Cannot access array value on possibly null variable ' . $array_var_id .
|
|
|
|
' of type ' . $array_type,
|
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
|
|
|
// fall through
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($array_access_type) {
|
|
|
|
$array_access_type = Type::combineUnionTypes($array_access_type, Type::getNull());
|
|
|
|
} else {
|
|
|
|
$array_access_type = Type::getNull();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($type instanceof TArray || $type instanceof ObjectLike) {
|
|
|
|
$has_array_access = true;
|
|
|
|
|
|
|
|
if ($in_assignment
|
|
|
|
&& $type instanceof TArray
|
|
|
|
&& $type->type_params[0]->isEmpty()
|
|
|
|
&& $key_value !== null
|
|
|
|
) {
|
|
|
|
// ok, type becomes an ObjectLike
|
|
|
|
|
|
|
|
$type = new ObjectLike([$key_value => new Type\Union([new TEmpty])]);
|
|
|
|
}
|
|
|
|
|
2018-05-03 19:56:30 +02:00
|
|
|
$offset_type = self::replaceOffsetTypeWithInts($offset_type);
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
if ($type instanceof TArray) {
|
|
|
|
// if we're assigning to an empty array with a key offset, refashion that array
|
|
|
|
if ($in_assignment) {
|
|
|
|
if ($type->type_params[0]->isEmpty()) {
|
|
|
|
$type->type_params[0] = $offset_type;
|
|
|
|
}
|
|
|
|
} elseif (!$type->type_params[0]->isEmpty()) {
|
2018-09-01 02:02:36 +02:00
|
|
|
if ((!TypeChecker::isContainedBy(
|
2018-02-01 06:50:01 +01:00
|
|
|
$project_checker->codebase,
|
2018-01-14 18:09:40 +01:00
|
|
|
$offset_type,
|
2018-09-01 02:02:36 +02:00
|
|
|
$type->type_params[0]->isMixed()
|
|
|
|
? new Type\Union([ new TInt, new TString ])
|
|
|
|
: $type->type_params[0],
|
2018-04-05 19:57:01 +02:00
|
|
|
true,
|
2018-05-18 17:02:50 +02:00
|
|
|
$offset_type->ignore_falsable_issues,
|
|
|
|
$has_scalar_match,
|
|
|
|
$type_coerced,
|
|
|
|
$type_coerced_from_mixed,
|
|
|
|
$to_string_cast,
|
|
|
|
$type_coerced_from_scalar
|
2018-09-01 02:02:36 +02:00
|
|
|
) && !$type_coerced_from_scalar)
|
|
|
|
|| $to_string_cast
|
2018-05-18 17:02:50 +02:00
|
|
|
) {
|
2018-05-05 23:30:18 +02:00
|
|
|
$expected_offset_types[] = $type->type_params[0]->getId();
|
2018-01-14 18:09:40 +01:00
|
|
|
} else {
|
|
|
|
$has_valid_offset = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-18 17:02:50 +02:00
|
|
|
if (!$stmt->dim && $type->count !== null) {
|
|
|
|
$type->count++;
|
2018-05-03 19:56:30 +02:00
|
|
|
}
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
if ($in_assignment && $replacement_type) {
|
|
|
|
$type->type_params[1] = Type::combineUnionTypes(
|
|
|
|
$type->type_params[1],
|
|
|
|
$replacement_type
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$array_access_type) {
|
|
|
|
$array_access_type = $type->type_params[1];
|
|
|
|
} else {
|
|
|
|
$array_access_type = Type::combineUnionTypes(
|
|
|
|
$array_access_type,
|
|
|
|
$type->type_params[1]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-07-12 18:08:15 +02:00
|
|
|
if ($array_access_type->isEmpty()
|
|
|
|
&& !$in_assignment
|
|
|
|
&& !$inside_isset
|
|
|
|
) {
|
2018-01-14 18:09:40 +01:00
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new EmptyArrayAccess(
|
|
|
|
'Cannot access value on empty array variable ' . $array_var_id,
|
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
2018-04-07 00:28:22 +02:00
|
|
|
return Type::getMixed(true);
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!IssueBuffer::isRecording()) {
|
2018-05-03 02:10:08 +02:00
|
|
|
$array_access_type = Type::getMixed(true);
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
}
|
2018-02-07 19:57:45 +01:00
|
|
|
} else {
|
2018-09-01 02:24:50 +02:00
|
|
|
$generic_key_type = $type->getGenericKeyType();
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
if ($key_value !== null) {
|
|
|
|
if (isset($type->properties[$key_value]) || $replacement_type) {
|
|
|
|
$has_valid_offset = true;
|
|
|
|
|
|
|
|
if ($replacement_type) {
|
|
|
|
if (isset($type->properties[$key_value])) {
|
|
|
|
$type->properties[$key_value] = Type::combineUnionTypes(
|
|
|
|
$type->properties[$key_value],
|
|
|
|
$replacement_type
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
$type->properties[$key_value] = $replacement_type;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$array_access_type) {
|
|
|
|
$array_access_type = clone $type->properties[$key_value];
|
|
|
|
} else {
|
|
|
|
$array_access_type = Type::combineUnionTypes(
|
|
|
|
$array_access_type,
|
|
|
|
$type->properties[$key_value]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} elseif ($in_assignment) {
|
|
|
|
$type->properties[$key_value] = new Type\Union([new TEmpty]);
|
|
|
|
|
|
|
|
if (!$array_access_type) {
|
|
|
|
$array_access_type = clone $type->properties[$key_value];
|
|
|
|
} else {
|
|
|
|
$array_access_type = Type::combineUnionTypes(
|
|
|
|
$array_access_type,
|
|
|
|
$type->properties[$key_value]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
2018-05-05 23:30:18 +02:00
|
|
|
if (!$inside_isset || $type->sealed) {
|
|
|
|
$object_like_keys = array_keys($type->properties);
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2018-05-05 23:30:18 +02:00
|
|
|
if (count($object_like_keys) === 1) {
|
|
|
|
$expected_keys_string = '\'' . $object_like_keys[0] . '\'';
|
|
|
|
} else {
|
|
|
|
$last_key = array_pop($object_like_keys);
|
|
|
|
$expected_keys_string = '\'' . implode('\', \'', $object_like_keys) .
|
|
|
|
'\' or \'' . $last_key . '\'';
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2018-05-05 23:30:18 +02:00
|
|
|
$expected_offset_types[] = $expected_keys_string;
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
|
|
|
|
$array_access_type = Type::getMixed();
|
|
|
|
}
|
2018-11-02 04:31:40 +01:00
|
|
|
} else {
|
|
|
|
$key_type = $generic_key_type->isMixed()
|
|
|
|
? new Type\Union([ new TInt, new TString ])
|
|
|
|
: $generic_key_type;
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2018-11-02 04:31:40 +01:00
|
|
|
$is_contained = TypeChecker::isContainedBy(
|
|
|
|
$codebase,
|
|
|
|
$offset_type,
|
|
|
|
$key_type,
|
|
|
|
true,
|
|
|
|
$offset_type->ignore_falsable_issues,
|
|
|
|
$has_scalar_match,
|
|
|
|
$type_coerced,
|
|
|
|
$type_coerced_from_mixed,
|
|
|
|
$to_string_cast,
|
|
|
|
$type_coerced_from_scalar
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($inside_isset && !$is_contained) {
|
|
|
|
$is_contained = TypeChecker::canBeContainedBy(
|
|
|
|
$codebase,
|
|
|
|
$offset_type,
|
|
|
|
$key_type,
|
|
|
|
true,
|
|
|
|
$offset_type->ignore_falsable_issues
|
2018-01-14 18:09:40 +01:00
|
|
|
);
|
2018-11-02 04:31:40 +01:00
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2018-11-02 04:31:40 +01:00
|
|
|
if (($is_contained
|
|
|
|
|| $type_coerced_from_scalar
|
|
|
|
|| $type_coerced_from_mixed
|
|
|
|
|| $in_assignment)
|
|
|
|
&& !$to_string_cast
|
|
|
|
) {
|
|
|
|
if ($replacement_type) {
|
|
|
|
$generic_params = Type::combineUnionTypes(
|
|
|
|
$type->getGenericValueType(),
|
|
|
|
$replacement_type
|
|
|
|
);
|
2018-05-03 19:56:30 +02:00
|
|
|
|
2018-11-02 04:31:40 +01:00
|
|
|
$new_key_type = Type::combineUnionTypes(
|
|
|
|
$generic_key_type,
|
|
|
|
$offset_type
|
|
|
|
);
|
2018-01-14 18:09:40 +01:00
|
|
|
|
2018-11-02 04:31:40 +01:00
|
|
|
$property_count = $type->sealed ? count($type->properties) : null;
|
2018-05-03 19:56:30 +02:00
|
|
|
|
2018-11-02 04:31:40 +01:00
|
|
|
$type = new TArray([
|
|
|
|
$new_key_type,
|
|
|
|
$generic_params,
|
|
|
|
]);
|
|
|
|
|
|
|
|
if (!$stmt->dim && $property_count) {
|
|
|
|
++$property_count;
|
|
|
|
$type->count = $property_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$array_access_type) {
|
|
|
|
$array_access_type = clone $generic_params;
|
|
|
|
} else {
|
|
|
|
$array_access_type = Type::combineUnionTypes(
|
|
|
|
$array_access_type,
|
|
|
|
$generic_params
|
|
|
|
);
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
} else {
|
2018-11-02 04:31:40 +01:00
|
|
|
if (!$array_access_type) {
|
|
|
|
$array_access_type = $type->getGenericValueType();
|
|
|
|
} else {
|
|
|
|
$array_access_type = Type::combineUnionTypes(
|
|
|
|
$array_access_type,
|
|
|
|
$type->getGenericValueType()
|
|
|
|
);
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
2018-11-02 04:31:40 +01:00
|
|
|
|
|
|
|
$has_valid_offset = true;
|
2018-01-14 18:09:40 +01:00
|
|
|
} else {
|
2018-11-02 04:31:40 +01:00
|
|
|
if (!$inside_isset || $type->sealed) {
|
|
|
|
$expected_offset_types[] = (string)$generic_key_type->getId();
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
|
2018-11-02 04:31:40 +01:00
|
|
|
$array_access_type = Type::getMixed();
|
2018-05-05 23:50:19 +02:00
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($type instanceof TString) {
|
2018-01-31 22:08:52 +01:00
|
|
|
if ($in_assignment && $replacement_type) {
|
|
|
|
if ($replacement_type->isMixed()) {
|
2018-05-30 22:19:18 +02:00
|
|
|
$codebase->analyzer->incrementMixedCount($statements_checker->getFilePath());
|
2018-01-31 22:08:52 +01:00
|
|
|
|
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new MixedStringOffsetAssignment(
|
|
|
|
'Right-hand-side of string offset assignment cannot be mixed',
|
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
|
|
|
// fall through
|
|
|
|
}
|
|
|
|
} else {
|
2018-05-30 22:19:18 +02:00
|
|
|
$codebase->analyzer->incrementNonMixedCount($statements_checker->getFilePath());
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-21 17:40:29 +02:00
|
|
|
if ($type instanceof TSingleLetter) {
|
|
|
|
$valid_offset_type = Type::getInt(false, 0);
|
|
|
|
} elseif ($type instanceof TLiteralString) {
|
|
|
|
$valid_offsets = [];
|
|
|
|
|
|
|
|
for ($i = 0, $l = strlen($type->value); $i < $l; $i++) {
|
|
|
|
$valid_offsets[] = new TLiteralInt($i);
|
|
|
|
}
|
|
|
|
|
|
|
|
$valid_offset_type = new Type\Union($valid_offsets);
|
|
|
|
} else {
|
|
|
|
$valid_offset_type = Type::getInt();
|
|
|
|
}
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
if (!TypeChecker::isContainedBy(
|
2018-02-01 06:50:01 +01:00
|
|
|
$project_checker->codebase,
|
2018-01-14 18:09:40 +01:00
|
|
|
$offset_type,
|
2018-08-21 17:40:29 +02:00
|
|
|
$valid_offset_type,
|
2018-01-14 18:09:40 +01:00
|
|
|
true
|
|
|
|
)) {
|
2018-08-21 17:40:29 +02:00
|
|
|
$expected_offset_types[] = $valid_offset_type->getId();
|
2018-01-14 18:09:40 +01:00
|
|
|
} else {
|
|
|
|
$has_valid_offset = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$array_access_type) {
|
2018-08-21 17:40:29 +02:00
|
|
|
$array_access_type = Type::getSingleLetter();
|
2018-01-14 18:09:40 +01:00
|
|
|
} else {
|
|
|
|
$array_access_type = Type::combineUnionTypes(
|
|
|
|
$array_access_type,
|
2018-08-21 17:40:29 +02:00
|
|
|
Type::getSingleLetter()
|
2018-01-14 18:09:40 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-05-03 19:56:30 +02:00
|
|
|
if ($type instanceof TMixed || $type instanceof TGenericParam || $type instanceof TEmpty) {
|
2018-05-30 22:19:18 +02:00
|
|
|
$codebase->analyzer->incrementMixedCount($statements_checker->getFilePath());
|
2018-01-31 22:08:52 +01:00
|
|
|
|
2018-05-31 21:07:03 +02:00
|
|
|
if (!$inside_isset) {
|
|
|
|
if ($in_assignment) {
|
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new MixedArrayAssignment(
|
|
|
|
'Cannot access array value on mixed variable ' . $array_var_id,
|
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
|
|
|
// fall through
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new MixedArrayAccess(
|
|
|
|
'Cannot access array value on mixed variable ' . $array_var_id,
|
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
|
|
|
// fall through
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$array_access_type = Type::getMixed();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2018-05-30 22:19:18 +02:00
|
|
|
$codebase->analyzer->incrementNonMixedCount($statements_checker->getFilePath());
|
2018-01-31 22:08:52 +01:00
|
|
|
|
2018-04-16 22:03:04 +02:00
|
|
|
if ($type instanceof Type\Atomic\TFalse && $array_type->ignore_falsable_issues) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
if ($type instanceof TNamedObject) {
|
|
|
|
if (strtolower($type->value) !== 'simplexmlelement'
|
2018-06-30 21:29:37 +02:00
|
|
|
&& strtolower($type->value) !== 'arrayaccess'
|
|
|
|
&& (($codebase->classExists($type->value)
|
|
|
|
&& !$codebase->classImplements($type->value, 'ArrayAccess'))
|
|
|
|
|| ($codebase->interfaceExists($type->value)
|
|
|
|
&& !$codebase->interfaceExtends($type->value, 'ArrayAccess'))
|
|
|
|
)
|
2018-01-14 18:09:40 +01:00
|
|
|
) {
|
|
|
|
$non_array_types[] = (string)$type;
|
|
|
|
} else {
|
|
|
|
$array_access_type = Type::getMixed();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$non_array_types[] = (string)$type;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($non_array_types) {
|
|
|
|
if ($has_array_access) {
|
|
|
|
if ($in_assignment) {
|
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new PossiblyInvalidArrayAssignment(
|
|
|
|
'Cannot access array value on non-array variable ' .
|
|
|
|
$array_var_id . ' of type ' . $non_array_types[0],
|
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)
|
|
|
|
) {
|
|
|
|
// do nothing
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new PossiblyInvalidArrayAccess(
|
|
|
|
'Cannot access array value on non-array variable ' .
|
|
|
|
$array_var_id . ' of type ' . $non_array_types[0],
|
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)
|
|
|
|
) {
|
|
|
|
// do nothing
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if ($in_assignment) {
|
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new InvalidArrayAssignment(
|
|
|
|
'Cannot access array value on non-array variable ' .
|
|
|
|
$array_var_id . ' of type ' . $non_array_types[0],
|
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
|
|
|
// fall through
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new InvalidArrayAccess(
|
|
|
|
'Cannot access array value on non-array variable ' .
|
|
|
|
$array_var_id . ' of type ' . $non_array_types[0],
|
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
|
|
|
// fall through
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$array_access_type = Type::getMixed();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($offset_type->isMixed()) {
|
2018-05-30 22:19:18 +02:00
|
|
|
$codebase->analyzer->incrementMixedCount($statements_checker->getFilePath());
|
2018-01-31 22:08:52 +01:00
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new MixedArrayOffset(
|
|
|
|
'Cannot access value on variable ' . $array_var_id . ' using mixed offset',
|
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
|
|
|
// fall through
|
|
|
|
}
|
2018-01-31 22:08:52 +01:00
|
|
|
} else {
|
2018-05-30 22:19:18 +02:00
|
|
|
$codebase->analyzer->incrementNonMixedCount($statements_checker->getFilePath());
|
2018-01-31 22:08:52 +01:00
|
|
|
|
2018-02-06 17:27:01 +01:00
|
|
|
if ($expected_offset_types) {
|
|
|
|
$invalid_offset_type = $expected_offset_types[0];
|
|
|
|
|
2018-05-05 23:30:18 +02:00
|
|
|
$used_offset = 'using a ' . $offset_type->getId() . ' offset';
|
2018-02-06 17:27:01 +01:00
|
|
|
|
|
|
|
if ($key_value !== null) {
|
|
|
|
$used_offset = 'using offset value of '
|
|
|
|
. (is_int($key_value) ? $key_value : '\'' . $key_value . '\'');
|
|
|
|
}
|
2018-01-31 22:08:52 +01:00
|
|
|
|
|
|
|
if ($has_valid_offset) {
|
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new PossiblyInvalidArrayOffset(
|
2018-02-06 17:27:01 +01:00
|
|
|
'Cannot access value on variable ' . $array_var_id . ' ' . $used_offset
|
|
|
|
. ', expecting ' . $invalid_offset_type,
|
2018-01-31 22:08:52 +01:00
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
|
|
|
// fall through
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new InvalidArrayOffset(
|
2018-02-06 17:27:01 +01:00
|
|
|
'Cannot access value on variable ' . $array_var_id . ' ' . $used_offset
|
|
|
|
. ', expecting ' . $invalid_offset_type,
|
2018-01-31 22:08:52 +01:00
|
|
|
new CodeLocation($statements_checker->getSource(), $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
|
|
|
// fall through
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($array_access_type === null) {
|
|
|
|
throw new \InvalidArgumentException('This is a bad place');
|
|
|
|
}
|
|
|
|
|
2018-03-15 19:25:04 +01:00
|
|
|
if ($in_assignment) {
|
|
|
|
$array_type->bustCache();
|
|
|
|
}
|
|
|
|
|
2018-01-14 18:09:40 +01:00
|
|
|
return $array_access_type;
|
|
|
|
}
|
2018-05-03 19:56:30 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Type\Union
|
|
|
|
*/
|
|
|
|
public static function replaceOffsetTypeWithInts(Type\Union $offset_type)
|
|
|
|
{
|
2018-05-18 17:02:50 +02:00
|
|
|
$offset_string_types = $offset_type->getLiteralStrings();
|
2018-05-03 19:56:30 +02:00
|
|
|
|
2018-05-18 17:02:50 +02:00
|
|
|
$offset_type = clone $offset_type;
|
2018-05-03 19:56:30 +02:00
|
|
|
|
2018-05-18 17:02:50 +02:00
|
|
|
foreach ($offset_string_types as $key => $offset_string_type) {
|
|
|
|
if (preg_match('/^(0|[1-9][0-9]*)$/', $offset_string_type->value)) {
|
|
|
|
$offset_type->addType(new Type\Atomic\TLiteralInt((int) $offset_string_type->value));
|
|
|
|
$offset_type->removeType($key);
|
2018-05-03 19:56:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $offset_type;
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|