2017-04-25 05:45:02 +02:00
|
|
|
<?php
|
2021-12-15 04:58:32 +01:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
namespace Psalm\Tests\Traits;
|
|
|
|
|
|
|
|
use Psalm\Config;
|
|
|
|
use Psalm\Context;
|
2021-06-08 04:55:21 +02:00
|
|
|
|
2021-12-03 21:07:25 +01:00
|
|
|
use function str_replace;
|
2019-06-26 22:52:29 +02:00
|
|
|
use function strlen;
|
2019-07-05 22:24:00 +02:00
|
|
|
use function strpos;
|
2021-12-03 21:07:25 +01:00
|
|
|
use function strtoupper;
|
2019-06-26 22:52:29 +02:00
|
|
|
use function substr;
|
2019-07-05 22:24:00 +02:00
|
|
|
use function version_compare;
|
2017-04-25 05:45:02 +02:00
|
|
|
|
2021-12-03 21:25:22 +01:00
|
|
|
use const PHP_OS;
|
2021-06-08 04:55:21 +02:00
|
|
|
use const PHP_VERSION;
|
2023-07-21 22:57:49 +02:00
|
|
|
use const PHP_VERSION_ID;
|
2021-06-08 04:55:21 +02:00
|
|
|
|
2018-11-06 03:57:36 +01:00
|
|
|
trait ValidCodeAnalysisTestTrait
|
2017-04-25 05:45:02 +02:00
|
|
|
{
|
|
|
|
/**
|
2022-12-18 20:13:36 +01:00
|
|
|
* @return iterable<
|
|
|
|
* string,
|
|
|
|
* array{
|
|
|
|
* code: string,
|
|
|
|
* assertions?: array<string, string>,
|
|
|
|
* ignored_issues?: list<string>,
|
|
|
|
* php_version?: string,
|
|
|
|
* }
|
|
|
|
* >
|
2017-04-25 05:45:02 +02:00
|
|
|
*/
|
2020-09-12 17:24:05 +02:00
|
|
|
abstract public function providerValidCodeParse(): iterable;
|
2017-04-25 05:45:02 +02:00
|
|
|
|
|
|
|
/**
|
2018-11-06 03:57:36 +01:00
|
|
|
* @dataProvider providerValidCodeParse
|
2017-06-29 16:25:41 +02:00
|
|
|
* @param array<string, string> $assertions
|
2022-11-05 22:34:42 +01:00
|
|
|
* @param list<string> $ignored_issues
|
2017-11-28 06:25:21 +01:00
|
|
|
* @small
|
2017-04-25 05:45:02 +02:00
|
|
|
*/
|
2019-02-07 18:25:57 +01:00
|
|
|
public function testValidCode(
|
2022-12-15 03:26:17 +01:00
|
|
|
string $code,
|
|
|
|
array $assertions = [],
|
2023-02-02 17:19:22 +01:00
|
|
|
array $ignored_issues = [],
|
2023-04-10 02:03:13 +02:00
|
|
|
string $php_version = '7.4'
|
2021-11-26 20:59:41 +01:00
|
|
|
): void {
|
2018-07-13 23:44:50 +02:00
|
|
|
$test_name = $this->getTestName();
|
2022-01-04 10:40:55 +01:00
|
|
|
if (strpos($test_name, 'PHP80-') !== false) {
|
2020-10-12 17:35:14 +02:00
|
|
|
if (version_compare(PHP_VERSION, '8.0.0', '<')) {
|
|
|
|
$this->markTestSkipped('Test case requires PHP 8.0.');
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
2022-02-22 22:41:53 +01:00
|
|
|
} elseif (strpos($test_name, 'PHP81-') !== false) {
|
|
|
|
if (version_compare(PHP_VERSION, '8.1.0', '<')) {
|
|
|
|
$this->markTestSkipped('Test case requires PHP 8.1.');
|
|
|
|
}
|
2017-04-25 05:45:02 +02:00
|
|
|
} elseif (strpos($test_name, 'SKIPPED-') !== false) {
|
|
|
|
$this->markTestSkipped('Skipped due to a bug.');
|
|
|
|
}
|
|
|
|
|
2023-03-03 07:50:11 +01:00
|
|
|
// sanity check - do we have a PHP tag?
|
|
|
|
if (strpos($code, '<?php') === false) {
|
|
|
|
$this->fail('Test case must have a <?php tag');
|
|
|
|
}
|
|
|
|
|
2022-11-05 22:34:42 +01:00
|
|
|
foreach ($ignored_issues as $issue_name) {
|
|
|
|
Config::getInstance()->setCustomErrorLevel($issue_name, Config::REPORT_SUPPRESS);
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
|
|
|
|
2021-12-03 21:25:22 +01:00
|
|
|
if (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN') {
|
2021-12-03 21:07:25 +01:00
|
|
|
$code = str_replace("\n", "\r\n", $code);
|
2020-03-10 17:52:18 +01:00
|
|
|
}
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
$context = new Context();
|
2019-02-07 18:25:57 +01:00
|
|
|
|
2021-11-27 01:06:33 +01:00
|
|
|
$this->project_analyzer->setPhpVersion($php_version, 'tests');
|
2017-04-25 05:45:02 +02:00
|
|
|
|
2020-10-30 00:41:10 +01:00
|
|
|
$codebase = $this->project_analyzer->getCodebase();
|
2023-02-19 07:51:51 +01:00
|
|
|
$codebase->enterServerMode();
|
2020-10-30 00:41:10 +01:00
|
|
|
$codebase->config->visitPreloadedStubFiles($codebase);
|
|
|
|
|
2023-07-19 23:52:25 +02:00
|
|
|
// avoid MethodSignatureMismatch for __unserialize/() when extending DateTime
|
2023-07-21 22:42:27 +02:00
|
|
|
if (PHP_VERSION_ID >= 8_02_00) {
|
2023-07-19 23:14:04 +02:00
|
|
|
$this->addStubFile(
|
|
|
|
'stubOne.phpstub',
|
|
|
|
'<?php
|
|
|
|
namespace {
|
|
|
|
interface DateTimeInterface {
|
|
|
|
public function __unserialize(mixed[] $data) {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-01-21 16:22:04 +01:00
|
|
|
$file_path = self::$src_dir_path . 'somefile.php';
|
|
|
|
|
2018-01-21 19:38:51 +01:00
|
|
|
$this->addFile($file_path, $code);
|
2018-01-21 16:22:04 +01:00
|
|
|
$this->analyzeFile($file_path, $context);
|
2017-04-25 05:45:02 +02:00
|
|
|
|
2017-08-15 01:30:11 +02:00
|
|
|
$actual_vars = [];
|
|
|
|
foreach ($assertions as $var => $_) {
|
2019-03-07 20:56:18 +01:00
|
|
|
$exact = false;
|
|
|
|
|
|
|
|
if ($var && strpos($var, '===') === strlen($var) - 3) {
|
|
|
|
$var = substr($var, 0, -3);
|
|
|
|
$exact = true;
|
|
|
|
}
|
|
|
|
|
2017-11-20 05:19:49 +01:00
|
|
|
if (isset($context->vars_in_scope[$var])) {
|
2022-01-16 16:07:56 +01:00
|
|
|
$value = $context->vars_in_scope[$var]->getId($exact);
|
2019-03-07 20:56:18 +01:00
|
|
|
if ($exact) {
|
2022-01-16 16:07:56 +01:00
|
|
|
$actual_vars[$var . '==='] = $value;
|
2019-03-07 20:56:18 +01:00
|
|
|
} else {
|
2022-01-16 16:07:56 +01:00
|
|
|
$actual_vars[$var] = $value;
|
2019-03-07 20:56:18 +01:00
|
|
|
}
|
2017-11-20 05:19:49 +01:00
|
|
|
}
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
2017-08-15 01:30:11 +02:00
|
|
|
|
|
|
|
$this->assertSame($assertions, $actual_vars);
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
|
|
|
}
|