1
0
mirror of https://github.com/danog/psalm.git synced 2024-11-30 04:39:00 +01:00
psalm/tests/ArrayAssignmentTest.php

514 lines
17 KiB
PHP
Raw Normal View History

2016-09-12 06:02:50 +02:00
<?php
namespace Psalm\Tests;
use PhpParser\ParserFactory;
use PHPUnit_Framework_TestCase;
2016-11-02 07:29:00 +01:00
use Psalm\Context;
use Psalm\Type;
2016-09-12 06:02:50 +02:00
class ArrayAssignmentTest extends PHPUnit_Framework_TestCase
{
2016-11-02 07:29:00 +01:00
protected static $parser;
2016-09-12 06:02:50 +02:00
public static function setUpBeforeClass()
{
2016-11-02 07:29:00 +01:00
self::$parser = (new ParserFactory)->create(ParserFactory::PREFER_PHP7);
2016-09-12 06:02:50 +02:00
$config = \Psalm\Config::getInstance();
$config->throw_exception = true;
$config->use_docblock_types = true;
}
public function setUp()
{
\Psalm\Checker\FileChecker::clearCache();
}
public function testImplicitIntArrayCreation()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-12 06:02:50 +02:00
$foo = [];
$foo[] = "hello";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
2016-11-13 17:24:25 +01:00
$this->assertEquals('array<int, string>', (string) $context->vars_in_scope['$foo']);
2016-09-12 06:02:50 +02:00
}
public function testImplicit2DIntArrayCreation()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-12 06:02:50 +02:00
$foo = [];
$foo[][] = "hello";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
2016-11-13 17:24:25 +01:00
$this->assertEquals('array<int, array<int, string>>', (string) $context->vars_in_scope['$foo']);
2016-09-12 06:02:50 +02:00
}
public function testImplicit3DIntArrayCreation()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-12 06:02:50 +02:00
$foo = [];
$foo[][][] = "hello";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
2016-11-13 17:24:25 +01:00
$this->assertEquals('array<int, array<int, array<int, string>>>', (string) $context->vars_in_scope['$foo']);
2016-09-12 06:02:50 +02:00
}
public function testImplicit4DIntArrayCreation()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-12 06:02:50 +02:00
$foo = [];
$foo[][][][] = "hello";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
2016-11-02 07:29:00 +01:00
$this->assertEquals(
2016-11-13 17:24:25 +01:00
'array<int, array<int, array<int, array<int, string>>>>',
2016-11-02 07:29:00 +01:00
(string) $context->vars_in_scope['$foo']
);
2016-09-12 06:02:50 +02:00
}
2016-09-15 20:21:51 +02:00
public function testImplicitIndexedIntArrayCreation()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-15 20:21:51 +02:00
$foo = [];
$foo[0] = "hello";
$foo[1] = "hello";
$foo[2] = "hello";
$bar = [0, 1, 2];
$bat = [];
foreach ($foo as $i => $text) {
$bat[$text] = $bar[$i];
}
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
2016-11-13 17:24:25 +01:00
$this->assertEquals('array<int, string>', (string) $context->vars_in_scope['$foo']);
$this->assertEquals('array<int, int>', (string) $context->vars_in_scope['$bar']);
$this->assertEquals('array<string, int>', (string) $context->vars_in_scope['$bat']);
2016-09-15 20:21:51 +02:00
}
2016-09-12 06:02:50 +02:00
public function testImplicitStringArrayCreation()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-12 06:02:50 +02:00
$foo = [];
$foo["bar"] = "hello";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
$this->assertEquals('array{bar:string}', (string) $context->vars_in_scope['$foo']);
2016-10-15 19:10:05 +02:00
$this->assertEquals('string', (string) $context->vars_in_scope['$foo[\'bar\']']);
2016-09-12 06:02:50 +02:00
}
public function testImplicit2DStringArrayCreation()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-12 06:02:50 +02:00
$foo = [];
$foo["bar"]["baz"] = "hello";
');
// check array access of baz on foo
// with some extra data if we need to create an array for type $foo["bar"],
2016-09-12 06:02:50 +02:00
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
$this->assertEquals('array{bar:array{baz:string}}', (string) $context->vars_in_scope['$foo']);
2016-10-15 19:10:05 +02:00
$this->assertEquals('string', (string) $context->vars_in_scope['$foo[\'bar\'][\'baz\']']);
2016-09-12 06:02:50 +02:00
}
public function testImplicit3DStringArrayCreation()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-12 06:02:50 +02:00
$foo = [];
$foo["bar"]["baz"]["bat"] = "hello";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
$this->assertEquals('array{bar:array{baz:array{bat:string}}}', (string) $context->vars_in_scope['$foo']);
2016-10-15 19:10:05 +02:00
$this->assertEquals('string', (string) $context->vars_in_scope['$foo[\'bar\'][\'baz\'][\'bat\']']);
2016-09-12 06:02:50 +02:00
}
public function testImplicit4DStringArrayCreation()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-12 06:02:50 +02:00
$foo = [];
$foo["bar"]["baz"]["bat"]["bap"] = "hello";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
2016-11-02 07:29:00 +01:00
$this->assertEquals(
'array{bar:array{baz:array{bat:array{bap:string}}}}',
(string) $context->vars_in_scope['$foo']
);
2016-10-15 19:10:05 +02:00
$this->assertEquals('string', (string) $context->vars_in_scope['$foo[\'bar\'][\'baz\'][\'bat\'][\'bap\']']);
2016-09-12 06:02:50 +02:00
}
2016-09-13 01:31:16 +02:00
2016-10-03 04:01:01 +02:00
public function test2Step2DStringArrayCreation()
2016-09-13 01:31:16 +02:00
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-13 01:31:16 +02:00
$foo = ["bar" => []];
$foo["bar"]["baz"] = "hello";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
$this->assertEquals('array{bar:array{baz:string}}', (string) $context->vars_in_scope['$foo']);
2016-10-15 19:10:05 +02:00
$this->assertEquals('string', (string) $context->vars_in_scope['$foo[\'bar\'][\'baz\']']);
2016-09-13 01:31:16 +02:00
}
2016-10-03 04:01:01 +02:00
public function test2StepImplicit3DStringArrayCreation()
2016-09-13 01:31:16 +02:00
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-13 01:31:16 +02:00
$foo = ["bar" => []];
$foo["bar"]["baz"]["bat"] = "hello";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
$this->assertEquals('array{bar:array{baz:array{bat:string}}}', (string) $context->vars_in_scope['$foo']);
2016-09-13 01:31:16 +02:00
}
2016-09-13 01:44:46 +02:00
public function testConflictingTypes()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-13 01:44:46 +02:00
$foo = [
"bar" => ["a" => "b"],
"baz" => [1]
];
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
2016-11-13 17:24:25 +01:00
$this->assertEquals('array{bar:array{a:string}, baz:array<int, int>}', (string) $context->vars_in_scope['$foo']);
2016-10-03 04:01:01 +02:00
}
public function testImplicitObjectLikeCreation()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-10-03 04:01:01 +02:00
$foo = [
"bar" => 1,
];
$foo["baz"] = "a";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
2016-11-13 17:24:25 +01:00
$this->assertEquals('array{bar:int, baz:string}', (string) $context->vars_in_scope['$foo']);
2016-09-13 01:44:46 +02:00
}
public function testConflictingTypesWithAssignment()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-13 01:44:46 +02:00
$foo = [
"bar" => ["a" => "b"],
"baz" => [1]
];
$foo["bar"]["bam"]["baz"] = "hello";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
2016-11-02 07:29:00 +01:00
$this->assertEquals(
2016-11-13 17:24:25 +01:00
'array{bar:array{a:string, bam:array{baz:string}}, baz:array<int, int>}',
2016-11-02 07:29:00 +01:00
(string) $context->vars_in_scope['$foo']
);
2016-09-13 01:44:46 +02:00
}
2016-09-13 06:18:41 +02:00
public function testConflictingTypesWithAssignment2()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-13 06:18:41 +02:00
$foo = [];
$foo["a"] = "hello";
$foo["b"][] = "goodbye";
2016-09-22 01:15:09 +02:00
$bar = $foo["a"];
2016-09-13 06:18:41 +02:00
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
2016-11-13 17:24:25 +01:00
$this->assertEquals('array{a:string, b:array<int, string>}', (string) $context->vars_in_scope['$foo']);
2016-10-15 19:10:05 +02:00
$this->assertEquals('string', (string) $context->vars_in_scope['$foo[\'a\']']);
2016-11-13 17:24:25 +01:00
$this->assertEquals('array<int, string>', (string) $context->vars_in_scope['$foo[\'b\']']);
2016-10-15 19:10:05 +02:00
$this->assertEquals('string', (string) $context->vars_in_scope['$bar']);
2016-09-13 06:18:41 +02:00
}
2016-09-13 01:44:46 +02:00
2016-09-13 06:18:41 +02:00
public function testConflictingTypesWithAssignment3()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-13 06:18:41 +02:00
$foo = [];
$foo["a"] = "hello";
$foo["b"]["c"]["d"] = "goodbye";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
2016-11-13 17:24:25 +01:00
$this->assertEquals('array{a:string, b:array{c:array{d:string}}}', (string) $context->vars_in_scope['$foo']);
2016-09-13 06:18:41 +02:00
}
2016-09-22 01:15:09 +02:00
2016-10-03 06:44:05 +02:00
public function testNestedObjectLikeAssignment()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-10-03 06:44:05 +02:00
$foo = [];
$foo["a"]["b"] = "hello";
$foo["a"]["c"] = 1;
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
2016-11-13 17:24:25 +01:00
$this->assertEquals('array{a:array{b:string, c:int}}', (string) $context->vars_in_scope['$foo']);
2016-10-03 06:44:05 +02:00
}
public function testConditionalObjectLikeAssignment()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-10-03 06:44:05 +02:00
$foo = ["a" => "hello"];
if (rand(0, 10) === 5) {
$foo["b"] = 1;
}
else {
$foo["b"] = 2;
}
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
2016-11-13 17:24:25 +01:00
$this->assertEquals('array{a:string, b:int}', (string) $context->vars_in_scope['$foo']);
2016-10-03 06:44:05 +02:00
}
2016-09-22 01:15:09 +02:00
public function testIssetKeyedOffset()
{
$file_checker = new \Psalm\Checker\FileChecker(
'somefile.php',
2016-11-02 07:29:00 +01:00
self::$parser->parse('<?php
2016-09-22 01:15:09 +02:00
if (!isset($foo["a"])) {
$foo["a"] = "hello";
}
')
);
$context = new Context('somefile.php');
2016-10-15 19:10:05 +02:00
$context->vars_in_scope['$foo'] = \Psalm\Type::getArray();
2016-09-22 01:15:09 +02:00
$file_checker->check(true, true, $context);
2016-10-15 19:10:05 +02:00
$this->assertEquals('mixed', (string) $context->vars_in_scope['$foo[\'a\']']);
2016-09-22 18:26:24 +02:00
}
public function testConditionalAssignment()
{
$file_checker = new \Psalm\Checker\FileChecker(
'somefile.php',
2016-11-02 07:29:00 +01:00
self::$parser->parse('<?php
2016-09-22 18:26:24 +02:00
if ($b) {
$foo["a"] = "hello";
}
')
);
2016-11-02 07:29:00 +01:00
2016-09-22 18:26:24 +02:00
$context = new Context('somefile.php');
2016-10-15 19:10:05 +02:00
$context->vars_in_scope['$b'] = \Psalm\Type::getBool();
$context->vars_in_scope['$foo'] = \Psalm\Type::getArray();
2016-09-22 18:26:24 +02:00
$file_checker->check(true, true, $context);
2016-10-15 19:10:05 +02:00
$this->assertFalse(isset($context->vars_in_scope['$foo[\'a\']']));
2016-09-22 19:45:47 +02:00
}
public function testImplementsArrayAccess()
{
2016-11-02 07:29:00 +01:00
$stmts = self::$parser->parse('<?php
2016-09-22 19:45:47 +02:00
class A implements \ArrayAccess { }
$a = new A();
$a["bar"] = "cool";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
2016-10-15 19:10:05 +02:00
$this->assertEquals('A', (string) $context->vars_in_scope['$a']);
$this->assertFalse(isset($context->vars_in_scope['$a[\'bar\']']));
2016-09-22 01:15:09 +02:00
}
2016-10-15 19:11:08 +02:00
public function testConditionalCheck()
{
$file_checker = new \Psalm\Checker\FileChecker(
'somefile.php',
2016-11-02 07:29:00 +01:00
self::$parser->parse('<?php
2016-10-15 19:11:08 +02:00
/**
* @param array{b:string} $a
2016-10-15 19:11:08 +02:00
* @return null|string
*/
function foo($a) {
if ($a["b"]) {
return $a["b"];
}
}
')
);
2016-11-02 07:29:00 +01:00
2016-10-15 19:11:08 +02:00
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
}
public function testArrayKey()
{
$file_checker = new \Psalm\Checker\FileChecker(
'somefile.php',
self::$parser->parse('<?php
$a = ["foo", "bar"];
$b = $a[0];
$c = ["a" => "foo", "b"=> "bar"];
$d = "a";
$e = $a[$d];
')
);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
$this->assertEquals('string', (string)$context->vars_in_scope['$b']);
$this->assertEquals('string', (string)$context->vars_in_scope['$e']);
}
2016-11-06 01:17:22 +01:00
public function testVariableKeyArrayCreate()
{
$stmts = self::$parser->parse('<?php
$a = [];
$b = "boop";
$a[$b][] = "bam";
$c = [];
$c[$b][$b][] = "bam";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
2016-11-13 17:24:25 +01:00
$this->assertEquals('array<string, array<int, string>>', (string) $context->vars_in_scope['$a']);
$this->assertEquals('array<string, array<string, array<int, string>>>', (string) $context->vars_in_scope['$c']);
2016-11-06 01:17:22 +01:00
}
public function testAssignExplicitValueToGeneric()
{
$stmts = self::$parser->parse('<?php
/** @var array<string, array<string, string>> */
$a = [];
$a["foo"] = ["bar" => "baz"];
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
$this->assertEquals('array<string, array<string, string>>', (string) $context->vars_in_scope['$a']);
}
2016-11-13 20:36:29 +01:00
public function testAdditionWithEmpty()
{
$stmts = self::$parser->parse('<?php
$a = [];
$a += ["bar"];
$b = [] + ["bar"];
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
$this->assertEquals('array<int, string>', (string) $context->vars_in_scope['$a']);
$this->assertEquals('array<int, string>', (string) $context->vars_in_scope['$b']);
}
public function testAdditionDifferentType()
{
$stmts = self::$parser->parse('<?php
$a = ["bar"];
$a += [1];
$b = ["bar"] + [1];
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
$this->assertEquals('array<int, string|int>', (string) $context->vars_in_scope['$a']);
$this->assertEquals('array<int, string|int>', (string) $context->vars_in_scope['$b']);
}
2016-12-05 01:33:42 +01:00
public function testPreset1DArrayTypeWithVarKeys()
{
$stmts = self::$parser->parse('<?php
/** @var array<string, array<int, string>> */
$a = [];
$foo = "foo";
$a[$foo][] = "bat";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
}
public function testPreset2DArrayTypeWithVarKeys()
{
$stmts = self::$parser->parse('<?php
/** @var array<string, array<string, array<int, string>>> */
$b = [];
$foo = "foo";
$bar = "bar";
$b[$foo][$bar][] = "bat";
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$context = new Context('somefile.php');
$file_checker->check(true, true, $context);
}
2016-12-12 05:41:11 +01:00
/**
* @expectedException \Psalm\Exception\CodeException
* @expectedExceptionMessage InvalidArrayAssignment
*/
public function testInvalidArrayAccess()
{
$context = new Context('somefile.php');
$stmts = self::$parser->parse('<?php
2016-12-12 05:41:11 +01:00
$a = 5;
$a[0] = 5;
');
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
$file_checker->check(true, true, $context);
}
2016-09-12 06:02:50 +02:00
}