1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-22 22:01:48 +01:00
psalm/tests/ConstValuesTest.php
Daniil Gentili 1986c8b4a8
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

252 lines
8.0 KiB
PHP

<?php
namespace Psalm\Tests;
use Psalm\Tests\Traits\InvalidCodeAnalysisTestTrait;
use Psalm\Tests\Traits\ValidCodeAnalysisTestTrait;
class ConstValuesTest extends TestCase
{
use InvalidCodeAnalysisTestTrait;
use ValidCodeAnalysisTestTrait;
/**
*
*/
public function providerValidCodeParse(): iterable
{
return [
'enumStringOrEnumIntCorrect' => [
'code' => '<?php
namespace Ns;
/** @psalm-param ( "foo\"with" | "bar" | 1 | 2 | 3 ) $s */
function foo($s) : void {}
foo("foo\"with");
foo("bar");
foo(1);
foo(2);
foo(3);',
],
'enumStringOrEnumIntWithoutSpacesCorrect' => [
'code' => '<?php
namespace Ns;
/** @psalm-param "foo\"with"|"bar"|1|2|3|4.0|4.1 $s */
function foo($s) : void {}
foo("foo\"with");
foo("bar");
foo(1);
foo(2);
foo(3);
foo(4.0);
foo(4.1);',
],
'noRedundantConditionWithSwitch' => [
'code' => '<?php
namespace Ns;
/**
* @psalm-param ( "foo" | "bar") $s
*/
function foo(string $s) : void {
switch ($s) {
case "foo":
break;
case "bar":
break;
}
}',
],
'classConstantCorrect' => [
'code' => '<?php
namespace Ns;
class C {
const A1 = "bat";
const B = "baz";
}
/** @psalm-param "foo"|"bar"|C::A1|C::B $s */
function foo($s) : void {}
foo("foo");
foo("bar");
foo("bat");
foo("baz");',
],
'selfClassConstGoodValue' => [
'code' => '<?php
class A {
const FOO = "foo";
const BAR = "bar";
/**
* @param (self::FOO | self::BAR) $s
*/
public static function foo(string $s) : void {}
}
A::foo("foo");',
],
'classConstants' => [
'code' => '<?php
namespace NS {
use OtherNS\C as E;
class C {}
class D {};
/** @psalm-param C::class|D::class|E::class $s */
function foo(string $s) : void {}
foo(C::class);
foo(D::class);
foo(E::class);
foo(\OtherNS\C::class);
}
namespace OtherNS {
class C {}
}',
],
];
}
/**
*
*/
public function providerInvalidCodeParse(): iterable
{
return [
'enumStringOrEnumIntIncorrectString' => [
'code' => '<?php
namespace Ns;
/** @psalm-param ( "foo" | "bar" | 1 | 2 | 3 ) $s */
function foo($s) : void {}
foo("bat");',
'error_message' => 'InvalidArgument',
],
'enumStringOrEnumIntIncorrectInt' => [
'code' => '<?php
namespace Ns;
/** @psalm-param ( "foo" | "bar" | 1 | 2 | 3 ) $s */
function foo($s) : void {}
foo(4);',
'error_message' => 'InvalidArgument',
],
'enumStringOrEnumIntWithoutSpacesIncorrect' => [
'code' => '<?php
namespace Ns;
/** @psalm-param "foo\"with"|"bar"|1|2|3 $s */
function foo($s) : void {}
foo(4);',
'error_message' => 'InvalidArgument',
],
'enumWrongFloat' => [
'code' => '<?php
namespace Ns;
/** @psalm-param 1.2|3.4|5.6 $s */
function foo($s) : void {}
foo(7.8);',
'error_message' => 'InvalidArgument',
],
'classConstantIncorrect' => [
'code' => '<?php
namespace Ns;
class C {
const A = "bat";
const B = "baz";
}
/** @psalm-param "foo"|"bar"|C::A|C::B $s */
function foo($s) : void {}
foo("for");',
'error_message' => 'InvalidArgument',
],
'classConstantNoClass' => [
'code' => '<?php
namespace Ns;
/** @psalm-param "foo"|"bar"|C::A|C::B $s */
function foo($s) : void {}',
'error_message' => 'UndefinedDocblockClass',
],
'selfClassConstBadValue' => [
'code' => '<?php
class A {
const FOO = "foo";
const BAR = "bar";
/**
* @param (self::FOO | self::BAR) $s
*/
public static function foo(string $s) : void {}
}
A::foo("for");',
'error_message' => 'InvalidArgument',
],
'selfClassConstBadConst' => [
'code' => '<?php
class A {
const FOO = "foo";
const BAR = "bar";
/**
* @param (self::1FOO | self::BAR) $s
*/
public static function foo(string $s) : void {}
}',
'error_message' => 'InvalidDocblock',
],
'classConstantInvalidValue' => [
'code' => '<?php
namespace NS {
use OtherNS\C as E;
class C {}
class D {};
class F {};
/** @psalm-param C::class|D::class|E::class $s */
function foo(string $s) : void {}
foo(F::class);
}
namespace OtherNS {
class C {}
}',
'error_message' => 'InvalidArgument',
],
'nonExistentConstantClass' => [
'code' => '<?php
/**
* @return Foo::HELLO|5
*/
function getVal()
{
return 5;
}',
'error_message' => 'UndefinedDocblockClass',
],
'nonExistentClassConstant' => [
'code' => '<?php
class Foo {}
/**
* @return Foo::HELLO|5
*/
function getVal()
{
return 5;
}',
'error_message' => 'UndefinedConstant',
],
'noIntToFloatEnum' => [
'code' => '<?php
/** @param 0.3|0.5 $p */
function f($p): void {}
f(1);',
'error_message' => 'InvalidArgument',
],
];
}
}