1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-09 06:28:36 +01:00
psalm/tests/TypeReconciliation/RedundantConditionTest.php

1575 lines
55 KiB
PHP
Raw Normal View History

<?php
2019-12-06 20:58:18 +01:00
namespace Psalm\Tests\TypeReconciliation;
2021-12-03 20:11:20 +01:00
use Psalm\Tests\TestCase;
use Psalm\Tests\Traits\InvalidCodeAnalysisTestTrait;
use Psalm\Tests\Traits\ValidCodeAnalysisTestTrait;
use const DIRECTORY_SEPARATOR;
2021-12-03 20:11:20 +01:00
class RedundantConditionTest extends TestCase
{
2021-12-03 20:11:20 +01:00
use ValidCodeAnalysisTestTrait;
use InvalidCodeAnalysisTestTrait;
public function providerValidCodeParse(): iterable
{
return [
'ignoreIssueAndAssign' => [
'code' => '<?php
2018-09-17 18:15:45 +02:00
function foo(): stdClass {
return new stdClass;
}
$b = null;
foreach ([0, 1] as $i) {
$a = foo();
if (!empty($a)) {
$b = $a;
}
}',
'assertions' => [
'$b' => 'null|stdClass',
],
'ignored_issues' => ['RedundantCondition'],
],
'byrefNoRedundantCondition' => [
'code' => '<?php
/**
* @param int $min ref
* @param int $other
*/
2018-01-11 21:50:45 +01:00
function testmin(&$min, int $other): void {
if (is_null($min)) {
$min = 3;
} elseif (!is_int($min)) {
$min = 5;
} elseif ($min < $other) {
$min = $other;
}
}',
'assertions' => [],
'ignored_issues' => [
'RedundantConditionGivenDocblockType',
'DocblockTypeContradiction',
],
],
'assignmentInIf' => [
'code' => '<?php
2018-01-11 21:50:45 +01:00
function test(int $x = null): int {
if (!$x && !($x = rand(0, 10))) {
echo "Failed to get non-empty x\n";
return -1;
}
return $x;
}',
],
'noRedundantConditionAfterAssignment' => [
'code' => '<?php
/** @param int $i */
2018-01-11 21:50:45 +01:00
function foo($i): void {
2020-12-01 23:25:45 +01:00
/** @psalm-suppress RedundantConditionGivenDocblockType */
if ($i !== null) {
2020-12-01 23:25:45 +01:00
/** @psalm-suppress RedundantCastGivenDocblockType */
$i = (int) $i;
if ($i) {}
}
}',
'assertions' => [],
],
'noRedundantConditionAfterDocblockTypeNullCheck' => [
'code' => '<?php
class A {
/** @var ?int */
public $foo;
}
class B {}
/**
* @param A|B $i
*/
2018-01-11 21:50:45 +01:00
function foo($i): void {
if (empty($i)) {
return;
}
switch (get_class($i)) {
2018-11-29 06:05:56 +01:00
case A::class:
if ($i->foo) {}
break;
default:
break;
}
}',
'assertions' => [],
'ignored_issues' => ['DocblockTypeContradiction'],
],
'noRedundantConditionTypeReplacementWithDocblock' => [
'code' => '<?php
class A {}
/**
* @return A
*/
function getA() {
return new A();
}
$maybe_a = rand(0, 1) ? new A : null;
if ($maybe_a === null) {
$maybe_a = getA();
}
if ($maybe_a === null) {}',
'assertions' => [],
'ignored_issues' => [
'DocblockTypeContradiction',
],
],
'noRedundantConditionAfterPossiblyNullCheck' => [
'code' => '<?php
if (rand(0, 1)) {
$a = "hello";
}
if ($a) {}',
'assertions' => [],
'ignored_issues' => ['PossiblyUndefinedGlobalVariable'],
],
'noRedundantConditionAfterFromDocblockRemoval' => [
'code' => '<?php
class A {
public function foo(): bool {
return (bool) rand(0, 1);
}
public function bar(): bool {
return (bool) rand(0, 1);
}
}
/** @return A */
function makeA() {
return new A;
}
$a = makeA();
if ($a === null) {
exit;
}
if ($a->foo() || $a->bar()) {}',
'assertions' => [],
'ignored_issues' => [
'DocblockTypeContradiction',
],
],
'noEmptyUndefinedArrayVar' => [
'code' => '<?php
if (rand(0,1)) {
/** @psalm-suppress UndefinedGlobalVariable */
$a = $b[0];
} else {
$a = null;
}
if ($a) {}',
'assertions' => [],
'ignored_issues' => ['MixedAssignment', 'MixedArrayAccess'],
],
2017-12-09 21:51:38 +01:00
'noComplaintWithIsNumericThenIsEmpty' => [
'code' => '<?php
2018-01-11 21:50:45 +01:00
function takesString(string $s): void {
2017-12-09 21:51:38 +01:00
if (!is_numeric($s) || empty($s)) {}
}',
],
'noRedundantConditionOnTryCatchVars' => [
'code' => '<?php
2018-01-11 21:50:45 +01:00
function trycatch(): void {
$value = null;
try {
if (rand() % 2 > 0) {
throw new RuntimeException("Failed");
}
$value = new stdClass();
if (rand() % 2 > 0) {
throw new RuntimeException("Failed");
}
} catch (Exception $e) {
if ($value) {
var_export($value);
}
}
if ($value) {}
}',
],
'noRedundantConditionInFalseCheck' => [
'code' => '<?php
$ch = curl_init();
if (!$ch) {}',
],
'noRedundantConditionInForCheck' => [
'code' => '<?php
class Node
{
/** @var Node|null */
public $next;
public function iterate(): void
{
for ($node = $this; $node !== null; $node = $node->next) {}
}
}',
],
'noRedundantConditionComparingBool' => [
'code' => '<?php
2018-01-11 21:50:45 +01:00
function getBool(): bool {
return (bool)rand(0, 1);
}
2018-01-11 21:50:45 +01:00
function takesBool(bool $b): void {
if ($b === getBool()) {}
}',
],
'evaluateElseifProperly' => [
'code' => '<?php
/** @param string $str */
2018-01-11 21:50:45 +01:00
function foo($str): int {
if (is_null($str)) {
return 1;
} else if (strlen($str) < 1) {
return 2;
}
return 2;
}',
'assertions' => [],
'ignored_issues' => [
'DocblockTypeContradiction',
'RedundantConditionGivenDocblockType',
],
],
'evaluateArrayCheck' => [
'code' => '<?php
2018-01-11 21:50:45 +01:00
function array_check(): void {
$data = ["f" => false];
while (rand(0, 1) > 0 && !$data["f"]) {
$data = ["f" => true];
}
}',
],
'mixedArrayAssignment' => [
'code' => '<?php
/** @param mixed $arr */
2018-01-11 21:50:45 +01:00
function foo($arr): void {
if ($arr["a"] === false) {
2020-06-25 07:33:02 +02:00
/** @psalm-suppress MixedArrayAssignment */
$arr["a"] = (bool) rand(0, 1);
if ($arr["a"] === false) {}
}
}',
'assertions' => [],
'ignored_issues' => ['MixedAssignment', 'MixedArrayAccess'],
],
'hardPhpTypeAssertionsOnDocblockBoolType' => [
'code' => '<?php
/** @param bool|null $bar */
function foo($bar): void {
if (!is_null($bar) && !is_bool($bar)) {
throw new \Exception("bad");
}
if ($bar !== null) {}
}',
'assertions' => [],
'ignored_issues' => ['DocblockTypeContradiction'],
],
'hardPhpTypeAssertionsOnDocblockStringType' => [
'code' => '<?php
/** @param string|null $bar */
2018-01-11 21:50:45 +01:00
function foo($bar): void {
if (!is_null($bar) && !is_string($bar)) {
throw new \Exception("bad");
}
if ($bar !== null) {}
}',
'assertions' => [],
'ignored_issues' => ['DocblockTypeContradiction'],
],
'isObjectAssertionOnDocblockType' => [
'code' => '<?php
class A {}
class B {}
/** @param A|B $a */
function foo($a) : void {
if (!is_object($a)) {
return;
}
if ($a instanceof A) {
} elseif ($a instanceof B) {
} else {
throw new \Exception("bad");
}
}',
'assertions' => [],
'ignored_issues' => ['RedundantConditionGivenDocblockType', 'DocblockTypeContradiction'],
],
'nullToMixedWithNullCheckWithArraykey' => [
'code' => '<?php
/** @return array<array-key, mixed> */
function getStrings(): array {
return ["hello", "world", 50];
}
$a = getStrings();
if (is_string($a[0]) && strlen($a[0]) > 3) {}',
Add support for strict arrays, fix type alias intersection, fix array_is_list assertion on non-lists (#8395) * Immutable CodeLocation * Remove excess clones * Remove external clones * Remove leftover clones * Fix final clone issue * Immutable storages * Refactoring * Fixes * Fixes * Fix * Fix * Fixes * Simplify * Fixes * Fix * Fixes * Update * Fix * Cache global types * Fix * Update * Update * Fixes * Fixes * Refactor * Fixes * Fix * Fix * More caching * Fix * Fix * Update * Update * Fix * Fixes * Update * Refactor * Update * Fixes * Break one more test * Fix * FIx * Fix * Fix * Fix * Fix * Improve performance and readability * Equivalent logic * Fixes * Revert * Revert "Revert" This reverts commit f9175100c8452c80559234200663fd4c4f4dd889. * Fix * Fix reference bug * Make default TypeVisitor immutable * Bugfix * Remove clones * Partial refactoring * Refactoring * Fixes * Fix * Fixes * Fixes * cs-fix * Fix final bugs * Add test * Misc fixes * Update * Fixes * Experiment with removing different property * revert "Experiment with removing different property" This reverts commit ac1156e077fc4ea633530d51096d27b6e88bfdf9. * Uniform naming * Uniform naming * Hack hotfix * Clean up $_FILES ref #8621 * Undo hack, try fixing properly * Helper method * Remove redundant call * Partially fix bugs * Cleanup * Change defaults * Fix bug * Fix (?, hope this doesn't break anything else) * cs-fix * Review fixes * Bugfix * Bugfix * Improve logic * Add support for list{} and callable-list{} types, properly implement array_is_list assertions (fixes #8389) * Default to sealed arrays * Fix array_merge bug * Fixes * Fix * Sealed type checks * Properly infer properties-of and get_object_vars on final classes * Fix array_map zipping * Fix tests * Fixes * Fixes * Fix more stuff * Recursively resolve type aliases * Fix typo * Fixes * Fix array_is_list assertion on keyed array * Add BC docs * Fixes * fix * Update * Update * Update * Update * Seal arrays with count assertions * Fix #8528 * Fix * Update * Improve sealed array foreach logic * get_object_vars on template properties * Fix sealed array assertion reconciler logic * Improved reconciler * Add tests * Single source of truth for test types * Fix tests * Fixup tests * Fixup tests * Fixup tests * Update * Fix tests * Fix tests * Final fixes * Fixes * Use list syntax only when needed * Fix tests * Cs-fix * Update docs * Update docs * Update docs * Update docs * Update docs * Document missing types * Update docs * Improve class-string-map docs * Update * Update * I love working on psalm :) * Keep arrays unsealed by default * Fixup tests * Fix syntax mistake * cs-fix * Fix typo * Re-import missing types * Keep strict types only in return types * argc/argv fixes * argc/argv fixes * Fix test * Comment-out valinor code, pinging @romm pls merge https://github.com/CuyZ/Valinor/pull/246 so we can add valinor to the psalm docs :)
2022-11-05 22:34:42 +01:00
'assertions' => [],
'ignored_issues' => [],
],
'nullToMixedWithNullCheckWithIntKey' => [
'code' => '<?php
/** @return array<int, mixed> */
function getStrings(): array {
return ["hello", "world", 50];
}
$a = getStrings();
if (is_string($a[0]) && strlen($a[0]) > 3) {}',
Add support for strict arrays, fix type alias intersection, fix array_is_list assertion on non-lists (#8395) * Immutable CodeLocation * Remove excess clones * Remove external clones * Remove leftover clones * Fix final clone issue * Immutable storages * Refactoring * Fixes * Fixes * Fix * Fix * Fixes * Simplify * Fixes * Fix * Fixes * Update * Fix * Cache global types * Fix * Update * Update * Fixes * Fixes * Refactor * Fixes * Fix * Fix * More caching * Fix * Fix * Update * Update * Fix * Fixes * Update * Refactor * Update * Fixes * Break one more test * Fix * FIx * Fix * Fix * Fix * Fix * Improve performance and readability * Equivalent logic * Fixes * Revert * Revert "Revert" This reverts commit f9175100c8452c80559234200663fd4c4f4dd889. * Fix * Fix reference bug * Make default TypeVisitor immutable * Bugfix * Remove clones * Partial refactoring * Refactoring * Fixes * Fix * Fixes * Fixes * cs-fix * Fix final bugs * Add test * Misc fixes * Update * Fixes * Experiment with removing different property * revert "Experiment with removing different property" This reverts commit ac1156e077fc4ea633530d51096d27b6e88bfdf9. * Uniform naming * Uniform naming * Hack hotfix * Clean up $_FILES ref #8621 * Undo hack, try fixing properly * Helper method * Remove redundant call * Partially fix bugs * Cleanup * Change defaults * Fix bug * Fix (?, hope this doesn't break anything else) * cs-fix * Review fixes * Bugfix * Bugfix * Improve logic * Add support for list{} and callable-list{} types, properly implement array_is_list assertions (fixes #8389) * Default to sealed arrays * Fix array_merge bug * Fixes * Fix * Sealed type checks * Properly infer properties-of and get_object_vars on final classes * Fix array_map zipping * Fix tests * Fixes * Fixes * Fix more stuff * Recursively resolve type aliases * Fix typo * Fixes * Fix array_is_list assertion on keyed array * Add BC docs * Fixes * fix * Update * Update * Update * Update * Seal arrays with count assertions * Fix #8528 * Fix * Update * Improve sealed array foreach logic * get_object_vars on template properties * Fix sealed array assertion reconciler logic * Improved reconciler * Add tests * Single source of truth for test types * Fix tests * Fixup tests * Fixup tests * Fixup tests * Update * Fix tests * Fix tests * Final fixes * Fixes * Use list syntax only when needed * Fix tests * Cs-fix * Update docs * Update docs * Update docs * Update docs * Update docs * Document missing types * Update docs * Improve class-string-map docs * Update * Update * I love working on psalm :) * Keep arrays unsealed by default * Fixup tests * Fix syntax mistake * cs-fix * Fix typo * Re-import missing types * Keep strict types only in return types * argc/argv fixes * argc/argv fixes * Fix test * Comment-out valinor code, pinging @romm pls merge https://github.com/CuyZ/Valinor/pull/246 so we can add valinor to the psalm docs :)
2022-11-05 22:34:42 +01:00
'assertions' => [],
'ignored_issues' => [],
],
'replaceFalseTypeWithTrueConditionalOnMixedEquality' => [
'code' => '<?php
function getData() {
return rand(0, 1) ? [1, 2, 3] : false;
}
$a = false;
while ($i = getData()) {
if (!$a && $i[0] === 2) {
$a = true;
}
if ($a === false) {}
}',
Add support for strict arrays, fix type alias intersection, fix array_is_list assertion on non-lists (#8395) * Immutable CodeLocation * Remove excess clones * Remove external clones * Remove leftover clones * Fix final clone issue * Immutable storages * Refactoring * Fixes * Fixes * Fix * Fix * Fixes * Simplify * Fixes * Fix * Fixes * Update * Fix * Cache global types * Fix * Update * Update * Fixes * Fixes * Refactor * Fixes * Fix * Fix * More caching * Fix * Fix * Update * Update * Fix * Fixes * Update * Refactor * Update * Fixes * Break one more test * Fix * FIx * Fix * Fix * Fix * Fix * Improve performance and readability * Equivalent logic * Fixes * Revert * Revert "Revert" This reverts commit f9175100c8452c80559234200663fd4c4f4dd889. * Fix * Fix reference bug * Make default TypeVisitor immutable * Bugfix * Remove clones * Partial refactoring * Refactoring * Fixes * Fix * Fixes * Fixes * cs-fix * Fix final bugs * Add test * Misc fixes * Update * Fixes * Experiment with removing different property * revert "Experiment with removing different property" This reverts commit ac1156e077fc4ea633530d51096d27b6e88bfdf9. * Uniform naming * Uniform naming * Hack hotfix * Clean up $_FILES ref #8621 * Undo hack, try fixing properly * Helper method * Remove redundant call * Partially fix bugs * Cleanup * Change defaults * Fix bug * Fix (?, hope this doesn't break anything else) * cs-fix * Review fixes * Bugfix * Bugfix * Improve logic * Add support for list{} and callable-list{} types, properly implement array_is_list assertions (fixes #8389) * Default to sealed arrays * Fix array_merge bug * Fixes * Fix * Sealed type checks * Properly infer properties-of and get_object_vars on final classes * Fix array_map zipping * Fix tests * Fixes * Fixes * Fix more stuff * Recursively resolve type aliases * Fix typo * Fixes * Fix array_is_list assertion on keyed array * Add BC docs * Fixes * fix * Update * Update * Update * Update * Seal arrays with count assertions * Fix #8528 * Fix * Update * Improve sealed array foreach logic * get_object_vars on template properties * Fix sealed array assertion reconciler logic * Improved reconciler * Add tests * Single source of truth for test types * Fix tests * Fixup tests * Fixup tests * Fixup tests * Update * Fix tests * Fix tests * Final fixes * Fixes * Use list syntax only when needed * Fix tests * Cs-fix * Update docs * Update docs * Update docs * Update docs * Update docs * Document missing types * Update docs * Improve class-string-map docs * Update * Update * I love working on psalm :) * Keep arrays unsealed by default * Fixup tests * Fix syntax mistake * cs-fix * Fix typo * Re-import missing types * Keep strict types only in return types * argc/argv fixes * argc/argv fixes * Fix test * Comment-out valinor code, pinging @romm pls merge https://github.com/CuyZ/Valinor/pull/246 so we can add valinor to the psalm docs :)
2022-11-05 22:34:42 +01:00
'assertions' => [],
'ignored_issues' => ['MixedAssignment', 'MissingReturnType', 'MixedArrayAccess'],
],
'nullCoalescePossiblyUndefined' => [
'code' => '<?php
if (rand(0,1)) {
$options = ["option" => true];
}
/** @psalm-suppress PossiblyUndefinedGlobalVariable */
$option = $options["option"] ?? false;
if ($option) {}',
Add support for strict arrays, fix type alias intersection, fix array_is_list assertion on non-lists (#8395) * Immutable CodeLocation * Remove excess clones * Remove external clones * Remove leftover clones * Fix final clone issue * Immutable storages * Refactoring * Fixes * Fixes * Fix * Fix * Fixes * Simplify * Fixes * Fix * Fixes * Update * Fix * Cache global types * Fix * Update * Update * Fixes * Fixes * Refactor * Fixes * Fix * Fix * More caching * Fix * Fix * Update * Update * Fix * Fixes * Update * Refactor * Update * Fixes * Break one more test * Fix * FIx * Fix * Fix * Fix * Fix * Improve performance and readability * Equivalent logic * Fixes * Revert * Revert "Revert" This reverts commit f9175100c8452c80559234200663fd4c4f4dd889. * Fix * Fix reference bug * Make default TypeVisitor immutable * Bugfix * Remove clones * Partial refactoring * Refactoring * Fixes * Fix * Fixes * Fixes * cs-fix * Fix final bugs * Add test * Misc fixes * Update * Fixes * Experiment with removing different property * revert "Experiment with removing different property" This reverts commit ac1156e077fc4ea633530d51096d27b6e88bfdf9. * Uniform naming * Uniform naming * Hack hotfix * Clean up $_FILES ref #8621 * Undo hack, try fixing properly * Helper method * Remove redundant call * Partially fix bugs * Cleanup * Change defaults * Fix bug * Fix (?, hope this doesn't break anything else) * cs-fix * Review fixes * Bugfix * Bugfix * Improve logic * Add support for list{} and callable-list{} types, properly implement array_is_list assertions (fixes #8389) * Default to sealed arrays * Fix array_merge bug * Fixes * Fix * Sealed type checks * Properly infer properties-of and get_object_vars on final classes * Fix array_map zipping * Fix tests * Fixes * Fixes * Fix more stuff * Recursively resolve type aliases * Fix typo * Fixes * Fix array_is_list assertion on keyed array * Add BC docs * Fixes * fix * Update * Update * Update * Update * Seal arrays with count assertions * Fix #8528 * Fix * Update * Improve sealed array foreach logic * get_object_vars on template properties * Fix sealed array assertion reconciler logic * Improved reconciler * Add tests * Single source of truth for test types * Fix tests * Fixup tests * Fixup tests * Fixup tests * Update * Fix tests * Fix tests * Final fixes * Fixes * Use list syntax only when needed * Fix tests * Cs-fix * Update docs * Update docs * Update docs * Update docs * Update docs * Document missing types * Update docs * Improve class-string-map docs * Update * Update * I love working on psalm :) * Keep arrays unsealed by default * Fixup tests * Fix syntax mistake * cs-fix * Fix typo * Re-import missing types * Keep strict types only in return types * argc/argv fixes * argc/argv fixes * Fix test * Comment-out valinor code, pinging @romm pls merge https://github.com/CuyZ/Valinor/pull/246 so we can add valinor to the psalm docs :)
2022-11-05 22:34:42 +01:00
'assertions' => [],
'ignored_issues' => ['MixedAssignment', 'MixedArrayAccess'],
],
'allowIntValueCheckAfterComparisonDueToOverflow' => [
'code' => '<?php
function foo(int $x) : void {
$x = $x + 1;
if (!is_int($x)) {
echo "Is a float.";
} else {
echo "Is an int.";
}
}
function bar(int $x) : void {
$x = $x + 1;
if (is_float($x)) {
echo "Is a float.";
} else {
echo "Is an int.";
}
}',
],
'allowIntValueCheckAfterComparisonDueToOverflowInc' => [
'code' => '<?php
function foo(int $x) : void {
$x++;
if (!is_int($x)) {
echo "Is a float.";
} else {
echo "Is an int.";
}
}
function bar(int $x) : void {
$x++;
if (is_float($x)) {
echo "Is a float.";
} else {
echo "Is an int.";
}
}',
],
'allowIntValueCheckAfterComparisonDueToConditionalOverflow' => [
'code' => '<?php
function foo(int $x) : void {
if (rand(0, 1)) {
$x = $x + 1;
}
if (is_float($x)) {
echo "Is a float.";
} else {
echo "Is an int.";
}
}',
],
'changeStringValue' => [
'code' => '<?php
$concat = "";
foreach (["x", "y"] as $v) {
if ($concat != "") {
$concat .= ", ";
}
$concat .= "($v)";
}',
],
2018-05-20 02:31:48 +02:00
'arrayCanBeEmpty' => [
'code' => '<?php
2018-05-20 02:31:48 +02:00
$x = ["key" => "value"];
if (rand(0, 1)) {
$x = [];
}
if ($x) {
var_export($x);
}',
],
'noRedundantConditionStringNotFalse' => [
'code' => '<?php
function foo(string $s) : void {
if ($s != false ) {}
}',
],
'noRedundantConditionStringNotTrue' => [
'code' => '<?php
function foo(string $s) : void {
if ($s != true ) {}
}',
],
'noRedundantConditionBoolNotFalse' => [
'code' => '<?php
function foo(bool $s) : void {
if ($s !== false ) {}
}',
],
'noRedundantConditionBoolNotTrue' => [
'code' => '<?php
function foo(bool $s) : void {
if ($s !== true ) {}
}',
],
'noRedundantConditionNullableBoolIsFalseOrTrue' => [
'code' => '<?php
function foo(?bool $s) : void {
if ($s === false ) {} elseif ($s === true) {}
}',
],
'noRedundantConditionNullableBoolIsTrueOrFalse' => [
'code' => '<?php
function foo(?bool $s) : void {
if ($s === true ) {} elseif ($s === false) {}
}',
],
'noRedundantConditionAfterCheckingMixedTwice' => [
'code' => '<?php
function foo($a) : void {
$b = $a ? 1 : 0;
$c = $a ? 1 : 0;
}',
'assertions' => [],
'ignored_issues' => ['MissingParamType'],
],
'notAlwaysTrueBinaryOp' => [
'code' => '<?php
function foo ($a) : void {
if (!$a) {}
$b = $a && rand(0, 1);
}',
'assertions' => [],
'ignored_issues' => ['MissingParamType'],
],
'noRedundantConditionAfterAssertingValue' => [
'code' => '<?php
function foo(string $t, bool $b) : void {
if (!$b && $t === "a") {
return;
}
if ($t === "c") {
if (!$b && bar($t)) {}
}
}
function bar(string $b) : bool {
return true;
}',
],
'noRedundantConditionBleed' => [
'code' => '<?php
$foo = getopt("i");
$i = $foo["i"];
/** @psalm-suppress TypeDoesNotContainNull */
if ($i === null) {
exit;
}
if ($i) {}',
],
'emptyWithoutKnowingArrayType' => [
'code' => '<?php
function foo(array $a) : void {
if (!empty($a["foo"])) {
foreach ($a["foo"] as $key => $_) {
if (rand(0, 1)) {
unset($a["foo"][$key]);
}
}
if (empty($a["foo"])) {}
}
}',
'assertions' => [],
'ignored_issues' => ['MixedAssignment', 'MixedArrayAccess', 'MixedArrayOffset'],
],
'emptyKnowingArrayType' => [
'code' => '<?php
/**
* @param array<string, array<string, int>> $a
*/
function foo(array $a) : void {
if (!empty($a["foo"])) {
foreach ($a["foo"] as $key => $_) {
if (rand(0, 1)) {
unset($a["foo"][$key]);
}
}
if (empty($a["foo"])) {}
}
}',
],
2019-03-16 02:50:16 +01:00
'suppressRedundantConditionAfterAssertNonEmpty' => [
'code' => '<?php
2019-03-16 02:50:16 +01:00
/**
* @param array<int> $a
*/
function process(array $a): void {
assert(!empty($a));
/** @psalm-suppress RedundantConditionGivenDocblockType */
assert(is_int($a[0]));
}',
],
'allowChecksOnFalsyIf' => [
'code' => '<?php
function foo(?string $s) : string {
if ($s == null) {
if ($s === null) {}
return "hello";
} else {
return $s;
}
2019-07-05 22:24:00 +02:00
}',
],
'updateArrayAfterUnset' => [
'code' => '<?php
2019-05-13 07:35:29 +02:00
/**
* @param string[] $arr
*/
function foo(string $s) : void {
$dict = ["a" => 1];
unset($dict[$s]);
if (count($dict)) {}
2019-07-05 22:24:00 +02:00
}',
2019-05-13 07:35:29 +02:00
],
'updateArrayAfterUnsetInLoop' => [
'code' => '<?php
/**
* @param string[] $arr
*/
function foo(array $arr) : void {
$dict = ["a" => 1, "b" => 2, "c" => 3];
foreach ($arr as $v) {
unset($dict[$v]);
}
if (count($dict)) {}
2019-07-05 22:24:00 +02:00
}',
],
'noRedundantConditionWhenAssertingOnIntersection' => [
'code' => '<?php
class A {}
interface I {}
class AChild extends A implements I {}
function isAChild(A $value): ?AChild {
if (!$value instanceof I) {
return null;
}
if (!$value instanceof AChild) {
return null;
}
return $value;
}',
],
'noRedundantConditionWhenAssertingOnIntersectionFlipped' => [
'code' => '<?php
class A {}
interface I {}
class AChild extends A implements I {}
/** @param I&A $value */
function isAChild(I $value): ?AChild {
if (!$value instanceof AChild) {
return null;
}
return $value;
}',
],
'noRedundantConditionWhenAssertingOnIntersectionOfInterfaces' => [
'code' => '<?php
interface A {}
interface I {}
class AChild implements I, A {}
function isAChild(A $value): ?AChild {
if (!$value instanceof I) {
return null;
}
if (!$value instanceof AChild) {
return null;
}
return $value;
}',
],
'noRedundantConditionWithUnionOfInterfaces' => [
'code' => '<?php
interface One {}
interface Two {}
/**
* @param One|Two $impl
*/
function a($impl) : void {
if ($impl instanceof One && $impl instanceof Two) {
throw new \Exception();
} elseif ($impl instanceof One) {}
}
/**
* @param One|Two $impl
*/
function b($impl) : void {
if ($impl instanceof One && $impl instanceof Two) {
throw new \Exception();
} else {
if ($impl instanceof One) {}
}
2022-12-18 17:15:15 +01:00
}',
],
'invalidateAfterPostIncrement' => [
'code' => '<?php
/**
* @param array<int, int> $tokens
*/
function propertyInUse(array $tokens, int $i): bool {
if ($tokens[$i] !== 1) {
return false;
}
$i++;
if ($tokens[$i] !== 2) {}
return false;
2022-12-18 17:15:15 +01:00
}',
],
'invalidateAfterAssignOp' => [
'code' => '<?php
/**
* @param array<int, int> $tokens
*/
function propertyInUse(array $tokens, int $i): bool {
if ($tokens[$i] !== 1) {
return false;
}
$i += 1;
if ($tokens[$i] !== 2) {}
return false;
2022-12-18 17:15:15 +01:00
}',
],
'invalidateAfterAssign' => [
'code' => '<?php
/**
* @param array<int, int> $tokens
*/
function propertyInUse(array $tokens, int $i): bool {
if ($tokens[$i] !== 1) {
return false;
}
$i = $i + 1;
if ($tokens[$i] !== 2) {}
return false;
2022-12-18 17:15:15 +01:00
}',
],
2020-01-11 00:05:45 +01:00
'numericNotString' => [
'code' => '<?php
2020-01-11 00:05:45 +01:00
/** @param mixed $value */
function test($value) : void {
if (!is_numeric($value)) {
throw new Exception("Invalid $value");
}
if (!is_string($value)) {}
2022-12-18 17:15:15 +01:00
}',
2020-01-11 00:05:45 +01:00
],
'checkClosedResource' => [
'code' => '<?php
$fp = tmpfile();
if ($fp) {
echo "foo", "\n";
} else {
echo "bar", "\n";
}
echo var_export([$fp, is_resource($fp), !! $fp], true);
fclose($fp);',
'assertions' => [
'$fp' => 'closed-resource',
2022-12-18 17:15:15 +01:00
],
],
2020-02-22 17:28:24 +01:00
'allowCheckOnReturnTypeUnion' => [
'code' => '<?php
2020-02-22 17:28:24 +01:00
/** @return int|string */
function returnsInt() {
return rand(0, 1) ? 1 : "hello";
}
if (is_int(returnsInt())) {}
if (!is_int(returnsInt())) {}',
],
'noRedundantConditionInClosureForProperty' => [
'code' => '<?php
class Queue {
private bool $closed = false;
public function enqueue(string $value): Closure {
if ($this->closed) {
return function() : void {
if ($this->closed) {}
};
}
return function() : void {};
}
2022-12-18 17:15:15 +01:00
}',
],
'noRedundantCastAfterCalculation' => [
'code' => '<?php
function x(string $x): int {
return (int) (hexdec($x) + 1);
}',
],
'unsetArrayWithKnownOffset' => [
'code' => '<?php
function bar(string $f) : void {
$filter = rand(0, 1) ? explode(",", $f) : [$f];
unset($filter[rand(0, 1)]);
if ($filter) {}
2022-12-18 17:15:15 +01:00
}',
],
'stringInScalar' => [
'code' => '<?php
/**
* @template T of scalar
* @param T $value
*/
function normalizeValue(bool|int|float|string $value): void
{
assert(is_string($value));
2022-01-22 01:00:45 +01:00
}',
'assertions' => [],
'ignored_issues' => [],
'php_version' => '8.0',
],
2021-04-08 04:02:13 +02:00
'NumericCanBeFalsy' => [
'code' => '<?php
2021-04-08 04:02:13 +02:00
function test(string|int|float|bool $value): bool {
if (is_numeric($value) || $value === true) {
if ($value) {
return true;
}
}
return false;
2022-01-22 01:00:45 +01:00
}',
'assertions' => [],
'ignored_issues' => [],
'php_version' => '8.0',
2021-04-08 04:02:13 +02:00
],
'NumericCanBeNotIntOrNotFloat' => [
'code' => '<?php
/** @param mixed $a */
function a($a): void{
if (is_numeric($a)) {
assert(!is_float($a));
}
}
/** @param mixed $a */
function b($a): void{
if (is_numeric($a)) {
assert(!is_int($a));
}
2022-12-18 17:15:15 +01:00
}',
2021-10-21 20:26:00 +02:00
],
'alwaysTrueAssignAllowedInsideAND' => [
'code' => '<?php
2021-10-21 20:26:00 +02:00
class A{
public function get(): stdClass{ return new stdClass;}
}
$a = new A();
if (($c = $a->get()) && rand(0,1)){
}
2022-12-18 17:15:15 +01:00
',
2021-10-21 20:26:00 +02:00
],
'alwaysTrueAssignAllowedInsideOr' => [
'code' => '<?php
2021-10-21 20:26:00 +02:00
class A{
public function get(): ?stdClass{ return new stdClass;}
}
$a = new A();
if ($a->get() || ($c = rand(0,1))){
}
2022-12-18 17:15:15 +01:00
',
],
'countWithNeverValuesInKeyedArray' => [
'code' => '<?php
/** @var non-empty-array $report_data */
$report_data = [];
if ( array_key_exists( "A", $report_data ) ) {
} elseif ( !empty( $report_data[0]["type"] ) && rand(0,1) ) {
if ( rand(0,1) ) {}
if ( count( $report_data ) === 1 ) {
}
2022-12-18 17:15:15 +01:00
}',
],
'countWithNeverValuesInKeyedList' => [
'code' => '<?php
/** @var non-empty-list $report_data */
$report_data = [];
if ( array_key_exists( 2, $report_data ) ) {
} elseif ( !empty( $report_data[0]["type"] ) && rand(0,1) ) {
if ( rand(0,1) ) {}
if ( count( $report_data ) === 1 ) {
}
2022-12-18 17:15:15 +01:00
}',
],
'secondFalsyTwiceWithChange' => [
'code' => '<?php
/**
* @param array{a?:int,b?:string} $p
*/
function f(array $p) : void {
if (!$p) {
throw new RuntimeException("");
}
if (rand(0, 1)) {
$p["a"] = 3;
}
assert(!!$p);
}',
],
];
}
public function providerInvalidCodeParse(): iterable
{
return [
'ifFalse' => [
'code' => '<?php
2018-09-17 18:15:45 +02:00
$y = false;
if ($y) {}',
'error_message' => 'TypeDoesNotContainType',
],
'ifNotTrue' => [
'code' => '<?php
2018-09-17 18:15:45 +02:00
$y = true;
if (!$y) {}',
'error_message' => 'TypeDoesNotContainType',
],
'ifTrue' => [
'code' => '<?php
2018-09-17 18:15:45 +02:00
$y = true;
if ($y) {}',
'error_message' => 'RedundantCondition',
],
'unnecessaryInstanceof' => [
'code' => '<?php
class One {
public function fooFoo() : void {}
}
$var = new One();
if ($var instanceof One) {
$var->fooFoo();
}',
'error_message' => 'RedundantCondition',
],
'failedTypeResolution' => [
'code' => '<?php
class A { }
/**
* @return void
*/
function fooFoo(A $a) {
if ($a instanceof A) {
}
}',
'error_message' => 'RedundantCondition',
],
'failedTypeResolutionWithDocblock' => [
'code' => '<?php
class A { }
/**
* @param A $a
* @return void
*/
function fooFoo(A $a) {
if ($a instanceof A) {
}
}',
'error_message' => 'RedundantCondition',
],
'typeResolutionFromDocblockAndInstanceof' => [
'code' => '<?php
class A { }
/**
* @param A $a
* @return void
* @psalm-suppress RedundantConditionGivenDocblockType
*/
function fooFoo($a) {
if ($a instanceof A) {
if ($a instanceof A) {
}
}
}',
'error_message' => 'RedundantCondition',
],
'typeResolutionRepeatingConditionWithSingleVar' => [
'code' => '<?php
$a = rand(0, 10) > 5;
if ($a && $a) {}',
'error_message' => 'RedundantCondition',
],
'typeResolutionRepeatingConditionWithVarInMiddle' => [
'code' => '<?php
$a = rand(0, 10) > 5;
$b = rand(0, 10) > 5;
if ($a && $b && $a) {}',
'error_message' => 'RedundantCondition',
],
2018-05-07 20:52:45 +02:00
'typeResolutionRepeatingOredConditionWithSingleVar' => [
'code' => '<?php
2018-05-07 20:52:45 +02:00
$a = rand(0, 10) > 5;
if ($a || $a) {}',
2021-09-22 19:47:37 +02:00
'error_message' => 'TypeDoesNotContainType',
2018-05-07 20:52:45 +02:00
],
'typeResolutionRepeatingOredConditionWithVarInMiddle' => [
'code' => '<?php
2018-05-07 20:52:45 +02:00
$a = rand(0, 10) > 5;
$b = rand(0, 10) > 5;
if ($a || $b || $a) {}',
2021-09-22 19:47:37 +02:00
'error_message' => 'TypeDoesNotContainType',
2018-05-07 20:52:45 +02:00
],
'typeResolutionIsIntAndIsNumeric' => [
'code' => '<?php
$c = rand(0, 10) > 5 ? "hello" : 3;
if (is_int($c) && is_numeric($c)) {}',
'error_message' => 'RedundantCondition',
],
'typeResolutionWithInstanceOfAndNotEmpty' => [
'code' => '<?php
$x = rand(0, 10) > 5 ? new stdClass : null;
if ($x instanceof stdClass && $x) {}',
'error_message' => 'RedundantCondition',
],
'methodWithMeaninglessCheck' => [
'code' => '<?php
class One {
/** @return void */
public function fooFoo() {}
}
class B {
/** @return void */
public function barBar(One $one) {
if (!$one) {
// do nothing
}
$one->fooFoo();
}
}',
'error_message' => 'TypeDoesNotContainType',
],
2018-05-07 07:26:06 +02:00
'twoVarLogicNotNestedWithElseifNegatedInIf' => [
'code' => '<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b): ?string {
if ($a) {
$a = null;
} elseif ($b) {
// do nothing here
} else {
return "bad";
}
if (!$a) return $b;
return $a;
}',
'error_message' => 'RedundantCondition',
],
'refineTypeInMethodCall' => [
'code' => '<?php
class A {}
/** @return ?A */
function getA() {
return rand(0, 1) ? new A : null;
}
2018-01-11 21:50:45 +01:00
function takesA(A $a): void {}
$a = getA();
if ($a instanceof A) {}
/** @psalm-suppress PossiblyNullArgument */
takesA($a);
if ($a instanceof A) {}',
2022-12-25 17:35:47 +01:00
'error_message' => 'RedundantCondition - src' . DIRECTORY_SEPARATOR . 'somefile.php:15',
],
'replaceFalseType' => [
'code' => '<?php
function foo(bool $b) : void {
if (!$b) {
$b = true;
}
if ($b) {}
}',
'error_message' => 'RedundantCondition',
],
'replaceTrueType' => [
'code' => '<?php
function foo(bool $b) : void {
if ($b) {
$b = false;
}
if ($b) {}
}',
'error_message' => 'TypeDoesNotContainType - src' . DIRECTORY_SEPARATOR . 'somefile.php:7',
],
2018-05-14 22:29:51 +02:00
'disallowFloatCheckAfterSettingToVar' => [
'code' => '<?php
function foo(int $x) : void {
if (rand(0, 1)) {
$x = 125;
}
if (is_float($x)) {
echo "Is a float.";
} else {
echo "Is an int.";
}
}',
'error_message' => 'TypeDoesNotContainType - src' . DIRECTORY_SEPARATOR . 'somefile.php:7',
],
'disallowTwoIntValueChecksDueToConditionalOverflow' => [
'code' => '<?php
function foo(int $x) : void {
$x = $x + 1;
if (is_int($x)) {
} elseif (is_int($x)) {}
}',
'error_message' => 'TypeDoesNotContainType - src' . DIRECTORY_SEPARATOR . 'somefile.php:6',
],
2018-05-20 02:31:48 +02:00
'redundantEmptyArray' => [
'code' => '<?php
2018-05-20 02:31:48 +02:00
$x = ["key" => "value"];
if ($x) {
var_export($x);
}',
'error_message' => 'RedundantCondition',
],
'redundantConditionStringNotFalse' => [
'code' => '<?php
function foo(string $s) : void {
if ($s !== false ) {}
}',
'error_message' => 'RedundantCondition',
],
'redundantConditionStringNotTrue' => [
'code' => '<?php
function foo(string $s) : void {
if ($s !== true ) {}
}',
'error_message' => 'RedundantCondition',
],
'redundantConditionAfterRemovingFalse' => [
'code' => '<?php
$s = rand(0, 1) ? rand(0, 5) : false;
if ($s !== false) {
if (is_int($s)) {}
}',
'error_message' => 'RedundantCondition',
],
'redundantConditionAfterRemovingTrue' => [
'code' => '<?php
$s = rand(0, 1) ? rand(0, 5) : true;
if ($s !== true) {
if (is_int($s)) {}
}',
'error_message' => 'RedundantCondition',
],
'impossibleNullEquality' => [
'code' => '<?php
$i = 5;
echo $i !== null;',
'error_message' => 'RedundantCondition',
],
'impossibleTrueEquality' => [
'code' => '<?php
$i = 5;
echo $i !== true;',
'error_message' => 'RedundantCondition',
],
'impossibleFalseEquality' => [
'code' => '<?php
$i = 5;
echo $i !== false;',
'error_message' => 'RedundantCondition',
],
'impossibleNumberEquality' => [
'code' => '<?php
$i = 5;
echo $i !== 3;',
'error_message' => 'RedundantCondition',
],
'alwaysTrueBinaryOp' => [
'code' => '<?php
function foo ($a) : void {
if (!$a) return;
$b = $a && rand(0, 1);
}',
'error_message' => 'RedundantCondition',
'ignored_issues' => ['MissingParamType'],
],
2019-03-16 17:34:48 +01:00
'negatedInstanceof' => [
'code' => '<?php
2019-03-16 17:34:48 +01:00
class A {}
class B {}
function foo(A $a) : void {
if (!$a instanceof B) {}
}',
'error_message' => 'RedundantCondition',
],
'redundantInstanceof' => [
'code' => '<?php
/** @param Exception $a */
function foo($a) : void {
if ($a instanceof \Exception) {}
}',
'error_message' => 'RedundantConditionGivenDocblockType',
],
'preventDocblockTypesBeingIdenticalToTrue' => [
'code' => '<?php
class A {}
/**
* @param A $a
*/
function foo($a, $b) : void {
if ($a === true) {}
}',
'error_message' => 'DocblockTypeContradiction',
],
'preventDocblockTypesBeingIdenticalToTrueReversed' => [
'code' => '<?php
class A {}
/**
* @param A $a
*/
function foo($a, $b) : void {
if (true === $a) {}
}',
'error_message' => 'DocblockTypeContradiction',
],
'preventDocblockTypesBeingIdenticalToFalse' => [
'code' => '<?php
class A {}
/**
* @param A $a
*/
function foo($a, $b) : void {
if ($a === false) {}
}',
'error_message' => 'DocblockTypeContradiction',
],
'preventDocblockTypesBeingIdenticalToFalseReversed' => [
'code' => '<?php
class A {}
/**
* @param A $a
*/
function foo($a, $b) : void {
if (false === $a) {}
}',
'error_message' => 'DocblockTypeContradiction',
],
'preventDocblockTypesBeingSameAsEmptyArrayReversed' => [
'code' => '<?php
class A {}
/**
* @param A $a
*/
function foo($a, $b) : void {
if ([] == $a) {}
}',
'error_message' => 'DocblockTypeContradiction',
],
'preventDocblockTypesBeingIdenticalToEmptyArrayReversed' => [
'code' => '<?php
class A {}
/**
* @param A $a
*/
function foo($a, $b) : void {
if ([] === $a) {}
}',
'error_message' => 'DocblockTypeContradiction',
],
'preventTypesBeingIdenticalToEmptyArrayReversed' => [
'code' => '<?php
class A {}
function foo(A $a, $b) : void {
if ([] === $a) {}
}',
'error_message' => 'TypeDoesNotContainType',
],
'SKIPPED-secondInterfaceAssertionIsRedundant' => [
'code' => '<?php
interface One {}
interface Two {}
/**
* @param One|Two $value
*/
function isOne($value): void {
if ($value instanceof One) {
if ($value instanceof One) {}
}
}',
'error_message' => 'RedundantConditionGivenDocblockType',
],
'errorAfterStatementThatCannotBeConvertedToAssertion' => [
'code' => '<?php
function a(float $b) : void {
if ($b === 0.0) {
return;
}
$a = new stdClass();
if ($a) {}
}',
'error_message' => 'RedundantCondition',
],
2020-10-17 18:30:19 +02:00
'noLongerWarnsAboutRedundancyHere' => [
'code' => '<?php
function a(bool $a, bool $b) : void {
if ($a || $b) {
if ($a) {
} elseif ($b) {
}
}
}',
'error_message' => 'RedundantCondition',
],
'prohibitFalsyChecksOnPropertiesWithMethodCall' => [
'code' => '<?php
class RequestHeaders {
public function has(string $s) : bool {
return true;
}
}
class Request {
public RequestHeaders $headers;
public function __construct(RequestHeaders $headers) {
$this->headers = $headers;
}
}
function lag(Request $req) : void {
if ($req->headers && $req->headers->has("foo")) {}
}',
'error_message' => 'RedundantCondition',
],
'prohibitFalsyChecksOnPropertiesWithoutMethodCall' => [
'code' => '<?php
class RequestHeaders {}
class Request {
public RequestHeaders $headers;
public function __construct(RequestHeaders $headers) {
$this->headers = $headers;
}
}
function lag(Request $req) : void {
if ($req->headers) {}
}',
'error_message' => 'RedundantCondition',
],
'checkResourceTwice' => [
'code' => '<?php
$fp = tmpfile();
if ($fp && is_resource($fp)) {
if (is_resource($fp)) {}
}',
'error_message' => 'RedundantCondition',
],
'preventAlwaysReturningInt' => [
'code' => '<?php
function returnsInt(): int {
return 3;
}
if (is_int(returnsInt())) {}',
'error_message' => 'RedundantCondition',
],
2020-02-22 17:28:24 +01:00
'preventAlwaysReturningSpecificInt' => [
'code' => '<?php
2020-02-22 17:28:24 +01:00
/**
* @return 3|4
*/
function returnsInt(): int {
return rand(0, 1) ? 3 : 4;
}
if (is_int(returnsInt())) {}',
'error_message' => 'RedundantConditionGivenDocblockType',
],
'preventNotAlwaysReturningInt' => [
'code' => '<?php
2020-02-22 17:28:24 +01:00
function returnsInt(): int {
return 3;
}
if (!is_int(returnsInt())) {}',
'error_message' => 'TypeDoesNotContainType',
],
2020-03-31 17:28:36 +02:00
'classAlwaysParent' => [
'code' => '<?php
2020-03-31 17:28:36 +02:00
class AParent {}
class A extends AParent {
public static function load() : A {
return new A();
}
}
$a = A::load();
if ($a instanceof AParent) {}',
'error_message' => 'RedundantCondition',
],
'staticClassIsAlwaysNull' => [
'code' => '<?php
/**
* @psalm-consistent-constructor
*/
2020-03-31 17:28:36 +02:00
class A {
/**
* @return ?static
*/
public static function load() {
return rand(0, 1)
? null
: new static();
}
}
$a = A::load();
if ($a && $a instanceof A) {}',
'error_message' => 'RedundantConditionGivenDocblockType',
],
'classStringNotEmpty' => [
'code' => '<?php
function foo(object $o) : void {
$oc = get_class($o);
if ($oc) {}
}',
'error_message' => 'RedundantCondition',
],
'leftCannotBeTrue' => [
'code' => '<?php
/** @psalm-type F = ""|"0" */
/**
* @param F $a
* @param F $b
*/
function foo(string $a, string $b): void {
if ($a || $b) {}
}',
'error_message' => 'DocblockTypeContradiction',
],
'rightCannotBeTrue' => [
'code' => '<?php
/** @param false $a */
function foo(bool $a): void {
if (rand(0, 1) || $a) {
echo "a or b";
}
}',
'error_message' => 'DocblockTypeContradiction',
],
2021-09-22 19:47:37 +02:00
'OrTrue' => [
'code' => '<?php
2021-09-22 19:47:37 +02:00
if(rand(0,1) || true){}',
'error_message' => 'RedundantCondition',
],
'AndTrue' => [
'code' => '<?php
2021-09-22 19:47:37 +02:00
if(rand(0,1) && true){}',
'error_message' => 'RedundantCondition',
],
'OrFalse' => [
'code' => '<?php
2021-09-22 19:47:37 +02:00
if(rand(0,1) || false){}',
'error_message' => 'TypeDoesNotContainType',
],
'AndFalse' => [
'code' => '<?php
2021-09-22 19:47:37 +02:00
if(rand(0,1) && false){}',
'error_message' => 'TypeDoesNotContainType',
2021-10-21 20:26:00 +02:00
],
'alwaysTrueAssign' => [
'code' => '<?php
2021-10-21 20:26:00 +02:00
class A{
public function get(): stdClass{ return new stdClass;}
}
$a = new A();
if ($c = $a->get()){
}
',
'error_message' => 'RedundantCondition',
],
'secondFalsyTwiceWithoutChange' => [
'code' => '<?php
/**
* @param array{a?:int,b?:string} $p
*/
function f(array $p) : void {
if (!$p) {
throw new RuntimeException("");
}
assert(!!$p);
}',
2022-12-18 17:15:15 +01:00
'error_message' => 'RedundantCondition',
],
'secondFalsyTwiceWithoutChangeWithElse' => [
'code' => '<?php
/**
* @param array{a?:int,b?:string} $p
*/
function f(array $p) : void {
if (!$p) {
throw new RuntimeException("");
} else {}
assert(!!$p);
}',
2022-12-18 17:15:15 +01:00
'error_message' => 'RedundantCondition',
],
2022-12-25 18:25:40 +01:00
'from_docblock should be kept when removing types' => [
2022-12-24 13:00:40 +01:00
'code' => '<?php
/**
2022-12-25 18:25:40 +01:00
* @see https://github.com/vimeo/psalm/issues/8932
*
2022-12-24 13:00:40 +01:00
* @param array|null $value
2022-12-25 18:25:40 +01:00
*
* @return null
2022-12-24 13:00:40 +01:00
*/
function reverseTransform($value)
{
if (null === $value) {
return null;
}
if (!\is_array($value)) {
throw new \Exception("array");
}
return null;
}',
'error_message' => 'DocblockTypeContradiction',
],
];
}
}