create(ParserFactory::PREFER_PHP7); $config = new TestConfig(); } public function setUp() { FileChecker::clearCache(); } public function testNotNull() { $this->assertEquals( 'MyObject', (string) TypeChecker::reconcileTypes('!null', Type::parseString('MyObject')) ); $this->assertEquals( 'MyObject', (string) TypeChecker::reconcileTypes('!null', Type::parseString('MyObject|null')) ); $this->assertEquals( 'MyObject|false', (string) TypeChecker::reconcileTypes('!null', Type::parseString('MyObject|false')) ); $this->assertEquals( 'mixed', (string) TypeChecker::reconcileTypes('!null', Type::parseString('mixed')) ); } public function testNotEmpty() { $this->assertEquals( 'MyObject', (string) TypeChecker::reconcileTypes('!empty', Type::parseString('MyObject')) ); $this->assertEquals( 'MyObject', (string) TypeChecker::reconcileTypes('!empty', Type::parseString('MyObject|null')) ); $this->assertEquals( 'MyObject', (string) TypeChecker::reconcileTypes('!empty', Type::parseString('MyObject|false')) ); $this->assertEquals( 'mixed', (string) TypeChecker::reconcileTypes('!empty', Type::parseString('mixed')) ); // @todo in the future this should also work /* $this->assertEquals( 'MyObject|true', (string) TypeChecker::reconcileTypes('!empty', Type::parseString('MyObject|bool')) ); */ } public function testNull() { $this->assertEquals( 'null', (string) TypeChecker::reconcileTypes('null', Type::parseString('MyObject|null')) ); $this->assertEquals( 'null', (string) TypeChecker::reconcileTypes('null', Type::parseString('mixed')) ); } public function testEmpty() { $this->assertEquals( 'null', (string) TypeChecker::reconcileTypes('empty', Type::parseString('MyObject')) ); $this->assertEquals( 'false', (string) TypeChecker::reconcileTypes('empty', Type::parseString('MyObject|false')) ); $this->assertEquals( 'false', (string) TypeChecker::reconcileTypes('empty', Type::parseString('MyObject|bool')) ); $this->assertEquals( 'mixed', (string) TypeChecker::reconcileTypes('empty', Type::parseString('mixed')) ); /** @var Type\Union */ $reconciled = TypeChecker::reconcileTypes('empty', Type::parseString('bool')); $this->assertEquals('false', (string) $reconciled); $this->assertInstanceOf('Psalm\Type\Atomic', $reconciled->types['false']); } public function testNotMyObject() { $this->assertEquals( 'bool', (string) TypeChecker::reconcileTypes('!MyObject', Type::parseString('MyObject|bool')) ); $this->assertEquals( 'null', (string) TypeChecker::reconcileTypes('!MyObject', Type::parseString('MyObject|null')) ); $this->assertEquals( 'MyObjectB', (string) TypeChecker::reconcileTypes('!MyObjectA', Type::parseString('MyObjectA|MyObjectB')) ); } public function testMyObject() { $this->assertEquals( 'MyObject', (string) TypeChecker::reconcileTypes('MyObject', Type::parseString('MyObject|bool')) ); $this->assertEquals( 'MyObjectA', (string) TypeChecker::reconcileTypes('MyObjectA', Type::parseString('MyObjectA|MyObjectB')) ); } public function testArrayContains() { $this->assertTrue( TypeChecker::isContainedBy( Type::parseString('array'), Type::parseString('array') ) ); $this->assertTrue( TypeChecker::isContainedBy( Type::parseString('array'), Type::parseString('array') ) ); $this->assertTrue( TypeChecker::isContainedBy( Type::parseString('array'), Type::parseString('array') ) ); $this->assertFalse( TypeChecker::isContainedBy( Type::parseString('array'), Type::parseString('array') ) ); } public function testNumeric() { $this->assertEquals( 'string', (string) TypeChecker::reconcileTypes('numeric', Type::parseString('string')) ); } /** * @expectedException \Psalm\Exception\CodeException * @expectedExceptionMessage TypeDoesNotContainType */ public function testMakeNonNullableNull() { $stmts = self::$parser->parse('check(true, true, $context); } /** * @expectedException \Psalm\Exception\CodeException * @expectedExceptionMessage TypeDoesNotContainType */ public function testMakeInstanceOfThingInElseif() { $stmts = self::$parser->parse(' 5 ? new A() : new B(); if ($a instanceof A) { } elseif ($a instanceof C) { } '); $file_checker = new FileChecker('somefile.php', $stmts); $context = new Context('somefile.php'); $file_checker->check(true, true, $context); } /** * @expectedException \Psalm\Exception\CodeException * @expectedExceptionMessage FailedTypeResolution */ public function testFailedTypeResolution() { $stmts = self::$parser->parse('check(true, true, $context); } public function testNotInstanceOf() { $stmts = self::$parser->parse('vars_in_scope['$a'] = Type::parseString('A'); $file_checker->check(true, true, $context); $this->assertEquals('null|A', (string) $context->vars_in_scope['$out']); } public function testNotInstanceOfProperty() { $stmts = self::$parser->parse('foo instanceof C) { // do something } else { $out = $a->foo; } '); $file_checker = new FileChecker('somefile.php', $stmts); $context = new Context('somefile.php'); $context->vars_in_scope['$a'] = Type::parseString('A'); $file_checker->check(true, true, $context); $this->assertEquals('null|B', (string) $context->vars_in_scope['$out']); } public function testNotInstanceOfPropertyElseif() { $stmts = self::$parser->parse('foo)) { } elseif ($a->foo instanceof C) { // do something } else { $out = $a->foo; } '); $file_checker = new FileChecker('somefile.php', $stmts); $context = new Context('somefile.php'); $context->vars_in_scope['$a'] = Type::parseString('A'); $file_checker->check(true, true, $context); $this->assertEquals('null|B', (string) $context->vars_in_scope['$out']); } public function testTypeArguments() { $stmts = self::$parser->parse('check(true, true, $context); $this->assertEquals('int', (string) $context->vars_in_scope['$a']); $this->assertEquals('int', (string) $context->vars_in_scope['$b']); $this->assertEquals('string', (string) $context->vars_in_scope['$c']); } /** * @expectedException \Psalm\Exception\CodeException * @expectedExceptionMessage TypeDoesNotContainType */ public function testTypeTransformation() { $stmts = self::$parser->parse('check(true, true, $context); } }