1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-06 04:59:14 +01:00
psalm/tests/Php71Test.php

372 lines
12 KiB
PHP
Raw Normal View History

2016-12-04 05:03:51 +01:00
<?php
2016-12-04 05:03:51 +01:00
namespace Psalm\Tests;
2021-12-04 21:55:53 +01:00
use Psalm\Tests\Traits\InvalidCodeAnalysisTestTrait;
use Psalm\Tests\Traits\ValidCodeAnalysisTestTrait;
class Php71Test extends TestCase
2016-12-04 05:03:51 +01:00
{
2021-12-04 21:55:53 +01:00
use InvalidCodeAnalysisTestTrait;
use ValidCodeAnalysisTestTrait;
2016-12-04 19:35:38 +01:00
2017-01-13 20:07:23 +01:00
/**
Add support for strict arrays, fix type alias intersection, fix array_is_list assertion on non-lists (#8395) * Immutable CodeLocation * Remove excess clones * Remove external clones * Remove leftover clones * Fix final clone issue * Immutable storages * Refactoring * Fixes * Fixes * Fix * Fix * Fixes * Simplify * Fixes * Fix * Fixes * Update * Fix * Cache global types * Fix * Update * Update * Fixes * Fixes * Refactor * Fixes * Fix * Fix * More caching * Fix * Fix * Update * Update * Fix * Fixes * Update * Refactor * Update * Fixes * Break one more test * Fix * FIx * Fix * Fix * Fix * Fix * Improve performance and readability * Equivalent logic * Fixes * Revert * Revert "Revert" This reverts commit f9175100c8452c80559234200663fd4c4f4dd889. * Fix * Fix reference bug * Make default TypeVisitor immutable * Bugfix * Remove clones * Partial refactoring * Refactoring * Fixes * Fix * Fixes * Fixes * cs-fix * Fix final bugs * Add test * Misc fixes * Update * Fixes * Experiment with removing different property * revert "Experiment with removing different property" This reverts commit ac1156e077fc4ea633530d51096d27b6e88bfdf9. * Uniform naming * Uniform naming * Hack hotfix * Clean up $_FILES ref #8621 * Undo hack, try fixing properly * Helper method * Remove redundant call * Partially fix bugs * Cleanup * Change defaults * Fix bug * Fix (?, hope this doesn't break anything else) * cs-fix * Review fixes * Bugfix * Bugfix * Improve logic * Add support for list{} and callable-list{} types, properly implement array_is_list assertions (fixes #8389) * Default to sealed arrays * Fix array_merge bug * Fixes * Fix * Sealed type checks * Properly infer properties-of and get_object_vars on final classes * Fix array_map zipping * Fix tests * Fixes * Fixes * Fix more stuff * Recursively resolve type aliases * Fix typo * Fixes * Fix array_is_list assertion on keyed array * Add BC docs * Fixes * fix * Update * Update * Update * Update * Seal arrays with count assertions * Fix #8528 * Fix * Update * Improve sealed array foreach logic * get_object_vars on template properties * Fix sealed array assertion reconciler logic * Improved reconciler * Add tests * Single source of truth for test types * Fix tests * Fixup tests * Fixup tests * Fixup tests * Update * Fix tests * Fix tests * Final fixes * Fixes * Use list syntax only when needed * Fix tests * Cs-fix * Update docs * Update docs * Update docs * Update docs * Update docs * Document missing types * Update docs * Improve class-string-map docs * Update * Update * I love working on psalm :) * Keep arrays unsealed by default * Fixup tests * Fix syntax mistake * cs-fix * Fix typo * Re-import missing types * Keep strict types only in return types * argc/argv fixes * argc/argv fixes * Fix test * Comment-out valinor code, pinging @romm pls merge https://github.com/CuyZ/Valinor/pull/246 so we can add valinor to the psalm docs :)
2022-11-05 22:34:42 +01:00
*
2017-01-13 20:07:23 +01:00
*/
public function providerValidCodeParse(): iterable
2016-12-04 19:35:38 +01:00
{
return [
'nullableReturnType' => [
'code' => '<?php
function a(): ?string
{
return rand(0, 10) ? "elePHPant" : null;
}
2017-06-29 16:22:49 +02:00
$a = a();',
'assertions' => [
'$a' => 'null|string',
2017-05-27 02:05:57 +02:00
],
],
'nullableReturnTypeInDocblock' => [
'code' => '<?php
/** @return ?string */
function a() {
return rand(0, 10) ? "elePHPant" : null;
}
2017-06-29 16:22:49 +02:00
$a = a();',
'assertions' => [
2017-06-29 16:22:49 +02:00
'$a' => 'null|string',
2017-05-27 02:05:57 +02:00
],
],
'nullableArgument' => [
'code' => '<?php
2018-01-11 21:50:45 +01:00
function test(?string $name): ?string
{
return $name;
}
2017-06-29 16:22:49 +02:00
test("elePHPant");
2017-05-27 02:05:57 +02:00
test(null);',
],
'protectedClassConst' => [
'code' => '<?php
class A
{
protected const IS_PROTECTED = 1;
}
2017-06-29 16:22:49 +02:00
class B extends A
{
2018-01-11 21:50:45 +01:00
function fooFoo(): int {
return A::IS_PROTECTED;
}
}',
],
'privateClassConst' => [
'code' => '<?php
class A
{
private const IS_PRIVATE = 1;
2017-06-29 16:22:49 +02:00
2018-01-11 21:50:45 +01:00
function fooFoo(): int {
return A::IS_PRIVATE;
}
2017-05-27 02:05:57 +02:00
}',
],
'publicClassConstFetch' => [
'code' => '<?php
class A
{
public const IS_PUBLIC = 1;
const IS_ALSO_PUBLIC = 2;
}
2017-06-29 16:22:49 +02:00
class B extends A
{
2018-01-11 21:50:45 +01:00
function fooFoo(): int {
echo A::IS_PUBLIC;
return A::IS_ALSO_PUBLIC;
}
}
2017-06-29 16:22:49 +02:00
echo A::IS_PUBLIC;
2017-05-27 02:05:57 +02:00
echo A::IS_ALSO_PUBLIC;',
],
'arrayDestructuringList' => [
'code' => '<?php
$data = [
[1, "Tom"],
[2, "Fred"],
];
2017-06-29 16:22:49 +02:00
// list() style
list($id1, $name1) = $data[0];
2017-06-29 16:22:49 +02:00
// [] style
[$id2, $name2] = $data[1];',
'assertions' => [
'$id1' => 'int',
'$name1' => 'string',
'$id2' => 'int',
'$name2' => 'string',
2017-05-27 02:05:57 +02:00
],
],
'arrayDestructuringInForeach' => [
'code' => '<?php
$data = [
[1, "Tom"],
[2, "Fred"],
];
2017-06-29 16:22:49 +02:00
// [] style
foreach ($data as [$id, $name]) {
echo $id;
echo $name;
2017-05-27 02:05:57 +02:00
}',
],
'arrayDestructuringWithKeys' => [
'code' => '<?php
$data = [
["id" => 1, "name" => "Tom"],
["id" => 2, "name" => "Fred"],
];
2017-06-29 16:22:49 +02:00
// list() style
list("id" => $id1, "name" => $name1) = $data[0];
2017-06-29 16:22:49 +02:00
// [] style
["id" => $id2, "name" => $name2] = $data[1];',
'assertions' => [
2017-06-29 16:22:49 +02:00
'$id1' => 'int',
'$name1' => 'string',
'$id2' => 'int',
'$name2' => 'string',
2017-05-27 02:05:57 +02:00
],
],
'arrayListDestructuringInForeachWithKeys' => [
'code' => '<?php
$data = [
["id" => 1, "name" => "Tom"],
["id" => 2, "name" => "Fred"],
];
2017-06-29 16:22:49 +02:00
// list() style
foreach ($data as list("id" => $id, "name" => $name)) {
$last_id = $id;
$last_name = $name;
}',
'assertions' => [
'$last_id' => 'int',
'$last_name' => 'string',
2017-05-27 02:05:57 +02:00
],
],
'arrayDestructuringInForeachWithKeys' => [
'code' => '<?php
$data = [
["id" => 1, "name" => "Tom"],
["id" => 2, "name" => "Fred"],
];
2017-06-29 16:22:49 +02:00
// [] style
foreach ($data as ["id" => $id, "name" => $name]) {
$last_id = $id;
$last_name = $name;
}',
'assertions' => [
'$last_id' => 'int',
'$last_name' => 'string',
2017-05-27 02:05:57 +02:00
],
],
'iterableArg' => [
'code' => '<?php
/**
* @param iterable<int, int> $iter
*/
2018-01-11 21:50:45 +01:00
function iterator(iterable $iter): void
{
foreach ($iter as $val) {
//
}
}
2017-06-29 16:22:49 +02:00
iterator([1, 2, 3, 4]);
2020-04-19 15:28:53 +02:00
/** @psalm-suppress MixedArgumentTypeCoercion */
2017-05-27 02:05:57 +02:00
iterator(new SplFixedArray(5));',
],
'traversableObject' => [
'code' => '<?php
/**
* @implements Iterator<0, mixed>
*/
class IteratorObj implements Iterator {
2018-01-11 21:50:45 +01:00
function rewind(): void {}
/** @return mixed */
function current() { return null; }
2018-01-11 21:50:45 +01:00
function key(): int { return 0; }
function next(): void {}
function valid(): bool { return false; }
}
2017-06-29 16:22:49 +02:00
2018-01-11 21:50:45 +01:00
function foo(\Traversable $t): void {
}
2017-06-29 16:22:49 +02:00
2017-05-27 02:05:57 +02:00
foo(new IteratorObj);',
],
'iterableIsArrayOrTraversable' => [
'code' => '<?php
function castToArray(iterable $arr): array {
if ($arr instanceof \Traversable) {
return iterator_to_array($arr, false);
}
return $arr;
}
function castToArray2(iterable $arr): array {
if (is_array($arr)) {
return $arr;
}
return iterator_to_array($arr, false);
}',
],
'substituteIterable' => [
'code' => '<?php
function foo(iterable $i): array {
if (!is_array($i)) {
$i = iterator_to_array($i, false);
}
return $i;
}',
],
'iterator_to_arrayMixedKey' => [
'code' => '<?php
/**
* @template TKey
* @template TValue
* @param Traversable<TKey, TValue> $traversable
* @return array<TValue>
*/
function toArray(Traversable $traversable): array
{
return iterator_to_array($traversable);
}',
],
'noReservedWordInDocblock' => [
'code' => '<?php
/**
* @param Closure():(resource|false) $op
* @return resource|false
*/
function create_resource($op) {
return $op();
2019-03-23 19:27:54 +01:00
}',
],
'arrayDestructuringOnArrayObject' => [
'code' => '<?php
$var = new ArrayObject([0 => "first", "dos" => "second"]);
[0 => $first, "dos" => $second] = $var;
echo $first;
echo $second;',
],
];
}
2017-01-13 20:07:23 +01:00
/**
Add support for strict arrays, fix type alias intersection, fix array_is_list assertion on non-lists (#8395) * Immutable CodeLocation * Remove excess clones * Remove external clones * Remove leftover clones * Fix final clone issue * Immutable storages * Refactoring * Fixes * Fixes * Fix * Fix * Fixes * Simplify * Fixes * Fix * Fixes * Update * Fix * Cache global types * Fix * Update * Update * Fixes * Fixes * Refactor * Fixes * Fix * Fix * More caching * Fix * Fix * Update * Update * Fix * Fixes * Update * Refactor * Update * Fixes * Break one more test * Fix * FIx * Fix * Fix * Fix * Fix * Improve performance and readability * Equivalent logic * Fixes * Revert * Revert "Revert" This reverts commit f9175100c8452c80559234200663fd4c4f4dd889. * Fix * Fix reference bug * Make default TypeVisitor immutable * Bugfix * Remove clones * Partial refactoring * Refactoring * Fixes * Fix * Fixes * Fixes * cs-fix * Fix final bugs * Add test * Misc fixes * Update * Fixes * Experiment with removing different property * revert "Experiment with removing different property" This reverts commit ac1156e077fc4ea633530d51096d27b6e88bfdf9. * Uniform naming * Uniform naming * Hack hotfix * Clean up $_FILES ref #8621 * Undo hack, try fixing properly * Helper method * Remove redundant call * Partially fix bugs * Cleanup * Change defaults * Fix bug * Fix (?, hope this doesn't break anything else) * cs-fix * Review fixes * Bugfix * Bugfix * Improve logic * Add support for list{} and callable-list{} types, properly implement array_is_list assertions (fixes #8389) * Default to sealed arrays * Fix array_merge bug * Fixes * Fix * Sealed type checks * Properly infer properties-of and get_object_vars on final classes * Fix array_map zipping * Fix tests * Fixes * Fixes * Fix more stuff * Recursively resolve type aliases * Fix typo * Fixes * Fix array_is_list assertion on keyed array * Add BC docs * Fixes * fix * Update * Update * Update * Update * Seal arrays with count assertions * Fix #8528 * Fix * Update * Improve sealed array foreach logic * get_object_vars on template properties * Fix sealed array assertion reconciler logic * Improved reconciler * Add tests * Single source of truth for test types * Fix tests * Fixup tests * Fixup tests * Fixup tests * Update * Fix tests * Fix tests * Final fixes * Fixes * Use list syntax only when needed * Fix tests * Cs-fix * Update docs * Update docs * Update docs * Update docs * Update docs * Document missing types * Update docs * Improve class-string-map docs * Update * Update * I love working on psalm :) * Keep arrays unsealed by default * Fixup tests * Fix syntax mistake * cs-fix * Fix typo * Re-import missing types * Keep strict types only in return types * argc/argv fixes * argc/argv fixes * Fix test * Comment-out valinor code, pinging @romm pls merge https://github.com/CuyZ/Valinor/pull/246 so we can add valinor to the psalm docs :)
2022-11-05 22:34:42 +01:00
*
2017-01-13 20:07:23 +01:00
*/
public function providerInvalidCodeParse(): iterable
{
return [
'invalidPrivateClassConstFetch' => [
'code' => '<?php
class A
{
private const IS_PRIVATE = 1;
}
2017-06-29 16:22:49 +02:00
echo A::IS_PRIVATE;',
2017-05-27 02:05:57 +02:00
'error_message' => 'InaccessibleClassConstant',
],
'invalidPrivateClassConstFetchFromSubclass' => [
'code' => '<?php
class A
{
private const IS_PRIVATE = 1;
}
2017-06-29 16:22:49 +02:00
class B extends A
{
2018-01-11 21:50:45 +01:00
function fooFoo(): int {
return A::IS_PRIVATE;
}
}',
2017-05-27 02:05:57 +02:00
'error_message' => 'InaccessibleClassConstant',
],
'invalidProtectedClassConstFetch' => [
'code' => '<?php
class A
{
protected const IS_PROTECTED = 1;
}
2017-06-29 16:22:49 +02:00
echo A::IS_PROTECTED;',
2017-05-27 02:05:57 +02:00
'error_message' => 'InaccessibleClassConstant',
],
'invalidIterableArg' => [
'code' => '<?php
/**
* @param iterable<string> $iter
*/
2018-01-11 21:50:45 +01:00
function iterator(iterable $iter): void
{
foreach ($iter as $val) {
//
}
}
2017-06-29 16:22:49 +02:00
class A {
}
2017-06-29 16:22:49 +02:00
iterator(new A());',
2017-05-27 02:05:57 +02:00
'error_message' => 'InvalidArgument',
],
'voidDoesntWorkIn70' => [
'code' => '<?php
function foo(): void {
}',
'error_message' => 'ReservedWord',
'ignored_issues' => [],
'php_version' => '7.0',
],
'objectDoesntWorkIn71' => [
'code' => '<?php
function foo(): object {
return new stdClass();
}',
'error_message' => 'ReservedWord',
'ignored_issues' => [],
'php_version' => '7.0',
],
'arrayDestructuringInvalidList' => [
'code' => '<?php
$a = 42;
list($id1, $name1) = $a;',
'error_message' => 'InvalidArrayOffset',
],
'arrayDestructuringInvalidArray' => [
'code' => '<?php
$a = 42;
[$id2, $name2] = $a;',
'error_message' => 'InvalidArrayOffset',
],
2016-12-04 19:35:38 +01:00
];
}
2016-12-04 05:03:51 +01:00
}