2017-02-17 20:50:47 -05:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class IssetTest extends TestCase
|
2017-02-17 20:50:47 -05:00
|
|
|
{
|
2017-04-24 23:45:02 -04:00
|
|
|
use Traits\FileCheckerValidCodeParseTestTrait;
|
2018-01-09 19:33:39 -05:00
|
|
|
use Traits\FileCheckerInvalidCodeParseTestTrait;
|
2017-02-17 20:50:47 -05:00
|
|
|
|
|
|
|
/**
|
2017-04-24 23:45:02 -04:00
|
|
|
* @return array
|
2017-02-17 20:50:47 -05:00
|
|
|
*/
|
2017-04-24 23:45:02 -04:00
|
|
|
public function providerFileCheckerValidCodeParse()
|
2017-02-17 20:50:47 -05:00
|
|
|
{
|
2017-04-24 23:45:02 -04:00
|
|
|
return [
|
|
|
|
'isset' => [
|
|
|
|
'<?php
|
|
|
|
$a = isset($b) ? $b : null;',
|
|
|
|
'assertions' => [
|
2017-06-29 10:22:49 -04:00
|
|
|
'$a' => 'mixed',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_levels' => ['MixedAssignment'],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'nullCoalesce' => [
|
|
|
|
'<?php
|
|
|
|
$a = $b ?? null;',
|
|
|
|
'assertions' => [
|
2017-06-29 10:22:49 -04:00
|
|
|
'$a' => 'mixed',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_levels' => ['MixedAssignment'],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'nullCoalesceWithGoodVariable' => [
|
|
|
|
'<?php
|
2017-11-28 00:46:41 -05:00
|
|
|
$b = rand(0, 10) > 5 ? "hello" : null;
|
2017-04-24 23:45:02 -04:00
|
|
|
$a = $b ?? null;',
|
|
|
|
'assertions' => [
|
2017-11-28 00:46:41 -05:00
|
|
|
'$a' => 'string|null',
|
2017-05-26 20:05:57 -04:00
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'issetKeyedOffset' => [
|
|
|
|
'<?php
|
|
|
|
if (!isset($foo["a"])) {
|
|
|
|
$foo["a"] = "hello";
|
|
|
|
}',
|
|
|
|
'assertions' => [
|
2017-06-29 10:22:49 -04:00
|
|
|
'$foo[\'a\']' => 'mixed',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'error_levels' => [],
|
|
|
|
'scope_vars' => [
|
2017-05-26 20:05:57 -04:00
|
|
|
'$foo' => \Psalm\Type::getArray(),
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'issetKeyedOffsetORFalse' => [
|
|
|
|
'<?php
|
|
|
|
/** @return void */
|
|
|
|
function takesString(string $str) {}
|
2017-06-29 10:22:49 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
$bar = rand(0, 1) ? ["foo" => "bar"] : false;
|
2017-06-29 10:22:49 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
if (isset($bar["foo"])) {
|
|
|
|
takesString($bar["foo"]);
|
|
|
|
}',
|
|
|
|
'assertions' => [],
|
2018-01-09 19:33:39 -05:00
|
|
|
'error_levels' => ['PossiblyInvalidArrayAccess'],
|
2017-04-24 23:45:02 -04:00
|
|
|
'scope_vars' => [
|
2017-05-26 20:05:57 -04:00
|
|
|
'$foo' => \Psalm\Type::getArray(),
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'nullCoalesceKeyedOffset' => [
|
|
|
|
'<?php
|
|
|
|
$foo["a"] = $foo["a"] ?? "hello";',
|
|
|
|
'assertions' => [
|
2017-06-29 10:22:49 -04:00
|
|
|
'$foo[\'a\']' => 'mixed',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'error_levels' => ['MixedAssignment'],
|
|
|
|
'scope_vars' => [
|
2017-05-26 20:05:57 -04:00
|
|
|
'$foo' => \Psalm\Type::getArray(),
|
|
|
|
],
|
|
|
|
],
|
2017-11-28 16:52:52 -05:00
|
|
|
'noRedundantConditionOnMixed' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function testarray(array $data): void {
|
2017-11-28 16:52:52 -05:00
|
|
|
foreach ($data as $item) {
|
|
|
|
if (isset($item["a"]) && isset($item["b"]) && isset($item["b"]["c"])) {
|
|
|
|
echo "Found\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'assertions' => [],
|
2018-01-09 19:33:39 -05:00
|
|
|
'error_levels' => ['MixedAssignment', 'MixedArrayAccess'],
|
2017-11-28 16:52:52 -05:00
|
|
|
],
|
2017-12-13 19:46:58 -05:00
|
|
|
'testUnset' => [
|
|
|
|
'<?php
|
|
|
|
$foo = ["a", "b", "c"];
|
|
|
|
foreach ($foo as $bar) {}
|
|
|
|
unset($foo, $bar);
|
|
|
|
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo(): void {
|
2017-12-13 19:46:58 -05:00
|
|
|
$foo = ["a", "b", "c"];
|
|
|
|
foreach ($foo as $bar) {}
|
|
|
|
unset($foo, $bar);
|
|
|
|
}',
|
|
|
|
],
|
2017-12-18 18:47:17 -05:00
|
|
|
'issetObjectLike' => [
|
|
|
|
'<?php
|
|
|
|
$arr = [
|
|
|
|
"profile" => [
|
|
|
|
"foo" => "bar",
|
|
|
|
],
|
|
|
|
"groups" => [
|
|
|
|
"foo" => "bar",
|
|
|
|
"hide" => rand() % 2 > 0,
|
|
|
|
],
|
|
|
|
];
|
|
|
|
|
|
|
|
foreach ($arr as $item) {
|
|
|
|
if (!isset($item["hide"]) || !$item["hide"]) {}
|
|
|
|
}',
|
|
|
|
],
|
2018-01-29 15:47:25 -05:00
|
|
|
'issetPropertyAffirmsObject' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/** @var ?int */
|
|
|
|
public $id;
|
|
|
|
}
|
|
|
|
|
|
|
|
function takesA(?A $a): A {
|
|
|
|
if (isset($a->id)) {
|
|
|
|
return $a;
|
|
|
|
}
|
|
|
|
|
|
|
|
return new A();
|
|
|
|
}',
|
|
|
|
],
|
2018-02-17 11:24:08 -05:00
|
|
|
'issetVariableKeysWithoutChange' => [
|
|
|
|
'<?php
|
|
|
|
$arr = [[1, 2, 3], null, [1, 2, 3], null];
|
|
|
|
$b = 2;
|
|
|
|
$c = 0;
|
|
|
|
if (isset($arr[$b][$c])) {
|
|
|
|
echo $arr[$b][$c];
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'issetNonNullArrayKey' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param array<int, int> $arr
|
|
|
|
*/
|
|
|
|
function foo(array $arr) : int {
|
|
|
|
$b = rand(0, 3);
|
|
|
|
if (!isset($arr[$b])) {
|
|
|
|
throw new \Exception("bad");
|
|
|
|
}
|
|
|
|
return $arr[$b];
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'issetArrayOffsetConditionalCreationWithInt' => [
|
|
|
|
'<?php
|
|
|
|
/** @param array<int, string> $arr */
|
|
|
|
function foo(array $arr) : string {
|
|
|
|
if (!isset($arr[0])) {
|
|
|
|
$arr[0] = "hello";
|
|
|
|
}
|
|
|
|
|
|
|
|
return $arr[0];
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'issetArrayOffsetConditionalCreationWithVariable' => [
|
|
|
|
'<?php
|
|
|
|
/** @param array<int, string> $arr */
|
|
|
|
function foo(array $arr) : string {
|
|
|
|
$b = 5;
|
|
|
|
|
|
|
|
if (!isset($arr[$b])) {
|
|
|
|
$arr[$b] = "hello";
|
|
|
|
}
|
|
|
|
|
|
|
|
return $arr[$b];
|
|
|
|
}',
|
|
|
|
],
|
2018-03-08 15:57:46 -05:00
|
|
|
'noExceptionOnBracketString' => [
|
|
|
|
'<?php
|
|
|
|
if (isset($foo["bar[]"])) {}',
|
|
|
|
],
|
2018-04-06 16:23:10 -04:00
|
|
|
'issetArrayOffsetAndProperty' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/** @var ?B */
|
|
|
|
public $b;
|
|
|
|
}
|
|
|
|
class B {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param A[] $arr
|
|
|
|
*/
|
|
|
|
function takesAList(array $arr) : B {
|
|
|
|
if (isset($arr[1]->b)) {
|
|
|
|
return $arr[1]->b;
|
|
|
|
}
|
|
|
|
throw new \Exception("bad");
|
|
|
|
}',
|
|
|
|
],
|
2018-04-06 18:28:22 -04:00
|
|
|
'allowUnknownAdditionToInt' => [
|
|
|
|
'<?php
|
|
|
|
$arr = [1, 1, 1, 1, 2, 5, 3, 2];
|
|
|
|
$cumulative = [];
|
|
|
|
|
|
|
|
foreach ($arr as $val) {
|
|
|
|
if (isset($cumulative[$val])) {
|
|
|
|
$cumulative[$val] = $cumulative[$val] + 1;
|
|
|
|
} else {
|
|
|
|
$cumulative[$val] = 1;
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'allowUnknownArrayMergeToInt' => [
|
|
|
|
'<?php
|
|
|
|
$arr = [1, 1, 1, 1, 2, 5, 3, 2];
|
|
|
|
$cumulative = [];
|
|
|
|
|
|
|
|
foreach ($arr as $val) {
|
|
|
|
if (isset($cumulative[$val])) {
|
|
|
|
$cumulative[$val] = array_merge($cumulative[$val], [$val]);
|
|
|
|
} else {
|
|
|
|
$cumulative[$val] = [$val];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($cumulative as $arr) {
|
|
|
|
foreach ($arr as $val) {
|
|
|
|
takesInt($val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function takesInt(int $i) : void {}',
|
|
|
|
],
|
2018-04-11 14:19:42 -04:00
|
|
|
'returnArrayWithDefinedKeys' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param array{bar?: int, foo: int|string} $arr
|
|
|
|
* @return array{bar: int, foo: string}|null
|
|
|
|
*/
|
|
|
|
function foo(array $arr) : ?array {
|
|
|
|
if (!isset($arr["bar"])) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_int($arr["foo"])) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $arr;
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'arrayAccessAfterTwoIssets' => [
|
|
|
|
'<?php
|
|
|
|
$arr = [];
|
|
|
|
|
|
|
|
foreach ([1, 2, 3] as $foo) {
|
|
|
|
if (!isset($arr["foo"])) {
|
|
|
|
$arr["foo"] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isset($arr["bar"])) {
|
|
|
|
$arr["bar"] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
echo $arr["bar"];
|
|
|
|
}',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
];
|
2017-02-17 20:50:47 -05:00
|
|
|
}
|
2018-01-09 19:33:39 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function providerFileCheckerInvalidCodeParse()
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
'complainAboutBadCallInIsset' => [
|
|
|
|
'<?php
|
|
|
|
class A {}
|
|
|
|
$a = isset(A::foo()[0]);',
|
|
|
|
'error_message' => 'UndefinedMethod',
|
|
|
|
],
|
2018-02-17 11:24:08 -05:00
|
|
|
'issetVariableKeysWithChange' => [
|
|
|
|
'<?php
|
|
|
|
$arr = [[1, 2, 3], null, [1, 2, 3], null];
|
|
|
|
$b = 2;
|
|
|
|
$c = 0;
|
|
|
|
if (isset($arr[$b][$c])) {
|
|
|
|
$b = 1;
|
|
|
|
echo $arr[$b][$c];
|
|
|
|
}',
|
|
|
|
'error_message' => 'PossiblyNullArrayAccess',
|
|
|
|
],
|
2018-01-09 19:33:39 -05:00
|
|
|
];
|
|
|
|
}
|
2017-02-17 20:50:47 -05:00
|
|
|
}
|