mirror of
https://github.com/danog/psalm.git
synced 2025-01-05 20:48:45 +01:00
282 lines
8.4 KiB
PHP
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 = [];
|
|
}
|
|
}
|