2017-02-23 00:25:28 -05:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class ReferenceConstraintTest extends TestCase
|
2017-02-23 00:25:28 -05:00
|
|
|
{
|
2018-11-05 21:57:36 -05:00
|
|
|
use Traits\InvalidCodeAnalysisTestTrait;
|
|
|
|
use Traits\ValidCodeAnalysisTestTrait;
|
2017-02-23 00:25:28 -05:00
|
|
|
|
|
|
|
/**
|
2017-04-24 23:45:02 -04:00
|
|
|
* @return array
|
2017-02-23 00:25:28 -05:00
|
|
|
*/
|
2018-11-05 21:57:36 -05:00
|
|
|
public function providerValidCodeParse()
|
2017-02-23 00:25:28 -05:00
|
|
|
{
|
2017-04-24 23:45:02 -04:00
|
|
|
return [
|
|
|
|
'functionParameterNoViolation' => [
|
|
|
|
'<?php
|
|
|
|
/** @return void */
|
|
|
|
function changeInt(int &$a) {
|
|
|
|
$a = 5;
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
|
|
|
],
|
2017-12-15 16:48:06 -05:00
|
|
|
'dontAllowByRefVarToBeAltered' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param ?string $str
|
|
|
|
* @psalm-suppress PossiblyNullArgument
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function nullable_ref_modifier(&$str): void {
|
2017-12-15 16:48:06 -05:00
|
|
|
if (strlen($str) > 5) {
|
|
|
|
$str = null;
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2018-01-11 23:18:13 -05:00
|
|
|
'trackFunctionReturnRefs' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/** @var string */
|
|
|
|
public $foo = "bar";
|
|
|
|
|
|
|
|
public function &getString() : string {
|
|
|
|
return $this->foo;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function useString(string &$s) : void {}
|
|
|
|
$a = new A();
|
|
|
|
|
|
|
|
useString($a->getString());',
|
|
|
|
],
|
2018-01-12 12:33:26 -05:00
|
|
|
'makeByRefUseMixed' => [
|
|
|
|
'<?php
|
|
|
|
function s(?string $p): void {}
|
|
|
|
|
|
|
|
$var = 1;
|
|
|
|
$callback = function() use(&$var): void {
|
|
|
|
s($var);
|
|
|
|
};
|
|
|
|
$var = null;
|
|
|
|
$callback();',
|
|
|
|
'assertions' => [],
|
|
|
|
'error_levels' => ['MixedArgument'],
|
|
|
|
],
|
2018-02-11 10:39:21 -05:00
|
|
|
'assignByRefToMixed' => [
|
|
|
|
'<?php
|
|
|
|
function testRef() : array {
|
|
|
|
$result = [];
|
|
|
|
foreach ([1, 2, 1] as $v) {
|
|
|
|
$x = &$result;
|
|
|
|
if (!isset($x[$v])) {
|
|
|
|
$x[$v] = 0;
|
|
|
|
}
|
|
|
|
$x[$v] ++;
|
|
|
|
}
|
|
|
|
return $result;
|
|
|
|
}',
|
|
|
|
'assertions' => [],
|
|
|
|
'error_levels' => [
|
|
|
|
'MixedAssignment',
|
|
|
|
'MixedArrayAccess',
|
|
|
|
'MixedReturnStatement',
|
|
|
|
'MixedInferredReturnType',
|
2018-06-18 13:16:51 -04:00
|
|
|
'MixedOperand',
|
2018-02-11 10:39:21 -05:00
|
|
|
],
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
];
|
2017-02-23 00:25:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-04-24 23:45:02 -04:00
|
|
|
* @return array
|
2017-02-23 00:25:28 -05:00
|
|
|
*/
|
2018-11-05 21:57:36 -05:00
|
|
|
public function providerInvalidCodeParse()
|
2017-02-23 00:25:28 -05:00
|
|
|
{
|
2017-04-24 23:45:02 -04:00
|
|
|
return [
|
|
|
|
'functionParameterViolation' => [
|
|
|
|
'<?php
|
|
|
|
/** @return void */
|
|
|
|
function changeInt(int &$a) {
|
|
|
|
$a = "hello";
|
|
|
|
}',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'ReferenceConstraintViolation',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'classMethodParameterViolation' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/** @var int */
|
|
|
|
private $foo;
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
public function __construct(int &$foo) {
|
|
|
|
$this->foo = &$foo;
|
|
|
|
$foo = "hello";
|
|
|
|
}
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
$bar = 5;
|
|
|
|
$a = new A($bar); // $bar is constrained to an int
|
|
|
|
$bar = null; // ReferenceConstraintViolation issue emitted',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'ReferenceConstraintViolation',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'classMethodParameterViolationInPostAssignment' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/** @var int */
|
|
|
|
private $foo;
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
public function __construct(int &$foo) {
|
|
|
|
$this->foo = &$foo;
|
|
|
|
}
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
$bar = 5;
|
|
|
|
$a = new A($bar);
|
|
|
|
$bar = null;',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'ReferenceConstraintViolation',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'contradictoryReferenceConstraints' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/** @var int */
|
|
|
|
private $foo;
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
public function __construct(int &$foo) {
|
|
|
|
$this->foo = &$foo;
|
|
|
|
}
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class B {
|
|
|
|
/** @var string */
|
|
|
|
private $bar;
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
public function __construct(string &$bar) {
|
|
|
|
$this->bar = &$bar;
|
|
|
|
}
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
if (rand(0, 1)) {
|
|
|
|
$v = 5;
|
|
|
|
$c = (new A($v)); // $v is constrained to an int
|
|
|
|
} else {
|
|
|
|
$v = "hello";
|
|
|
|
$c = (new B($v)); // $v is constrained to a string
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
$v = 8;',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'ConflictingReferenceConstraint',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
];
|
2017-02-23 00:25:28 -05:00
|
|
|
}
|
|
|
|
}
|