mirror of
https://github.com/danog/psalm.git
synced 2024-12-13 01:37:23 +01:00
397 lines
12 KiB
PHP
397 lines
12 KiB
PHP
<?php
|
|
namespace Psalm\Tests;
|
|
|
|
class Php70Test extends TestCase
|
|
{
|
|
use Traits\InvalidCodeAnalysisTestTrait;
|
|
use Traits\ValidCodeAnalysisTestTrait;
|
|
|
|
/**
|
|
* @return iterable<string,array{string,assertions?:array<string,string>,error_levels?:string[]}>
|
|
*/
|
|
public function providerValidCodeParse()
|
|
{
|
|
return [
|
|
'functionTypeHints' => [
|
|
'<?php
|
|
function indexof(string $haystack, string $needle): int
|
|
{
|
|
$pos = strpos($haystack, $needle);
|
|
|
|
if ($pos === false) {
|
|
return -1;
|
|
}
|
|
|
|
return $pos;
|
|
}
|
|
|
|
$a = indexof("arr", "a");',
|
|
'assertions' => [
|
|
'$a' => 'int',
|
|
],
|
|
],
|
|
'methodTypeHints' => [
|
|
'<?php
|
|
class Foo {
|
|
public static function indexof(string $haystack, string $needle): int
|
|
{
|
|
$pos = strpos($haystack, $needle);
|
|
|
|
if ($pos === false) {
|
|
return -1;
|
|
}
|
|
|
|
return $pos;
|
|
}
|
|
}
|
|
|
|
$a = Foo::indexof("arr", "a");',
|
|
'assertions' => [
|
|
'$a' => 'int',
|
|
],
|
|
],
|
|
'nullCoalesce' => [
|
|
'<?php
|
|
$arr = ["hello", "goodbye"];
|
|
$a = $arr[rand(0, 10)] ?? null;',
|
|
'assertions' => [
|
|
'$a' => 'string|null',
|
|
],
|
|
],
|
|
'nullCoalesceWithNullableOnLeft' => [
|
|
'<?php
|
|
/** @return ?string */
|
|
function foo() {
|
|
return rand(0, 10) > 5 ? "hello" : null;
|
|
}
|
|
$a = foo() ?? "goodbye";',
|
|
'assertions' => [
|
|
'$a' => 'string',
|
|
],
|
|
],
|
|
'nullCoalesceWithReference' => [
|
|
'<?php
|
|
$var = 0;
|
|
($a =& $var) ?? "hello";',
|
|
'assertions' => [
|
|
'$a' => 'mixed',
|
|
],
|
|
],
|
|
'spaceship' => [
|
|
'<?php
|
|
$a = 1 <=> 1;',
|
|
'assertions' => [
|
|
'$a' => 'int',
|
|
],
|
|
],
|
|
'defineArray' => [
|
|
'<?php
|
|
define("ANIMALS", [
|
|
"dog",
|
|
"cat",
|
|
"bird"
|
|
]);
|
|
|
|
$a = ANIMALS[1];',
|
|
'assertions' => [
|
|
'$a' => 'string',
|
|
],
|
|
],
|
|
'anonymousClassLogger' => [
|
|
'<?php
|
|
interface Logger {
|
|
/** @return void */
|
|
public function log(string $msg);
|
|
}
|
|
|
|
class Application {
|
|
/** @var Logger|null */
|
|
private $logger;
|
|
|
|
/** @return void */
|
|
public function setLogger(Logger $logger) {
|
|
$this->logger = $logger;
|
|
}
|
|
}
|
|
|
|
$app = new Application;
|
|
$app->setLogger(new class implements Logger {
|
|
/** @return void */
|
|
public function log(string $msg) {
|
|
echo $msg;
|
|
}
|
|
});',
|
|
],
|
|
'anonymousClassFunctionReturnType' => [
|
|
'<?php
|
|
$class = new class {
|
|
public function f(): int {
|
|
return 42;
|
|
}
|
|
};
|
|
|
|
function g(int $i): int {
|
|
return $i;
|
|
}
|
|
|
|
$x = g($class->f());',
|
|
],
|
|
'anonymousClassStatement' => [
|
|
'<?php
|
|
new class {};',
|
|
],
|
|
'anonymousClassTwoFunctions' => [
|
|
'<?php
|
|
interface I {}
|
|
|
|
class A
|
|
{
|
|
/** @var ?I */
|
|
protected $i;
|
|
|
|
public function foo(): void
|
|
{
|
|
$this->i = new class implements I {};
|
|
}
|
|
|
|
public function foo2(): void {} // commenting this line out fixes
|
|
}',
|
|
],
|
|
'anonymousClassExtendsWithThis' => [
|
|
'<?php
|
|
class A {
|
|
public function foo() : void {}
|
|
}
|
|
$class = new class extends A {
|
|
public function f(): int {
|
|
$this->foo();
|
|
return 42;
|
|
}
|
|
};',
|
|
],
|
|
'returnAnonymousClass' => [
|
|
'<?php
|
|
/** @return object */
|
|
function getNewAnonymousClass() {
|
|
return new class {};
|
|
}',
|
|
],
|
|
'returnAnonymousClassInClass' => [
|
|
'<?php
|
|
class A {
|
|
/** @return object */
|
|
public function getNewAnonymousClass() {
|
|
return new class {};
|
|
}
|
|
}',
|
|
],
|
|
'generatorWithReturn' => [
|
|
'<?php
|
|
/**
|
|
* @return Generator<int,int>
|
|
* @psalm-generator-return string
|
|
*/
|
|
function fooFoo(int $i): Generator {
|
|
if ($i === 1) {
|
|
return "bash";
|
|
}
|
|
|
|
yield 1;
|
|
}',
|
|
],
|
|
'generatorDelegation' => [
|
|
'<?php
|
|
/**
|
|
* @return Generator<int, int, mixed, int>
|
|
*/
|
|
function count_to_ten(): Generator {
|
|
yield 1;
|
|
yield 2;
|
|
yield from [3, 4];
|
|
yield from new ArrayIterator([5, 6]);
|
|
yield from seven_eight();
|
|
return yield from nine_ten();
|
|
}
|
|
|
|
/**
|
|
* @return Generator<int, int>
|
|
*/
|
|
function seven_eight(): Generator {
|
|
yield 7;
|
|
yield from eight();
|
|
}
|
|
|
|
/**
|
|
* @return Generator<int,int>
|
|
*/
|
|
function eight(): Generator {
|
|
yield 8;
|
|
}
|
|
|
|
/**
|
|
* @return Generator<int,int, mixed, int>
|
|
*/
|
|
function nine_ten(): Generator {
|
|
yield 9;
|
|
return 10;
|
|
}
|
|
|
|
$gen = count_to_ten();
|
|
foreach ($gen as $num) {
|
|
echo "$num ";
|
|
}
|
|
$gen2 = $gen->getReturn();',
|
|
'assertions' => [
|
|
'$gen' => 'Generator<int, int, mixed, int>',
|
|
'$gen2' => 'int',
|
|
],
|
|
'error_levels' => ['MixedAssignment'],
|
|
],
|
|
'yieldFromArray' => [
|
|
'<?php
|
|
/**
|
|
* @return Generator<int, int, mixed, void>
|
|
*/
|
|
function Bar() : Generator {
|
|
yield from [1];
|
|
}',
|
|
],
|
|
'generatorWithNestedYield' => [
|
|
'<?php
|
|
function other_generator(): Generator {
|
|
yield "traffic";
|
|
return 1;
|
|
}
|
|
function foo(): Generator {
|
|
/** @var int */
|
|
$value = yield from other_generator();
|
|
var_export($value);
|
|
}',
|
|
],
|
|
'multipleUse' => [
|
|
'<?php
|
|
namespace Name\Space {
|
|
class A {
|
|
|
|
}
|
|
|
|
class B {
|
|
|
|
}
|
|
}
|
|
|
|
namespace Noom\Spice {
|
|
use Name\Space\{
|
|
A,
|
|
B
|
|
};
|
|
|
|
new A();
|
|
new B();
|
|
}',
|
|
],
|
|
'generatorVoidReturn' => [
|
|
'<?php
|
|
/**
|
|
* @return Generator
|
|
*/
|
|
function generator2() : Generator {
|
|
if (rand(0,1)) {
|
|
return;
|
|
}
|
|
yield 2;
|
|
}',
|
|
],
|
|
'returnType' => [
|
|
'<?php
|
|
function takesInt(int $i) : void {
|
|
echo $i;
|
|
}
|
|
|
|
function takesString(string $s) : void {
|
|
echo $s;
|
|
}
|
|
|
|
/**
|
|
* @return Generator<int, string, mixed, int>
|
|
*/
|
|
function other_generator() : Generator {
|
|
yield "traffic";
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
* @return Generator<int, string>
|
|
*/
|
|
function foo() : Generator {
|
|
$a = yield from other_generator();
|
|
takesInt($a);
|
|
}
|
|
|
|
foreach (foo() as $s) {
|
|
takesString($s);
|
|
}',
|
|
],
|
|
'expectNonNullableTypeWithYield' => [
|
|
'<?php
|
|
function example() : Generator {
|
|
yield from [2];
|
|
return null;
|
|
}',
|
|
],
|
|
'yieldFromIterable' => [
|
|
'<?php
|
|
/**
|
|
* @param iterable<int, string> $s
|
|
* @return Generator<int, string>
|
|
*/
|
|
function foo(iterable $s) : Traversable {
|
|
yield from $s;
|
|
}',
|
|
],
|
|
];
|
|
}
|
|
|
|
/**
|
|
* @return iterable<string,array{string,error_message:string,2?:string[],3?:bool,4?:string}>
|
|
*/
|
|
public function providerInvalidCodeParse()
|
|
{
|
|
return [
|
|
'anonymousClassWithBadStatement' => [
|
|
'<?php
|
|
$foo = new class {
|
|
public function a() {
|
|
new B();
|
|
}
|
|
};',
|
|
'error_message' => 'UndefinedClass',
|
|
],
|
|
'anonymousClassWithInvalidFunctionReturnType' => [
|
|
'<?php
|
|
$foo = new class {
|
|
public function a(): string {
|
|
return 5;
|
|
}
|
|
};',
|
|
'error_message' => 'InvalidReturnStatement',
|
|
],
|
|
'expectNonNullableTypeWithNullReturn' => [
|
|
'<?php
|
|
function example() : Generator {
|
|
yield from [2];
|
|
return null;
|
|
}
|
|
|
|
function example2() : Generator {
|
|
if (rand(0, 1)) {
|
|
return example();
|
|
}
|
|
return null;
|
|
}',
|
|
'error_message' => 'NullableReturnStatement',
|
|
],
|
|
];
|
|
}
|
|
}
|