2016-12-11 23:41:11 -05:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
2019-06-26 22:52:29 +02:00
|
|
|
use const DIRECTORY_SEPARATOR;
|
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class FunctionCallTest extends TestCase
|
2016-12-11 23:41:11 -05:00
|
|
|
{
|
2018-11-05 21:57:36 -05:00
|
|
|
use Traits\InvalidCodeAnalysisTestTrait;
|
|
|
|
use Traits\ValidCodeAnalysisTestTrait;
|
2017-02-13 19:16:25 -05:00
|
|
|
|
|
|
|
/**
|
2019-03-01 22:55:20 +02:00
|
|
|
* @return iterable<string,array{string,assertions?:array<string,string>,error_levels?:string[]}>
|
2017-07-25 16:11:02 -04:00
|
|
|
*/
|
2020-09-12 17:24:05 +02:00
|
|
|
public function providerValidCodeParse(): iterable
|
2017-07-25 16:11:02 -04:00
|
|
|
{
|
2018-01-21 10:22:04 -05:00
|
|
|
return [
|
2019-12-29 18:05:08 +02:00
|
|
|
'preg_grep' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param array<int,string> $strings
|
|
|
|
* @return array<int,string>
|
|
|
|
*/
|
|
|
|
function filter(array $strings): array {
|
|
|
|
return preg_grep("/search/", $strings, PREG_GREP_INVERT);
|
|
|
|
}
|
|
|
|
'
|
|
|
|
],
|
2018-06-15 10:33:51 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
'typedArrayWithDefault' => [
|
|
|
|
'<?php
|
|
|
|
class A {}
|
2017-05-05 18:53:45 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
/** @param array<A> $a */
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo(array $a = []): void {
|
2017-05-05 18:53:45 -04:00
|
|
|
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2018-01-18 17:16:50 -05:00
|
|
|
'abs' => [
|
2017-07-14 18:09:25 -04:00
|
|
|
'<?php
|
|
|
|
$a = abs(-5);
|
2018-03-14 18:05:36 -04:00
|
|
|
$b = abs(-7.5);
|
|
|
|
$c = $_GET["c"];
|
|
|
|
$c = is_numeric($c) ? abs($c) : null;',
|
2017-07-14 18:09:25 -04:00
|
|
|
'assertions' => [
|
|
|
|
'$a' => 'int',
|
2018-03-14 18:05:36 -04:00
|
|
|
'$b' => 'float',
|
2020-04-07 16:48:29 -04:00
|
|
|
'$c' => 'float|int|null',
|
2017-07-14 18:09:25 -04:00
|
|
|
],
|
2018-03-14 18:05:36 -04:00
|
|
|
'error_levels' => ['MixedAssignment', 'MixedArgument'],
|
2017-07-14 18:09:25 -04:00
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'validDocblockParamDefault' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param int|false $p
|
|
|
|
* @return void
|
|
|
|
*/
|
2017-05-26 20:05:57 -04:00
|
|
|
function f($p = false) {}',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2018-11-21 12:38:43 -05:00
|
|
|
'byRefNewString' => [
|
2017-04-24 23:45:02 -04:00
|
|
|
'<?php
|
2019-02-20 17:43:12 -05:00
|
|
|
function fooFoo(?string &$v): void {}
|
2017-05-26 20:05:57 -04:00
|
|
|
fooFoo($a);',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2018-11-21 12:38:43 -05:00
|
|
|
'byRefVariableFunctionExistingArray' => [
|
|
|
|
'<?php
|
|
|
|
$arr = [];
|
|
|
|
function fooFoo(array &$v): void {}
|
|
|
|
$function = "fooFoo";
|
|
|
|
$function($arr);
|
|
|
|
if ($arr) {}',
|
|
|
|
],
|
|
|
|
'byRefProperty' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/** @var string */
|
|
|
|
public $foo = "hello";
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
|
|
|
|
|
|
|
function fooFoo(string &$v): void {}
|
|
|
|
|
|
|
|
fooFoo($a->foo);',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'namespaced' => [
|
|
|
|
'<?php
|
|
|
|
namespace A;
|
2017-05-05 18:53:45 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
/** @return void */
|
|
|
|
function f(int $p) {}
|
2017-05-26 20:05:57 -04:00
|
|
|
f(5);',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'namespacedRootFunctionCall' => [
|
|
|
|
'<?php
|
|
|
|
namespace {
|
|
|
|
/** @return void */
|
|
|
|
function foo() { }
|
|
|
|
}
|
|
|
|
namespace A\B\C {
|
|
|
|
foo();
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'namespacedAliasedFunctionCall' => [
|
|
|
|
'<?php
|
|
|
|
namespace Aye {
|
|
|
|
/** @return void */
|
|
|
|
function foo() { }
|
|
|
|
}
|
|
|
|
namespace Bee {
|
|
|
|
use Aye as A;
|
2017-05-05 18:53:45 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
A\foo();
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2018-12-19 16:15:19 -05:00
|
|
|
'noRedundantConditionAfterArrayObjectCountCheck' => [
|
|
|
|
'<?php
|
|
|
|
/** @var ArrayObject<int, int> */
|
|
|
|
$a = [];
|
|
|
|
$b = 5;
|
|
|
|
if (count($a)) {}',
|
|
|
|
],
|
|
|
|
'noRedundantConditionAfterMixedOrEmptyArrayCountCheck' => [
|
|
|
|
'<?php
|
|
|
|
function foo(string $s) : void {
|
2020-04-04 11:51:24 -04:00
|
|
|
$a = $_GET["s"] ?: [];
|
2018-12-19 16:15:19 -05:00
|
|
|
if (count($a)) {}
|
|
|
|
if (!count($a)) {}
|
|
|
|
}',
|
|
|
|
'assertions' => [],
|
2019-03-23 14:27:54 -04:00
|
|
|
'error_levels' => ['MixedAssignment', 'MixedArgument'],
|
2018-12-19 16:15:19 -05:00
|
|
|
],
|
|
|
|
'objectLikeArrayAssignmentInConditional' => [
|
|
|
|
'<?php
|
|
|
|
$a = [];
|
|
|
|
|
|
|
|
if (rand(0, 1)) {
|
|
|
|
$a["a"] = 5;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count($a)) {}
|
|
|
|
if (!count($a)) {}',
|
|
|
|
],
|
|
|
|
'noRedundantConditionAfterCheckingExplodeLength' => [
|
|
|
|
'<?php
|
|
|
|
/** @var string */
|
|
|
|
$s = "hello";
|
|
|
|
$segments = explode(".", $s);
|
|
|
|
if (count($segments) === 1) {}',
|
|
|
|
],
|
|
|
|
'arrayPopNonEmptyAfterThreeAssertions' => [
|
|
|
|
'<?php
|
|
|
|
class A {}
|
|
|
|
class B extends A {
|
|
|
|
/** @var array<int, string> */
|
|
|
|
public $arr = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @var array<A> */
|
|
|
|
$replacement_stmts = [];
|
|
|
|
|
|
|
|
if (!$replacement_stmts
|
|
|
|
|| !$replacement_stmts[0] instanceof B
|
|
|
|
|| count($replacement_stmts[0]->arr) > 1
|
|
|
|
) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$b = $replacement_stmts[0]->arr;',
|
|
|
|
'assertions' => [
|
|
|
|
'$b' => 'array<int, string>',
|
|
|
|
],
|
|
|
|
],
|
2019-02-21 17:17:10 -05:00
|
|
|
'countMoreThan0CanBeInverted' => [
|
|
|
|
'<?php
|
|
|
|
$a = [];
|
|
|
|
|
|
|
|
if (rand(0, 1)) {
|
|
|
|
$a[] = "hello";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count($a) > 0) {
|
|
|
|
exit;
|
|
|
|
}',
|
|
|
|
'assertions' => [
|
|
|
|
'$a' => 'array<empty, empty>',
|
|
|
|
],
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'byRefAfterCallable' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param callable $callback
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function route($callback) {
|
|
|
|
if (!is_callable($callback)) { }
|
|
|
|
$a = preg_match("", "", $b);
|
|
|
|
if ($b[0]) {}
|
|
|
|
}',
|
|
|
|
'assertions' => [],
|
|
|
|
'error_levels' => [
|
|
|
|
'MixedAssignment',
|
2017-05-26 20:05:57 -04:00
|
|
|
'MixedArrayAccess',
|
2018-02-07 15:20:47 -05:00
|
|
|
'RedundantConditionGivenDocblockType',
|
2017-05-26 20:05:57 -04:00
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2018-01-24 17:07:03 -05:00
|
|
|
'ignoreNullablePregReplace' => [
|
|
|
|
'<?php
|
|
|
|
function foo(string $s): string {
|
|
|
|
$s = preg_replace("/hello/", "", $s);
|
|
|
|
if ($s === null) {
|
|
|
|
return "hello";
|
|
|
|
}
|
|
|
|
return $s;
|
|
|
|
}
|
|
|
|
function bar(string $s): string {
|
|
|
|
$s = preg_replace("/hello/", "", $s);
|
|
|
|
return $s;
|
|
|
|
}
|
|
|
|
function bat(string $s): ?string {
|
|
|
|
$s = preg_replace("/hello/", "", $s);
|
|
|
|
return $s;
|
|
|
|
}',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'extractVarCheck' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function takesString(string $str): void {}
|
2017-05-05 18:53:45 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
$foo = null;
|
|
|
|
$a = ["$foo" => "bar"];
|
|
|
|
extract($a);
|
|
|
|
takesString($foo);',
|
|
|
|
'assertions' => [],
|
|
|
|
'error_levels' => [
|
|
|
|
'MixedAssignment',
|
2017-05-26 20:05:57 -04:00
|
|
|
'MixedArrayAccess',
|
2018-04-07 15:16:46 -04:00
|
|
|
'MixedArgument',
|
2017-05-26 20:05:57 -04:00
|
|
|
],
|
2017-05-05 18:53:45 -04:00
|
|
|
],
|
2017-11-08 09:37:58 -05:00
|
|
|
'compact' => [
|
|
|
|
'<?php
|
2020-04-21 18:18:11 -06:00
|
|
|
/**
|
|
|
|
* @return array<string, mixed>
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function test(): array {
|
2017-11-08 09:37:58 -05:00
|
|
|
return compact(["val"]);
|
|
|
|
}',
|
|
|
|
],
|
2017-11-19 13:05:35 -05:00
|
|
|
'objectLikeKeyChecksAgainstGeneric' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param array<string, string> $b
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function a($b): string
|
2017-11-19 13:05:35 -05:00
|
|
|
{
|
|
|
|
return $b["a"];
|
|
|
|
}
|
|
|
|
|
|
|
|
a(["a" => "hello"]);',
|
|
|
|
],
|
2020-08-30 11:44:14 -04:00
|
|
|
'objectLikeKeyChecksAgainstTKeyedArray' => [
|
2017-11-19 13:05:35 -05:00
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param array{a: string} $b
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function a($b): string
|
2017-11-19 13:05:35 -05:00
|
|
|
{
|
|
|
|
return $b["a"];
|
|
|
|
}
|
|
|
|
|
|
|
|
a(["a" => "hello"]);',
|
|
|
|
],
|
2018-01-08 00:09:22 -05:00
|
|
|
'getenv' => [
|
|
|
|
'<?php
|
|
|
|
$a = getenv();
|
|
|
|
$b = getenv("some_key");',
|
|
|
|
'assertions' => [
|
2020-05-11 11:36:50 -04:00
|
|
|
'$a' => 'array<string, string>',
|
2019-10-16 22:14:33 -07:00
|
|
|
'$b' => 'false|string',
|
2018-01-08 00:09:22 -05:00
|
|
|
],
|
|
|
|
],
|
2018-01-25 13:07:36 -05:00
|
|
|
'ignoreFalsableFileGetContents' => [
|
|
|
|
'<?php
|
|
|
|
function foo(string $s): string {
|
|
|
|
return file_get_contents($s);
|
|
|
|
}
|
|
|
|
function bar(string $s): string {
|
|
|
|
$a = file_get_contents($s);
|
|
|
|
if ($a === false) {
|
|
|
|
return "hello";
|
|
|
|
}
|
|
|
|
return $a;
|
|
|
|
}
|
|
|
|
/**
|
2019-10-16 22:14:33 -07:00
|
|
|
* @return false|string
|
2018-01-25 13:07:36 -05:00
|
|
|
*/
|
|
|
|
function bat(string $s) {
|
|
|
|
return file_get_contents($s);
|
|
|
|
}',
|
|
|
|
],
|
2018-03-02 00:49:53 -05:00
|
|
|
'validCallables' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
public static function b() : void {}
|
|
|
|
}
|
|
|
|
|
|
|
|
function c() : void {}
|
|
|
|
|
|
|
|
["a", "b"]();
|
|
|
|
"A::b"();
|
2019-03-23 14:27:54 -04:00
|
|
|
"c"();',
|
2018-03-02 00:49:53 -05:00
|
|
|
],
|
2018-03-08 14:04:00 -05:00
|
|
|
'noInvalidOperandForCoreFunctions' => [
|
|
|
|
'<?php
|
|
|
|
function foo(string $a, string $b) : int {
|
|
|
|
$aTime = strtotime($a);
|
|
|
|
$bTime = strtotime($b);
|
|
|
|
|
|
|
|
return $aTime - $bTime;
|
|
|
|
}',
|
|
|
|
],
|
2018-03-15 17:32:03 -04:00
|
|
|
'strposIntSecondParam' => [
|
|
|
|
'<?php
|
|
|
|
function hasZeroByteOffset(string $s) : bool {
|
|
|
|
return strpos($s, 0) !== false;
|
2019-03-23 14:27:54 -04:00
|
|
|
}',
|
2018-03-15 17:32:03 -04:00
|
|
|
],
|
2018-03-18 17:11:14 -04:00
|
|
|
'functionCallInGlobalScope' => [
|
|
|
|
'<?php
|
|
|
|
$a = function() use ($argv) : void {};',
|
|
|
|
],
|
2018-04-02 22:19:58 -04:00
|
|
|
'varExport' => [
|
|
|
|
'<?php
|
|
|
|
$a = var_export(["a"], true);',
|
|
|
|
'assertions' => [
|
|
|
|
'$a' => 'string',
|
|
|
|
],
|
|
|
|
],
|
2019-01-21 16:55:50 -05:00
|
|
|
'varExportConstFetch' => [
|
|
|
|
'<?php
|
|
|
|
class Foo {
|
|
|
|
const BOOL_VAR_EXPORT_RETURN = true;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param mixed $mixed
|
|
|
|
*/
|
|
|
|
public static function Baz($mixed) : string {
|
|
|
|
return var_export($mixed, self::BOOL_VAR_EXPORT_RETURN);
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2019-11-02 10:10:38 +01:00
|
|
|
'explode' => [
|
2018-04-16 16:03:04 -04:00
|
|
|
'<?php
|
2019-11-02 10:10:38 +01:00
|
|
|
/** @var string $string */
|
|
|
|
$elements = explode(" ", $string);',
|
|
|
|
'assertions' => [
|
|
|
|
'$elements' => 'non-empty-list<string>',
|
|
|
|
],
|
|
|
|
],
|
|
|
|
'explodeWithPositiveLimit' => [
|
|
|
|
'<?php
|
|
|
|
/** @var string $string */
|
|
|
|
$elements = explode(" ", $string, 5);',
|
|
|
|
'assertions' => [
|
|
|
|
'$elements' => 'non-empty-list<string>',
|
|
|
|
],
|
|
|
|
],
|
|
|
|
'explodeWithNegativeLimit' => [
|
|
|
|
'<?php
|
|
|
|
/** @var string $string */
|
|
|
|
$elements = explode(" ", $string, -5);',
|
|
|
|
'assertions' => [
|
|
|
|
'$elements' => 'list<string>',
|
|
|
|
],
|
|
|
|
],
|
|
|
|
'explodeWithDynamicLimit' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @var string $string
|
|
|
|
* @var int $limit
|
|
|
|
*/
|
|
|
|
$elements = explode(" ", $string, $limit);',
|
|
|
|
'assertions' => [
|
|
|
|
'$elements' => 'list<string>',
|
|
|
|
],
|
|
|
|
],
|
|
|
|
'explodeWithDynamicDelimiter' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @var string $delim
|
|
|
|
* @var string $string
|
|
|
|
*/
|
|
|
|
$elements = explode($delim, $string);',
|
|
|
|
'assertions' => [
|
|
|
|
'$elements' => 'false|non-empty-list<string>',
|
|
|
|
],
|
|
|
|
],
|
|
|
|
'explodeWithDynamicDelimiterAndPositiveLimit' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @var string $delim
|
|
|
|
* @var string $string
|
|
|
|
*/
|
|
|
|
$elements = explode($delim, $string, 5);',
|
|
|
|
'assertions' => [
|
|
|
|
'$elements' => 'false|non-empty-list<string>',
|
|
|
|
],
|
|
|
|
],
|
|
|
|
'explodeWithDynamicDelimiterAndNegativeLimit' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @var string $delim
|
|
|
|
* @var string $string
|
|
|
|
*/
|
|
|
|
$elements = explode($delim, $string, -5);',
|
|
|
|
'assertions' => [
|
|
|
|
'$elements' => 'false|list<string>',
|
|
|
|
],
|
2018-04-16 16:03:04 -04:00
|
|
|
],
|
2019-11-02 10:10:38 +01:00
|
|
|
'explodeWithDynamicDelimiterAndLimit' => [
|
2019-07-17 14:13:28 -04:00
|
|
|
'<?php
|
2019-11-02 10:10:38 +01:00
|
|
|
/**
|
|
|
|
* @var string $delim
|
|
|
|
* @var string $string
|
|
|
|
* @var int $limit
|
|
|
|
*/
|
|
|
|
$elements = explode($delim, $string, $limit);',
|
|
|
|
'assertions' => [
|
|
|
|
'$elements' => 'false|list<string>',
|
|
|
|
],
|
|
|
|
],
|
2021-03-11 07:09:15 +02:00
|
|
|
'explodeWithDynamicNonEmptyDelimiter' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @var non-empty-string $delim
|
|
|
|
* @var string $string
|
|
|
|
*/
|
|
|
|
$elements = explode($delim, $string);',
|
|
|
|
'assertions' => [
|
|
|
|
'$elements' => 'non-empty-list<string>',
|
|
|
|
],
|
|
|
|
],
|
|
|
|
'explodeWithLiteralNonEmptyDelimiter' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @var string $string
|
|
|
|
*/
|
|
|
|
$elements = explode(" ", $string);',
|
|
|
|
'assertions' => [
|
|
|
|
'$elements' => 'non-empty-list<string>',
|
|
|
|
],
|
|
|
|
],
|
|
|
|
'explodeWithLiteralEmptyDelimiter' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @var string $string
|
|
|
|
*/
|
|
|
|
$elements = explode("", $string);',
|
|
|
|
'assertions' => [
|
|
|
|
'$elements' => 'false',
|
|
|
|
],
|
|
|
|
],
|
2019-11-02 10:10:38 +01:00
|
|
|
'explodeWithPossiblyFalse' => [
|
|
|
|
'<?php
|
|
|
|
/** @return non-empty-list<string> */
|
|
|
|
function exploder(string $d, string $s) : array {
|
|
|
|
return explode($d, $s);
|
|
|
|
}',
|
2019-07-17 14:13:28 -04:00
|
|
|
],
|
2018-04-19 12:16:00 -04:00
|
|
|
'allowPossiblyUndefinedClassInClassExists' => [
|
|
|
|
'<?php
|
2019-03-23 14:27:54 -04:00
|
|
|
if (class_exists(Foo::class)) {}',
|
2018-04-19 12:16:00 -04:00
|
|
|
],
|
2018-11-12 12:03:55 -05:00
|
|
|
'allowConstructorAfterClassExists' => [
|
|
|
|
'<?php
|
|
|
|
function foo(string $s) : void {
|
|
|
|
if (class_exists($s)) {
|
|
|
|
new $s();
|
|
|
|
}
|
2019-01-02 06:58:49 -05:00
|
|
|
}',
|
|
|
|
'assertions' => [],
|
|
|
|
'error_levels' => ['MixedMethodCall'],
|
2018-11-12 12:03:55 -05:00
|
|
|
],
|
2018-05-13 19:20:05 -04:00
|
|
|
'next' => [
|
|
|
|
'<?php
|
|
|
|
$arr = ["one", "two", "three"];
|
|
|
|
$n = next($arr);',
|
|
|
|
'assertions' => [
|
2019-10-16 22:14:33 -07:00
|
|
|
'$n' => 'false|string',
|
2018-05-13 19:20:05 -04:00
|
|
|
],
|
|
|
|
],
|
2018-05-18 18:10:10 -04:00
|
|
|
'iteratorToArray' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @return Generator<stdClass>
|
|
|
|
*/
|
|
|
|
function generator(): Generator {
|
|
|
|
yield new stdClass;
|
|
|
|
}
|
|
|
|
|
2019-01-27 14:50:05 -05:00
|
|
|
$a = iterator_to_array(generator());',
|
|
|
|
'assertions' => [
|
2020-02-21 01:26:51 -05:00
|
|
|
'$a' => 'array<array-key, stdClass>',
|
2019-01-27 14:50:05 -05:00
|
|
|
],
|
|
|
|
],
|
|
|
|
'iteratorToArrayWithGetIterator' => [
|
|
|
|
'<?php
|
|
|
|
class C implements IteratorAggregate {
|
|
|
|
/**
|
|
|
|
* @return Traversable<int,string>
|
|
|
|
*/
|
|
|
|
public function getIterator() {
|
|
|
|
yield 1 => "1";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$a = iterator_to_array(new C);',
|
|
|
|
'assertions' => [
|
|
|
|
'$a' => 'array<int, string>',
|
|
|
|
],
|
2018-05-18 18:10:10 -04:00
|
|
|
],
|
2019-10-10 23:47:11 -04:00
|
|
|
'iteratorToArrayWithGetIteratorReturningList' => [
|
|
|
|
'<?php
|
|
|
|
class C implements IteratorAggregate {
|
|
|
|
/**
|
|
|
|
* @return Traversable<int,string>
|
|
|
|
*/
|
|
|
|
public function getIterator() {
|
|
|
|
yield 1 => "1";
|
|
|
|
}
|
|
|
|
}
|
2020-01-31 13:58:02 -05:00
|
|
|
$a = iterator_to_array(new C, false);',
|
2018-06-28 16:01:53 -04:00
|
|
|
'assertions' => [
|
2020-01-31 13:58:02 -05:00
|
|
|
'$a' => 'list<string>',
|
2018-06-28 16:01:53 -04:00
|
|
|
],
|
|
|
|
],
|
2020-01-31 13:58:02 -05:00
|
|
|
'strtrWithPossiblyFalseFirstArg' => [
|
2018-06-28 16:01:53 -04:00
|
|
|
'<?php
|
2020-01-31 13:58:02 -05:00
|
|
|
/**
|
|
|
|
* @param false|string $str
|
|
|
|
* @param array<string, string> $replace_pairs
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
function strtr_wrapper($str, array $replace_pairs) {
|
|
|
|
/** @psalm-suppress PossiblyFalseArgument */
|
|
|
|
return strtr($str, $replace_pairs);
|
|
|
|
}',
|
2018-06-28 16:01:53 -04:00
|
|
|
],
|
2018-07-06 13:09:05 -04:00
|
|
|
'versionCompare' => [
|
|
|
|
'<?php
|
2020-01-27 23:52:06 -05:00
|
|
|
/** @return "="|"==" */
|
2018-07-06 13:09:05 -04:00
|
|
|
function getString() : string {
|
2020-01-27 23:52:06 -05:00
|
|
|
return rand(0, 1) ? "==" : "=";
|
2018-07-06 13:09:05 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
$a = version_compare("5.0.0", "7.0.0");
|
|
|
|
$b = version_compare("5.0.0", "7.0.0", "==");
|
|
|
|
$c = version_compare("5.0.0", "7.0.0", getString());
|
|
|
|
',
|
|
|
|
'assertions' => [
|
|
|
|
'$a' => 'int',
|
|
|
|
'$b' => 'bool',
|
2020-01-27 23:52:06 -05:00
|
|
|
'$c' => 'bool',
|
2018-07-06 13:09:05 -04:00
|
|
|
],
|
|
|
|
],
|
2018-07-07 20:35:24 -04:00
|
|
|
'getTimeOfDay' => [
|
|
|
|
'<?php
|
|
|
|
$a = gettimeofday(true) - gettimeofday(true);
|
|
|
|
$b = gettimeofday();
|
|
|
|
$c = gettimeofday(false);',
|
|
|
|
'assertions' => [
|
|
|
|
'$a' => 'float',
|
|
|
|
'$b' => 'array<string, int>',
|
|
|
|
'$c' => 'array<string, int>',
|
|
|
|
],
|
|
|
|
],
|
2018-07-09 14:37:14 -04:00
|
|
|
'parseUrlArray' => [
|
|
|
|
'<?php
|
|
|
|
function foo(string $s) : string {
|
2020-11-25 20:04:57 -05:00
|
|
|
$parts = parse_url($s);
|
|
|
|
return $parts["host"] ?? "";
|
2018-07-09 14:37:14 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
function hereisanotherone(string $s) : string {
|
|
|
|
$parsed = parse_url($s);
|
|
|
|
|
2020-03-18 20:17:24 +01:00
|
|
|
if (isset($parsed["host"])) {
|
2018-07-09 14:37:14 -04:00
|
|
|
return $parsed["host"];
|
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
function hereisthelastone(string $s) : string {
|
|
|
|
$parsed = parse_url($s);
|
|
|
|
|
2020-03-18 20:17:24 +01:00
|
|
|
if (isset($parsed["host"])) {
|
2018-07-09 14:37:14 -04:00
|
|
|
return $parsed["host"];
|
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
2018-07-10 05:08:53 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
function portisint(string $s) : int {
|
|
|
|
$parsed = parse_url($s);
|
|
|
|
|
|
|
|
if (isset($parsed["port"])) {
|
|
|
|
return $parsed["port"];
|
|
|
|
}
|
|
|
|
|
|
|
|
return 80;
|
|
|
|
}
|
|
|
|
|
|
|
|
function portismaybeint(string $s) : ? int {
|
|
|
|
$parsed = parse_url($s);
|
|
|
|
|
|
|
|
return $parsed["port"] ?? null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$porta = parse_url("", PHP_URL_PORT);
|
|
|
|
$porte = parse_url("localhost:443", PHP_URL_PORT);',
|
|
|
|
'assertions' => [
|
2020-05-24 15:19:58 +02:00
|
|
|
'$porta' => 'false|int|null',
|
|
|
|
'$porte' => 'false|int|null',
|
2018-07-10 05:08:53 +01:00
|
|
|
],
|
2018-07-09 14:37:14 -04:00
|
|
|
'error_levels' => ['MixedReturnStatement', 'MixedInferredReturnType'],
|
|
|
|
],
|
|
|
|
'parseUrlComponent' => [
|
|
|
|
'<?php
|
|
|
|
function foo(string $s) : string {
|
|
|
|
return parse_url($s, PHP_URL_HOST) ?? "";
|
|
|
|
}
|
|
|
|
|
|
|
|
function bar(string $s) : string {
|
|
|
|
return parse_url($s, PHP_URL_HOST);
|
|
|
|
}
|
|
|
|
|
|
|
|
function bag(string $s) : string {
|
|
|
|
$host = parse_url($s, PHP_URL_HOST);
|
|
|
|
|
|
|
|
if (is_string($host)) {
|
|
|
|
return $host;
|
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
|
|
|
}',
|
|
|
|
],
|
2020-03-18 20:17:24 +01:00
|
|
|
'parseUrlTypes' => [
|
|
|
|
'<?php
|
|
|
|
$url = "foo";
|
|
|
|
$components = parse_url($url);
|
|
|
|
$scheme = parse_url($url, PHP_URL_SCHEME);
|
|
|
|
$host = parse_url($url, PHP_URL_HOST);
|
|
|
|
$port = parse_url($url, PHP_URL_PORT);
|
|
|
|
$user = parse_url($url, PHP_URL_USER);
|
|
|
|
$pass = parse_url($url, PHP_URL_PASS);
|
|
|
|
$path = parse_url($url, PHP_URL_PATH);
|
|
|
|
$query = parse_url($url, PHP_URL_QUERY);
|
|
|
|
$fragment = parse_url($url, PHP_URL_FRAGMENT);',
|
|
|
|
'assertions' => [
|
|
|
|
'$components' => 'array{fragment?: string, host?: string, pass?: string, path?: string, port?: int, query?: string, scheme?: string, user?: string}|false',
|
2020-05-24 15:19:58 +02:00
|
|
|
'$scheme' => 'false|null|string',
|
|
|
|
'$host' => 'false|null|string',
|
|
|
|
'$port' => 'false|int|null',
|
|
|
|
'$user' => 'false|null|string',
|
|
|
|
'$pass' => 'false|null|string',
|
|
|
|
'$path' => 'false|null|string',
|
|
|
|
'$query' => 'false|null|string',
|
|
|
|
'$fragment' => 'false|null|string',
|
2020-03-18 20:17:24 +01:00
|
|
|
],
|
|
|
|
],
|
2018-07-13 18:36:25 -04:00
|
|
|
'triggerUserError' => [
|
|
|
|
'<?php
|
|
|
|
function mightLeave() : string {
|
|
|
|
if (rand(0, 1)) {
|
|
|
|
trigger_error("bad", E_USER_ERROR);
|
|
|
|
} else {
|
|
|
|
return "here";
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2018-08-02 13:43:13 -05:00
|
|
|
'getParentClass' => [
|
|
|
|
'<?php
|
|
|
|
class A {}
|
|
|
|
class B extends A {}
|
|
|
|
|
|
|
|
$b = get_parent_class(new A());
|
|
|
|
if ($b === false) {}
|
|
|
|
$c = new $b();',
|
2019-01-02 06:58:49 -05:00
|
|
|
'assertions' => [],
|
|
|
|
'error_levels' => ['MixedMethodCall'],
|
2018-08-02 13:43:13 -05:00
|
|
|
],
|
2018-09-04 13:57:05 -04:00
|
|
|
'suppressError' => [
|
|
|
|
'<?php
|
|
|
|
$a = @file_get_contents("foo");',
|
|
|
|
'assertions' => [
|
2019-10-16 22:14:33 -07:00
|
|
|
'$a' => 'false|string',
|
2018-09-04 13:57:05 -04:00
|
|
|
],
|
|
|
|
],
|
2020-07-05 05:55:42 -07:00
|
|
|
'echo' => [
|
|
|
|
'<?php
|
|
|
|
echo false;',
|
|
|
|
],
|
2018-10-10 10:58:47 -04:00
|
|
|
'printrOutput' => [
|
|
|
|
'<?php
|
|
|
|
function foo(string $s) : void {
|
|
|
|
echo $s;
|
|
|
|
}
|
|
|
|
|
|
|
|
foo(print_r(1, true));',
|
|
|
|
],
|
2018-10-10 16:03:00 -04:00
|
|
|
'microtime' => [
|
|
|
|
'<?php
|
|
|
|
$a = microtime(true);
|
|
|
|
$b = microtime();
|
2019-06-30 11:14:05 -04:00
|
|
|
/** @psalm-suppress InvalidScalarArgument */
|
2018-10-10 16:03:00 -04:00
|
|
|
$c = microtime(1);
|
|
|
|
$d = microtime(false);',
|
|
|
|
'assertions' => [
|
|
|
|
'$a' => 'float',
|
|
|
|
'$b' => 'string',
|
|
|
|
'$c' => 'float|string',
|
|
|
|
'$d' => 'string',
|
|
|
|
],
|
|
|
|
],
|
2018-10-23 14:38:36 -04:00
|
|
|
'filterVar' => [
|
|
|
|
'<?php
|
|
|
|
function filterInt(string $s) : int {
|
|
|
|
$filtered = filter_var($s, FILTER_VALIDATE_INT);
|
|
|
|
if ($filtered === false) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return $filtered;
|
|
|
|
}
|
|
|
|
function filterNullableInt(string $s) : ?int {
|
2019-01-02 17:04:01 +02:00
|
|
|
return filter_var($s, FILTER_VALIDATE_INT, ["options" => ["default" => null]]);
|
2018-10-23 14:38:36 -04:00
|
|
|
}
|
|
|
|
function filterIntWithDefault(string $s) : int {
|
2019-01-02 17:04:01 +02:00
|
|
|
return filter_var($s, FILTER_VALIDATE_INT, ["options" => ["default" => 5]]);
|
2018-10-23 14:38:36 -04:00
|
|
|
}
|
|
|
|
function filterBool(string $s) : bool {
|
|
|
|
return filter_var($s, FILTER_VALIDATE_BOOLEAN);
|
|
|
|
}
|
|
|
|
function filterNullableBool(string $s) : ?bool {
|
|
|
|
return filter_var($s, FILTER_VALIDATE_BOOLEAN, FILTER_NULL_ON_FAILURE);
|
|
|
|
}
|
|
|
|
function filterNullableBoolWithFlagsArray(string $s) : ?bool {
|
|
|
|
return filter_var($s, FILTER_VALIDATE_BOOLEAN, ["flags" => FILTER_NULL_ON_FAILURE]);
|
|
|
|
}
|
|
|
|
function filterFloat(string $s) : float {
|
|
|
|
$filtered = filter_var($s, FILTER_VALIDATE_FLOAT);
|
|
|
|
if ($filtered === false) {
|
|
|
|
return 0.0;
|
|
|
|
}
|
|
|
|
return $filtered;
|
|
|
|
}
|
|
|
|
function filterFloatWithDefault(string $s) : float {
|
2019-01-02 17:04:01 +02:00
|
|
|
return filter_var($s, FILTER_VALIDATE_FLOAT, ["options" => ["default" => 5.0]]);
|
2018-10-23 14:38:36 -04:00
|
|
|
}',
|
|
|
|
],
|
2019-01-02 09:00:45 -05:00
|
|
|
'callVariableVar' => [
|
|
|
|
'<?php
|
|
|
|
class Foo
|
|
|
|
{
|
|
|
|
public static function someInt(): int
|
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return int
|
|
|
|
*/
|
|
|
|
function makeInt()
|
|
|
|
{
|
|
|
|
$fooClass = Foo::class;
|
|
|
|
return $fooClass::someInt();
|
|
|
|
}',
|
|
|
|
],
|
2019-01-04 14:54:40 -05:00
|
|
|
'expectsIterable' => [
|
|
|
|
'<?php
|
|
|
|
function foo(iterable $i) : void {}
|
|
|
|
function bar(array $a) : void {
|
|
|
|
foo($a);
|
|
|
|
}',
|
|
|
|
],
|
2019-01-05 16:23:18 -05:00
|
|
|
'getTypeHasValues' => [
|
2019-01-05 14:50:11 -05:00
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param mixed $maybe
|
|
|
|
*/
|
|
|
|
function matchesTypes($maybe) : void {
|
|
|
|
$t = gettype($maybe);
|
|
|
|
if ($t === "object") {}
|
2019-03-23 14:27:54 -04:00
|
|
|
}',
|
2019-01-05 14:50:11 -05:00
|
|
|
],
|
2021-07-14 12:49:46 +02:00
|
|
|
'getTypeSwitchClosedResource' => [
|
|
|
|
'<?php
|
|
|
|
$data = "foo";
|
|
|
|
switch (gettype($data)) {
|
|
|
|
case "resource (closed)":
|
|
|
|
case "unknown type":
|
|
|
|
return "foo";
|
|
|
|
}',
|
|
|
|
],
|
2019-01-05 14:50:11 -05:00
|
|
|
'functionResolutionInNamespace' => [
|
|
|
|
'<?php
|
|
|
|
namespace Foo;
|
|
|
|
function sort(int $_) : void {}
|
2019-03-23 14:27:54 -04:00
|
|
|
sort(5);',
|
2019-01-05 14:50:11 -05:00
|
|
|
],
|
2019-01-06 13:02:46 -05:00
|
|
|
'rangeWithIntStep' => [
|
|
|
|
'<?php
|
|
|
|
|
|
|
|
function foo(int $bar) : string {
|
|
|
|
return (string) $bar;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (range(1, 10, 1) as $x) {
|
|
|
|
foo($x);
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'rangeWithNoStep' => [
|
|
|
|
'<?php
|
|
|
|
|
|
|
|
function foo(int $bar) : string {
|
|
|
|
return (string) $bar;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (range(1, 10) as $x) {
|
|
|
|
foo($x);
|
|
|
|
}',
|
|
|
|
],
|
2019-02-07 10:50:42 -05:00
|
|
|
'rangeWithNoStepAndString' => [
|
|
|
|
'<?php
|
|
|
|
|
|
|
|
function foo(string $bar) : void {}
|
|
|
|
|
|
|
|
foreach (range("a", "z") as $x) {
|
|
|
|
foo($x);
|
|
|
|
}',
|
|
|
|
],
|
2019-01-06 13:02:46 -05:00
|
|
|
'rangeWithFloatStep' => [
|
|
|
|
'<?php
|
|
|
|
|
|
|
|
function foo(float $bar) : string {
|
|
|
|
return (string) $bar;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (range(1, 10, .3) as $x) {
|
|
|
|
foo($x);
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'rangeWithFloatStart' => [
|
|
|
|
'<?php
|
|
|
|
|
|
|
|
function foo(float $bar) : string {
|
|
|
|
return (string) $bar;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (range(1.5, 10) as $x) {
|
|
|
|
foo($x);
|
|
|
|
}',
|
|
|
|
],
|
2021-04-09 10:28:48 -05:00
|
|
|
'rangeWithIntOrFloatStep' => [
|
|
|
|
'<?php
|
|
|
|
/** @var int|float */
|
|
|
|
$step = 1;
|
|
|
|
$a = range(1, 10, $step);
|
|
|
|
|
|
|
|
/** @var int */
|
|
|
|
$step = 1;
|
|
|
|
$b = range(1, 10, $step);
|
|
|
|
|
|
|
|
/** @var float */
|
|
|
|
$step = 1.;
|
|
|
|
$c = range(1, 10, $step);
|
|
|
|
',
|
|
|
|
'assertions' => [
|
|
|
|
'$a' => 'non-empty-list<float|int>',
|
|
|
|
'$b' => 'non-empty-list<int>',
|
|
|
|
'$c' => 'non-empty-list<float>',
|
|
|
|
],
|
|
|
|
],
|
2019-01-06 16:40:44 -05:00
|
|
|
'duplicateNamespacedFunction' => [
|
|
|
|
'<?php
|
|
|
|
namespace Bar;
|
|
|
|
|
|
|
|
function sort() : void {}',
|
|
|
|
],
|
2019-01-15 17:53:23 -05:00
|
|
|
'arrayMapAfterFunctionMissingFile' => [
|
|
|
|
'<?php
|
|
|
|
require_once(FOO);
|
|
|
|
$urls = array_map("strval", [1, 2, 3]);',
|
|
|
|
[],
|
|
|
|
'error_levels' => ['UndefinedConstant', 'UnresolvableInclude'],
|
|
|
|
],
|
2019-01-24 10:41:06 -05:00
|
|
|
'noNamespaceClash' => [
|
|
|
|
'<?php
|
|
|
|
namespace FunctionNamespace {
|
|
|
|
function foo() : void {}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace ClassNamespace {
|
|
|
|
class Foo {}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
use ClassNamespace\Foo;
|
|
|
|
use function FunctionNamespace\foo;
|
|
|
|
|
|
|
|
new Foo();
|
|
|
|
|
|
|
|
foo();
|
|
|
|
}',
|
|
|
|
],
|
2019-02-07 12:25:57 -05:00
|
|
|
'hashInit70' => [
|
|
|
|
'<?php
|
|
|
|
$h = hash_init("sha256");',
|
|
|
|
[
|
|
|
|
'$h' => 'resource',
|
|
|
|
],
|
|
|
|
[],
|
2019-03-23 14:27:54 -04:00
|
|
|
'7.1',
|
2019-02-07 12:25:57 -05:00
|
|
|
],
|
|
|
|
'hashInit71' => [
|
|
|
|
'<?php
|
|
|
|
$h = hash_init("sha256");',
|
|
|
|
[
|
|
|
|
'$h' => 'resource',
|
|
|
|
],
|
|
|
|
[],
|
2019-03-23 14:27:54 -04:00
|
|
|
'7.1',
|
2019-02-07 12:25:57 -05:00
|
|
|
],
|
|
|
|
'hashInit72' => [
|
|
|
|
'<?php
|
|
|
|
$h = hash_init("sha256");',
|
|
|
|
[
|
2020-08-18 15:35:31 +02:00
|
|
|
'$h' => 'HashContext|false',
|
2019-02-07 12:25:57 -05:00
|
|
|
],
|
|
|
|
[],
|
2019-03-23 14:27:54 -04:00
|
|
|
'7.2',
|
2019-02-07 12:25:57 -05:00
|
|
|
],
|
|
|
|
'hashInit73' => [
|
|
|
|
'<?php
|
|
|
|
$h = hash_init("sha256");',
|
|
|
|
[
|
2020-08-18 15:35:31 +02:00
|
|
|
'$h' => 'HashContext|false',
|
2019-02-07 12:25:57 -05:00
|
|
|
],
|
|
|
|
[],
|
2019-03-23 14:27:54 -04:00
|
|
|
'7.3',
|
2019-02-20 17:43:12 -05:00
|
|
|
],
|
|
|
|
'nullableByRef' => [
|
|
|
|
'<?php
|
|
|
|
function foo(?string &$s) : void {}
|
|
|
|
|
|
|
|
function bar() : void {
|
|
|
|
foo($bar);
|
2019-03-23 14:27:54 -04:00
|
|
|
}',
|
2019-02-20 17:43:12 -05:00
|
|
|
],
|
2019-03-02 18:02:11 -05:00
|
|
|
'getClassNewInstance' => [
|
|
|
|
'<?php
|
|
|
|
interface I {}
|
|
|
|
class C implements I {}
|
|
|
|
|
|
|
|
class Props {
|
|
|
|
/** @var class-string<I>[] */
|
|
|
|
public $arr = [];
|
|
|
|
}
|
|
|
|
|
2019-03-23 14:27:54 -04:00
|
|
|
(new Props)->arr[] = get_class(new C);',
|
2019-03-02 18:02:11 -05:00
|
|
|
],
|
|
|
|
'getClassVariable' => [
|
|
|
|
'<?php
|
|
|
|
interface I {}
|
|
|
|
class C implements I {}
|
|
|
|
$c_instance = new C;
|
|
|
|
|
|
|
|
class Props {
|
|
|
|
/** @var class-string<I>[] */
|
|
|
|
public $arr = [];
|
|
|
|
}
|
|
|
|
|
2019-03-23 14:27:54 -04:00
|
|
|
(new Props)->arr[] = get_class($c_instance);',
|
2019-03-02 18:02:11 -05:00
|
|
|
],
|
|
|
|
'getClassAnonymousNewInstance' => [
|
|
|
|
'<?php
|
|
|
|
interface I {}
|
|
|
|
|
|
|
|
class Props {
|
|
|
|
/** @var class-string<I>[] */
|
|
|
|
public $arr = [];
|
|
|
|
}
|
|
|
|
|
2019-03-23 14:27:54 -04:00
|
|
|
(new Props)->arr[] = get_class(new class implements I{});',
|
2019-03-02 18:02:11 -05:00
|
|
|
],
|
|
|
|
'getClassAnonymousVariable' => [
|
|
|
|
'<?php
|
|
|
|
interface I {}
|
|
|
|
$anon_instance = new class implements I {};
|
|
|
|
|
|
|
|
class Props {
|
|
|
|
/** @var class-string<I>[] */
|
|
|
|
public $arr = [];
|
|
|
|
}
|
|
|
|
|
2019-03-23 14:27:54 -04:00
|
|
|
(new Props)->arr[] = get_class($anon_instance);',
|
2019-03-02 18:02:11 -05:00
|
|
|
],
|
2019-03-29 10:46:40 -04:00
|
|
|
'mktime' => [
|
|
|
|
'<?php
|
|
|
|
/** @psalm-suppress InvalidScalarArgument */
|
|
|
|
$a = mktime("foo");
|
|
|
|
/** @psalm-suppress MixedArgument */
|
|
|
|
$b = mktime($_GET["foo"]);
|
|
|
|
$c = mktime(1, 2, 3);',
|
|
|
|
'assertions' => [
|
2019-10-16 22:14:33 -07:00
|
|
|
'$a' => 'false|int',
|
|
|
|
'$b' => 'false|int',
|
2019-03-29 10:46:40 -04:00
|
|
|
'$c' => 'int',
|
2019-07-05 16:24:00 -04:00
|
|
|
],
|
2019-03-29 10:46:40 -04:00
|
|
|
],
|
2019-04-03 17:08:37 -04:00
|
|
|
'PHP73-hrtime' => [
|
|
|
|
'<?php
|
|
|
|
$a = hrtime(true);
|
|
|
|
$b = hrtime();
|
2019-09-08 14:44:59 -04:00
|
|
|
/** @psalm-suppress InvalidScalarArgument */
|
2019-04-03 17:08:37 -04:00
|
|
|
$c = hrtime(1);
|
|
|
|
$d = hrtime(false);',
|
|
|
|
'assertions' => [
|
|
|
|
'$a' => 'int',
|
2019-06-16 09:42:34 -04:00
|
|
|
'$b' => 'array{0: int, 1: int}',
|
|
|
|
'$c' => 'array{0: int, 1: int}|int',
|
|
|
|
'$d' => 'array{0: int, 1: int}',
|
2019-04-03 17:08:37 -04:00
|
|
|
],
|
|
|
|
],
|
|
|
|
'PHP73-hrtimeCanBeFloat' => [
|
|
|
|
'<?php
|
|
|
|
$a = hrtime(true);
|
|
|
|
|
|
|
|
if (is_int($a)) {}
|
|
|
|
if (is_float($a)) {}',
|
|
|
|
],
|
2019-04-09 14:48:51 -04:00
|
|
|
'min' => [
|
|
|
|
'<?php
|
|
|
|
$a = min(0, 1);
|
|
|
|
$b = min([0, 1]);
|
|
|
|
$c = min("a", "b");
|
|
|
|
$d = min(1, 2, 3, 4);
|
|
|
|
$e = min(1, 2, 3, 4, 5);
|
|
|
|
$f = min(...[1, 2, 3]);',
|
|
|
|
'assertions' => [
|
|
|
|
'$a' => 'int',
|
|
|
|
'$b' => 'int',
|
|
|
|
'$c' => 'string',
|
|
|
|
'$d' => 'int',
|
|
|
|
'$e' => 'int',
|
|
|
|
'$f' => 'int',
|
|
|
|
],
|
|
|
|
],
|
|
|
|
'minUnpackedArg' => [
|
|
|
|
'<?php
|
|
|
|
$f = min(...[1, 2, 3]);',
|
|
|
|
'assertions' => [
|
|
|
|
'$f' => 'int',
|
|
|
|
],
|
|
|
|
],
|
|
|
|
'sscanf' => [
|
|
|
|
'<?php
|
|
|
|
sscanf("10:05:03", "%d:%d:%d", $hours, $minutes, $seconds);',
|
|
|
|
'assertions' => [
|
2021-06-07 20:19:17 +03:00
|
|
|
'$hours' => 'float|int|null|string',
|
|
|
|
'$minutes' => 'float|int|null|string',
|
|
|
|
'$seconds' => 'float|int|null|string',
|
2019-04-09 14:48:51 -04:00
|
|
|
],
|
|
|
|
],
|
2019-05-21 11:51:41 -04:00
|
|
|
'noImplicitAssignmentToStringFromMixedWithDocblockTypes' => [
|
|
|
|
'<?php
|
|
|
|
/** @param string $s */
|
|
|
|
function takesString($s) : void {}
|
|
|
|
function takesInt(int $i) : void {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param mixed $s
|
|
|
|
* @psalm-suppress MixedArgument
|
|
|
|
*/
|
|
|
|
function bar($s) : void {
|
|
|
|
takesString($s);
|
|
|
|
takesInt($s);
|
|
|
|
}',
|
|
|
|
],
|
2019-05-21 12:59:06 -04:00
|
|
|
'ignoreNullableIssuesAfterMixedCoercion' => [
|
|
|
|
'<?php
|
|
|
|
function takesNullableString(?string $s) : void {}
|
|
|
|
function takesString(string $s) : void {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param mixed $s
|
|
|
|
* @psalm-suppress MixedArgument
|
|
|
|
*/
|
|
|
|
function bar($s) : void {
|
|
|
|
takesNullableString($s);
|
|
|
|
takesString($s);
|
|
|
|
}',
|
|
|
|
],
|
2019-05-29 07:56:06 -04:00
|
|
|
'countableSimpleXmlElement' => [
|
|
|
|
'<?php
|
|
|
|
$xml = new SimpleXMLElement("<?xml version=\"1.0\"?><a><b></b><b></b></a>");
|
2019-07-05 16:24:00 -04:00
|
|
|
echo count($xml);',
|
2019-05-29 07:56:06 -04:00
|
|
|
],
|
2019-08-12 16:17:55 -04:00
|
|
|
'countableCallableArray' => [
|
|
|
|
'<?php
|
|
|
|
/** @param callable|false $x */
|
|
|
|
function example($x) : void {
|
|
|
|
if (is_array($x)) {
|
|
|
|
echo "Count is: " . count($x);
|
|
|
|
}
|
|
|
|
}'
|
|
|
|
],
|
2021-01-13 15:48:38 +01:00
|
|
|
'countNonEmptyArrayShouldBePositiveInt' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @psalm-pure
|
|
|
|
* @param non-empty-list $x
|
|
|
|
* @return positive-int
|
|
|
|
*/
|
|
|
|
function example($x) : int {
|
|
|
|
return count($x);
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'countListShouldBeZeroOrPositive' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @psalm-pure
|
|
|
|
* @param list $x
|
|
|
|
* @return positive-int|0
|
|
|
|
*/
|
|
|
|
function example($x) : int {
|
|
|
|
return count($x);
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'countArrayShouldBeZeroOrPositive' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @psalm-pure
|
|
|
|
* @param array $x
|
|
|
|
* @return positive-int|0
|
|
|
|
*/
|
|
|
|
function example($x) : int {
|
|
|
|
return count($x);
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'countEmptyArrayShouldBeZero' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @psalm-pure
|
|
|
|
* @param array<empty, empty> $x
|
|
|
|
* @return 0
|
|
|
|
*/
|
|
|
|
function example($x) : int {
|
|
|
|
return count($x);
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'countConstantSizeArrayShouldBeConstantInteger' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @psalm-pure
|
|
|
|
* @param array{int, int, string} $x
|
|
|
|
* @return 3
|
|
|
|
*/
|
|
|
|
function example($x) : int {
|
|
|
|
return count($x);
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'countCallableArrayShouldBe2' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @psalm-pure
|
|
|
|
* @return 2
|
|
|
|
*/
|
|
|
|
function example(callable $x) : int {
|
|
|
|
assert(is_array($x));
|
|
|
|
return count($x);
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'countOnPureObjectIsPure' => [
|
|
|
|
'<?php
|
|
|
|
class PureCountable implements \Countable {
|
|
|
|
/** @psalm-pure */
|
|
|
|
public function count(): int { return 1; }
|
|
|
|
}
|
|
|
|
/** @psalm-pure */
|
|
|
|
function example(PureCountable $x) : int {
|
|
|
|
return count($x);
|
|
|
|
}',
|
|
|
|
],
|
2019-05-31 09:43:46 -04:00
|
|
|
'refineWithTraitExists' => [
|
|
|
|
'<?php
|
|
|
|
function foo(string $s) : void {
|
|
|
|
if (trait_exists($s)) {
|
|
|
|
new ReflectionClass($s);
|
|
|
|
}
|
2019-07-05 16:24:00 -04:00
|
|
|
}',
|
2019-05-31 09:43:46 -04:00
|
|
|
],
|
|
|
|
'refineWithClassExistsOrTraitExists' => [
|
|
|
|
'<?php
|
|
|
|
function foo(string $s) : void {
|
|
|
|
if (trait_exists($s) || class_exists($s)) {
|
|
|
|
new ReflectionClass($s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function bar(string $s) : void {
|
|
|
|
if (class_exists($s) || trait_exists($s)) {
|
|
|
|
new ReflectionClass($s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function baz(string $s) : void {
|
|
|
|
if (class_exists($s) || interface_exists($s) || trait_exists($s)) {
|
|
|
|
new ReflectionClass($s);
|
|
|
|
}
|
2019-07-05 16:24:00 -04:00
|
|
|
}',
|
2019-05-31 09:43:46 -04:00
|
|
|
],
|
2019-06-03 10:33:54 -04:00
|
|
|
'minSingleArg' => [
|
|
|
|
'<?php
|
|
|
|
/** @psalm-suppress TooFewArguments */
|
|
|
|
min(0);',
|
|
|
|
],
|
2019-06-07 15:49:10 -04:00
|
|
|
'PHP73-allowIsCountableToInformType' => [
|
|
|
|
'<?php
|
|
|
|
function getObject() : iterable{
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
$iterableObject = getObject();
|
|
|
|
|
|
|
|
if (is_countable($iterableObject)) {
|
|
|
|
if (count($iterableObject) === 0) {}
|
|
|
|
}',
|
|
|
|
],
|
2019-06-15 16:10:48 -04:00
|
|
|
'versionCompareAsCallable' => [
|
|
|
|
'<?php
|
|
|
|
$a = ["1.0", "2.0"];
|
2020-01-07 17:40:23 +01:00
|
|
|
usort($a, "version_compare");',
|
2019-06-15 16:10:48 -04:00
|
|
|
],
|
2019-06-26 00:14:06 -04:00
|
|
|
'coerceToObjectAfterBeingCalled' => [
|
|
|
|
'<?php
|
|
|
|
class Foo {
|
|
|
|
public function bar() : void {}
|
|
|
|
}
|
|
|
|
|
|
|
|
function takesFoo(Foo $foo) : void {}
|
|
|
|
|
|
|
|
/** @param mixed $f */
|
|
|
|
function takesMixed($f) : void {
|
|
|
|
if (rand(0, 1)) {
|
|
|
|
$f = new Foo();
|
|
|
|
}
|
|
|
|
/** @psalm-suppress MixedArgument */
|
|
|
|
takesFoo($f);
|
|
|
|
$f->bar();
|
2019-07-05 16:24:00 -04:00
|
|
|
}',
|
2019-06-26 00:14:06 -04:00
|
|
|
],
|
2019-06-28 10:48:30 -04:00
|
|
|
'functionExists' => [
|
|
|
|
'<?php
|
|
|
|
if (!function_exists("in_array")) {
|
|
|
|
function in_array($a, $b) {
|
|
|
|
return true;
|
|
|
|
}
|
2019-07-05 16:24:00 -04:00
|
|
|
}',
|
2019-06-28 10:48:30 -04:00
|
|
|
],
|
2019-08-15 16:28:03 +02:00
|
|
|
'pregMatch' => [
|
|
|
|
'<?php
|
|
|
|
function takesInt(int $i) : void {}
|
2019-08-25 23:51:36 -04:00
|
|
|
|
2019-08-15 16:28:03 +02:00
|
|
|
takesInt(preg_match("{foo}", "foo"));',
|
|
|
|
],
|
|
|
|
'pregMatchWithMatches' => [
|
|
|
|
'<?php
|
|
|
|
/** @param string[] $matches */
|
|
|
|
function takesMatches(array $matches) : void {}
|
2019-08-25 23:51:36 -04:00
|
|
|
|
2019-08-15 16:28:03 +02:00
|
|
|
preg_match("{foo}", "foo", $matches);
|
2019-08-25 23:51:36 -04:00
|
|
|
|
2019-08-15 16:28:03 +02:00
|
|
|
takesMatches($matches);',
|
|
|
|
],
|
|
|
|
'pregMatchWithOffset' => [
|
|
|
|
'<?php
|
|
|
|
/** @param string[] $matches */
|
|
|
|
function takesMatches(array $matches) : void {}
|
2019-08-25 23:51:36 -04:00
|
|
|
|
2019-08-15 16:28:03 +02:00
|
|
|
preg_match("{foo}", "foo", $matches, 0, 10);
|
2019-08-25 23:51:36 -04:00
|
|
|
|
2019-08-15 16:28:03 +02:00
|
|
|
takesMatches($matches);',
|
|
|
|
],
|
|
|
|
'pregMatchWithFlags' => [
|
|
|
|
'<?php
|
|
|
|
function takesInt(int $i) : void {}
|
2019-08-25 23:51:36 -04:00
|
|
|
|
2019-08-15 16:28:03 +02:00
|
|
|
if (preg_match("{foo}", "this is foo", $matches, PREG_OFFSET_CAPTURE)) {
|
|
|
|
/**
|
|
|
|
* @psalm-suppress MixedArrayAccess
|
|
|
|
* @psalm-suppress MixedArgument
|
|
|
|
*/
|
|
|
|
takesInt($matches[0][1]);
|
|
|
|
}',
|
|
|
|
],
|
2019-07-05 10:40:05 -04:00
|
|
|
'pregReplaceCallback' => [
|
|
|
|
'<?php
|
|
|
|
function foo(string $s) : string {
|
|
|
|
return preg_replace_callback(
|
|
|
|
\'/<files (psalm-version="[^"]+") (?:php-version="(.+)">\n)/\',
|
2021-04-09 15:19:24 +02:00
|
|
|
/** @param string[] $matches */
|
2019-07-05 10:40:05 -04:00
|
|
|
function (array $matches) : string {
|
|
|
|
return $matches[1];
|
|
|
|
},
|
|
|
|
$s
|
|
|
|
);
|
|
|
|
}',
|
|
|
|
],
|
2020-06-22 20:24:34 -04:00
|
|
|
'pregReplaceCallbackWithArray' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param string[] $ids
|
|
|
|
* @psalm-suppress MissingClosureReturnType
|
|
|
|
* @psalm-suppress MixedArgumentTypeCoercion
|
|
|
|
*/
|
|
|
|
function(array $ids): array {
|
|
|
|
return \preg_replace_callback(
|
|
|
|
"",
|
|
|
|
fn (array $matches) => $matches[4],
|
|
|
|
$ids
|
|
|
|
);
|
2020-08-23 17:32:07 +03:00
|
|
|
};',
|
|
|
|
'assertions' => [],
|
|
|
|
'error_levels' => [],
|
|
|
|
'7.4'
|
2020-06-22 20:24:34 -04:00
|
|
|
],
|
2019-07-24 17:24:23 -04:00
|
|
|
'compactDefinedVariable' => [
|
|
|
|
'<?php
|
2020-04-21 18:18:11 -06:00
|
|
|
/**
|
|
|
|
* @return array<string, mixed>
|
|
|
|
*/
|
2019-07-24 17:24:23 -04:00
|
|
|
function foo(int $a, string $b, bool $c) : array {
|
|
|
|
return compact("a", "b", "c");
|
|
|
|
}',
|
|
|
|
],
|
2019-07-28 16:40:35 -04:00
|
|
|
'PHP73-setCookiePhp73' => [
|
|
|
|
'<?php
|
|
|
|
setcookie(
|
|
|
|
"name",
|
|
|
|
"value",
|
|
|
|
[
|
|
|
|
"path" => "/",
|
|
|
|
"expires" => 0,
|
|
|
|
"httponly" => true,
|
|
|
|
"secure" => true,
|
|
|
|
"samesite" => "Lax"
|
|
|
|
]
|
|
|
|
);',
|
|
|
|
],
|
2019-09-08 14:44:59 -04:00
|
|
|
'printrBadArg' => [
|
|
|
|
'<?php
|
|
|
|
/** @psalm-suppress InvalidScalarArgument */
|
|
|
|
$a = print_r([], 1);
|
|
|
|
echo $a;',
|
|
|
|
],
|
2019-09-11 13:52:37 -04:00
|
|
|
'dontCoerceCallMapArgs' => [
|
|
|
|
'<?php
|
|
|
|
function getStr() : ?string {
|
|
|
|
return rand(0,1) ? "test" : null;
|
|
|
|
}
|
|
|
|
|
|
|
|
function test() : void {
|
|
|
|
$g = getStr();
|
|
|
|
/** @psalm-suppress PossiblyNullArgument */
|
|
|
|
$x = strtoupper($g);
|
|
|
|
$c = "prefix " . (strtoupper($g ?? "") === "x" ? "xa" : "ya");
|
|
|
|
echo "$x, $c\n";
|
|
|
|
}'
|
|
|
|
],
|
2019-10-04 11:07:28 -06:00
|
|
|
'mysqliRealConnectFunctionAllowsNullParameters' => [
|
|
|
|
'<?php
|
|
|
|
$mysqli = mysqli_init();
|
|
|
|
mysqli_real_connect($mysqli, null, \'test\', null);',
|
|
|
|
],
|
2020-01-24 13:34:50 -05:00
|
|
|
'callUserFunc' => [
|
|
|
|
'<?php
|
|
|
|
$func = function(int $arg1, int $arg2) : int {
|
|
|
|
return $arg1 * $arg2;
|
|
|
|
};
|
|
|
|
|
|
|
|
$a = call_user_func($func, 2, 4);',
|
|
|
|
[
|
|
|
|
'$a' => 'int',
|
|
|
|
]
|
|
|
|
],
|
|
|
|
'callUserFuncArray' => [
|
|
|
|
'<?php
|
|
|
|
$func = function(int $arg1, int $arg2) : int {
|
|
|
|
return $arg1 * $arg2;
|
|
|
|
};
|
|
|
|
|
|
|
|
$a = call_user_func_array($func, [2, 4]);',
|
|
|
|
[
|
|
|
|
'$a' => 'int',
|
|
|
|
]
|
|
|
|
],
|
2020-04-14 21:07:44 -04:00
|
|
|
'dateTest' => [
|
|
|
|
'<?php
|
|
|
|
$y = date("Y");
|
|
|
|
$m = date("m");
|
|
|
|
$F = date("F");
|
|
|
|
$y2 = date("Y", 10000);
|
|
|
|
$F2 = date("F", 10000);
|
|
|
|
/** @psalm-suppress MixedArgument */
|
|
|
|
$F3 = date("F", $_GET["F3"]);',
|
|
|
|
[
|
|
|
|
'$y' => 'numeric-string',
|
|
|
|
'$m' => 'numeric-string',
|
|
|
|
'$F' => 'string',
|
|
|
|
'$y2' => 'numeric-string',
|
|
|
|
'$F2' => 'string',
|
|
|
|
'$F3' => 'false|string',
|
|
|
|
]
|
|
|
|
],
|
2020-04-26 23:36:44 +02:00
|
|
|
'sscanfReturnTypeWithTwoParameters' => [
|
|
|
|
'<?php
|
|
|
|
$data = sscanf("42 psalm road", "%s %s");',
|
|
|
|
[
|
2021-06-07 20:19:17 +03:00
|
|
|
'$data' => 'list<float|int|null|string>|null',
|
2020-04-26 23:36:44 +02:00
|
|
|
]
|
|
|
|
],
|
|
|
|
'sscanfReturnTypeWithMoreThanTwoParameters' => [
|
|
|
|
'<?php
|
|
|
|
$n = sscanf("42 psalm road", "%s %s", $p1, $p2);',
|
|
|
|
[
|
|
|
|
'$n' => 'int',
|
2021-06-07 20:19:17 +03:00
|
|
|
'$p1' => 'float|int|null|string',
|
|
|
|
'$p2' => 'float|int|null|string',
|
|
|
|
],
|
2020-04-26 23:36:44 +02:00
|
|
|
],
|
2020-06-24 11:51:24 -04:00
|
|
|
'writeArgsAllowed' => [
|
|
|
|
'<?php
|
|
|
|
/** @return false|int */
|
|
|
|
function safeMatch(string $pattern, string $subject, ?array $matches = null, int $flags = 0) {
|
|
|
|
return \preg_match($pattern, $subject, $matches, $flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
safeMatch("/a/", "b");'
|
|
|
|
],
|
2020-07-30 14:21:42 -04:00
|
|
|
'fgetcsv' => [
|
|
|
|
'<?php
|
|
|
|
$headers = fgetcsv(fopen("test.txt", "r"));
|
|
|
|
if (empty($headers)) {
|
|
|
|
throw new Exception("invalid headers");
|
|
|
|
}
|
|
|
|
print_r(array_map("strval", $headers));'
|
|
|
|
],
|
2020-08-30 11:32:01 -04:00
|
|
|
'allowListEqualToRange' => [
|
|
|
|
'<?php
|
|
|
|
/** @param array<int, int> $two */
|
|
|
|
function collectCommit(array $one, array $two) : void {
|
|
|
|
if ($one && array_values($one) === array_values($two)) {}
|
|
|
|
}'
|
|
|
|
],
|
2020-09-04 18:10:14 -04:00
|
|
|
'pregMatchAll' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
2020-09-04 18:31:50 -04:00
|
|
|
* @return array<list<string>>
|
2020-09-04 18:10:14 -04:00
|
|
|
*/
|
|
|
|
function extractUsernames(string $input): array {
|
2020-09-04 20:33:02 -04:00
|
|
|
preg_match_all(\'/([a-zA-Z])*/\', $input, $matches);
|
2020-09-04 18:10:14 -04:00
|
|
|
|
|
|
|
return $matches;
|
|
|
|
}'
|
|
|
|
],
|
2020-09-04 20:33:02 -04:00
|
|
|
'pregMatchAllOffsetCapture' => [
|
|
|
|
'<?php
|
|
|
|
function foo(string $input): array {
|
|
|
|
preg_match_all(\'/([a-zA-Z])*/\', $input, $matches, PREG_OFFSET_CAPTURE);
|
|
|
|
|
|
|
|
return $matches[0];
|
|
|
|
}'
|
|
|
|
],
|
2020-10-15 01:59:07 +02:00
|
|
|
'pregMatchAllReturnsFalse' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @return int|false
|
|
|
|
*/
|
|
|
|
function badpattern() {
|
|
|
|
return @preg_match_all("foo", "foo", $matches);
|
|
|
|
}'
|
|
|
|
],
|
2020-09-13 21:42:44 -04:00
|
|
|
'strposAllowDictionary' => [
|
|
|
|
'<?php
|
|
|
|
function sayHello(string $format): void {
|
|
|
|
if (strpos("abcdefghijklmno", $format)) {}
|
|
|
|
}',
|
|
|
|
],
|
2020-10-12 09:57:11 -04:00
|
|
|
'curlInitIsResourceAllowedIn7x' => [
|
|
|
|
'<?php
|
|
|
|
$ch = curl_init();
|
|
|
|
if (!is_resource($ch)) {}',
|
|
|
|
[],
|
|
|
|
[],
|
|
|
|
'7.4'
|
|
|
|
],
|
2020-10-12 13:04:28 -04:00
|
|
|
'pregSplit' => [
|
|
|
|
'<?php
|
|
|
|
/** @return non-empty-list */
|
|
|
|
function foo(string $s) {
|
|
|
|
return preg_split("/ /", $s);
|
|
|
|
}'
|
|
|
|
],
|
2020-10-12 20:25:46 -04:00
|
|
|
'pregSplitWithFlags' => [
|
|
|
|
'<?php
|
|
|
|
/** @return list<string> */
|
|
|
|
function foo(string $s) {
|
|
|
|
return preg_split("/ /", $s, -1, PREG_SPLIT_NO_EMPTY | PREG_SPLIT_DELIM_CAPTURE);
|
|
|
|
}'
|
|
|
|
],
|
2020-10-12 13:46:43 -04:00
|
|
|
'mbConvertEncodingWithArray' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param array<int, string> $str
|
|
|
|
* @return array<int, string>
|
|
|
|
*/
|
|
|
|
function test2(array $str): array {
|
|
|
|
return mb_convert_encoding($str, "UTF-8", "UTF-8");
|
|
|
|
}'
|
|
|
|
],
|
2020-10-14 17:30:08 -04:00
|
|
|
'getDebugType' => [
|
|
|
|
'<?php
|
|
|
|
function foo(mixed $var) : void {
|
|
|
|
switch (get_debug_type($var)) {
|
|
|
|
case "string":
|
|
|
|
echo "a string";
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Exception::class;
|
|
|
|
echo "an Exception with message " . $var->getMessage();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
[],
|
|
|
|
[],
|
|
|
|
'8.0'
|
|
|
|
],
|
2020-12-14 23:08:07 -05:00
|
|
|
'getTypeDoubleThenInt' => [
|
|
|
|
'<?php
|
|
|
|
function safe_float(mixed $val): bool {
|
|
|
|
switch (gettype($val)) {
|
|
|
|
case "double":
|
|
|
|
case "integer":
|
|
|
|
return true;
|
|
|
|
// ... more cases omitted
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
[],
|
|
|
|
[],
|
|
|
|
'8.0'
|
|
|
|
],
|
2021-03-11 06:13:17 +01:00
|
|
|
'maxWithFloats' => [
|
|
|
|
'<?php
|
|
|
|
function foo(float $_float): void
|
|
|
|
{}
|
|
|
|
|
|
|
|
foo(max(1.1, 1.2));',
|
|
|
|
],
|
|
|
|
'maxWithObjects' => [
|
|
|
|
'<?php
|
|
|
|
function foo(DateTimeImmutable $fooDate): string
|
|
|
|
{
|
|
|
|
return $fooDate->format("Y");
|
|
|
|
}
|
|
|
|
|
|
|
|
foo(max(new DateTimeImmutable(), new DateTimeImmutable()));',
|
|
|
|
],
|
|
|
|
'maxWithMisc' => [
|
|
|
|
'<?php
|
|
|
|
$a = max(new DateTimeImmutable(), 1.2);',
|
|
|
|
[
|
|
|
|
'$a' => 'DateTimeImmutable|float',
|
|
|
|
],
|
|
|
|
],
|
2021-03-31 22:16:21 -05:00
|
|
|
'strtolowerEmptiness' => [
|
|
|
|
'<?php
|
|
|
|
/** @param non-empty-string $s */
|
|
|
|
function foo(string $s) : void {
|
|
|
|
$s = strtolower($s);
|
|
|
|
|
|
|
|
foo($s);
|
|
|
|
}',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
];
|
2017-02-16 20:51:53 -05:00
|
|
|
}
|
2017-02-27 10:52:43 -05:00
|
|
|
|
|
|
|
/**
|
2021-03-19 20:44:44 -05:00
|
|
|
* @return iterable<string,array{string,error_message:string,1?:string[],2?:bool,3?:string}>
|
2017-02-27 10:52:43 -05:00
|
|
|
*/
|
2020-09-12 17:24:05 +02:00
|
|
|
public function providerInvalidCodeParse(): iterable
|
2017-02-27 10:52:43 -05:00
|
|
|
{
|
2017-04-24 23:45:02 -04:00
|
|
|
return [
|
|
|
|
'invalidScalarArgument' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo(int $a): void {}
|
2017-04-24 23:45:02 -04:00
|
|
|
fooFoo("string");',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'InvalidScalarArgument',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2018-08-28 12:37:25 -04:00
|
|
|
'invalidArgumentWithDeclareStrictTypes' => [
|
|
|
|
'<?php declare(strict_types=1);
|
|
|
|
function fooFoo(int $a): void {}
|
|
|
|
fooFoo("string");',
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
],
|
2018-08-28 17:42:39 -04:00
|
|
|
'builtinFunctioninvalidArgumentWithWeakTypes' => [
|
|
|
|
'<?php
|
|
|
|
$s = substr(5, 4);',
|
|
|
|
'error_message' => 'InvalidScalarArgument',
|
|
|
|
],
|
|
|
|
'builtinFunctioninvalidArgumentWithDeclareStrictTypes' => [
|
|
|
|
'<?php declare(strict_types=1);
|
|
|
|
$s = substr(5, 4);',
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
],
|
|
|
|
'builtinFunctioninvalidArgumentWithDeclareStrictTypesInClass' => [
|
|
|
|
'<?php declare(strict_types=1);
|
|
|
|
class A {
|
|
|
|
public function foo() : void {
|
|
|
|
$s = substr(5, 4);
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'mixedArgument' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo(int $a): void {}
|
2017-04-24 23:45:02 -04:00
|
|
|
/** @var mixed */
|
|
|
|
$a = "hello";
|
|
|
|
fooFoo($a);',
|
|
|
|
'error_message' => 'MixedArgument',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_levels' => ['MixedAssignment'],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'nullArgument' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo(int $a): void {}
|
2017-04-24 23:45:02 -04:00
|
|
|
fooFoo(null);',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'NullArgument',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'tooFewArguments' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo(int $a): void {}
|
2017-04-24 23:45:02 -04:00
|
|
|
fooFoo();',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'TooFewArguments',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'tooManyArguments' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo(int $a): void {}
|
2017-04-24 23:45:02 -04:00
|
|
|
fooFoo(5, "dfd");',
|
2020-03-15 13:44:00 -04:00
|
|
|
'error_message' => 'TooManyArguments - src' . DIRECTORY_SEPARATOR . 'somefile.php:3:21 - Too many arguments for fooFoo '
|
2018-02-02 11:26:55 -05:00
|
|
|
. '- expecting 1 but saw 2',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2017-05-24 23:11:40 -04:00
|
|
|
'tooManyArgumentsForConstructor' => [
|
|
|
|
'<?php
|
|
|
|
class A { }
|
|
|
|
new A("hello");',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'TooManyArguments',
|
2017-05-24 23:11:40 -04:00
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'typeCoercion' => [
|
|
|
|
'<?php
|
|
|
|
class A {}
|
|
|
|
class B extends A{}
|
2017-05-05 18:53:45 -04:00
|
|
|
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo(B $b): void {}
|
2017-04-24 23:45:02 -04:00
|
|
|
fooFoo(new A());',
|
2019-04-25 18:02:19 -04:00
|
|
|
'error_message' => 'ArgumentTypeCoercion',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'arrayTypeCoercion' => [
|
|
|
|
'<?php
|
|
|
|
class A {}
|
|
|
|
class B extends A{}
|
2017-05-05 18:53:45 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
/**
|
|
|
|
* @param B[] $b
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function fooFoo(array $b) {}
|
|
|
|
fooFoo([new A()]);',
|
2019-04-25 18:02:19 -04:00
|
|
|
'error_message' => 'ArgumentTypeCoercion',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'duplicateParam' => [
|
|
|
|
'<?php
|
2017-09-02 11:18:56 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
2017-04-24 23:45:02 -04:00
|
|
|
function f($p, $p) {}',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'DuplicateParam',
|
2018-01-28 20:03:47 -05:00
|
|
|
'error_levels' => ['MissingParamType'],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'invalidParamDefault' => [
|
|
|
|
'<?php
|
|
|
|
function f(int $p = false) {}',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'InvalidParamDefault',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'invalidDocblockParamDefault' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param int $p
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function f($p = false) {}',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'InvalidParamDefault',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2017-09-16 13:16:21 -04:00
|
|
|
'badByRef' => [
|
2017-04-24 23:45:02 -04:00
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo(string &$v): void {}
|
2017-04-24 23:45:02 -04:00
|
|
|
fooFoo("a");',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'InvalidPassByReference',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2017-12-22 15:21:23 +01:00
|
|
|
'badArrayByRef' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo(array &$a): void {}
|
2017-12-22 15:21:23 +01:00
|
|
|
fooFoo([1, 2, 3]);',
|
|
|
|
'error_message' => 'InvalidPassByReference',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'invalidArgAfterCallable' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param callable $callback
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
function route($callback) {
|
|
|
|
if (!is_callable($callback)) { }
|
|
|
|
takes_int("string");
|
|
|
|
}
|
2017-05-05 18:53:45 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
function takes_int(int $i) {}',
|
|
|
|
'error_message' => 'InvalidScalarArgument',
|
|
|
|
'error_levels' => [
|
|
|
|
'MixedAssignment',
|
2017-05-26 20:05:57 -04:00
|
|
|
'MixedArrayAccess',
|
2018-02-07 15:20:47 -05:00
|
|
|
'RedundantConditionGivenDocblockType',
|
2017-05-26 20:05:57 -04:00
|
|
|
],
|
|
|
|
],
|
2017-10-23 11:47:00 -04:00
|
|
|
'undefinedFunctionInArrayMap' => [
|
|
|
|
'<?php
|
|
|
|
array_map(
|
|
|
|
"undefined_function",
|
|
|
|
[1, 2, 3]
|
|
|
|
);',
|
|
|
|
'error_message' => 'UndefinedFunction',
|
|
|
|
],
|
2017-11-19 13:05:35 -05:00
|
|
|
'objectLikeKeyChecksAgainstDifferentGeneric' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param array<string, int> $b
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function a($b): int
|
2017-11-19 13:05:35 -05:00
|
|
|
{
|
|
|
|
return $b["a"];
|
|
|
|
}
|
|
|
|
|
|
|
|
a(["a" => "hello"]);',
|
|
|
|
'error_message' => 'InvalidScalarArgument',
|
|
|
|
],
|
2020-08-30 11:44:14 -04:00
|
|
|
'objectLikeKeyChecksAgainstDifferentTKeyedArray' => [
|
2017-11-19 13:05:35 -05:00
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param array{a: int} $b
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function a($b): int
|
2017-11-19 13:05:35 -05:00
|
|
|
{
|
|
|
|
return $b["a"];
|
|
|
|
}
|
|
|
|
|
|
|
|
a(["a" => "hello"]);',
|
2018-02-10 23:30:40 -05:00
|
|
|
'error_message' => 'InvalidArgument',
|
2017-11-19 13:05:35 -05:00
|
|
|
],
|
2018-01-22 00:17:16 -05:00
|
|
|
'possiblyNullFunctionCall' => [
|
|
|
|
'<?php
|
|
|
|
$a = rand(0, 1) ? function(): void {} : null;
|
|
|
|
$a();',
|
|
|
|
'error_message' => 'PossiblyNullFunctionCall',
|
|
|
|
],
|
|
|
|
'possiblyInvalidFunctionCall' => [
|
|
|
|
'<?php
|
2018-02-11 20:56:34 -05:00
|
|
|
$a = rand(0, 1) ? function(): void {} : 23515;
|
2018-01-22 00:17:16 -05:00
|
|
|
$a();',
|
|
|
|
'error_message' => 'PossiblyInvalidFunctionCall',
|
|
|
|
],
|
2018-04-02 22:19:58 -04:00
|
|
|
'varExportAssignmentToVoid' => [
|
|
|
|
'<?php
|
|
|
|
$a = var_export(["a"]);',
|
|
|
|
'error_message' => 'AssignmentToVoid',
|
|
|
|
],
|
2018-04-16 16:03:04 -04:00
|
|
|
'explodeWithEmptyString' => [
|
|
|
|
'<?php
|
|
|
|
function exploder(string $s) : array {
|
|
|
|
return explode("", $s);
|
|
|
|
}',
|
|
|
|
'error_message' => 'FalsableReturnStatement',
|
|
|
|
],
|
2018-04-30 13:17:09 -04:00
|
|
|
'complainAboutArrayToIterable' => [
|
|
|
|
'<?php
|
|
|
|
class A {}
|
|
|
|
class B {}
|
|
|
|
/**
|
|
|
|
* @param iterable<mixed,A> $p
|
|
|
|
*/
|
|
|
|
function takesIterableOfA(iterable $p): void {}
|
|
|
|
|
2018-04-30 22:18:41 -04:00
|
|
|
takesIterableOfA([new B]); // should complain',
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
],
|
|
|
|
'complainAboutArrayToIterableSingleParam' => [
|
|
|
|
'<?php
|
|
|
|
class A {}
|
|
|
|
class B {}
|
|
|
|
/**
|
|
|
|
* @param iterable<A> $p
|
|
|
|
*/
|
|
|
|
function takesIterableOfA(iterable $p): void {}
|
|
|
|
|
2018-04-30 13:17:09 -04:00
|
|
|
takesIterableOfA([new B]); // should complain',
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
],
|
2018-05-20 13:14:31 -04:00
|
|
|
'putInvalidTypeMessagesFirst' => [
|
|
|
|
'<?php
|
|
|
|
$q = rand(0,1) ? new stdClass : false;
|
|
|
|
strlen($q);',
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
],
|
2019-01-05 16:23:18 -05:00
|
|
|
'getTypeInvalidValue' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param mixed $maybe
|
|
|
|
*/
|
|
|
|
function matchesTypes($maybe) : void {
|
|
|
|
$t = gettype($maybe);
|
|
|
|
if ($t === "bool") {}
|
|
|
|
}',
|
|
|
|
'error_message' => 'TypeDoesNotContainType',
|
|
|
|
],
|
2019-01-06 13:02:46 -05:00
|
|
|
'rangeWithFloatStep' => [
|
|
|
|
'<?php
|
|
|
|
|
|
|
|
function foo(int $bar) : string {
|
|
|
|
return (string) $bar;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (range(1, 10, .3) as $x) {
|
|
|
|
foo($x);
|
|
|
|
}',
|
|
|
|
'error_message' => 'InvalidScalarArgument',
|
|
|
|
],
|
|
|
|
'rangeWithFloatStart' => [
|
|
|
|
'<?php
|
|
|
|
|
|
|
|
function foo(int $bar) : string {
|
|
|
|
return (string) $bar;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (range(1.4, 10) as $x) {
|
|
|
|
foo($x);
|
|
|
|
}',
|
|
|
|
'error_message' => 'InvalidScalarArgument',
|
|
|
|
],
|
2019-01-06 16:40:44 -05:00
|
|
|
'duplicateFunction' => [
|
|
|
|
'<?php
|
|
|
|
function f() : void {}
|
|
|
|
function f() : void {}',
|
|
|
|
'error_message' => 'DuplicateFunction',
|
|
|
|
],
|
|
|
|
'duplicateCoreFunction' => [
|
|
|
|
'<?php
|
|
|
|
function sort() : void {}',
|
|
|
|
'error_message' => 'DuplicateFunction',
|
|
|
|
],
|
2019-02-10 16:15:52 -05:00
|
|
|
'functionCallOnMixed' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @var mixed $s
|
|
|
|
* @psalm-suppress MixedAssignment
|
|
|
|
*/
|
|
|
|
$s = 1;
|
|
|
|
$s();',
|
|
|
|
'error_message' => 'MixedFunctionCall',
|
|
|
|
],
|
2019-02-18 16:35:23 -05:00
|
|
|
'iterableOfObjectCannotAcceptIterableOfInt' => [
|
|
|
|
'<?php
|
|
|
|
/** @param iterable<string,object> $_p */
|
|
|
|
function accepts(iterable $_p): void {}
|
|
|
|
|
|
|
|
/** @return iterable<int,int> */
|
|
|
|
function iterable() { yield 1; }
|
|
|
|
|
|
|
|
accepts(iterable());',
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
],
|
|
|
|
'iterableOfObjectCannotAcceptTraversableOfInt' => [
|
|
|
|
'<?php
|
|
|
|
/** @param iterable<string,object> $_p */
|
|
|
|
function accepts(iterable $_p): void {}
|
|
|
|
|
|
|
|
/** @return Traversable<int,int> */
|
|
|
|
function traversable() { yield 1; }
|
|
|
|
|
|
|
|
accepts(traversable());',
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
],
|
|
|
|
'iterableOfObjectCannotAcceptGeneratorOfInt' => [
|
|
|
|
'<?php
|
|
|
|
/** @param iterable<string,object> $_p */
|
|
|
|
function accepts(iterable $_p): void {}
|
|
|
|
|
|
|
|
/** @return Generator<int,int,mixed,void> */
|
|
|
|
function generator() { yield 1; }
|
|
|
|
|
|
|
|
accepts(generator());',
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
],
|
|
|
|
'iterableOfObjectCannotAcceptArrayOfInt' => [
|
|
|
|
'<?php
|
|
|
|
/** @param iterable<string,object> $_p */
|
|
|
|
function accepts(iterable $_p): void {}
|
|
|
|
|
|
|
|
/** @return array<int,int> */
|
|
|
|
function arr() { return [1]; }
|
|
|
|
|
|
|
|
accepts(arr());',
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
],
|
2019-02-20 17:43:12 -05:00
|
|
|
'nonNullableByRef' => [
|
|
|
|
'<?php
|
|
|
|
function foo(string &$s) : void {}
|
|
|
|
|
|
|
|
function bar() : void {
|
|
|
|
foo($bar);
|
|
|
|
}',
|
2019-04-25 18:02:19 -04:00
|
|
|
'error_message' => 'NullReference',
|
2019-02-20 17:43:12 -05:00
|
|
|
],
|
2019-02-21 13:26:37 -05:00
|
|
|
'intCastByRef' => [
|
|
|
|
'<?php
|
|
|
|
function foo(int &$i) : void {}
|
|
|
|
|
|
|
|
$a = rand(0, 1) ? null : 5;
|
|
|
|
/** @psalm-suppress MixedArgument */
|
|
|
|
foo((int) $a);',
|
2019-05-21 11:51:41 -04:00
|
|
|
'error_message' => 'InvalidPassByReference',
|
|
|
|
],
|
|
|
|
'implicitAssignmentToStringFromMixed' => [
|
|
|
|
'<?php
|
|
|
|
/** @param "a"|"b" $s */
|
|
|
|
function takesString(string $s) : void {}
|
|
|
|
function takesInt(int $i) : void {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param mixed $s
|
|
|
|
* @psalm-suppress MixedArgument
|
|
|
|
*/
|
|
|
|
function bar($s) : void {
|
|
|
|
takesString($s);
|
|
|
|
takesInt($s);
|
|
|
|
}',
|
2019-07-05 16:24:00 -04:00
|
|
|
'error_message' => 'InvalidScalarArgument',
|
2019-02-21 13:26:37 -05:00
|
|
|
],
|
2019-06-19 09:04:01 -04:00
|
|
|
'tooFewArgsAccurateCount' => [
|
|
|
|
'<?php
|
|
|
|
preg_match(\'/adsf/\');',
|
2020-03-15 13:44:00 -04:00
|
|
|
'error_message' => 'TooFewArguments - src' . DIRECTORY_SEPARATOR . 'somefile.php:2:21 - Too few arguments for preg_match - expecting 2 but saw 1',
|
2019-07-05 16:24:00 -04:00
|
|
|
],
|
2019-07-24 17:24:23 -04:00
|
|
|
'compactUndefinedVariable' => [
|
|
|
|
'<?php
|
2020-04-21 18:18:11 -06:00
|
|
|
/**
|
|
|
|
* @return array<string, mixed>
|
|
|
|
*/
|
2019-07-24 17:24:23 -04:00
|
|
|
function foo() : array {
|
|
|
|
return compact("a", "b", "c");
|
|
|
|
}',
|
|
|
|
'error_message' => 'UndefinedVariable',
|
|
|
|
],
|
2019-08-12 16:17:55 -04:00
|
|
|
'countCallableArrayShouldBeTwo' => [
|
|
|
|
'<?php
|
|
|
|
/** @param callable|false $x */
|
|
|
|
function example($x) : void {
|
|
|
|
if (is_array($x)) {
|
|
|
|
$c = count($x);
|
|
|
|
if ($c !== 2) {}
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'error_message' => 'TypeDoesNotContainType',
|
|
|
|
],
|
2021-01-13 15:48:38 +01:00
|
|
|
'countOnObjectCannotBePositive' => [
|
|
|
|
'<?php
|
|
|
|
/** @return positive-int|0 */
|
|
|
|
function example(\Countable $x) : int {
|
|
|
|
return count($x);
|
|
|
|
}',
|
|
|
|
'error_message' => 'LessSpecificReturnStatement',
|
|
|
|
],
|
|
|
|
'countOnUnknownObjectCannotBePure' => [
|
|
|
|
'<?php
|
|
|
|
/** @psalm-pure */
|
|
|
|
function example(\Countable $x) : int {
|
|
|
|
return count($x);
|
|
|
|
}',
|
|
|
|
'error_message' => 'ImpureFunctionCall',
|
|
|
|
],
|
2019-09-11 13:52:37 -04:00
|
|
|
'coerceCallMapArgsInStrictMode' => [
|
|
|
|
'<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
function getStr() : ?string {
|
|
|
|
return rand(0,1) ? "test" : null;
|
|
|
|
}
|
|
|
|
|
|
|
|
function test() : void {
|
|
|
|
$g = getStr();
|
|
|
|
/** @psalm-suppress PossiblyNullArgument */
|
|
|
|
$x = strtoupper($g);
|
|
|
|
$c = "prefix " . (strtoupper($g ?? "") === "x" ? "xa" : "ya");
|
|
|
|
echo "$x, $c\n";
|
|
|
|
}',
|
2020-11-25 20:04:57 -05:00
|
|
|
'error_message' => 'RedundantCondition',
|
2019-09-11 13:52:37 -04:00
|
|
|
],
|
2019-10-01 08:45:36 -04:00
|
|
|
'noCrashOnEmptyArrayPush' => [
|
|
|
|
'<?php
|
|
|
|
array_push();',
|
|
|
|
'error_message' => 'TooFewArguments',
|
|
|
|
],
|
2019-11-08 13:01:34 +01:00
|
|
|
'printOnlyString' => [
|
|
|
|
'<?php
|
|
|
|
print [];',
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
],
|
|
|
|
'printReturns1' => [
|
|
|
|
'<?php
|
|
|
|
(print "test") === 2;',
|
|
|
|
'error_message' => 'TypeDoesNotContainType',
|
|
|
|
],
|
2020-03-02 21:33:03 +01:00
|
|
|
'sodiumMemzeroNullifyString' => [
|
|
|
|
'<?php
|
|
|
|
function returnsStr(): string {
|
|
|
|
$str = "x";
|
|
|
|
sodium_memzero($str);
|
|
|
|
return $str;
|
|
|
|
}',
|
|
|
|
'error_message' => 'NullableReturnStatement'
|
2020-03-09 10:56:37 -04:00
|
|
|
],
|
|
|
|
'noCrashWithPattern' => [
|
|
|
|
'<?php
|
|
|
|
echo !\is_callable($loop_callback)
|
|
|
|
|| (\is_array($loop_callback)
|
|
|
|
&& !\method_exists(...$loop_callback));',
|
|
|
|
'error_message' => 'UndefinedGlobalVariable'
|
|
|
|
],
|
2020-03-18 20:17:24 +01:00
|
|
|
'parseUrlPossiblyUndefined' => [
|
|
|
|
'<?php
|
|
|
|
function bar(string $s) : string {
|
|
|
|
$parsed = parse_url($s);
|
|
|
|
|
|
|
|
return $parsed["host"];
|
|
|
|
}',
|
|
|
|
'error_message' => 'PossiblyUndefinedArrayOffset',
|
|
|
|
],
|
|
|
|
'parseUrlPossiblyUndefined2' => [
|
|
|
|
'<?php
|
|
|
|
function bag(string $s) : string {
|
|
|
|
$parsed = parse_url($s);
|
|
|
|
|
|
|
|
if (is_string($parsed["host"] ?? false)) {
|
|
|
|
return $parsed["host"];
|
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
|
|
|
}',
|
|
|
|
'error_message' => 'PossiblyUndefinedArrayOffset',
|
|
|
|
],
|
2020-09-10 18:28:34 -04:00
|
|
|
'strposNoSetFirstParam' => [
|
|
|
|
'<?php
|
|
|
|
function sayHello(string $format): void {
|
|
|
|
if (strpos("u", $format)) {}
|
|
|
|
}',
|
2020-09-10 22:44:35 -04:00
|
|
|
'error_message' => 'InvalidLiteralArgument',
|
2020-09-10 18:28:34 -04:00
|
|
|
],
|
2020-10-12 09:57:11 -04:00
|
|
|
'curlInitIsResourceFailsIn8x' => [
|
|
|
|
'<?php
|
|
|
|
$ch = curl_init();
|
|
|
|
if (!is_resource($ch)) {}',
|
|
|
|
'error_message' => 'RedundantCondition',
|
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.0'
|
|
|
|
],
|
2020-10-12 12:09:08 -04:00
|
|
|
'maxCallWithArray' => [
|
|
|
|
'<?php
|
|
|
|
function foo(array $a) {
|
|
|
|
max($a);
|
|
|
|
}',
|
|
|
|
'error_message' => 'ArgumentTypeCoercion',
|
|
|
|
],
|
2020-10-12 13:04:28 -04:00
|
|
|
'pregSplitNoEmpty' => [
|
|
|
|
'<?php
|
|
|
|
/** @return non-empty-list */
|
|
|
|
function foo(string $s) {
|
|
|
|
return preg_split("/ /", $s, -1, PREG_SPLIT_NO_EMPTY);
|
|
|
|
}',
|
|
|
|
'error_message' => 'InvalidReturnStatement'
|
|
|
|
],
|
2021-03-11 06:13:17 +01:00
|
|
|
'maxWithMixed' => [
|
|
|
|
'<?php
|
|
|
|
/** @var mixed $b */;
|
|
|
|
/** @var mixed $c */;
|
|
|
|
$a = max($b, $c);',
|
|
|
|
'error_message' => 'MixedAssignment'
|
|
|
|
],
|
2021-07-13 00:06:36 +02:00
|
|
|
'literalFalseArgument' => [
|
|
|
|
'<?php
|
|
|
|
function takesAString(string $s): void{
|
|
|
|
echo $s;
|
|
|
|
}
|
|
|
|
|
|
|
|
takesAString(false);',
|
|
|
|
'error_message' => 'InvalidArgument'
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
];
|
2017-02-27 10:52:43 -05:00
|
|
|
}
|
2021-07-19 19:40:17 +02:00
|
|
|
|
|
|
|
public function testTriggerErrorDefault(): void
|
|
|
|
{
|
|
|
|
$config = \Psalm\Config::getInstance();
|
|
|
|
$config->trigger_error_exits = 'default';
|
|
|
|
|
|
|
|
$this->addFile(
|
|
|
|
'somefile.php',
|
|
|
|
'<?php
|
|
|
|
/** @return true */
|
|
|
|
function returnsTrue(): bool {
|
|
|
|
return trigger_error("", E_USER_NOTICE);
|
|
|
|
}
|
|
|
|
/** @return never */
|
|
|
|
function returnsNever(): void {
|
|
|
|
trigger_error("", E_USER_ERROR);
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* @psalm-suppress ArgumentTypeCoercion
|
|
|
|
* @return mixed
|
|
|
|
*/
|
|
|
|
function returnsNeverOrBool(int $i) {
|
|
|
|
return trigger_error("", $i);
|
|
|
|
}'
|
|
|
|
);
|
|
|
|
|
|
|
|
//will only pass if no exception is thrown
|
|
|
|
$this->assertTrue(true);
|
|
|
|
|
|
|
|
$this->analyzeFile('somefile.php', new \Psalm\Context());
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testTriggerErrorAlways(): void
|
|
|
|
{
|
|
|
|
$config = \Psalm\Config::getInstance();
|
|
|
|
$config->trigger_error_exits = 'always';
|
|
|
|
|
|
|
|
$this->addFile(
|
|
|
|
'somefile.php',
|
|
|
|
'<?php
|
|
|
|
/** @return never */
|
|
|
|
function returnsNever1(): void {
|
|
|
|
trigger_error("", E_USER_NOTICE);
|
|
|
|
}
|
|
|
|
/** @return never */
|
|
|
|
function returnsNever2(): void {
|
|
|
|
trigger_error("", E_USER_ERROR);
|
|
|
|
}'
|
|
|
|
);
|
|
|
|
|
|
|
|
//will only pass if no exception is thrown
|
|
|
|
$this->assertTrue(true);
|
|
|
|
|
|
|
|
$this->analyzeFile('somefile.php', new \Psalm\Context());
|
|
|
|
}
|
|
|
|
|
2021-07-20 23:55:49 +02:00
|
|
|
public function testTriggerErrorNever(): void
|
2021-07-19 19:40:17 +02:00
|
|
|
{
|
|
|
|
$config = \Psalm\Config::getInstance();
|
2021-07-20 23:53:04 +02:00
|
|
|
$config->trigger_error_exits = 'never';
|
2021-07-19 19:40:17 +02:00
|
|
|
|
|
|
|
$this->addFile(
|
|
|
|
'somefile.php',
|
|
|
|
'<?php
|
|
|
|
/** @return true */
|
|
|
|
function returnsTrue1(): bool {
|
|
|
|
return trigger_error("", E_USER_NOTICE);
|
|
|
|
}
|
|
|
|
/** @return true */
|
|
|
|
function returnsTrue2(): bool {
|
|
|
|
return trigger_error("", E_USER_ERROR);
|
|
|
|
}'
|
|
|
|
);
|
|
|
|
|
|
|
|
//will only pass if no exception is thrown
|
|
|
|
$this->assertTrue(true);
|
|
|
|
|
|
|
|
$this->analyzeFile('somefile.php', new \Psalm\Context());
|
|
|
|
}
|
2016-12-11 23:41:11 -05:00
|
|
|
}
|