1
0
mirror of https://github.com/danog/psalm.git synced 2024-11-27 04:45:20 +01:00
psalm/tests/TypeAlgebraTest.php

944 lines
32 KiB
PHP
Raw Normal View History

<?php
namespace Psalm\Tests;
class TypeAlgebraTest extends TestCase
{
use Traits\FileCheckerInvalidCodeParseTestTrait;
use Traits\FileCheckerValidCodeParseTestTrait;
/**
* @return array
*/
public function providerFileCheckerValidCodeParse()
{
return [
'twoVarLogic' => [
'<?php
2018-01-11 21:50:45 +01:00
function takesString(string $s): void {}
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b): void {
if ($a !== null || $b !== null) {
if ($a !== null) {
$c = $a;
} else {
$c = $b;
}
takesString($c);
}
2017-05-27 02:05:57 +02:00
}',
],
'threeVarLogic' => [
'<?php
2018-01-11 21:50:45 +01:00
function takesString(string $s): void {}
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b, ?string $c): void {
if ($a !== null || $b !== null || $c !== null) {
if ($a !== null) {
$d = $a;
} elseif ($b !== null) {
$d = $b;
} else {
$d = $c;
}
takesString($d);
}
2017-05-27 02:05:57 +02:00
}',
],
'twoVarLogicNotNested' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b): string {
if (!$a && !$b) return "bad";
if (!$a) return $b;
return $a;
2017-05-27 02:05:57 +02:00
}',
],
'twoVarLogicNotNestedWithAllPathsReturning' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b): string {
if (!$a && !$b) {
return "bad";
} else {
if (!$a) {
return $b;
} else {
return $a;
}
}
2017-05-27 02:05:57 +02:00
}',
],
'twoVarLogicNotNestedWithAssignmentBeforeReturn' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b): string {
if (!$a && !$b) {
$a = 5;
return "bad";
}
if (!$a) {
$a = 7;
return $b;
}
return $a;
2017-05-27 02:05:57 +02:00
}',
],
'invertedTwoVarLogicNotNested' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b): string {
if ($a || $b) {
// do nothing
} else {
return "bad";
}
if (!$a) return $b;
return $a;
2017-05-27 02:05:57 +02:00
}',
],
'invertedTwoVarLogicNotNestedWithAssignmentBeforeReturn' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b): string {
if ($a || $b) {
// do nothing
} else {
$a = 5;
return "bad";
}
if (!$a) return $b;
return $a;
2017-05-27 02:05:57 +02:00
}',
],
'twoVarLogicNotNestedWithElseifAndNoNegations' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b): string {
if ($a) {
// do nothing
} elseif ($b) {
// do nothing here
} else {
return "bad";
}
if (!$a) return $b;
return $a;
2017-05-27 02:05:57 +02:00
}',
],
2018-05-07 07:26:06 +02:00
'threeVarLogicNotNestedWithNoRedefinitionsWithClasses' => [
'<?php
function foo(?stdClass $a, ?stdClass $b, ?stdClass $c): stdClass {
if ($a) {
// do nothing
} elseif ($b) {
// do nothing here
} elseif ($c) {
// do nothing here
} else {
return new stdClass;
}
if (!$a && !$b) return $c;
if (!$a) return $b;
return $a;
}',
],
'threeVarLogicNotNestedWithNoRedefinitionsWithStrings' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b, ?string $c): string {
if ($a) {
// do nothing
} elseif ($b) {
// do nothing here
} elseif ($c) {
// do nothing here
} else {
return "bad";
}
if (!$a && !$b) return $c;
if (!$a) return $b;
return $a;
2017-05-27 02:05:57 +02:00
}',
],
'threeVarLogicNotNestedAndOrWithNoRedefinitions' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b, ?string $c): string {
if ($a) {
// do nothing
} elseif ($b || $c) {
// do nothing here
} else {
return "bad";
}
if (!$a && !$b) return $c;
if (!$a) return $b;
return $a;
2017-05-27 02:05:57 +02:00
}',
],
'twoVarLogicNotNestedWithElseifCorrectlyNegatedInElseIf' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b): string {
if ($a) {
// do nothing here
} elseif ($b) {
$a = null;
} else {
return "bad";
}
if (!$a) return $b;
return $a;
2017-05-27 02:05:57 +02:00
}',
],
'nestedReassignment' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a): void {
if ($a === null) {
$a = "blah-blah";
} else {
$a = rand(0, 1) ? "blah" : null;
if ($a === null) {
}
}
2017-05-27 02:05:57 +02:00
}',
],
'twoVarLogicNotNestedWithElseifCorrectlyReinforcedInIf' => [
'<?php
class A {}
class B extends A {}
2018-01-11 21:50:45 +01:00
function foo(?A $a, ?A $b): A {
if ($a) {
$a = new B;
} elseif ($b) {
// do nothing
} else {
return new A;
}
if (!$a) return $b;
return $a;
2017-05-27 02:05:57 +02:00
}',
],
'differentValueChecks' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(string $a): void {
if ($a === "foo") {
// do something
} elseif ($a === "bar") {
// can never get here
}
2017-05-27 02:05:57 +02:00
}',
],
'repeatedSet' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(): void {
if ($a = rand(0, 1) ? "" : null) {
return;
}
if (rand(0, 1)) {
$a = rand(0, 1) ? "hello" : null;
if ($a) {
}
}
2017-05-27 02:05:57 +02:00
}',
],
'repeatedSetInsideWhile' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(): void {
if ($a = rand(0, 1) ? "" : null) {
return;
} else {
while (rand(0, 1)) {
$a = rand(0, 1) ? "hello" : null;
}
if ($a) {
}
}
2017-05-27 02:05:57 +02:00
}',
],
'byRefAssignment' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(): void {
preg_match("/hello/", "hello molly", $matches);
if (!$matches) {
return;
}
preg_match("/hello/", "hello dolly", $matches);
if (!$matches) {
}
}',
],
'orConditionalAfterAndConditional' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(string $a, string $b): void {
if ($a && $b) {
echo "a";
} elseif ($a || $b) {
echo "b";
}
}',
],
'issetOnOneStringAfterAnother' => [
'<?php
/** @param string[] $arr */
2018-01-11 21:50:45 +01:00
function foo(array $arr): void {
$a = "a";
if (!isset($arr[$a])) {
return;
}
foreach ([0, 1, 2, 3] as $i) {
if (!isset($arr[$a . $i])) {
echo "a";
}
$a = "hello";
}
2017-05-27 02:05:57 +02:00
}',
],
'noParadoxInLoop' => [
'<?php
2018-01-11 21:50:45 +01:00
function paradox2(): void {
$condition = rand() % 2 > 0;
if (!$condition) {
foreach ([1, 2] as $value) {
if ($condition) { }
$condition = true;
}
}
}',
],
'noParadoxInListAssignment' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(string $a): void {
if (!$a) {
list($a) = explode(":", "a:b");
if ($a) { }
}
}',
],
'noParadoxAfterAssignment' => [
'<?php
2018-01-11 21:50:45 +01:00
function get_bool(): bool {
return rand() % 2 > 0;
}
2018-01-11 21:50:45 +01:00
function leftover(): bool {
$res = get_bool();
if ($res === false) {
return true;
}
$res = get_bool();
if ($res === false) {
return false;
}
return true;
2017-10-12 20:02:06 +02:00
}',
],
'noParadoxAfterArrayAppending' => [
'<?php
/** @return array|false */
function array_append2(array $errors) {
if ($errors) {
return $errors;
}
$errors[] = "deterministic";
if ($errors) {
return false;
}
return $errors;
}
/** @return array|false */
function array_append(array $errors) {
if ($errors) {
return $errors;
}
if (rand() % 2 > 0) {
$errors[] = "unlucky";
}
if ($errors) {
return false;
}
return $errors;
2017-10-12 20:02:06 +02:00
}',
],
'noParadoxInCatch' => [
'<?php
2018-01-11 21:50:45 +01:00
function maybe_returns_array(): ?array {
if (rand() % 2 > 0) {
return ["key" => "value"];
}
if (rand() % 3 > 0) {
throw new Exception("An exception occurred");
}
return null;
}
2018-01-11 21:50:45 +01:00
function try_catch_check(): array {
$arr = null;
try {
$arr = maybe_returns_array();
if (!$arr) { return []; }
} catch (Exception $e) {
if (!$arr) { return []; }
}
return $arr;
2017-10-12 20:02:06 +02:00
}',
],
'lotsaTruthyStatements' => [
'<?php
class A {
/**
* @var ?string
*/
public $a = null;
/**
* @var ?string
*/
public $b = null;
}
function f(A $obj): string {
if (($obj->a !== null) == true) {
return $obj->a; // definitely not null
} elseif (!is_null($obj->b) == true) {
return $obj->b;
} else {
throw new \InvalidArgumentException("$obj->a or $obj->b must be set");
}
}',
],
'lotsaFalsyStatements' => [
'<?php
class A {
/**
* @var ?string
*/
public $a = null;
/**
* @var ?string
*/
public $b = null;
}
function f(A $obj): string {
if (($obj->a === null) == false) {
return $obj->a; // definitely not null
} elseif (is_null($obj->b) == false) {
return $obj->b;
} else {
throw new \InvalidArgumentException("$obj->a or $obj->b must be set");
}
}',
],
'ifGetClass' => [
'<?php
class A {}
class B extends A {
2018-01-11 21:50:45 +01:00
public function foo(): void {}
}
2018-01-11 21:50:45 +01:00
function takesA(A $a): void {
if (get_class($a) === "B") {
$a->foo();
}
}',
],
'ifNotEqualsGetClass' => [
'<?php
class A {}
class B extends A {
2018-01-11 21:50:45 +01:00
public function foo(): void {}
}
2018-01-11 21:50:45 +01:00
function takesA(A $a): void {
if (get_class($a) !== "B") {
// do nothing
} else {
$a->foo();
}
}',
],
'nestedCheckWithSingleVarPerLevel' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?stdClass $a, ?stdClass $b): void {
if ($a) {
if ($b) {}
}
}',
],
'nestedCheckWithTwoVarsPerLevel' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?stdClass $a, ?stdClass $b, ?stdClass $c, ?stdClass $d): void {
if ($a && $b) {
if ($c && $d) {}
}
}',
],
'nestedCheckWithReturn' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?stdClass $a, ?stdClass $b): void {
if ($a === null) {
return;
}
if ($b) {
echo "hello";
}
}',
],
'propertyFetchAfterNotNullCheck' => [
'<?php
class A {
/** @var ?string */
public $foo;
}
$a = new A;
if ($a->foo === null) {
$a->foo = "hello";
exit;
}
if ($a->foo === "somestring") {}',
],
'propertyFetchAfterNotNullCheckInElseif' => [
'<?php
class A {
/** @var ?string */
public $foo;
}
if (rand(0, 10) > 5) {
} elseif (($a = new A) && $a->foo) {}',
],
2017-12-11 02:21:21 +01:00
'noParadoxForGetopt' => [
'<?php
$options = getopt("t:");
try {
if (!isset($options["t"])) {
throw new Exception("bad");
}
} catch (Exception $e) {}',
],
2018-05-06 02:32:04 +02:00
'instanceofInOr' => [
2017-12-18 06:06:23 +01:00
'<?php
class A {}
class B extends A {}
class C extends A {}
2018-01-11 21:50:45 +01:00
function takesA(A $a): void {}
2017-12-18 06:06:23 +01:00
2018-01-11 21:50:45 +01:00
function foo(?A $a): void {
2017-12-18 06:06:23 +01:00
if ($a instanceof B
|| ($a instanceof C && rand(0, 1))
) {
takesA($a);
}
}',
],
2018-05-06 02:32:04 +02:00
'instanceofInOrNegated' => [
'<?php
class A {}
class B extends A {}
class C extends A {}
function takesA(A $a): void {}
function foo(?A $a, ?A $b, ?A $c): void {
2018-05-07 07:26:06 +02:00
if (!$a || ($b && $c)) {
2018-05-06 02:32:04 +02:00
return;
}
takesA($a);
}',
],
'instanceofInBothOrs' => [
'<?php
class A {}
class B extends A {}
class C extends A {}
function takesA(A $a): void {}
function foo(?A $a): void {
if (($a instanceof B && rand(0, 1))
|| ($a instanceof C && rand(0, 1))
) {
takesA($a);
}
}',
],
'instanceofInBothOrsWithSecondVar' => [
'<?php
class A {}
class B extends A {}
class C extends A {}
function takesA(A $a): void {}
function foo(?A $a, ?A $b): void {
if (($a instanceof B && $b instanceof B)
|| ($a instanceof C && $b instanceof C)
) {
takesA($a);
takesA($b);
}
}',
],
'explosionOfCNF' => [
'<?php
class A {
/** @var ?string */
public $foo;
/** @var ?string */
public $bar;
}
$a1 = rand(0, 1) ? new A() : null;
$a4 = rand(0, 1) ? new A() : null;
$a5 = rand(0, 1) ? new A() : null;
$a7 = rand(0, 1) ? new A() : null;
$a8 = rand(0, 1) ? new A() : null;
if ($a1 || (($a4 && $a5) || ($a7 && $a8))) {}',
],
'instanceofInCNFOr' => [
'<?php
class A {}
class B extends A {}
class C extends A {}
2018-01-11 21:50:45 +01:00
function takesA(A $a): void {}
2018-01-11 21:50:45 +01:00
function foo(?A $a): void {
$c = rand(0, 1);
if (($a instanceof B || $a instanceof C)
&& ($a instanceof B || $c)
) {
takesA($a);
}
}',
],
'reconcileNestedOrsInElse' => [
'<?php
class A {}
class B {}
2018-01-11 21:50:45 +01:00
function takesA(A $a): void {}
2018-01-11 21:50:45 +01:00
function foo(?A $a, ?B $b): void {
if ($a === null || $b === null || rand(0, 1)) {
// do nothing
} else {
takesA($a);
}
}',
],
'getClassComparison' => [
'<?php
class Foo {
public function bar() : void {}
}
class Bar extends Foo{
public function bar() : void {}
}
class Baz {
public function test(Foo $foo) : void {
if (get_class($foo) !== Foo::class) {
// do nothing
} else {
$foo->bar();
}
}
}',
],
'noParadoxAfterConditionalAssignment' => [
'<?php
if ($a = rand(0, 5)) {
echo $a;
} elseif ($a = rand(0, 5)) {
echo $a;
}',
],
2018-05-06 02:52:10 +02:00
'callWithNonNullInTernary' => [
'<?php
function sayHello(?int $a, ?int $b): void {
if ($a === null && $b === null) {
throw new \LogicException();
}
takesInt($a !== null ? $a : $b);
}
function takesInt(int $c) : void {}',
],
'callWithNonNullInIf' => [
'<?php
function sayHello(?int $a, ?int $b): void {
if ($a === null && $b === null) {
throw new \LogicException();
}
if ($a !== null) {
takesInt($a);
} else {
takesInt($b);
}
}
function takesInt(int $c) : void {}',
],
'callWithNonNullInIfWithCallInElseif' => [
'<?php
function sayHello(?int $a, ?int $b): void {
if ($a === null && $b === null) {
throw new \LogicException();
}
if ($a !== null) {
takesInt($a);
} elseif (rand(0, 1)) {
takesInt($b);
}
}
function takesInt(int $c) : void {}',
],
'typeSimplification' => [
'<?php
class A {}
class B extends A {}
function foo(A $a, A $b) : ?B {
if (($a instanceof B || !$b instanceof B) && $a instanceof B && $b instanceof B) {
return $a;
}
return null;
}',
],
2018-05-07 07:26:06 +02:00
'instanceofNoRedundant' => [
'<?php
function logic(Bar $a, ?Bar $b) : void {
if ((!$a instanceof Foo || !$b instanceof Foo)
&& (!$a instanceof Foo || !$b instanceof Bar)
&& (!$a instanceof Bar || !$b instanceof Foo)
&& (!$a instanceof Bar || !$b instanceof Bar)
) {
} else {
if ($b instanceof Foo) {}
}
}
class Foo {}
class Bar extends Foo {}
class Bat extends Foo {}',
],
];
}
/**
* @return array
*/
public function providerFileCheckerInvalidCodeParse()
{
return [
'threeVarLogicWithChange' => [
'<?php
2018-01-11 21:50:45 +01:00
function takesString(string $s): void {}
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b, ?string $c): void {
if ($a !== null || $b !== null || $c !== null) {
$c = null;
if ($a !== null) {
$d = $a;
} elseif ($b !== null) {
$d = $b;
} else {
$d = $c;
}
takesString($d);
}
}',
'error_message' => 'PossiblyNullArgument',
],
'threeVarLogicWithException' => [
'<?php
2018-01-11 21:50:45 +01:00
function takesString(string $s): void {}
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b, ?string $c): void {
if ($a !== null || $b !== null || $c !== null) {
if ($c !== null) {
throw new \Exception("bad");
}
if ($a !== null) {
$d = $a;
} elseif ($b !== null) {
$d = $b;
} else {
$d = $c;
}
takesString($d);
}
}',
'error_message' => 'PossiblyNullArgument',
],
'invertedTwoVarLogicNotNestedWithVarChange' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b): string {
if ($a !== null || $b !== null) {
$b = null;
} else {
return "bad";
}
if ($a !== null) return $b;
return $a;
}',
'error_message' => 'NullableReturnStatement',
],
'invertedTwoVarLogicNotNestedWithElseif' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b): string {
if (rand(0, 1)) {
// do nothing
} elseif ($a || $b) {
// do nothing here
} else {
return "bad";
}
if (!$a) return $b;
return $a;
}',
'error_message' => 'NullableReturnStatement',
],
'threeVarLogicWithElseifAndAnd' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b, ?string $c): string {
if ($a) {
// do nothing
} elseif ($b && $c) {
// do nothing here
} else {
return "bad";
}
if (!$a && !$b) return $c;
if (!$a) return $b;
return $a;
}',
2018-05-06 02:32:04 +02:00
'error_message' => 'ParadoxicalCondition',
],
'twoVarLogicNotNestedWithElseifIncorrectlyReinforcedInIf' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a, ?string $b): string {
if ($a) {
$a = "";
} elseif ($b) {
// do nothing
} else {
return "bad";
}
if (!$a) return $b;
return $a;
}',
2018-05-07 07:26:06 +02:00
'error_message' => 'RedundantCondition',
],
'repeatedIfStatements' => [
'<?php
/** @return string|null */
function foo(?string $a) {
if ($a) {
return $a;
}
if ($a) {
}
}',
2017-05-27 02:05:57 +02:00
'error_message' => 'ParadoxicalCondition',
],
'repeatedConditionals' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(?string $a): void {
if ($a) {
// do something
} elseif ($a) {
// can never get here
}
}',
2017-05-27 02:05:57 +02:00
'error_message' => 'ParadoxicalCondition',
],
'repeatedAndConditional' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(string $a, string $b): void {
if ($a && $b) {
echo "a";
} elseif ($a && $b) {
echo "b";
}
}',
'error_message' => 'ParadoxicalCondition',
],
'andConditionalAfterOrConditional' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(string $a, string $b): void {
if ($a || $b) {
echo "a";
} elseif ($a && $b) {
echo "b";
}
}',
'error_message' => 'ParadoxicalCondition',
],
'repeatedVarFromOrConditional' => [
'<?php
2018-01-11 21:50:45 +01:00
function foo(string $a, string $b): void {
if ($a || $b) {
echo "a";
} elseif ($a) {
echo "b";
}
}',
'error_message' => 'ParadoxicalCondition',
],
'typeDoesntEqualType' => [
'<?php
$a = "hello";
$b = 5;
if ($a !== $b) {}',
'error_message' => 'RedundantCondition',
],
];
2017-04-02 23:37:56 +02:00
}
}