2016-09-09 16:38:32 -04:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
2019-06-26 22:52:29 +02:00
|
|
|
use function function_exists;
|
|
|
|
use function print_r;
|
2019-07-05 16:24:00 -04:00
|
|
|
use Psalm\Type;
|
|
|
|
use function stripos;
|
2016-09-09 16:38:32 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class TypeParseTest extends TestCase
|
2016-09-09 16:38:32 -04:00
|
|
|
{
|
2017-12-18 22:07:52 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
2019-05-16 18:36:36 -04:00
|
|
|
public function setUp() : void
|
2017-12-18 22:07:52 -05:00
|
|
|
{
|
2019-02-11 08:41:48 -05:00
|
|
|
$this->file_provider = new \Psalm\Tests\Internal\Provider\FakeFileProvider();
|
|
|
|
|
|
|
|
$config = new TestConfig();
|
|
|
|
|
|
|
|
$providers = new \Psalm\Internal\Provider\Providers(
|
|
|
|
$this->file_provider,
|
|
|
|
new \Psalm\Tests\Internal\Provider\FakeParserCacheProvider()
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->project_analyzer = new \Psalm\Internal\Analyzer\ProjectAnalyzer(
|
|
|
|
$config,
|
2019-06-09 12:37:28 -04:00
|
|
|
$providers
|
2019-02-11 08:41:48 -05:00
|
|
|
);
|
2017-12-18 22:07:52 -05:00
|
|
|
}
|
|
|
|
|
2018-03-22 17:55:36 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testThisToStatic()
|
|
|
|
{
|
|
|
|
$this->assertSame('static', (string) Type::parseString('$this'));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testThisToStaticUnion()
|
|
|
|
{
|
2019-10-16 22:14:33 -07:00
|
|
|
$this->assertSame('A|static', (string) Type::parseString('$this|A'));
|
2018-03-22 17:55:36 -04:00
|
|
|
}
|
|
|
|
|
2017-01-13 14:07:23 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
2016-09-09 16:38:32 -04:00
|
|
|
public function testIntOrString()
|
|
|
|
{
|
2017-05-26 20:05:57 -04:00
|
|
|
$this->assertSame('int|string', (string) Type::parseString('int|string'));
|
2016-09-22 00:31:07 -04:00
|
|
|
}
|
2016-09-09 16:38:32 -04:00
|
|
|
|
2018-03-22 22:28:06 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testBracketedIntOrString()
|
|
|
|
{
|
|
|
|
$this->assertSame('int|string', (string) Type::parseString('(int|string)'));
|
|
|
|
}
|
|
|
|
|
2018-03-20 20:19:26 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testBoolOrIntOrString()
|
|
|
|
{
|
|
|
|
$this->assertSame('bool|int|string', (string) Type::parseString('bool|int|string'));
|
|
|
|
}
|
|
|
|
|
2017-11-20 00:32:40 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testNullable()
|
|
|
|
{
|
|
|
|
$this->assertSame('null|string', (string) Type::parseString('?string'));
|
|
|
|
}
|
|
|
|
|
2018-08-29 16:03:16 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testNullableUnion()
|
|
|
|
{
|
2019-10-16 22:14:33 -07:00
|
|
|
$this->assertSame('int|null|string', (string) Type::parseString('?(string|int)'));
|
2018-08-29 16:03:16 -04:00
|
|
|
}
|
|
|
|
|
2018-05-21 12:55:44 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testNullableFullyQualified()
|
|
|
|
{
|
|
|
|
$this->assertSame('null|stdClass', (string) Type::parseString('?\\stdClass'));
|
|
|
|
}
|
|
|
|
|
2018-06-08 23:54:07 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testNullableOrNullable()
|
|
|
|
{
|
2019-10-16 22:14:33 -07:00
|
|
|
$this->assertSame('int|null|string', (string) Type::parseString('?string|?int'));
|
2018-06-08 23:54:07 -04:00
|
|
|
}
|
|
|
|
|
2017-01-13 14:07:23 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
2019-06-15 17:57:40 -04:00
|
|
|
public function testArrayWithClosingBracket()
|
2016-09-22 00:31:07 -04:00
|
|
|
{
|
2017-05-26 20:05:57 -04:00
|
|
|
$this->assertSame('array<int, int>', (string) Type::parseString('array<int, int>'));
|
2019-06-15 17:57:40 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testArrayWithoutClosingBracket()
|
|
|
|
{
|
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
|
|
|
Type::parseString('array<int, int');
|
2018-03-20 20:19:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testArrayWithSingleArg()
|
|
|
|
{
|
2019-01-05 00:15:53 -05:00
|
|
|
$this->assertSame('array<array-key, int>', (string) Type::parseString('array<int>'));
|
2018-03-20 20:19:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testArrayWithNestedSingleArg()
|
|
|
|
{
|
2019-01-05 00:15:53 -05:00
|
|
|
$this->assertSame('array<array-key, array<array-key, int>>', (string) Type::parseString('array<array<int>>'));
|
2018-03-20 20:19:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testArrayWithUnion()
|
|
|
|
{
|
2017-05-26 20:05:57 -04:00
|
|
|
$this->assertSame('array<int|string, string>', (string) Type::parseString('array<int|string, string>'));
|
2016-09-22 00:31:07 -04:00
|
|
|
}
|
2016-09-09 16:38:32 -04:00
|
|
|
|
2019-02-27 09:08:27 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testNonEmptyArrray()
|
|
|
|
{
|
|
|
|
$this->assertSame('non-empty-array<array-key, int>', (string) Type::parseString('non-empty-array<int>'));
|
|
|
|
}
|
|
|
|
|
2017-01-13 14:07:23 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
2016-09-22 00:31:07 -04:00
|
|
|
public function testGeneric()
|
|
|
|
{
|
2017-05-26 20:05:57 -04:00
|
|
|
$this->assertSame('B<int>', (string) Type::parseString('B<int>'));
|
2016-09-09 16:38:32 -04:00
|
|
|
}
|
|
|
|
|
2018-03-18 13:57:04 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testIntersection()
|
|
|
|
{
|
2018-09-07 11:09:07 -04:00
|
|
|
$this->assertSame('I1&I2&I3', (string) Type::parseString('I1&I2&I3'));
|
2018-03-18 13:57:04 -04:00
|
|
|
}
|
|
|
|
|
2018-03-22 17:55:36 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testIntersectionOrNull()
|
|
|
|
{
|
2019-10-16 22:14:33 -07:00
|
|
|
$this->assertSame('I1&I2|null', (string) Type::parseString('I1&I2|null'));
|
2018-03-22 17:55:36 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testNullOrIntersection()
|
|
|
|
{
|
2019-10-16 22:14:33 -07:00
|
|
|
$this->assertSame('I1&I2|null', (string) Type::parseString('null|I1&I2'));
|
2018-03-22 17:55:36 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testInteratorAndTraversable()
|
|
|
|
{
|
2019-10-30 19:00:27 +01:00
|
|
|
$this->assertSame('Iterator<mixed, int>&Traversable', (string) Type::parseString('Iterator<int>&Traversable'));
|
2018-03-22 17:55:36 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testTraversableAndIteratorOrNull()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
2019-10-30 19:00:27 +01:00
|
|
|
'Traversable&Iterator<mixed, int>|null',
|
2018-03-22 17:55:36 -04:00
|
|
|
(string) Type::parseString('Traversable&Iterator<int>|null')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-09-07 11:09:07 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testIntersectionAfterGeneric()
|
|
|
|
{
|
2018-12-05 10:36:42 -05:00
|
|
|
$this->assertSame('Countable&iterable<mixed, int>&I', (string) Type::parseString('Countable&iterable<int>&I'));
|
2018-09-07 11:09:07 -04:00
|
|
|
}
|
|
|
|
|
2019-05-29 19:58:54 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testIntersectionOfIterables()
|
|
|
|
{
|
|
|
|
$this->assertSame('iterable<mixed, A>&iterable<mixed, B>', (string) Type::parseString('iterable<A>&iterable<B>'));
|
|
|
|
}
|
|
|
|
|
2019-06-29 21:56:34 -04:00
|
|
|
public function testIterableContainingObjectLike() : void
|
2019-06-29 21:46:01 -04:00
|
|
|
{
|
|
|
|
$this->assertSame('iterable<string, array{0: int}>', Type::parseString('iterable<string, array{int}>')->getId());
|
|
|
|
}
|
|
|
|
|
2017-01-13 14:07:23 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
2018-03-20 20:19:26 -04:00
|
|
|
public function testPhpDocSimpleArray()
|
2016-10-29 20:57:03 -04:00
|
|
|
{
|
2019-01-05 00:15:53 -05:00
|
|
|
$this->assertSame('array<array-key, A>', (string) Type::parseString('A[]'));
|
2018-03-20 20:19:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testPhpDocUnionArray()
|
|
|
|
{
|
2019-01-05 00:15:53 -05:00
|
|
|
$this->assertSame('array<array-key, A|B>', (string) Type::parseString('(A|B)[]'));
|
2018-03-20 20:19:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testPhpDocMultiDimensionalArray()
|
|
|
|
{
|
2019-01-05 00:15:53 -05:00
|
|
|
$this->assertSame('array<array-key, array<array-key, A>>', (string) Type::parseString('A[][]'));
|
2018-03-20 20:19:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testPhpDocMultidimensionalUnionArray()
|
|
|
|
{
|
2019-01-05 00:15:53 -05:00
|
|
|
$this->assertSame('array<array-key, array<array-key, A|B>>', (string) Type::parseString('(A|B)[][]'));
|
2018-03-20 20:19:26 -04:00
|
|
|
}
|
|
|
|
|
2018-03-22 21:21:04 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testPhpDocObjectLikeArray()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
2019-06-16 09:42:34 -04:00
|
|
|
'array<array-key, array{b: bool, d: string}>',
|
|
|
|
(string) Type::parseString('array{b: bool, d: string}[]')
|
2018-03-22 21:21:04 -04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-03-20 20:19:26 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testPhpDocUnionOfArrays()
|
|
|
|
{
|
2019-01-05 00:15:53 -05:00
|
|
|
$this->assertSame('array<array-key, A|B>', (string) Type::parseString('A[]|B[]'));
|
2018-03-20 20:19:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testPhpDocUnionOfArraysOrObject()
|
|
|
|
{
|
2019-10-16 22:14:33 -07:00
|
|
|
$this->assertSame('C|array<array-key, A|B>', (string) Type::parseString('A[]|B[]|C'));
|
2016-10-29 20:57:03 -04:00
|
|
|
}
|
|
|
|
|
2018-03-05 16:06:06 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testPsalmOnlyAtomic()
|
|
|
|
{
|
|
|
|
$this->assertSame('class-string', (string) Type::parseString('class-string'));
|
|
|
|
}
|
|
|
|
|
2019-01-02 09:00:45 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testParamaterizedClassString()
|
|
|
|
{
|
|
|
|
$this->assertSame('class-string<A>', (string) Type::parseString('class-string<A>'));
|
|
|
|
}
|
|
|
|
|
2019-06-27 11:44:35 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testParamaterizedClassStringUnion()
|
|
|
|
{
|
|
|
|
$this->assertSame('class-string<A>|class-string<B>', (string) Type::parseString('class-string<A>|class-string<B>'));
|
|
|
|
}
|
|
|
|
|
2017-10-12 14:02:06 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testInvalidType()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2017-10-12 14:02:06 -04:00
|
|
|
Type::parseString('array(A)');
|
|
|
|
}
|
|
|
|
|
2018-03-22 22:28:06 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testBracketedUnionAndIntersection()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-03-22 22:28:06 -04:00
|
|
|
Type::parseString('(A|B)&C');
|
|
|
|
}
|
|
|
|
|
2018-06-30 14:25:32 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
2018-06-30 13:09:05 -04:00
|
|
|
public function testBracketInUnion()
|
|
|
|
{
|
|
|
|
Type::parseString('null|(scalar|array|object)');
|
|
|
|
}
|
|
|
|
|
2017-01-13 14:07:23 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
2018-03-20 20:19:26 -04:00
|
|
|
public function testObjectLikeWithSimpleArgs()
|
2016-09-22 00:31:07 -04:00
|
|
|
{
|
2019-06-16 09:42:34 -04:00
|
|
|
$this->assertSame('array{a: int, b: string}', (string) Type:: parseString('array{a: int, b: string}'));
|
2018-03-20 20:19:26 -04:00
|
|
|
}
|
|
|
|
|
2019-06-15 17:57:40 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testObjectLikeWithoutClosingBracket()
|
|
|
|
{
|
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2019-06-16 09:42:34 -04:00
|
|
|
Type::parseString('array{a: int, b: string');
|
2019-06-15 17:57:40 -04:00
|
|
|
}
|
|
|
|
|
2019-01-18 00:56:24 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testObjectWithSimpleArgs()
|
|
|
|
{
|
|
|
|
$this->assertSame('object{a:int, b:string}', (string) Type::parseString('object{a:int, b:string}'));
|
|
|
|
}
|
|
|
|
|
2019-08-08 18:01:08 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testObjectWithDollarArgs()
|
|
|
|
{
|
|
|
|
$this->assertSame('object{a:int, $b:string}', (string) Type::parseString('object{a:int, $b:string}'));
|
|
|
|
}
|
|
|
|
|
2018-03-20 20:19:26 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testObjectLikeWithUnionArgs()
|
|
|
|
{
|
2017-05-26 20:05:57 -04:00
|
|
|
$this->assertSame(
|
2019-06-16 09:42:34 -04:00
|
|
|
'array{a: int|string, b: string}',
|
|
|
|
(string) Type::parseString('array{a: int|string, b: string}')
|
2016-11-02 02:29:00 -04:00
|
|
|
);
|
2018-03-20 20:19:26 -04:00
|
|
|
}
|
2016-11-02 02:29:00 -04:00
|
|
|
|
2018-03-20 20:19:26 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testObjectLikeWithGenericArgs()
|
|
|
|
{
|
2017-05-26 20:05:57 -04:00
|
|
|
$this->assertSame(
|
2019-10-16 22:14:33 -07:00
|
|
|
'array{a: array<int, int|string>, b: string}',
|
2019-06-16 09:42:34 -04:00
|
|
|
(string) Type::parseString('array{a: array<int, string|int>, b: string}')
|
2016-11-02 02:29:00 -04:00
|
|
|
);
|
2018-03-20 20:19:26 -04:00
|
|
|
}
|
2017-12-18 22:07:52 -05:00
|
|
|
|
2018-03-20 20:19:26 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testObjectLikeWithIntKeysAndUnionArgs()
|
|
|
|
{
|
2017-12-18 22:07:52 -05:00
|
|
|
$this->assertSame(
|
2019-06-16 09:42:34 -04:00
|
|
|
'array{0: null|stdClass}',
|
2017-12-18 22:07:52 -05:00
|
|
|
(string)Type::parseString('array{stdClass|null}')
|
|
|
|
);
|
2018-03-20 20:19:26 -04:00
|
|
|
}
|
2017-12-18 22:07:52 -05:00
|
|
|
|
2018-03-20 20:19:26 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testObjectLikeWithIntKeysAndGenericArgs()
|
|
|
|
{
|
2017-12-18 22:07:52 -05:00
|
|
|
$this->assertSame(
|
2019-06-16 09:42:34 -04:00
|
|
|
'array{0: array<array-key, mixed>}',
|
2017-12-18 22:07:52 -05:00
|
|
|
(string)Type::parseString('array{array}')
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertSame(
|
2019-06-16 09:42:34 -04:00
|
|
|
'array{0: array<int, string>}',
|
2017-12-18 22:07:52 -05:00
|
|
|
(string)Type::parseString('array{array<int, string>}')
|
|
|
|
);
|
2018-03-16 23:37:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testObjectLikeOptional()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
2019-06-16 09:42:34 -04:00
|
|
|
'array{a: int, b?: int}',
|
|
|
|
(string)Type::parseString('array{a: int, b?: int}')
|
2018-03-16 23:37:10 -04:00
|
|
|
);
|
2016-09-22 00:31:07 -04:00
|
|
|
}
|
2018-03-26 22:13:10 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
2019-12-19 00:28:32 +00:00
|
|
|
public function testSimpleCallable()
|
2018-03-26 22:13:10 -04:00
|
|
|
{
|
|
|
|
$this->assertSame(
|
2018-04-08 12:57:56 -04:00
|
|
|
'callable(int, string):void',
|
2018-03-26 22:13:10 -04:00
|
|
|
(string)Type::parseString('callable(int, string) : void')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-06-15 17:57:40 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCallableWithoutClosingBracket()
|
|
|
|
{
|
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
|
|
|
Type::parseString('callable(int, string');
|
|
|
|
}
|
|
|
|
|
2019-02-21 16:00:18 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCallableWithParamNames()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
|
|
|
'callable(int, string):void',
|
|
|
|
(string)Type::parseString('callable(int $foo, string $bar) : void')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-12-08 14:10:06 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCallableReturningIntersection()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
|
|
|
'callable(int, string):I1&I2',
|
|
|
|
(string)Type::parseString('callable(int, string) : (I1&I2)')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-03-28 10:53:19 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testEmptyCallable()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
2018-04-08 12:57:56 -04:00
|
|
|
'callable():void',
|
2018-03-28 10:53:19 -04:00
|
|
|
(string)Type::parseString('callable() : void')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-03-26 22:13:10 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCallableWithUnionLastType()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
2018-04-08 12:57:56 -04:00
|
|
|
'callable(int, int|string):void',
|
2018-03-26 22:13:10 -04:00
|
|
|
(string)Type::parseString('callable(int, int|string) : void')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCallableWithVariadic()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
2018-04-08 12:57:56 -04:00
|
|
|
'callable(int, string...):void',
|
2018-03-27 00:12:41 -04:00
|
|
|
(string)Type::parseString('callable(int, string...) : void')
|
2018-03-26 22:13:10 -04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-04-18 13:23:48 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCallableThatReturnsACallable()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
|
|
|
'callable():callable():string',
|
|
|
|
(string)Type::parseString('callable() : callable() : string')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-11-06 08:53:14 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCallableThatReturnsACallableThatReturnsACallable()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
|
|
|
'callable():callable():callable():string',
|
|
|
|
(string)Type::parseString('callable() : callable() : callable() : string')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-04-18 19:04:06 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
2018-04-18 19:00:08 -04:00
|
|
|
public function testCallableOrInt()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
|
|
|
'callable(string):void|int',
|
|
|
|
(string)Type::parseString('callable(string):void|int')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-08-08 18:01:08 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCallableWithGoodVariadic()
|
|
|
|
{
|
|
|
|
Type::parseString('callable(int, string...) : void');
|
|
|
|
Type::parseString('callable(int,string...) : void');
|
|
|
|
}
|
|
|
|
|
2018-03-26 22:13:10 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
2019-12-19 00:28:32 +00:00
|
|
|
public function testCallableWithSpreadBefore()
|
2018-03-26 22:13:10 -04:00
|
|
|
{
|
2019-12-19 00:28:32 +00:00
|
|
|
$this->assertSame(
|
|
|
|
'callable(int, string...):void',
|
|
|
|
(string)Type::parseString('callable(int, ...string):void')
|
|
|
|
);
|
2018-03-27 00:12:41 -04:00
|
|
|
}
|
|
|
|
|
2018-04-15 18:16:31 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCallableWithTrailingColon()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-04-15 18:16:31 -04:00
|
|
|
Type::parseString('callable(int):');
|
|
|
|
}
|
|
|
|
|
2018-03-27 00:12:41 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCallableWithAnotherBadVariadic()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-03-27 00:12:41 -04:00
|
|
|
Type::parseString('callable(int, string..) : void');
|
2018-03-26 22:13:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCallableWithVariadicAndDefault()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-03-27 00:12:41 -04:00
|
|
|
Type::parseString('callable(int, string...=) : void');
|
2018-03-26 22:13:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testBadVariadic()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-03-27 00:12:41 -04:00
|
|
|
Type::parseString('string...');
|
2018-03-26 22:13:10 -04:00
|
|
|
}
|
|
|
|
|
2018-03-27 14:43:39 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testBadFullStop()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-03-27 14:43:39 -04:00
|
|
|
Type::parseString('string.');
|
|
|
|
}
|
|
|
|
|
2018-04-05 14:11:57 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testBadSemicolon()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-04-05 14:11:57 -04:00
|
|
|
Type::parseString('string;');
|
|
|
|
}
|
|
|
|
|
2019-01-29 10:34:31 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testBadGenericString()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2019-01-29 10:34:31 -05:00
|
|
|
Type::parseString('string<T>');
|
|
|
|
}
|
|
|
|
|
2018-06-11 17:23:28 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testBadAmpersand()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-06-11 17:23:28 -04:00
|
|
|
Type::parseString('&array');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testBadColon()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-06-11 17:23:28 -04:00
|
|
|
Type::parseString(':array');
|
|
|
|
}
|
|
|
|
|
2018-09-05 22:36:32 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testBadBrackets()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-09-05 22:36:32 -04:00
|
|
|
Type::parseString('max(a)');
|
|
|
|
}
|
|
|
|
|
2018-09-05 22:40:52 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testMoreBadBrackets()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-09-05 22:40:52 -04:00
|
|
|
Type::parseString('max(a):void');
|
|
|
|
}
|
|
|
|
|
2018-09-06 00:41:07 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testGeneratorWithWBadBrackets()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-09-06 00:41:07 -04:00
|
|
|
Type::parseString('Generator{string, A}');
|
|
|
|
}
|
|
|
|
|
2018-06-11 17:23:28 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testBadEquals()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-06-11 17:23:28 -04:00
|
|
|
Type::parseString('=array');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testBadBar()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-06-11 17:23:28 -04:00
|
|
|
Type::parseString('|array');
|
|
|
|
}
|
|
|
|
|
2018-06-12 09:11:13 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testBadColonDash()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-06-12 09:11:13 -04:00
|
|
|
Type::parseString('array|string:-');
|
|
|
|
}
|
|
|
|
|
2018-03-29 02:20:19 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testDoubleBar()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
$this->expectException(\Psalm\Exception\TypeParseTreeException::class);
|
2018-03-29 02:20:19 -04:00
|
|
|
Type::parseString('PDO||Closure|numeric');
|
|
|
|
}
|
|
|
|
|
2018-03-26 22:13:10 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCallableWithDefault()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
2018-04-08 12:57:56 -04:00
|
|
|
'callable(int, string=):void',
|
2018-03-26 22:13:10 -04:00
|
|
|
(string)Type::parseString('callable(int, string=) : void')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-11-06 08:53:14 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testNestedCallable()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
|
|
|
'callable(callable(A):B):C',
|
|
|
|
(string)Type::parseString('callable(callable(A):B):C')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-03-26 22:13:10 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCallableWithoutReturn()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
|
|
|
'callable(int, string)',
|
|
|
|
(string)Type::parseString('callable(int, string)')
|
|
|
|
);
|
|
|
|
}
|
2018-04-18 19:00:08 -04:00
|
|
|
|
2019-01-29 10:34:31 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCombineLiteralStringWithClassString()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
2019-11-30 12:57:18 -05:00
|
|
|
'class-string|string(array)',
|
|
|
|
Type::parseString('"array"|class-string')->getId()
|
2019-01-29 10:34:31 -05:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCombineLiteralClassStringWithClassString()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
2019-12-01 23:09:34 -05:00
|
|
|
'class-string',
|
2019-11-30 12:57:18 -05:00
|
|
|
Type::parseString('A::class|class-string')->getId()
|
2019-01-29 10:34:31 -05:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-05-23 22:27:36 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testKeyOfClassConstant()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
|
|
|
'key-of<Foo\Baz::BAR>',
|
|
|
|
(string)Type::parseString('key-of<Foo\Baz::BAR>')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testKeyOfTemplate()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
|
|
|
'key-of<T>',
|
|
|
|
(string)Type::parseString('key-of<T>', null, ['T' => ['' => [Type::getArray()]]])
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testIndexedAccess()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
|
|
|
'T[K]',
|
2020-02-06 16:38:45 -05:00
|
|
|
(string) Type::parseString(
|
2019-05-23 22:27:36 -04:00
|
|
|
'T[K]',
|
|
|
|
null,
|
|
|
|
[
|
|
|
|
'T' => ['' => [Type::getArray()]],
|
2020-02-06 16:38:45 -05:00
|
|
|
'K' => ['' => [new Type\Union([
|
|
|
|
new Type\Atomic\TTemplateKeyOf('T', 'fn-foo', Type::getMixed())
|
|
|
|
])]],
|
2019-05-23 22:27:36 -04:00
|
|
|
]
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-05-28 10:44:04 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testValueOfClassConstant()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
|
|
|
'value-of<Foo\Baz::BAR>',
|
|
|
|
(string)Type::parseString('value-of<Foo\Baz::BAR>')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-12-27 12:49:28 -05:00
|
|
|
public function testClassStringMap() : void
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
|
|
|
'class-string-map<T as Foo, T>',
|
|
|
|
(string)Type::parseString('class-string-map<T as Foo, T>')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-04-20 15:22:48 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testVeryLargeType()
|
|
|
|
{
|
2019-10-16 22:14:33 -07:00
|
|
|
$very_large_type = 'array{a: Closure():(array<mixed, mixed>|null), b?: Closure():array<mixed, mixed>, c?: Closure():array<mixed, mixed>, d?: Closure():array<mixed, mixed>, e?: Closure():(array{f: null|string, g: null|string, h: null|string, i: string, j: mixed, k: mixed, l: mixed, m: mixed, n: bool, o?: array{0: string}}|null), p?: Closure():(array{f: null|string, g: null|string, h: null|string, i: string, j: mixed, k: mixed, l: mixed, m: mixed, n: bool, o?: array{0: string}}|null), q: string, r?: Closure():(array<mixed, mixed>|null), s: array<mixed, mixed>}|null';
|
2018-04-20 15:22:48 -04:00
|
|
|
|
|
|
|
$this->assertSame(
|
|
|
|
$very_large_type,
|
|
|
|
(string) Type::parseString($very_large_type)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-05-20 17:19:53 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testEnum()
|
|
|
|
{
|
2019-01-27 18:34:13 -05:00
|
|
|
$docblock_type = Type::parseString('( \'foo\\\'with\' | "bar\"bar" | "baz" | "bat\\\\" | \'bang bang\' | 1 | 2 | 3 | 4.5)');
|
2018-05-20 17:19:53 -04:00
|
|
|
|
|
|
|
$resolved_type = new Type\Union([
|
|
|
|
new Type\Atomic\TLiteralString('foo\'with'),
|
|
|
|
new Type\Atomic\TLiteralString('bar"bar'),
|
|
|
|
new Type\Atomic\TLiteralString('baz'),
|
|
|
|
new Type\Atomic\TLiteralString('bat\\'),
|
|
|
|
new Type\Atomic\TLiteralString('bang bang'),
|
|
|
|
new Type\Atomic\TLiteralInt(1),
|
|
|
|
new Type\Atomic\TLiteralInt(2),
|
2019-01-27 18:34:13 -05:00
|
|
|
new Type\Atomic\TLiteralInt(3),
|
2019-03-23 14:27:54 -04:00
|
|
|
new Type\Atomic\TLiteralFloat(4.5),
|
2018-05-20 17:19:53 -04:00
|
|
|
]);
|
|
|
|
|
|
|
|
$this->assertSame($resolved_type->getId(), $docblock_type->getId());
|
|
|
|
}
|
|
|
|
|
2019-12-11 11:29:26 -05:00
|
|
|
public function testEmptyString() : void
|
2019-12-11 10:52:46 -05:00
|
|
|
{
|
|
|
|
$docblock_type = Type::parseString('""|"admin"|"fun"');
|
|
|
|
|
|
|
|
$resolved_type = new Type\Union([
|
|
|
|
new Type\Atomic\TLiteralString(''),
|
|
|
|
new Type\Atomic\TLiteralString('admin'),
|
|
|
|
new Type\Atomic\TLiteralString('fun'),
|
|
|
|
]);
|
|
|
|
|
|
|
|
$this->assertSame($resolved_type->getId(), $docblock_type->getId());
|
|
|
|
|
|
|
|
$docblock_type = Type::parseString('"admin"|""|"fun"');
|
|
|
|
|
|
|
|
$resolved_type = new Type\Union([
|
|
|
|
new Type\Atomic\TLiteralString('admin'),
|
|
|
|
new Type\Atomic\TLiteralString(''),
|
|
|
|
new Type\Atomic\TLiteralString('fun'),
|
|
|
|
]);
|
|
|
|
|
|
|
|
$this->assertSame($resolved_type->getId(), $docblock_type->getId());
|
|
|
|
|
|
|
|
$docblock_type = Type::parseString('"admin"|"fun"|""');
|
|
|
|
|
|
|
|
$resolved_type = new Type\Union([
|
|
|
|
new Type\Atomic\TLiteralString('admin'),
|
|
|
|
new Type\Atomic\TLiteralString('fun'),
|
|
|
|
new Type\Atomic\TLiteralString(''),
|
|
|
|
]);
|
|
|
|
|
|
|
|
$this->assertSame($resolved_type->getId(), $docblock_type->getId());
|
|
|
|
}
|
|
|
|
|
2018-05-21 00:46:56 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testEnumWithoutSpaces()
|
|
|
|
{
|
2019-01-27 18:34:13 -05:00
|
|
|
$docblock_type = Type::parseString('\'foo\\\'with\'|"bar\"bar"|"baz"|"bat\\\\"|\'bang bang\'|1|2|3|4.5');
|
2018-05-21 00:46:56 -04:00
|
|
|
|
|
|
|
$resolved_type = new Type\Union([
|
|
|
|
new Type\Atomic\TLiteralString('foo\'with'),
|
|
|
|
new Type\Atomic\TLiteralString('bar"bar'),
|
|
|
|
new Type\Atomic\TLiteralString('baz'),
|
|
|
|
new Type\Atomic\TLiteralString('bat\\'),
|
|
|
|
new Type\Atomic\TLiteralString('bang bang'),
|
|
|
|
new Type\Atomic\TLiteralInt(1),
|
|
|
|
new Type\Atomic\TLiteralInt(2),
|
2019-01-27 18:34:13 -05:00
|
|
|
new Type\Atomic\TLiteralInt(3),
|
2019-03-23 14:27:54 -04:00
|
|
|
new Type\Atomic\TLiteralFloat(4.5),
|
2018-05-21 00:46:56 -04:00
|
|
|
]);
|
|
|
|
|
|
|
|
$this->assertSame($resolved_type->getId(), $docblock_type->getId());
|
|
|
|
}
|
|
|
|
|
2019-11-21 10:44:24 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testSingleLiteralString()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
|
|
|
'string',
|
|
|
|
(string)Type::parseString('"var"')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testSingleLiteralInt()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
|
|
|
'int',
|
|
|
|
(string)Type::parseString('6')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testSingleLiteralFloat()
|
|
|
|
{
|
|
|
|
$this->assertSame(
|
|
|
|
'float',
|
|
|
|
(string)Type::parseString('6.315')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-05-21 00:46:56 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testEnumWithClassConstants()
|
|
|
|
{
|
|
|
|
$docblock_type = Type::parseString('("baz" | One2::TWO_THREE | Foo::BAR_BAR | Bat\Bar::BAZ_BAM)');
|
|
|
|
|
|
|
|
$resolved_type = new Type\Union([
|
|
|
|
new Type\Atomic\TLiteralString('baz'),
|
|
|
|
new Type\Atomic\TScalarClassConstant('One2', 'TWO_THREE'),
|
|
|
|
new Type\Atomic\TScalarClassConstant('Foo', 'BAR_BAR'),
|
|
|
|
new Type\Atomic\TScalarClassConstant('Bat\\Bar', 'BAZ_BAM'),
|
|
|
|
]);
|
|
|
|
|
|
|
|
$this->assertSame($resolved_type->getId(), $docblock_type->getId());
|
|
|
|
}
|
|
|
|
|
2019-01-27 17:16:22 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testReflectionTypeParse()
|
|
|
|
{
|
2019-05-16 18:36:36 -04:00
|
|
|
if (!function_exists('Psalm\Tests\someFunction')) {
|
|
|
|
/** @psalm-suppress UnusedParam */
|
|
|
|
function someFunction(string $param, array $param2, int $param3 = null) : string
|
|
|
|
{
|
|
|
|
return 'hello';
|
|
|
|
}
|
2019-01-27 17:16:22 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
$reflectionFunc = new \ReflectionFunction('Psalm\Tests\someFunction');
|
|
|
|
$reflectionParams = $reflectionFunc->getParameters();
|
|
|
|
|
|
|
|
$this->assertSame(
|
|
|
|
'string',
|
2019-01-27 17:26:32 -05:00
|
|
|
(string) \Psalm\Codebase::getPsalmTypeFromReflection($reflectionParams[0]->getType())
|
2019-01-27 17:16:22 -05:00
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertSame(
|
|
|
|
'array<array-key, mixed>',
|
2019-01-27 17:26:32 -05:00
|
|
|
(string) \Psalm\Codebase::getPsalmTypeFromReflection($reflectionParams[1]->getType())
|
2019-01-27 17:16:22 -05:00
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertSame(
|
2019-01-27 17:27:12 -05:00
|
|
|
'int|null',
|
2019-01-27 17:26:32 -05:00
|
|
|
(string) \Psalm\Codebase::getPsalmTypeFromReflection($reflectionParams[2]->getType())
|
2019-01-27 17:16:22 -05:00
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertSame(
|
|
|
|
'string',
|
2019-01-27 17:26:32 -05:00
|
|
|
(string) \Psalm\Codebase::getPsalmTypeFromReflection($reflectionFunc->getReturnType())
|
2019-01-27 17:16:22 -05:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-04-18 19:00:08 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
2019-02-07 12:25:57 -05:00
|
|
|
public function testValidCallMapType()
|
|
|
|
{
|
|
|
|
$callmap_types = \Psalm\Internal\Codebase\CallMap::getCallMap();
|
2019-01-08 21:44:50 -05:00
|
|
|
|
2019-02-07 12:25:57 -05:00
|
|
|
foreach ($callmap_types as $signature) {
|
|
|
|
$return_type = $signature[0] ?? null;
|
|
|
|
$param_type_1 = $signature[1] ?? null;
|
|
|
|
$param_type_2 = $signature[2] ?? null;
|
|
|
|
$param_type_3 = $signature[3] ?? null;
|
|
|
|
$param_type_4 = $signature[4] ?? null;
|
2018-04-18 19:00:08 -04:00
|
|
|
|
2019-02-07 12:25:57 -05:00
|
|
|
if ($return_type && $return_type !== 'void') {
|
|
|
|
if (stripos($return_type, 'oci-') !== false) {
|
2019-04-09 16:52:32 -04:00
|
|
|
continue;
|
2019-02-07 12:25:57 -05:00
|
|
|
}
|
|
|
|
|
2019-04-11 11:02:46 -04:00
|
|
|
try {
|
|
|
|
\Psalm\Type::parseString($return_type);
|
|
|
|
} catch (\Psalm\Exception\TypeParseTreeException $e) {
|
|
|
|
self::assertTrue(false, $e . ' | ' . print_r($signature, true));
|
|
|
|
}
|
2018-04-18 19:00:08 -04:00
|
|
|
}
|
|
|
|
|
2019-02-07 12:25:57 -05:00
|
|
|
if ($param_type_1 && $param_type_1 !== 'mixed') {
|
|
|
|
if (stripos($param_type_1, 'oci-') !== false) {
|
2019-04-09 16:52:32 -04:00
|
|
|
continue;
|
2019-02-07 12:25:57 -05:00
|
|
|
}
|
2018-04-18 19:00:08 -04:00
|
|
|
|
2019-04-11 11:02:46 -04:00
|
|
|
try {
|
|
|
|
\Psalm\Type::parseString($param_type_1);
|
|
|
|
} catch (\Psalm\Exception\TypeParseTreeException $e) {
|
|
|
|
self::assertTrue(false, $e . ' | ' . print_r($signature, true));
|
|
|
|
}
|
2019-02-07 12:25:57 -05:00
|
|
|
}
|
2018-04-18 19:00:08 -04:00
|
|
|
|
2019-02-07 12:25:57 -05:00
|
|
|
if ($param_type_2 && $param_type_2 !== 'mixed') {
|
2019-04-11 11:02:46 -04:00
|
|
|
try {
|
|
|
|
\Psalm\Type::parseString($param_type_2);
|
|
|
|
} catch (\Psalm\Exception\TypeParseTreeException $e) {
|
|
|
|
self::assertTrue(false, $e . ' | ' . print_r($signature, true));
|
|
|
|
}
|
2019-02-07 12:25:57 -05:00
|
|
|
}
|
2018-04-18 19:00:08 -04:00
|
|
|
|
2019-02-07 12:25:57 -05:00
|
|
|
if ($param_type_3 && $param_type_3 !== 'mixed') {
|
2019-04-11 11:02:46 -04:00
|
|
|
try {
|
|
|
|
\Psalm\Type::parseString($param_type_3);
|
|
|
|
} catch (\Psalm\Exception\TypeParseTreeException $e) {
|
|
|
|
self::assertTrue(false, $e . ' | ' . print_r($signature, true));
|
|
|
|
}
|
2019-02-07 12:25:57 -05:00
|
|
|
}
|
2018-04-18 19:00:08 -04:00
|
|
|
|
2019-02-07 12:25:57 -05:00
|
|
|
if ($param_type_4 && $param_type_4 !== 'mixed') {
|
2019-04-11 11:02:46 -04:00
|
|
|
try {
|
|
|
|
\Psalm\Type::parseString($param_type_4);
|
|
|
|
} catch (\Psalm\Exception\TypeParseTreeException $e) {
|
|
|
|
self::assertTrue(false, $e . ' | ' . print_r($signature, true));
|
|
|
|
}
|
2019-02-07 12:25:57 -05:00
|
|
|
}
|
|
|
|
}
|
2018-04-18 19:00:08 -04:00
|
|
|
}
|
2016-09-09 16:38:32 -04:00
|
|
|
}
|