1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-22 22:01:48 +01:00
psalm/tests/TypeAlgebraTest.php

751 lines
25 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 15:50:45 -05:00
function takesString(string $s): void {}
2018-01-11 15:50:45 -05: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-26 20:05:57 -04:00
}',
],
'threeVarLogic' => [
'<?php
2018-01-11 15:50:45 -05:00
function takesString(string $s): void {}
2018-01-11 15:50:45 -05: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-26 20:05:57 -04:00
}',
],
'twoVarLogicNotNested' => [
'<?php
2018-01-11 15:50:45 -05:00
function foo(?string $a, ?string $b): string {
if (!$a && !$b) return "bad";
if (!$a) return $b;
return $a;
2017-05-26 20:05:57 -04:00
}',
],
'twoVarLogicNotNestedWithAllPathsReturning' => [
'<?php
2018-01-11 15:50:45 -05:00
function foo(?string $a, ?string $b): string {
if (!$a && !$b) {
return "bad";
} else {
if (!$a) {
return $b;
} else {
return $a;
}
}
2017-05-26 20:05:57 -04:00
}',
],
'twoVarLogicNotNestedWithAssignmentBeforeReturn' => [
'<?php
2018-01-11 15:50:45 -05: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-26 20:05:57 -04:00
}',
],
'invertedTwoVarLogicNotNested' => [
'<?php
2018-01-11 15:50:45 -05:00
function foo(?string $a, ?string $b): string {
if ($a || $b) {
// do nothing
} else {
return "bad";
}
if (!$a) return $b;
return $a;
2017-05-26 20:05:57 -04:00
}',
],
'invertedTwoVarLogicNotNestedWithAssignmentBeforeReturn' => [
'<?php
2018-01-11 15:50:45 -05: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-26 20:05:57 -04:00
}',
],
'twoVarLogicNotNestedWithElseifAndNoNegations' => [
'<?php
2018-01-11 15:50:45 -05: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-26 20:05:57 -04:00
}',
],
'threeVarLogicNotNestedWithNoRedefinitions' => [
'<?php
2018-01-11 15:50:45 -05: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-26 20:05:57 -04:00
}',
],
'threeVarLogicNotNestedAndOrWithNoRedefinitions' => [
'<?php
2018-01-11 15:50:45 -05: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-26 20:05:57 -04:00
}',
],
'twoVarLogicNotNestedWithElseifCorrectlyNegatedInElseIf' => [
'<?php
2018-01-11 15:50:45 -05: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-26 20:05:57 -04:00
}',
],
'nestedReassignment' => [
'<?php
2018-01-11 15:50:45 -05:00
function foo(?string $a): void {
if ($a === null) {
$a = "blah-blah";
} else {
$a = rand(0, 1) ? "blah" : null;
if ($a === null) {
}
}
2017-05-26 20:05:57 -04:00
}',
],
'twoVarLogicNotNestedWithElseifCorrectlyReinforcedInIf' => [
'<?php
class A {}
class B extends A {}
2018-01-11 15:50:45 -05: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-26 20:05:57 -04:00
}',
],
'differentValueChecks' => [
'<?php
2018-01-11 15:50:45 -05:00
function foo(string $a): void {
if ($a === "foo") {
// do something
} elseif ($a === "bar") {
// can never get here
}
2017-05-26 20:05:57 -04:00
}',
],
'repeatedSet' => [
'<?php
2018-01-11 15:50:45 -05: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-26 20:05:57 -04:00
}',
],
'repeatedSetInsideWhile' => [
'<?php
2018-01-11 15:50:45 -05: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-26 20:05:57 -04:00
}',
],
'byRefAssignment' => [
'<?php
2018-01-11 15:50:45 -05: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 15:50:45 -05: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 15:50:45 -05: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-26 20:05:57 -04:00
}',
],
'noParadoxInLoop' => [
'<?php
2018-01-11 15:50:45 -05:00
function paradox2(): void {
$condition = rand() % 2 > 0;
if (!$condition) {
foreach ([1, 2] as $value) {
if ($condition) { }
$condition = true;
}
}
}',
],
'noParadoxInListAssignment' => [
'<?php
2018-01-11 15:50:45 -05:00
function foo(string $a): void {
if (!$a) {
list($a) = explode(":", "a:b");
if ($a) { }
}
}',
],
'noParadoxAfterAssignment' => [
'<?php
2018-01-11 15:50:45 -05:00
function get_bool(): bool {
return rand() % 2 > 0;
}
2018-01-11 15:50:45 -05: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 14:02:06 -04: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 14:02:06 -04:00
}',
],
'noParadoxInCatch' => [
'<?php
2018-01-11 15:50:45 -05: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 15:50:45 -05: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 14:02:06 -04: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 15:50:45 -05:00
public function foo(): void {}
}
2018-01-11 15:50:45 -05:00
function takesA(A $a): void {
if (get_class($a) === "B") {
$a->foo();
}
}',
],
'ifNotEqualsGetClass' => [
'<?php
class A {}
class B extends A {
2018-01-11 15:50:45 -05:00
public function foo(): void {}
}
2018-01-11 15:50:45 -05:00
function takesA(A $a): void {
if (get_class($a) !== "B") {
// do nothing
} else {
$a->foo();
}
}',
],
'nestedCheckWithSingleVarPerLevel' => [
'<?php
2018-01-11 15:50:45 -05:00
function foo(?stdClass $a, ?stdClass $b): void {
if ($a) {
if ($b) {}
}
}',
],
'nestedCheckWithTwoVarsPerLevel' => [
'<?php
2018-01-11 15:50:45 -05:00
function foo(?stdClass $a, ?stdClass $b, ?stdClass $c, ?stdClass $d): void {
if ($a && $b) {
if ($c && $d) {}
}
}',
],
'nestedCheckWithReturn' => [
'<?php
2018-01-11 15:50:45 -05: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-10 20:21:21 -05:00
'noParadoxForGetopt' => [
'<?php
$options = getopt("t:");
try {
if (!isset($options["t"])) {
throw new Exception("bad");
}
} catch (Exception $e) {}',
],
2017-12-18 00:06:23 -05:00
// because we only support expressions in CNF atm
'SKIPPED-instanceofInOr' => [
'<?php
class A {}
class B extends A {}
class C extends A {}
2018-01-11 15:50:45 -05:00
function takesA(A $a): void {}
2017-12-18 00:06:23 -05:00
2018-01-11 15:50:45 -05:00
function foo(?A $a): void {
2017-12-18 00:06:23 -05:00
if ($a instanceof B
|| ($a instanceof C && rand(0, 1))
) {
takesA($a);
}
}',
],
'instanceofInCNFOr' => [
'<?php
class A {}
class B extends A {}
class C extends A {}
2018-01-11 15:50:45 -05:00
function takesA(A $a): void {}
2018-01-11 15:50:45 -05: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 15:50:45 -05:00
function takesA(A $a): void {}
2018-01-11 15:50:45 -05:00
function foo(?A $a, ?B $b): void {
if ($a === null || $b === null || rand(0, 1)) {
// do nothing
} else {
takesA($a);
}
}',
],
];
}
/**
* @return array
*/
public function providerFileCheckerInvalidCodeParse()
{
return [
'threeVarLogicWithChange' => [
'<?php
2018-01-11 15:50:45 -05:00
function takesString(string $s): void {}
2018-01-11 15:50:45 -05: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 15:50:45 -05:00
function takesString(string $s): void {}
2018-01-11 15:50:45 -05: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 15:50:45 -05: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 15:50:45 -05: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 15:50:45 -05: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;
}',
'error_message' => 'NullableReturnStatement',
],
'twoVarLogicNotNestedWithElseifIncorrectlyReinforcedInIf' => [
'<?php
2018-01-11 15:50:45 -05:00
function foo(?string $a, ?string $b): string {
if ($a) {
$a = "";
} elseif ($b) {
// do nothing
} else {
return "bad";
}
if (!$a) return $b;
return $a;
}',
'error_message' => 'NullableReturnStatement',
],
'repeatedIfStatements' => [
'<?php
/** @return string|null */
function foo(?string $a) {
if ($a) {
return $a;
}
if ($a) {
}
}',
2017-05-26 20:05:57 -04:00
'error_message' => 'ParadoxicalCondition',
],
'repeatedConditionals' => [
'<?php
2018-01-11 15:50:45 -05:00
function foo(?string $a): void {
if ($a) {
// do something
} elseif ($a) {
// can never get here
}
}',
2017-05-26 20:05:57 -04:00
'error_message' => 'ParadoxicalCondition',
],
'repeatedAndConditional' => [
'<?php
2018-01-11 15:50:45 -05: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 15:50:45 -05: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 15:50:45 -05:00
function foo(string $a, string $b): void {
if ($a || $b) {
echo "a";
} elseif ($a) {
echo "b";
}
}',
'error_message' => 'ParadoxicalCondition',
],
];
2017-04-02 17:37:56 -04:00
}
}