2021-09-09 16:04:12 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
2023-03-16 07:08:48 +01:00
|
|
|
use Psalm\Tests\Traits\InvalidCodeAnalysisTestTrait;
|
2021-12-04 21:55:53 +01:00
|
|
|
use Psalm\Tests\Traits\ValidCodeAnalysisTestTrait;
|
|
|
|
|
2021-09-09 16:04:12 +02:00
|
|
|
class CoreStubsTest extends TestCase
|
|
|
|
{
|
2021-12-04 21:55:53 +01:00
|
|
|
use ValidCodeAnalysisTestTrait;
|
2023-03-16 07:08:48 +01:00
|
|
|
use InvalidCodeAnalysisTestTrait;
|
2021-09-09 16:04:12 +02:00
|
|
|
|
|
|
|
public function providerValidCodeParse(): iterable
|
|
|
|
{
|
|
|
|
yield 'RecursiveArrayIterator::CHILD_ARRAYS_ONLY (#6464)' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-09-09 16:04:12 +02:00
|
|
|
|
2022-12-18 17:15:15 +01:00
|
|
|
new RecursiveArrayIterator([], RecursiveArrayIterator::CHILD_ARRAYS_ONLY);',
|
2021-09-09 16:04:12 +02:00
|
|
|
];
|
2022-01-20 13:54:37 +01:00
|
|
|
yield 'proc_open() named arguments' => [
|
2022-01-22 18:05:32 +01:00
|
|
|
'code' => '<?php
|
2022-01-20 13:54:37 +01:00
|
|
|
|
|
|
|
proc_open(
|
|
|
|
command: "ls",
|
|
|
|
descriptor_spec: [],
|
|
|
|
pipes: $pipes,
|
|
|
|
cwd: null,
|
|
|
|
env_vars: null,
|
|
|
|
options: null
|
|
|
|
);',
|
|
|
|
'assertions' => [],
|
2022-01-22 18:05:32 +01:00
|
|
|
'ignored_issues' => [],
|
2022-01-20 13:54:37 +01:00
|
|
|
'php_version' => '8.0',
|
|
|
|
];
|
2022-07-25 15:37:49 +02:00
|
|
|
yield 'Iterating over \DatePeriod (#5954) PHP7 Traversable' => [
|
2022-10-17 10:21:26 +02:00
|
|
|
'code' => '<?php
|
2022-07-25 15:37:49 +02:00
|
|
|
|
|
|
|
$period = new DatePeriod(
|
|
|
|
new DateTimeImmutable("now"),
|
|
|
|
DateInterval::createFromDateString("1 day"),
|
|
|
|
new DateTime("+1 week")
|
|
|
|
);
|
|
|
|
$dt = null;
|
|
|
|
foreach ($period as $dt) {
|
|
|
|
echo $dt->format("Y-m-d");
|
|
|
|
}',
|
|
|
|
'assertions' => [
|
|
|
|
'$period' => 'DatePeriod<DateTimeImmutable>',
|
2022-12-18 17:15:15 +01:00
|
|
|
'$dt' => 'DateTimeInterface|null',
|
2022-07-25 15:37:49 +02:00
|
|
|
],
|
2022-11-05 22:34:42 +01:00
|
|
|
'ignored_issues' => [],
|
2022-07-25 15:37:49 +02:00
|
|
|
'php_version' => '7.3',
|
|
|
|
];
|
|
|
|
yield 'Iterating over \DatePeriod (#5954) PHP8 IteratorAggregate' => [
|
2022-10-17 10:21:26 +02:00
|
|
|
'code' => '<?php
|
2022-07-22 15:03:45 +02:00
|
|
|
|
|
|
|
$period = new DatePeriod(
|
|
|
|
new DateTimeImmutable("now"),
|
|
|
|
DateInterval::createFromDateString("1 day"),
|
|
|
|
new DateTime("+1 week")
|
|
|
|
);
|
|
|
|
$dt = null;
|
|
|
|
foreach ($period as $dt) {
|
|
|
|
echo $dt->format("Y-m-d");
|
|
|
|
}',
|
|
|
|
'assertions' => [
|
|
|
|
'$period' => 'DatePeriod<DateTimeImmutable>',
|
2022-12-18 17:15:15 +01:00
|
|
|
'$dt' => 'DateTimeImmutable|null',
|
2022-07-22 15:03:45 +02:00
|
|
|
],
|
2022-11-05 22:34:42 +01:00
|
|
|
'ignored_issues' => [],
|
2022-07-25 15:37:49 +02:00
|
|
|
'php_version' => '8.0',
|
2022-07-22 15:03:45 +02:00
|
|
|
];
|
|
|
|
yield 'Iterating over \DatePeriod (#5954), ISO string' => [
|
2022-10-17 10:21:26 +02:00
|
|
|
'code' => '<?php
|
2022-07-22 15:03:45 +02:00
|
|
|
|
|
|
|
$period = new DatePeriod("R4/2012-07-01T00:00:00Z/P7D");
|
|
|
|
$dt = null;
|
|
|
|
foreach ($period as $dt) {
|
|
|
|
echo $dt->format("Y-m-d");
|
|
|
|
}',
|
|
|
|
'assertions' => [
|
|
|
|
'$period' => 'DatePeriod<string>',
|
2022-12-18 17:15:15 +01:00
|
|
|
'$dt' => 'DateTime|null',
|
2022-07-22 15:03:45 +02:00
|
|
|
],
|
2022-11-05 22:34:42 +01:00
|
|
|
'ignored_issues' => [],
|
2022-07-25 15:37:49 +02:00
|
|
|
'php_version' => '8.0',
|
|
|
|
];
|
|
|
|
yield 'DatePeriod implements only Traversable on PHP 7' => [
|
2022-10-17 10:21:26 +02:00
|
|
|
'code' => '<?php
|
2022-07-25 15:37:49 +02:00
|
|
|
|
|
|
|
$period = new DatePeriod("R4/2012-07-01T00:00:00Z/P7D");
|
|
|
|
if ($period instanceof IteratorAggregate) {}',
|
|
|
|
'assertions' => [],
|
2022-11-05 22:34:42 +01:00
|
|
|
'ignored_issues' => [],
|
2022-07-25 15:37:49 +02:00
|
|
|
'php_version' => '7.3',
|
|
|
|
];
|
|
|
|
yield 'DatePeriod implements IteratorAggregate on PHP 8' => [
|
2022-10-17 10:21:26 +02:00
|
|
|
'code' => '<?php
|
2022-07-25 15:37:49 +02:00
|
|
|
|
|
|
|
$period = new DatePeriod("R4/2012-07-01T00:00:00Z/P7D");
|
|
|
|
if ($period instanceof IteratorAggregate) {}',
|
|
|
|
'assertions' => [],
|
2022-11-05 22:34:42 +01:00
|
|
|
'ignored_issues' => ['RedundantCondition'],
|
2022-07-25 15:37:49 +02:00
|
|
|
'php_version' => '8.0',
|
2022-07-22 15:03:45 +02:00
|
|
|
];
|
2022-12-18 11:04:16 +01:00
|
|
|
yield 'sprintf yields a non-empty-string for non-empty-string value' => [
|
2022-12-18 11:26:43 +01:00
|
|
|
'code' => '<?php
|
2022-12-18 10:12:49 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param non-empty-string $foo
|
|
|
|
* @return non-empty-string
|
|
|
|
*/
|
|
|
|
function foo(string $foo): string
|
|
|
|
{
|
|
|
|
return sprintf("%s", $foo);
|
|
|
|
}
|
|
|
|
',
|
|
|
|
];
|
2022-12-18 11:04:16 +01:00
|
|
|
yield 'sprintf yields a string for possible empty string param' => [
|
2022-12-18 11:26:43 +01:00
|
|
|
'code' => '<?php
|
2022-12-18 11:04:16 +01:00
|
|
|
|
|
|
|
$a = sprintf("%s", "");
|
|
|
|
',
|
|
|
|
'assertions' => [
|
|
|
|
'$a===' => 'string',
|
2022-12-18 11:26:43 +01:00
|
|
|
],
|
2022-12-18 11:04:16 +01:00
|
|
|
];
|
2023-03-10 06:30:59 +01:00
|
|
|
yield 'json_encode returns a non-empty-string provided JSON_THROW_ON_ERROR | JSON_UNESCAPED_UNICODE' => [
|
|
|
|
'code' => '<?php
|
2023-03-10 06:44:53 +01:00
|
|
|
$a = json_encode([], JSON_UNESCAPED_UNICODE | JSON_THROW_ON_ERROR);
|
2023-03-10 06:30:59 +01:00
|
|
|
',
|
2023-03-10 06:44:53 +01:00
|
|
|
'assertions' => [
|
|
|
|
'$a===' => 'non-empty-string',
|
|
|
|
],
|
2023-03-10 06:30:59 +01:00
|
|
|
];
|
2023-03-16 07:08:07 +01:00
|
|
|
yield 'json_encode returns a non-empty-string with JSON_THROW_ON_ERROR' => [
|
|
|
|
'code' => '<?php
|
|
|
|
$a = json_encode([], JSON_THROW_ON_ERROR | JSON_HEX_TAG);
|
|
|
|
$b = json_encode([], JSON_THROW_ON_ERROR | JSON_HEX_TAG | JSON_HEX_APOS | JSON_HEX_QUOT | JSON_HEX_AMP | JSON_UNESCAPED_UNICODE);
|
|
|
|
$c = json_encode([], JSON_THROW_ON_ERROR | JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE);
|
|
|
|
$d = json_encode([], JSON_THROW_ON_ERROR | JSON_PRESERVE_ZERO_FRACTION);
|
2023-03-16 07:42:36 +01:00
|
|
|
$e = json_encode([], JSON_PRESERVE_ZERO_FRACTION);
|
|
|
|
$f = json_encode([], JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE);
|
2023-03-16 07:08:07 +01:00
|
|
|
',
|
|
|
|
'assertions' => [
|
|
|
|
'$a===' => 'non-empty-string',
|
|
|
|
'$b===' => 'non-empty-string',
|
|
|
|
'$c===' => 'non-empty-string',
|
|
|
|
'$d===' => 'non-empty-string',
|
2023-03-16 07:42:36 +01:00
|
|
|
'$e===' => 'false|non-empty-string',
|
|
|
|
'$f===' => 'false|non-empty-string',
|
2023-03-16 07:08:07 +01:00
|
|
|
],
|
|
|
|
];
|
2023-03-17 11:39:04 +01:00
|
|
|
yield 'str_starts_with/str_ends_with/str_contains redundant condition detection' => [
|
|
|
|
'code' => '<?php
|
|
|
|
$a1 = str_starts_with(uniqid(), "");
|
|
|
|
/** @psalm-suppress InvalidLiteralArgument */
|
|
|
|
$b1 = str_starts_with("", "random string");
|
|
|
|
$c1 = str_starts_with(uniqid(), "random string");
|
|
|
|
|
|
|
|
$a2 = str_ends_with(uniqid(), "");
|
|
|
|
/** @psalm-suppress InvalidLiteralArgument */
|
|
|
|
$b2 = str_ends_with("", "random string");
|
|
|
|
$c2 = str_ends_with(uniqid(), "random string");
|
|
|
|
|
|
|
|
$a3 = str_contains(uniqid(), "");
|
|
|
|
/** @psalm-suppress InvalidLiteralArgument */
|
|
|
|
$b3 = str_contains("", "random string");
|
|
|
|
$c3 = str_contains(uniqid(), "random string");
|
|
|
|
',
|
|
|
|
'assertions' => [
|
|
|
|
'$a1===' => 'true',
|
|
|
|
'$b1===' => 'false',
|
|
|
|
'$c1===' => 'bool',
|
|
|
|
'$a2===' => 'true',
|
|
|
|
'$b2===' => 'false',
|
|
|
|
'$c2===' => 'bool',
|
|
|
|
'$a3===' => 'true',
|
|
|
|
'$b3===' => 'false',
|
|
|
|
'$c3===' => 'bool',
|
|
|
|
],
|
|
|
|
];
|
2023-03-17 12:58:44 +01:00
|
|
|
yield 'PHP8 str_* function assert non-empty-string' => [
|
|
|
|
'code' => '<?php
|
|
|
|
/** @return non-empty-string */
|
|
|
|
function after_str_contains(): string
|
|
|
|
{
|
|
|
|
$string = file_get_contents("");
|
|
|
|
if (str_contains($string, "foo")) {
|
|
|
|
return $string;
|
|
|
|
}
|
|
|
|
throw new RuntimeException();
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return non-empty-string */
|
|
|
|
function after_str_starts_with(): string
|
|
|
|
{
|
|
|
|
$string = file_get_contents("");
|
|
|
|
if (str_starts_with($string, "foo")) {
|
|
|
|
return $string;
|
|
|
|
}
|
|
|
|
throw new RuntimeException();
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return non-empty-string */
|
|
|
|
function after_str_ends_with(): string
|
|
|
|
{
|
|
|
|
$string = file_get_contents("");
|
|
|
|
if (str_ends_with($string, "foo")) {
|
|
|
|
return $string;
|
|
|
|
}
|
|
|
|
throw new RuntimeException();
|
|
|
|
}
|
|
|
|
$a = after_str_contains();
|
|
|
|
$b = after_str_starts_with();
|
|
|
|
$c = after_str_ends_with();
|
|
|
|
',
|
|
|
|
'assertions' => [
|
|
|
|
'$a===' => 'non-empty-string',
|
|
|
|
'$b===' => 'non-empty-string',
|
|
|
|
'$c===' => 'non-empty-string',
|
|
|
|
],
|
|
|
|
];
|
|
|
|
yield "PHP8 str_* function doesn't subtract string after assertion" => [
|
|
|
|
'code' => '<?php
|
|
|
|
/** @return false|string */
|
|
|
|
function after_str_contains()
|
|
|
|
{
|
|
|
|
$string = file_get_contents("");
|
|
|
|
if (!str_contains($string, "foo")) {
|
|
|
|
return $string;
|
|
|
|
}
|
|
|
|
throw new RuntimeException();
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return false|string */
|
|
|
|
function after_str_starts_with()
|
|
|
|
{
|
|
|
|
$string = file_get_contents("");
|
|
|
|
if (!str_starts_with($string, "foo")) {
|
|
|
|
return $string;
|
|
|
|
}
|
|
|
|
throw new RuntimeException();
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return false|string */
|
|
|
|
function after_str_ends_with()
|
|
|
|
{
|
|
|
|
$string = file_get_contents("");
|
|
|
|
if (!str_ends_with($string, "foo")) {
|
|
|
|
return $string;
|
|
|
|
}
|
|
|
|
throw new RuntimeException();
|
|
|
|
}
|
|
|
|
$a = after_str_contains();
|
|
|
|
$b = after_str_starts_with();
|
|
|
|
$c = after_str_ends_with();
|
|
|
|
',
|
|
|
|
'assertions' => [
|
|
|
|
'$a===' => 'false|string',
|
|
|
|
'$b===' => 'false|string',
|
|
|
|
'$c===' => 'false|string',
|
|
|
|
],
|
|
|
|
];
|
2023-03-17 19:05:44 +01:00
|
|
|
yield "str_contains doesn't yield InvalidLiteralArgument for __DIR__" => [
|
|
|
|
'code' => '<?php
|
|
|
|
$d = __DIR__;
|
|
|
|
echo str_contains($d, "psalm");
|
|
|
|
',
|
|
|
|
];
|
2021-09-09 16:04:12 +02:00
|
|
|
}
|
2023-03-16 07:08:48 +01:00
|
|
|
|
|
|
|
public function providerInvalidCodeParse(): iterable
|
|
|
|
{
|
|
|
|
yield 'json_decode invalid depth' => [
|
|
|
|
'code' => '<?php
|
|
|
|
json_decode("true", depth: -1);
|
|
|
|
',
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
];
|
|
|
|
yield 'json_encode invalid depth' => [
|
|
|
|
'code' => '<?php
|
|
|
|
json_encode([], depth: 439877348953739);
|
|
|
|
',
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
];
|
2023-03-17 11:41:36 +01:00
|
|
|
yield 'str_contains literal haystack' => [
|
|
|
|
'code' => '<?php
|
|
|
|
str_contains("literal", "");
|
|
|
|
',
|
|
|
|
'error_message' => 'InvalidLiteralArgument',
|
|
|
|
];
|
|
|
|
yield 'str_starts_with literal haystack' => [
|
|
|
|
'code' => '<?php
|
|
|
|
str_starts_with("literal", "");
|
|
|
|
',
|
|
|
|
'error_message' => 'InvalidLiteralArgument',
|
|
|
|
];
|
|
|
|
yield 'str_ends_with literal haystack' => [
|
|
|
|
'code' => '<?php
|
|
|
|
str_ends_with("literal", "");
|
|
|
|
',
|
|
|
|
'error_message' => 'InvalidLiteralArgument',
|
|
|
|
];
|
2023-03-16 07:08:48 +01:00
|
|
|
}
|
2021-09-09 16:04:12 +02:00
|
|
|
}
|