1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-21 21:31:13 +01:00

Fix bugs in array key checks

This commit is contained in:
Matthew Brown 2016-09-09 18:36:35 -04:00
parent 9bb0b18a62
commit f455851f89
6 changed files with 57 additions and 21 deletions

View File

@ -241,7 +241,7 @@ class FunctionChecker extends FunctionLikeChecker
else {
$inner_type = new Type\Union($closure_return_types);
return new Type\Union([new Type\Generic('array', [$inner_type])]);
return new Type\Union([new Type\Generic('array', [Type::getInt(), $inner_type])]);
}
}
elseif ($call_args[0]->value instanceof PhpParser\Node\Scalar\String_) {
@ -250,7 +250,7 @@ class FunctionChecker extends FunctionLikeChecker
if (isset($call_map[$mapped_function_id][0])) {
if ($call_map[$mapped_function_id][0]) {
$mapped_function_return = Type::parseString($call_map[$mapped_function_id][0]);
return new Type\Union([new Type\Generic('array', [$mapped_function_return])]);
return new Type\Union([new Type\Generic('array', [Type::getInt(), $mapped_function_return])]);
}
}
else {
@ -272,7 +272,7 @@ class FunctionChecker extends FunctionLikeChecker
$inner_value_types = [];
$inner_key_types = [];
foreach ($call_args as $call_arg) {
foreach ($call_args as $offset => $call_arg) {
if (!isset($call_arg->value->inferredType)) {
return Type::getArray();
}
@ -290,7 +290,7 @@ class FunctionChecker extends FunctionLikeChecker
$inner_value_types = array_merge(array_values($type_part->type_params[1]->types), $inner_value_types);
}
if ($inner_types) {
if ($inner_value_types) {
return new Type\Union([
new Type\Generic('array',
[
@ -306,7 +306,7 @@ class FunctionChecker extends FunctionLikeChecker
}
if ($call_map_key === 'explode') {
return Type::parseString('array<string>');
return Type::parseString('array<int, string>');
}
if (!isset($call_map[$call_map_key]) || !$call_map[$call_map_key][0]) {

View File

@ -683,8 +683,6 @@ class StatementsChecker
$context->update($old_else_context, $else_context, $has_leaving_statements, array_keys($negatable_if_types), $updated_vars);
}
if (!$has_ending_statements) {
$vars = array_diff_key($else_context->vars_possibly_in_scope, $context->vars_possibly_in_scope);
@ -1793,7 +1791,7 @@ class StatementsChecker
}
if ($value_index) {
$key_type_part = $return_type->type_params[$key_index];
$key_type_part = $return_type->type_params[0];
if (!$key_type) {
$key_type = $key_type_part;
@ -2286,7 +2284,7 @@ class StatementsChecker
if ($type->type_params[1]->isEmpty()) {
// boil this down to a regular array
if ($assignment_value_type->isMixed()) {
return Type::getArray();
return Type::getArray()->types['array'];
}
$type->type_params[0] = $assignment_key_type;
@ -3496,6 +3494,10 @@ class StatementsChecker
}
}
if (!isset($stmt->inferredType)) {
$stmt->inferredType = Type::getMixed();
}
if (!$key_type) {
$key_type = new Type\Union([
new Type\Atomic('int'),
@ -3510,7 +3512,10 @@ class StatementsChecker
if (isset($stmt->dim->inferredType) && $key_type) {
foreach ($stmt->dim->inferredType->types as $at) {
if (!$at->isIn($key_type)) {
if ($at->isMixed()) {
// @todo emit issue
}
elseif (!$at->isIn($key_type)) {
$var_id = self::getVarId($stmt->var);
if (IssueBuffer::accepts(

View File

@ -727,6 +727,11 @@ class TypeChecker
$line_number,
$suppressed_issues
);
// special case if result is just a simple array
if ((string) $result_type === 'array') {
$result_type = Type::getArray();
}
}
//echo((string) $new_types[$key] . ' and ' . (isset($existing_types[$key]) ? (string) $existing_types[$key] : '') . ' => ' . $result_type . PHP_EOL);

View File

@ -27,6 +27,10 @@ abstract class Type
if (count($type_tokens) === 1) {
$type_tokens[0] = self::fixScalarTerms($type_tokens[0]);
if ($type_tokens[0] === 'array') {
return Type::getArray();
}
return new Union([new Atomic($type_tokens[0])]);
}
@ -547,7 +551,10 @@ abstract class Type
}
}
else {
$key_types[$type->value][(string) $type] = null;
if ($type->value === 'array') {
throw new \InvalidArgumentException('Cannot have a non-generic array');
}
$value_types[$type->value][(string) $type] = null;
}
}
@ -561,15 +568,21 @@ abstract class Type
$new_types = [];
foreach ($value_types as $generic_type => $value_type) {
$key_type = $key_types[$generic_type];
$key_type = isset($key_types[$generic_type]) ? $key_types[$generic_type] : [];
$expanded_key_types = [];
foreach ($key_type as $expandable_key_type) {
$expanded_key_types = array_merge($expanded_key_types, array_values($expandable_key_type->types));
}
if (count($value_type) === 1) {
$value_type_param = array_values($value_type)[0];
$generic_type_params = [$value_type_param];
// if we're continuing, also add the correspoinding key type param if it exists
if (count($key_type) === 1) {
array_unshift($generic_type_params, array_values($key_type)[0]);
if ($expanded_key_types) {
array_unshift($generic_type_params, self::combineTypes($expanded_key_types));
}
$new_types[] = $value_type_param ? new Generic($generic_type, $generic_type_params) : new Atomic($generic_type);
@ -582,12 +595,6 @@ abstract class Type
$expanded_value_types = array_merge($expanded_value_types, array_values($expandable_value_type->types));
}
$expanded_key_types = [];
foreach ($key_type as $expandable_key_type) {
$expanded_key_types = array_merge($expanded_key_types, array_values($expandable_key_type->types));
}
$generic_type_params = [self::combineTypes($expanded_value_types)];
if ($expanded_key_types) {

View File

@ -74,10 +74,18 @@ class ParseTree
case ',':
$current_parent = $current_leaf->parent;
if (!$current_parent || $current_parent->value !== self::GENERIC) {
if (!$current_parent) {
throw new \InvalidArgumentException('Cannot parse comma in non-generic type');
}
if ($current_parent->value !== self::GENERIC) {
if (!$current_parent->parent->value) {
throw new \InvalidArgumentException('Cannot parse comma in non-generic type');
}
$current_leaf = $current_leaf->parent;
}
break;
case '|':

View File

@ -87,6 +87,17 @@ class TypeCombinationTest extends PHPUnit_Framework_TestCase
);
}
public function testArrayMixedOrStringKeys()
{
$this->assertEquals(
'array<mixed,string>',
(string) Type::combineTypes([
self::getAtomic('array<int|string,string>'),
self::getAtomic('array<mixed,string>')
])
);
}
public function testArrayMixedOrEmpty()
{
$this->assertEquals(