1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-05 20:48:45 +01:00
psalm/src/Psalm/Internal/FileManipulation/PropertyDocblockManipulator.php
2022-01-03 04:09:59 +02:00

282 lines
8.4 KiB
PHP

<?php
namespace Psalm\Internal\FileManipulation;
use PhpParser\Node\Stmt\Property;
use Psalm\Config;
use Psalm\DocComment;
use Psalm\FileManipulation;
use Psalm\Internal\Analyzer\ProjectAnalyzer;
use Psalm\Internal\Scanner\ParsedDocblock;
use UnexpectedValueException;
use function array_shift;
use function count;
use function ltrim;
use function str_replace;
use function strlen;
use function strrpos;
use function substr;
use function substr_count;
/**
* @internal
*/
class PropertyDocblockManipulator
{
/**
* @var array<string, array<int, self>>
*/
private static $manipulators = [];
/** @var Property */
private $stmt;
/** @var int */
private $docblock_start;
/** @var int */
private $docblock_end;
/** @var null|int */
private $typehint_start;
/** @var int */
private $typehint_area_start;
/** @var null|int */
private $typehint_end;
/** @var null|string */
private $new_php_type;
/** @var bool */
private $type_is_php_compatible = false;
/** @var null|string */
private $new_phpdoc_type;
/** @var null|string */
private $new_psalm_type;
/** @var string */
private $indentation;
/** @var bool */
private $add_newline = false;
/** @var string|null */
private $type_description;
public static function getForProperty(
ProjectAnalyzer $project_analyzer,
string $file_path,
Property $stmt
): self {
if (isset(self::$manipulators[$file_path][$stmt->getLine()])) {
return self::$manipulators[$file_path][$stmt->getLine()];
}
$manipulator
= self::$manipulators[$file_path][$stmt->getLine()]
= new self($project_analyzer, $stmt, $file_path);
return $manipulator;
}
private function __construct(
ProjectAnalyzer $project_analyzer,
Property $stmt,
string $file_path
) {
$this->stmt = $stmt;
$docblock = $stmt->getDocComment();
$this->docblock_start = $docblock ? $docblock->getStartFilePos() : (int)$stmt->getAttribute('startFilePos');
$this->docblock_end = (int)$stmt->getAttribute('startFilePos');
$codebase = $project_analyzer->getCodebase();
$file_contents = $codebase->getFileContents($file_path);
if (count($stmt->props) > 1) {
$config = Config::getInstance();
if ($config->isInProjectDirs($file_path)) {
throw new UnexpectedValueException('Cannot replace multiple inline properties in ' . $file_path);
}
$this->indentation = '';
return;
}
$prop = $stmt->props[0];
if ($stmt->type) {
$this->typehint_start = (int)$stmt->type->getAttribute('startFilePos');
$this->typehint_end = (int)$stmt->type->getAttribute('endFilePos');
}
$this->typehint_area_start = (int)$prop->getAttribute('startFilePos') - 1;
$preceding_newline_pos = strrpos($file_contents, "\n", $this->docblock_end - strlen($file_contents));
if ($preceding_newline_pos === false) {
$this->indentation = '';
return;
}
if (!$docblock) {
$preceding_semicolon_pos = strrpos($file_contents, ";", $preceding_newline_pos - strlen($file_contents));
if ($preceding_semicolon_pos) {
$preceding_space = substr(
$file_contents,
$preceding_semicolon_pos + 1,
$preceding_newline_pos - $preceding_semicolon_pos - 1
);
if (!substr_count($preceding_space, "\n")) {
$this->add_newline = true;
}
}
}
$first_line = substr($file_contents, $preceding_newline_pos + 1, $this->docblock_end - $preceding_newline_pos);
$this->indentation = str_replace(ltrim($first_line), '', $first_line);
}
public function setType(
?string $php_type,
string $new_type,
string $phpdoc_type,
bool $is_php_compatible,
?string $description = null
): void {
$new_type = str_replace(['<mixed, mixed>', '<array-key, mixed>', '<never, never>'], '', $new_type);
$this->new_php_type = $php_type;
$this->new_phpdoc_type = $phpdoc_type;
$this->new_psalm_type = $new_type;
$this->type_is_php_compatible = $is_php_compatible;
$this->type_description = $description;
}
/**
* Gets a new docblock given the existing docblock, if one exists, and the updated return types
* and/or parameters
*
*/
private function getDocblock(): string
{
$docblock = $this->stmt->getDocComment();
if ($docblock) {
$parsed_docblock = DocComment::parsePreservingLength($docblock);
} else {
$parsed_docblock = new ParsedDocblock('', []);
}
$modified_docblock = false;
$old_phpdoc_type = null;
if (isset($parsed_docblock->tags['var'])) {
$old_phpdoc_type = array_shift($parsed_docblock->tags['var']);
}
if ($this->new_phpdoc_type
&& $this->new_phpdoc_type !== $old_phpdoc_type
) {
$modified_docblock = true;
$parsed_docblock->tags['var'] = [
$this->new_phpdoc_type
. ($this->type_description ? (' ' . $this->type_description) : ''),
];
}
$old_psalm_type = null;
if (isset($parsed_docblock->tags['psalm-var'])) {
$old_psalm_type = array_shift($parsed_docblock->tags['psalm-var']);
}
if ($this->new_psalm_type
&& $this->new_phpdoc_type !== $this->new_psalm_type
&& $this->new_psalm_type !== $old_psalm_type
) {
$modified_docblock = true;
$parsed_docblock->tags['psalm-var'] = [$this->new_psalm_type];
}
if (!$parsed_docblock->tags && !$parsed_docblock->description) {
return '';
}
if (!$modified_docblock) {
return (string)$docblock . "\n" . $this->indentation;
}
return $parsed_docblock->render($this->indentation);
}
/**
* @return array<int, FileManipulation>
*/
public static function getManipulationsForFile(string $file_path): array
{
if (!isset(self::$manipulators[$file_path])) {
return [];
}
$file_manipulations = [];
foreach (self::$manipulators[$file_path] as $manipulator) {
if ($manipulator->new_php_type) {
if ($manipulator->typehint_start && $manipulator->typehint_end) {
$file_manipulations[$manipulator->typehint_start] = new FileManipulation(
$manipulator->typehint_start,
$manipulator->typehint_end,
$manipulator->new_php_type
);
} else {
$file_manipulations[$manipulator->typehint_area_start] = new FileManipulation(
$manipulator->typehint_area_start,
$manipulator->typehint_area_start,
' ' . $manipulator->new_php_type
);
}
} elseif ($manipulator->new_php_type === ''
&& $manipulator->new_phpdoc_type
&& $manipulator->typehint_start
&& $manipulator->typehint_end
) {
$file_manipulations[$manipulator->typehint_start] = new FileManipulation(
$manipulator->typehint_start,
$manipulator->typehint_end,
''
);
}
if (!$manipulator->new_php_type
|| !$manipulator->type_is_php_compatible
|| $manipulator->docblock_start !== $manipulator->docblock_end
) {
$file_manipulations[$manipulator->docblock_start] = new FileManipulation(
$manipulator->docblock_start
- ($manipulator->add_newline ? strlen($manipulator->indentation) : 0),
$manipulator->docblock_end,
($manipulator->add_newline ? "\n" . $manipulator->indentation : '')
. $manipulator->getDocblock()
);
}
}
return $file_manipulations;
}
public static function clearCache(): void
{
self::$manipulators = [];
}
}