2018-01-28 23:41:11 -05:00
|
|
|
|
<?php
|
2020-11-28 04:48:16 +02:00
|
|
|
|
|
2023-10-19 13:12:06 +02:00
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
2018-01-28 23:41:11 -05:00
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
|
2020-11-28 04:48:16 +02:00
|
|
|
|
use DOMAttr;
|
|
|
|
|
use DOMDocument;
|
|
|
|
|
use DOMXPath;
|
|
|
|
|
use PHPUnit\Framework\Constraint\Constraint;
|
|
|
|
|
use Psalm\Config;
|
2021-12-03 20:11:20 +01:00
|
|
|
|
use Psalm\Config\IssueHandler;
|
2020-11-28 04:48:16 +02:00
|
|
|
|
use Psalm\Context;
|
|
|
|
|
use Psalm\DocComment;
|
2021-12-03 20:29:06 +01:00
|
|
|
|
use Psalm\Exception\CodeException;
|
2021-12-03 20:11:20 +01:00
|
|
|
|
use Psalm\Internal\Analyzer\ProjectAnalyzer;
|
2021-07-02 02:10:21 +03:00
|
|
|
|
use Psalm\Internal\Provider\FakeFileProvider;
|
2021-12-03 20:11:20 +01:00
|
|
|
|
use Psalm\Internal\Provider\Providers;
|
2020-11-28 04:48:16 +02:00
|
|
|
|
use Psalm\Internal\RuntimeCaches;
|
2023-01-17 19:30:43 -05:00
|
|
|
|
use Psalm\Issue\UnusedBaselineEntry;
|
2021-12-04 21:55:53 +01:00
|
|
|
|
use Psalm\Tests\Internal\Provider\FakeParserCacheProvider;
|
2021-12-03 21:40:18 +01:00
|
|
|
|
use UnexpectedValueException;
|
2020-11-28 04:48:16 +02:00
|
|
|
|
|
2022-01-12 14:25:07 -06:00
|
|
|
|
use function array_diff;
|
2020-11-28 04:48:16 +02:00
|
|
|
|
use function array_filter;
|
2019-07-05 16:24:00 -04:00
|
|
|
|
use function array_keys;
|
2022-01-12 14:25:07 -06:00
|
|
|
|
use function array_map;
|
2020-11-28 04:48:16 +02:00
|
|
|
|
use function array_shift;
|
2023-10-09 23:38:24 +01:00
|
|
|
|
use function assert;
|
2019-07-05 16:24:00 -04:00
|
|
|
|
use function count;
|
|
|
|
|
use function dirname;
|
|
|
|
|
use function explode;
|
2022-01-12 14:25:07 -06:00
|
|
|
|
use function file;
|
2019-07-05 16:24:00 -04:00
|
|
|
|
use function file_exists;
|
|
|
|
|
use function file_get_contents;
|
2020-11-28 04:48:16 +02:00
|
|
|
|
use function glob;
|
2019-07-05 16:24:00 -04:00
|
|
|
|
use function implode;
|
2020-11-28 04:48:16 +02:00
|
|
|
|
use function in_array;
|
2022-01-12 14:25:07 -06:00
|
|
|
|
use function preg_match;
|
2019-07-05 16:24:00 -04:00
|
|
|
|
use function preg_quote;
|
2022-01-12 14:25:07 -06:00
|
|
|
|
use function scandir;
|
2019-06-26 22:52:29 +02:00
|
|
|
|
use function sort;
|
2020-11-28 04:48:16 +02:00
|
|
|
|
use function str_replace;
|
2022-01-12 14:25:07 -06:00
|
|
|
|
use function strlen;
|
2021-06-08 05:55:21 +03:00
|
|
|
|
use function strpos;
|
2019-07-05 16:24:00 -04:00
|
|
|
|
use function substr;
|
|
|
|
|
use function trim;
|
2022-01-12 14:25:07 -06:00
|
|
|
|
use function usort;
|
2020-08-16 23:26:54 +03:00
|
|
|
|
use function var_export;
|
2018-01-28 23:41:11 -05:00
|
|
|
|
|
2020-11-28 04:48:16 +02:00
|
|
|
|
use const DIRECTORY_SEPARATOR;
|
2022-01-12 14:25:07 -06:00
|
|
|
|
use const FILE_IGNORE_NEW_LINES;
|
|
|
|
|
use const FILE_SKIP_EMPTY_LINES;
|
2020-11-28 04:48:16 +02:00
|
|
|
|
use const LIBXML_NONET;
|
|
|
|
|
|
2018-01-28 23:41:11 -05:00
|
|
|
|
class DocumentationTest extends TestCase
|
|
|
|
|
{
|
2020-11-28 04:48:16 +02:00
|
|
|
|
/**
|
|
|
|
|
* a list of all files containing annotation documentation
|
|
|
|
|
*/
|
|
|
|
|
private const ANNOTATION_DOCS = [
|
|
|
|
|
'docs/annotating_code/supported_annotations.md',
|
|
|
|
|
'docs/annotating_code/templated_annotations.md',
|
|
|
|
|
'docs/annotating_code/adding_assertions.md',
|
|
|
|
|
'docs/security_analysis/annotations.md',
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* annotations that we don’t want documented
|
|
|
|
|
*/
|
|
|
|
|
private const INTENTIONALLY_UNDOCUMENTED_ANNOTATIONS = [
|
2021-10-28 10:44:37 +02:00
|
|
|
|
'@psalm-self-out', // Not documented as it's a legacy alias of @psalm-this-out
|
2020-11-28 04:48:16 +02:00
|
|
|
|
'@psalm-variadic',
|
|
|
|
|
];
|
2021-06-07 17:46:26 +03:00
|
|
|
|
|
2020-11-28 04:48:16 +02:00
|
|
|
|
/**
|
|
|
|
|
* These should be documented
|
|
|
|
|
*/
|
|
|
|
|
private const WALL_OF_SHAME = [
|
|
|
|
|
'@psalm-assert-untainted',
|
|
|
|
|
'@psalm-flow',
|
2022-04-27 01:46:13 -04:00
|
|
|
|
'@psalm-generator-return',
|
2020-11-28 04:48:16 +02:00
|
|
|
|
'@psalm-override-method-visibility',
|
|
|
|
|
'@psalm-override-property-visibility',
|
|
|
|
|
'@psalm-scope-this',
|
|
|
|
|
'@psalm-seal-methods',
|
|
|
|
|
'@psalm-stub-override',
|
|
|
|
|
];
|
|
|
|
|
|
2022-12-16 12:58:47 -06:00
|
|
|
|
protected ProjectAnalyzer $project_analyzer;
|
2018-01-28 23:41:11 -05:00
|
|
|
|
|
2022-12-16 12:58:47 -06:00
|
|
|
|
private static string $docContents = '';
|
2020-11-28 04:48:16 +02:00
|
|
|
|
|
2018-01-28 23:41:11 -05:00
|
|
|
|
/**
|
|
|
|
|
* @return array<string, array<int, string>>
|
|
|
|
|
*/
|
2020-09-12 17:24:05 +02:00
|
|
|
|
private static function getCodeBlocksFromDocs(): array
|
2018-01-28 23:41:11 -05:00
|
|
|
|
{
|
2022-01-12 15:22:21 -06:00
|
|
|
|
$issues_dir = dirname(__DIR__) . DIRECTORY_SEPARATOR . 'docs' . DIRECTORY_SEPARATOR . 'running_psalm' . DIRECTORY_SEPARATOR . 'issues';
|
2018-01-28 23:41:11 -05:00
|
|
|
|
|
2020-03-19 12:32:49 -04:00
|
|
|
|
if (!file_exists($issues_dir)) {
|
2021-12-03 21:40:18 +01:00
|
|
|
|
throw new UnexpectedValueException('docs not found');
|
2018-01-28 23:41:11 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$issue_code = [];
|
2023-10-09 23:38:24 +01:00
|
|
|
|
$files = glob($issues_dir . '/*.md');
|
|
|
|
|
assert($files !== false);
|
2018-01-28 23:41:11 -05:00
|
|
|
|
|
2023-10-09 23:38:24 +01:00
|
|
|
|
foreach ($files as $file_path) {
|
2020-03-19 12:32:49 -04:00
|
|
|
|
$file_contents = file_get_contents($file_path);
|
2023-10-09 23:38:24 +01:00
|
|
|
|
assert($file_contents !== false);
|
2018-01-28 23:41:11 -05:00
|
|
|
|
|
2020-03-19 12:32:49 -04:00
|
|
|
|
$file_lines = explode("\n", $file_contents);
|
2018-01-28 23:41:11 -05:00
|
|
|
|
|
2020-03-19 12:32:49 -04:00
|
|
|
|
$current_issue = str_replace('# ', '', array_shift($file_lines));
|
2018-01-28 23:41:11 -05:00
|
|
|
|
|
2020-03-19 12:32:49 -04:00
|
|
|
|
for ($i = 0, $j = count($file_lines); $i < $j; ++$i) {
|
|
|
|
|
$current_line = $file_lines[$i];
|
2018-01-28 23:41:11 -05:00
|
|
|
|
|
2020-03-19 12:32:49 -04:00
|
|
|
|
if (substr($current_line, 0, 6) === '```php' && $current_issue) {
|
|
|
|
|
$current_block = '';
|
2018-01-28 23:41:11 -05:00
|
|
|
|
++$i;
|
|
|
|
|
|
2020-03-19 12:32:49 -04:00
|
|
|
|
do {
|
|
|
|
|
$current_block .= $file_lines[$i] . "\n";
|
|
|
|
|
++$i;
|
|
|
|
|
} while (substr($file_lines[$i], 0, 3) !== '```' && $i < $j);
|
|
|
|
|
|
|
|
|
|
$issue_code[$current_issue][] = trim($current_block);
|
|
|
|
|
|
|
|
|
|
continue 2;
|
|
|
|
|
}
|
2018-01-28 23:41:11 -05:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return $issue_code;
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-05 18:51:26 +01:00
|
|
|
|
public function setUp(): void
|
2018-01-28 23:41:11 -05:00
|
|
|
|
{
|
2020-08-23 17:32:07 +03:00
|
|
|
|
RuntimeCaches::clearAll();
|
2018-01-28 23:41:11 -05:00
|
|
|
|
|
2021-07-02 02:10:21 +03:00
|
|
|
|
$this->file_provider = new FakeFileProvider();
|
2018-01-28 23:41:11 -05:00
|
|
|
|
|
2021-12-03 20:11:20 +01:00
|
|
|
|
$this->project_analyzer = new ProjectAnalyzer(
|
2018-01-28 23:41:11 -05:00
|
|
|
|
new TestConfig(),
|
2021-12-03 20:11:20 +01:00
|
|
|
|
new Providers(
|
2018-09-28 16:18:45 -04:00
|
|
|
|
$this->file_provider,
|
2022-12-18 10:15:15 -06:00
|
|
|
|
new FakeParserCacheProvider(),
|
|
|
|
|
),
|
2018-01-28 23:41:11 -05:00
|
|
|
|
);
|
2019-02-07 15:27:43 -05:00
|
|
|
|
|
2021-11-27 02:06:33 +02:00
|
|
|
|
$this->project_analyzer->setPhpVersion('8.0', 'tests');
|
2018-01-28 23:41:11 -05:00
|
|
|
|
}
|
|
|
|
|
|
2020-06-23 23:56:39 +03:00
|
|
|
|
public function testAllIssuesCoveredInConfigSchema(): void
|
|
|
|
|
{
|
2021-12-03 20:11:20 +01:00
|
|
|
|
$all_issues = IssueHandler::getAllIssueTypes();
|
2020-06-23 23:56:39 +03:00
|
|
|
|
$all_issues[] = 'PluginIssue'; // not an ordinary issue
|
|
|
|
|
sort($all_issues);
|
|
|
|
|
|
|
|
|
|
$schema = new DOMDocument();
|
|
|
|
|
$schema->load(__DIR__ . '/../config.xsd', LIBXML_NONET);
|
|
|
|
|
|
|
|
|
|
$xpath = new DOMXPath($schema);
|
|
|
|
|
$xpath->registerNamespace('xs', 'http://www.w3.org/2001/XMLSchema');
|
|
|
|
|
|
|
|
|
|
/** @var iterable<mixed, DOMAttr> $handlers */
|
|
|
|
|
$handlers = $xpath->query('//xs:complexType[@name="IssueHandlersType"]/xs:choice/xs:element/@name');
|
|
|
|
|
$handler_types = [];
|
|
|
|
|
foreach ($handlers as $handler) {
|
|
|
|
|
$handler_types[] = $handler->value;
|
|
|
|
|
}
|
|
|
|
|
sort($handler_types);
|
|
|
|
|
|
|
|
|
|
$this->assertSame(implode("\n", $all_issues), implode("\n", $handler_types));
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
|
public function testAllIssuesCovered(): void
|
2018-01-29 00:39:21 -05:00
|
|
|
|
{
|
2021-12-03 20:11:20 +01:00
|
|
|
|
$all_issues = IssueHandler::getAllIssueTypes();
|
2019-05-03 15:29:44 -04:00
|
|
|
|
$all_issues[] = 'ParseError';
|
|
|
|
|
$all_issues[] = 'PluginIssue';
|
|
|
|
|
|
2018-01-29 00:39:21 -05:00
|
|
|
|
sort($all_issues);
|
|
|
|
|
|
|
|
|
|
$code_blocks = self::getCodeBlocksFromDocs();
|
|
|
|
|
|
|
|
|
|
// these cannot have code
|
|
|
|
|
$code_blocks['UnrecognizedExpression'] = true;
|
|
|
|
|
$code_blocks['UnrecognizedStatement'] = true;
|
2019-01-07 08:38:56 -05:00
|
|
|
|
$code_blocks['PluginIssue'] = true;
|
2019-08-04 10:37:36 -04:00
|
|
|
|
$code_blocks['TaintedInput'] = true;
|
2020-11-17 12:44:31 -05:00
|
|
|
|
$code_blocks['TaintedCustom'] = true;
|
2020-11-27 17:02:37 -05:00
|
|
|
|
$code_blocks['ComplexFunction'] = true;
|
|
|
|
|
$code_blocks['ComplexMethod'] = true;
|
2021-06-07 17:46:26 +03:00
|
|
|
|
$code_blocks['ConfigIssue'] = true;
|
2018-01-29 00:39:21 -05:00
|
|
|
|
|
|
|
|
|
$documented_issues = array_keys($code_blocks);
|
|
|
|
|
sort($documented_issues);
|
|
|
|
|
|
2018-04-13 01:42:24 +02:00
|
|
|
|
$this->assertSame(implode("\n", $all_issues), implode("\n", $documented_issues));
|
2018-01-29 00:39:21 -05:00
|
|
|
|
}
|
|
|
|
|
|
2018-01-28 23:41:11 -05:00
|
|
|
|
/**
|
2018-11-05 21:57:36 -05:00
|
|
|
|
* @dataProvider providerInvalidCodeParse
|
2018-01-28 23:41:11 -05:00
|
|
|
|
* @small
|
2022-11-05 22:34:42 +01:00
|
|
|
|
* @param array<string> $ignored_issues
|
2018-01-28 23:41:11 -05:00
|
|
|
|
*/
|
2022-12-14 20:26:17 -06:00
|
|
|
|
public function testInvalidCode(string $code, string $error_message, array $ignored_issues = [], bool $check_references = false, string $php_version = '8.0'): void
|
2018-01-28 23:41:11 -05:00
|
|
|
|
{
|
2018-07-14 00:44:50 +03:00
|
|
|
|
if (strpos($this->getTestName(), 'SKIPPED-') !== false) {
|
2018-01-28 23:41:11 -05:00
|
|
|
|
$this->markTestSkipped();
|
|
|
|
|
}
|
|
|
|
|
|
2021-11-27 02:06:33 +02:00
|
|
|
|
$this->project_analyzer->setPhpVersion($php_version, 'tests');
|
2021-09-05 20:12:24 +03:00
|
|
|
|
|
2018-02-03 18:52:35 -05:00
|
|
|
|
if ($check_references) {
|
2018-11-11 12:01:14 -05:00
|
|
|
|
$this->project_analyzer->getCodebase()->reportUnusedCode();
|
2019-08-18 14:27:50 -04:00
|
|
|
|
$this->project_analyzer->trackUnusedSuppressions();
|
2018-02-03 18:52:35 -05:00
|
|
|
|
}
|
2018-01-28 23:41:11 -05:00
|
|
|
|
|
2020-11-17 12:44:31 -05:00
|
|
|
|
$is_taint_test = strpos($error_message, 'Tainted') !== false;
|
|
|
|
|
|
2019-11-11 09:59:56 -05:00
|
|
|
|
$is_array_offset_test = strpos($error_message, 'ArrayOffset') && strpos($error_message, 'PossiblyUndefined') !== false;
|
|
|
|
|
|
|
|
|
|
$this->project_analyzer->getConfig()->ensure_array_string_offsets_exist = $is_array_offset_test;
|
|
|
|
|
$this->project_analyzer->getConfig()->ensure_array_int_offsets_exist = $is_array_offset_test;
|
|
|
|
|
|
2022-11-05 22:34:42 +01:00
|
|
|
|
foreach ($ignored_issues as $error_level) {
|
2018-11-11 12:01:14 -05:00
|
|
|
|
$this->project_analyzer->getCodebase()->config->setCustomErrorLevel($error_level, Config::REPORT_SUPPRESS);
|
2018-01-28 23:41:11 -05:00
|
|
|
|
}
|
|
|
|
|
|
2021-12-03 20:29:06 +01:00
|
|
|
|
$this->expectException(CodeException::class);
|
2022-01-19 19:29:16 +01:00
|
|
|
|
$this->expectExceptionMessageMatches('/\b' . preg_quote($error_message, '/') . '\b/');
|
2018-01-28 23:41:11 -05:00
|
|
|
|
|
2020-10-29 19:41:10 -04:00
|
|
|
|
$codebase = $this->project_analyzer->getCodebase();
|
|
|
|
|
$codebase->config->visitPreloadedStubFiles($codebase);
|
|
|
|
|
|
2018-01-28 23:41:11 -05:00
|
|
|
|
$file_path = self::$src_dir_path . 'somefile.php';
|
|
|
|
|
|
|
|
|
|
$this->addFile($file_path, $code);
|
|
|
|
|
|
2020-11-17 12:44:31 -05:00
|
|
|
|
if ($is_taint_test) {
|
|
|
|
|
$this->project_analyzer->trackTaintedInputs();
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-28 16:30:56 -04:00
|
|
|
|
$this->analyzeFile($file_path, new Context());
|
2018-01-28 23:41:11 -05:00
|
|
|
|
|
|
|
|
|
if ($check_references) {
|
2019-12-02 15:24:01 -05:00
|
|
|
|
$this->project_analyzer->consolidateAnalyzedData();
|
2018-01-28 23:41:11 -05:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2022-11-11 20:14:21 -05:00
|
|
|
|
* @return array<string,array{string,string,string[],bool,string}>
|
2018-01-28 23:41:11 -05:00
|
|
|
|
*/
|
2020-09-12 17:24:05 +02:00
|
|
|
|
public function providerInvalidCodeParse(): array
|
2018-01-28 23:41:11 -05:00
|
|
|
|
{
|
|
|
|
|
$invalid_code_data = [];
|
|
|
|
|
|
|
|
|
|
foreach (self::getCodeBlocksFromDocs() as $issue_name => $blocks) {
|
2021-09-05 20:12:24 +03:00
|
|
|
|
$php_version = '8.0';
|
|
|
|
|
$ignored_issues = [];
|
2018-01-28 23:41:11 -05:00
|
|
|
|
switch ($issue_name) {
|
2018-03-06 12:19:50 -05:00
|
|
|
|
case 'InvalidStringClass':
|
2022-02-13 17:19:25 -06:00
|
|
|
|
case 'MissingThrowsDocblock':
|
2019-01-07 08:38:56 -05:00
|
|
|
|
case 'PluginClass':
|
2020-07-30 10:25:59 -04:00
|
|
|
|
case 'RedundantIdentityWithTrue':
|
2020-10-03 23:22:26 -04:00
|
|
|
|
case 'TraitMethodSignatureMismatch':
|
2022-02-13 17:19:25 -06:00
|
|
|
|
case 'UncaughtThrowInGlobalScope':
|
2023-01-17 19:30:43 -05:00
|
|
|
|
case UnusedBaselineEntry::getIssueType():
|
2020-10-03 23:22:26 -04:00
|
|
|
|
continue 2;
|
|
|
|
|
|
2022-01-31 23:38:15 +02:00
|
|
|
|
/** @todo reinstate this test when the issue is restored */
|
|
|
|
|
case 'MethodSignatureMustProvideReturnType':
|
|
|
|
|
continue 2;
|
|
|
|
|
|
2018-01-28 23:41:11 -05:00
|
|
|
|
case 'InvalidFalsableReturnType':
|
|
|
|
|
$ignored_issues = ['FalsableReturnStatement'];
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 'InvalidNullableReturnType':
|
|
|
|
|
$ignored_issues = ['NullableReturnStatement'];
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 'InvalidReturnType':
|
|
|
|
|
$ignored_issues = ['InvalidReturnStatement'];
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 'MixedInferredReturnType':
|
|
|
|
|
$ignored_issues = ['MixedReturnStatement'];
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 'MixedStringOffsetAssignment':
|
|
|
|
|
$ignored_issues = ['MixedAssignment'];
|
|
|
|
|
break;
|
|
|
|
|
|
2018-05-08 16:34:08 -04:00
|
|
|
|
case 'ParadoxicalCondition':
|
|
|
|
|
$ignored_issues = ['MissingParamType'];
|
|
|
|
|
break;
|
|
|
|
|
|
2018-01-28 23:41:11 -05:00
|
|
|
|
case 'UnusedClass':
|
|
|
|
|
case 'UnusedMethod':
|
|
|
|
|
$ignored_issues = ['UnusedVariable'];
|
|
|
|
|
break;
|
|
|
|
|
|
2022-01-25 15:49:09 -06:00
|
|
|
|
case 'AmbiguousConstantInheritance':
|
2022-02-13 17:19:25 -06:00
|
|
|
|
case 'DeprecatedConstant':
|
2021-09-06 02:57:12 +03:00
|
|
|
|
case 'DuplicateEnumCase':
|
|
|
|
|
case 'DuplicateEnumCaseValue':
|
2022-02-13 17:19:25 -06:00
|
|
|
|
case 'InvalidEnumBackingType':
|
|
|
|
|
case 'InvalidEnumCaseValue':
|
2022-12-12 03:12:55 -04:00
|
|
|
|
case 'InvalidEnumMethod':
|
2021-11-06 22:22:38 +02:00
|
|
|
|
case 'NoEnumProperties':
|
2022-02-13 17:19:25 -06:00
|
|
|
|
case 'OverriddenFinalConstant':
|
2023-02-12 02:42:59 -04:00
|
|
|
|
case 'InvalidInterfaceImplementation':
|
2021-09-05 20:12:24 +03:00
|
|
|
|
$php_version = '8.1';
|
|
|
|
|
break;
|
2018-01-28 23:41:11 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$invalid_code_data[$issue_name] = [
|
2020-03-20 19:15:06 -04:00
|
|
|
|
$blocks[0],
|
2018-01-28 23:41:11 -05:00
|
|
|
|
$issue_name,
|
|
|
|
|
$ignored_issues,
|
2019-09-19 11:59:43 -04:00
|
|
|
|
strpos($issue_name, 'Unused') !== false
|
|
|
|
|
|| strpos($issue_name, 'Unevaluated') !== false
|
|
|
|
|
|| strpos($issue_name, 'Unnecessary') !== false,
|
2022-12-18 10:15:15 -06:00
|
|
|
|
$php_version,
|
2018-01-28 23:41:11 -05:00
|
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return $invalid_code_data;
|
|
|
|
|
}
|
2020-08-16 23:26:54 +03:00
|
|
|
|
|
|
|
|
|
public function testShortcodesAreUnique(): void
|
|
|
|
|
{
|
2021-12-03 20:11:20 +01:00
|
|
|
|
$all_issues = IssueHandler::getAllIssueTypes();
|
2020-08-16 23:26:54 +03:00
|
|
|
|
$all_shortcodes = [];
|
|
|
|
|
|
|
|
|
|
foreach ($all_issues as $issue_type) {
|
2023-02-15 03:02:34 -04:00
|
|
|
|
/** @var class-string $issue_class */
|
2020-08-16 23:26:54 +03:00
|
|
|
|
$issue_class = '\\Psalm\\Issue\\' . $issue_type;
|
|
|
|
|
/** @var int $shortcode */
|
|
|
|
|
$shortcode = $issue_class::SHORTCODE;
|
|
|
|
|
$all_shortcodes[$shortcode][] = $issue_type;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$duplicate_shortcodes = array_filter(
|
|
|
|
|
$all_shortcodes,
|
2023-10-21 19:02:17 +02:00
|
|
|
|
static fn($issues): bool => count($issues) > 1
|
2020-08-16 23:26:54 +03:00
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
$this->assertEquals(
|
|
|
|
|
[],
|
|
|
|
|
$duplicate_shortcodes,
|
2022-12-18 10:15:15 -06:00
|
|
|
|
"Duplicate shortcodes found: \n" . var_export($duplicate_shortcodes, true),
|
2020-08-16 23:26:54 +03:00
|
|
|
|
);
|
|
|
|
|
}
|
2020-11-28 04:48:16 +02:00
|
|
|
|
|
|
|
|
|
/** @dataProvider knownAnnotations */
|
|
|
|
|
public function testAllAnnotationsAreDocumented(string $annotation): void
|
|
|
|
|
{
|
|
|
|
|
if ('' === self::$docContents) {
|
|
|
|
|
foreach (self::ANNOTATION_DOCS as $file) {
|
2023-10-09 23:38:24 +01:00
|
|
|
|
$file_contents = file_get_contents(__DIR__ . '/../' . $file);
|
|
|
|
|
assert($file_contents !== false);
|
|
|
|
|
self::$docContents .= $file_contents;
|
2020-11-28 04:48:16 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$this->assertThat(
|
|
|
|
|
self::$docContents,
|
|
|
|
|
$this->conciseExpected($this->stringContains('@psalm-' . $annotation)),
|
2022-12-18 10:15:15 -06:00
|
|
|
|
"'@psalm-$annotation' is not present in the docs",
|
2020-11-28 04:48:16 +02:00
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2022-11-11 20:14:21 -05:00
|
|
|
|
/** @return iterable<string, array{string}> */
|
2020-11-28 04:48:16 +02:00
|
|
|
|
public function knownAnnotations(): iterable
|
|
|
|
|
{
|
|
|
|
|
foreach (DocComment::PSALM_ANNOTATIONS as $annotation) {
|
|
|
|
|
if (in_array('@psalm-' . $annotation, self::INTENTIONALLY_UNDOCUMENTED_ANNOTATIONS, true)) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (in_array('@psalm-' . $annotation, self::WALL_OF_SHAME, true)) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
yield $annotation => [$annotation];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Creates a constraint wrapper that displays the expected value in a concise form
|
|
|
|
|
*/
|
|
|
|
|
public function conciseExpected(Constraint $inner): Constraint
|
|
|
|
|
{
|
|
|
|
|
return new class ($inner) extends Constraint
|
|
|
|
|
{
|
2022-12-16 12:58:47 -06:00
|
|
|
|
private Constraint $inner;
|
2020-11-28 04:48:16 +02:00
|
|
|
|
|
|
|
|
|
public function __construct(Constraint $inner)
|
|
|
|
|
{
|
|
|
|
|
$this->inner = $inner;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public function toString(): string
|
|
|
|
|
{
|
|
|
|
|
return $this->inner->toString();
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-24 10:48:32 +02:00
|
|
|
|
protected function matches(mixed $other): bool
|
2020-11-28 04:48:16 +02:00
|
|
|
|
{
|
|
|
|
|
return $this->inner->matches($other);
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-24 10:48:32 +02:00
|
|
|
|
protected function failureDescription(mixed $other): string
|
2020-11-28 04:48:16 +02:00
|
|
|
|
{
|
|
|
|
|
return $this->exporter()->shortenedExport($other) . ' ' . $this->toString();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
2022-01-12 14:25:07 -06:00
|
|
|
|
|
2022-01-12 15:22:21 -06:00
|
|
|
|
/**
|
|
|
|
|
* Tests that issues.md contains the expected links to issue documentation.
|
|
|
|
|
* issues.md can be generated automatically with bin/generate_documentation_issues_list.php.
|
|
|
|
|
*/
|
2022-01-12 14:25:07 -06:00
|
|
|
|
public function testIssuesIndex(): void
|
|
|
|
|
{
|
2022-01-12 15:22:21 -06:00
|
|
|
|
$docs_dir = dirname(__DIR__) . DIRECTORY_SEPARATOR . "docs" . DIRECTORY_SEPARATOR . "running_psalm" . DIRECTORY_SEPARATOR;
|
2022-01-12 14:25:07 -06:00
|
|
|
|
$issues_index = "{$docs_dir}issues.md";
|
|
|
|
|
$issues_dir = "{$docs_dir}issues";
|
|
|
|
|
|
|
|
|
|
if (!file_exists($issues_dir)) {
|
|
|
|
|
throw new UnexpectedValueException("Issues documentation not found");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!file_exists($issues_index)) {
|
|
|
|
|
throw new UnexpectedValueException("Issues index not found");
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-12 03:12:55 -04:00
|
|
|
|
$issues_index_contents = file($issues_index, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES);
|
2022-10-03 15:13:47 +02:00
|
|
|
|
if ($issues_index_contents === false) {
|
|
|
|
|
throw new UnexpectedValueException("Issues index returned false");
|
|
|
|
|
}
|
2022-01-12 14:25:07 -06:00
|
|
|
|
array_shift($issues_index_contents); // Remove title
|
|
|
|
|
|
|
|
|
|
$issues_index_list = array_map(function (string $issues_line) {
|
|
|
|
|
preg_match('/^ - \[([^\]]*)\]\(issues\/\1\.md\)$/', $issues_line, $matches);
|
|
|
|
|
$this->assertCount(2, $matches, "Invalid format in issues index: $issues_line");
|
|
|
|
|
return $matches[1];
|
|
|
|
|
}, $issues_index_contents);
|
|
|
|
|
|
2023-10-09 23:38:24 +01:00
|
|
|
|
$dir_contents = scandir($issues_dir);
|
|
|
|
|
assert($dir_contents !== false);
|
2022-01-12 14:25:07 -06:00
|
|
|
|
$issue_files = array_filter(array_map(function (string $issue_file) {
|
|
|
|
|
if ($issue_file === "." || $issue_file === "..") {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
$this->assertStringEndsWith(".md", $issue_file, "Invalid file in issues documentation: $issue_file");
|
|
|
|
|
return substr($issue_file, 0, strlen($issue_file) - 3);
|
2023-10-09 23:38:24 +01:00
|
|
|
|
}, $dir_contents));
|
2022-01-12 14:25:07 -06:00
|
|
|
|
|
|
|
|
|
$unlisted_issues = array_diff($issue_files, $issues_index_list);
|
|
|
|
|
$this->assertEmpty($unlisted_issues, "Issue documentation missing from issues.md: " . implode(", ", $unlisted_issues));
|
|
|
|
|
|
|
|
|
|
$missing_documentation = array_diff($issues_index_list, $issue_files);
|
|
|
|
|
$this->assertEmpty($missing_documentation, "issues.md has link to non-existent documentation for: " . implode(", ", $missing_documentation));
|
|
|
|
|
|
|
|
|
|
$sorted = $issues_index_list;
|
|
|
|
|
usort($sorted, "strcasecmp");
|
|
|
|
|
for ($i = 0; $i < count($sorted); ++$i) {
|
|
|
|
|
$this->assertEquals($sorted[$i], $issues_index_list[$i], "issues.md out of order, expected {$sorted[$i]} before {$issues_index_list[$i]}");
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-01-28 23:41:11 -05:00
|
|
|
|
}
|