2017-12-05 12:14:10 -05:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
|
|
|
class EmptyTest extends TestCase
|
|
|
|
{
|
2018-04-30 22:13:13 -04:00
|
|
|
use Traits\FileCheckerInvalidCodeParseTestTrait;
|
2017-12-05 12:14:10 -05:00
|
|
|
use Traits\FileCheckerValidCodeParseTestTrait;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function providerFileCheckerValidCodeParse()
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
'empty' => [
|
|
|
|
'<?php
|
|
|
|
$a = !empty($b) ? $b : null;',
|
|
|
|
'assertions' => [
|
|
|
|
'$a' => 'mixed',
|
|
|
|
],
|
|
|
|
'error_levels' => ['MixedAssignment'],
|
|
|
|
],
|
|
|
|
'emptyArrayVar' => [
|
|
|
|
'<?php
|
|
|
|
function a(array $in): void
|
|
|
|
{
|
|
|
|
$r = [];
|
|
|
|
foreach ($in as $entry) {
|
|
|
|
if (!empty($entry["a"])) {
|
|
|
|
$r[] = [];
|
|
|
|
}
|
|
|
|
if (empty($entry["a"])) {
|
|
|
|
$r[] = [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function b(array $in): void
|
|
|
|
{
|
|
|
|
$i = 0;
|
|
|
|
foreach ($in as $entry) {
|
|
|
|
if (!empty($entry["a"])) {
|
|
|
|
$i--;
|
|
|
|
}
|
|
|
|
if (empty($entry["a"])) {
|
|
|
|
$i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function c(array $in): void
|
|
|
|
{
|
|
|
|
foreach ($in as $entry) {
|
|
|
|
if (!empty($entry["a"])) {}
|
|
|
|
}
|
|
|
|
foreach ($in as $entry) {
|
|
|
|
if (empty($entry["a"])) {}
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'assertions' => [],
|
2018-01-09 19:33:39 -05:00
|
|
|
'error_levels' => ['MixedAssignment', 'MixedArrayAccess'],
|
2017-12-05 12:14:10 -05:00
|
|
|
],
|
|
|
|
'removeEmptyArray' => [
|
|
|
|
'<?php
|
|
|
|
$arr_or_string = [];
|
|
|
|
|
|
|
|
if (rand(0, 1)) {
|
|
|
|
$arr_or_string = "hello";
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return void **/
|
|
|
|
function foo(string $s) {}
|
|
|
|
|
|
|
|
if (!empty($arr_or_string)) {
|
|
|
|
foo($arr_or_string);
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'emptyArrayReconciliationThenIf' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param string|string[] $a
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo($a): string {
|
2017-12-05 12:14:10 -05:00
|
|
|
if (is_string($a)) {
|
|
|
|
return $a;
|
|
|
|
} elseif (empty($a)) {
|
|
|
|
return "goodbye";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($a[0])) {
|
|
|
|
return $a[0];
|
|
|
|
};
|
|
|
|
|
|
|
|
return "not found";
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'emptyStringReconciliationThenIf' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param Exception|string|string[] $a
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo($a): string {
|
2017-12-05 12:14:10 -05:00
|
|
|
if (is_array($a)) {
|
|
|
|
return "hello";
|
|
|
|
} elseif (empty($a)) {
|
|
|
|
return "goodbye";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_string($a)) {
|
|
|
|
return $a;
|
|
|
|
};
|
|
|
|
|
|
|
|
return "an exception";
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'emptyExceptionReconciliationAfterIf' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param Exception|null $a
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo($a): string {
|
2017-12-05 12:14:10 -05:00
|
|
|
if ($a && $a->getMessage() === "hello") {
|
|
|
|
return "hello";
|
|
|
|
} elseif (empty($a)) {
|
|
|
|
return "goodbye";
|
|
|
|
}
|
|
|
|
|
|
|
|
return $a->getMessage();
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'noFalsyLeak' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo(string $s): void {
|
2017-12-05 12:14:10 -05:00
|
|
|
if (empty($s) || $s === "hello") {}
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'noRedundantConditionOnMixed' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function testarray(array $data): void {
|
2017-12-05 12:14:10 -05:00
|
|
|
foreach ($data as $item) {
|
|
|
|
if (!empty($item["a"]) && !empty($item["b"]) && !empty($item["b"]["c"])) {
|
|
|
|
echo "Found\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'assertions' => [],
|
2018-01-09 19:33:39 -05:00
|
|
|
'error_levels' => ['MixedAssignment', 'MixedArrayAccess'],
|
2017-12-05 12:14:10 -05:00
|
|
|
],
|
2017-12-10 14:56:51 -05:00
|
|
|
'dontBleedEmptyAfterExtract' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo(array $args): void {
|
2017-12-10 14:56:51 -05:00
|
|
|
extract($args);
|
|
|
|
if ((empty($arr) && empty($a)) || $c === 0) {
|
|
|
|
} else {
|
|
|
|
foreach ($arr as $b) {}
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'assertions' => [],
|
2018-04-07 15:16:46 -04:00
|
|
|
'error_levels' => ['MixedAssignment', 'MixedArgument'],
|
2017-12-10 14:56:51 -05:00
|
|
|
],
|
2018-03-17 19:28:01 -04:00
|
|
|
'emptyObjectLike' => [
|
|
|
|
'<?php
|
|
|
|
$arr = [
|
|
|
|
"profile" => [
|
|
|
|
"foo" => "bar",
|
|
|
|
],
|
|
|
|
"groups" => [
|
|
|
|
"foo" => "bar",
|
|
|
|
"hide" => rand(0, 5),
|
|
|
|
],
|
|
|
|
];
|
|
|
|
|
|
|
|
foreach ($arr as $item) {
|
|
|
|
if (empty($item["hide"]) || $item["hide"] === 3) {}
|
|
|
|
}',
|
|
|
|
],
|
2018-04-09 10:19:23 -04:00
|
|
|
'alwaysBoolResult' => [
|
|
|
|
'<?php
|
|
|
|
function takesBool(bool $p): void {}
|
|
|
|
takesBool(empty($q));'
|
|
|
|
],
|
2018-05-08 16:34:08 -04:00
|
|
|
'noRedundantConditionAfterFalsyIntChecks' => [
|
|
|
|
'<?php
|
|
|
|
function foo(int $t) : void {
|
|
|
|
if (!$t) {
|
|
|
|
foreach ([0, 1, 2] as $a) {
|
|
|
|
if (!$t) {
|
|
|
|
$t = $a;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'noRedundantConditionAfterEmptyMixedChecks' => [
|
2018-05-08 00:57:18 -04:00
|
|
|
'<?php
|
|
|
|
function foo($t) : void {
|
|
|
|
if (empty($t)) {
|
|
|
|
foreach ($_GET["u"] as $a) {
|
|
|
|
if (empty($t)) {
|
|
|
|
$t = $a;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'assertions' => [],
|
|
|
|
'error_levels' => ['MixedAssignment', 'MissingParamType'],
|
|
|
|
],
|
2018-05-31 15:07:03 -04:00
|
|
|
'noReconciliationForMixed' => [
|
|
|
|
'<?php
|
|
|
|
function foo(array $arr) : void {
|
|
|
|
$a = empty($arr["a"]) ? "" : $arr["a"];
|
|
|
|
|
|
|
|
if ($a) {}
|
|
|
|
}',
|
|
|
|
'assertions' => [],
|
|
|
|
'error_levels' => ['MixedAssignment', 'MissingParamType'],
|
|
|
|
],
|
2017-12-05 12:14:10 -05:00
|
|
|
];
|
|
|
|
}
|
2018-04-30 22:13:13 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function providerFileCheckerInvalidCodeParse()
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
'preventImpossibleEmpty' => [
|
|
|
|
'<?php
|
|
|
|
function foo(array $arr) : void {
|
|
|
|
if (empty($ar)) {
|
|
|
|
// do something
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'error_message' => 'UndefinedVariable',
|
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
2017-12-05 12:14:10 -05:00
|
|
|
}
|