1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-07 13:42:11 +01:00
psalm/examples/plugins/FunctionCasingChecker.php
2020-10-14 18:51:15 -04:00

123 lines
3.9 KiB
PHP

<?php
namespace Psalm\Example\Plugin;
use PhpParser;
use PhpParser\Node\Expr;
use PhpParser\Node\Expr\FuncCall;
use PhpParser\Node\Expr\MethodCall;
use PhpParser\Node\Expr\StaticCall;
use Psalm\Checker;
use Psalm\Checker\StatementsChecker;
use Psalm\Codebase;
use Psalm\CodeLocation;
use Psalm\Context;
use Psalm\FileManipulation;
use Psalm\Plugin\Hook\AfterFunctionCallAnalysisInterface;
use Psalm\Plugin\Hook\AfterMethodCallAnalysisInterface;
use Psalm\StatementsSource;
use Psalm\Type\Union;
/**
* Prevents any assignment to a float value
*/
class FunctionCasingChecker implements AfterFunctionCallAnalysisInterface, AfterMethodCallAnalysisInterface
{
/**
* @param MethodCall|StaticCall $expr
* @param FileManipulation[] $file_replacements
*/
public static function afterMethodCallAnalysis(
Expr $expr,
string $method_id,
string $appearing_method_id,
string $declaring_method_id,
Context $context,
StatementsSource $statements_source,
Codebase $codebase,
array &$file_replacements = [],
Union &$return_type_candidate = null
): void {
if (!$expr->name instanceof PhpParser\Node\Identifier) {
return;
}
try {
/** @psalm-suppress ArgumentTypeCoercion */
$method_id = new \Psalm\Internal\MethodIdentifier(...explode('::', $declaring_method_id));
$function_storage = $codebase->methods->getStorage($method_id);
if ($function_storage->cased_name === '__call') {
return;
}
if ($function_storage->cased_name === '__callStatic') {
return;
}
if ($function_storage->cased_name !== (string)$expr->name) {
if (\Psalm\IssueBuffer::accepts(
new IncorrectFunctionCasing(
'Function is incorrectly cased, expecting ' . $function_storage->cased_name,
new CodeLocation($statements_source, $expr->name)
),
$statements_source->getSuppressedIssues()
)) {
// fall through
}
}
} catch (\Exception $e) {
// can throw if storage is missing
}
}
/**
* @param non-empty-string $function_id
* @param FileManipulation[] $file_replacements
*/
public static function afterFunctionCallAnalysis(
FuncCall $expr,
string $function_id,
Context $context,
StatementsSource $statements_source,
Codebase $codebase,
Union $return_type_candidate,
array &$file_replacements
): void {
if ($expr->name instanceof PhpParser\Node\Expr) {
return;
}
try {
$function_storage = $codebase->functions->getStorage(
$statements_source instanceof \Psalm\Internal\Analyzer\StatementsAnalyzer
? $statements_source
: null,
strtolower($function_id)
);
if (!$function_storage->cased_name) {
return;
}
$function_name_parts = explode('\\', $function_storage->cased_name);
if (end($function_name_parts) !== end($expr->name->parts)) {
if (\Psalm\IssueBuffer::accepts(
new IncorrectFunctionCasing(
'Function is incorrectly cased, expecting ' . $function_storage->cased_name,
new CodeLocation($statements_source, $expr->name)
),
$statements_source->getSuppressedIssues()
)) {
// fall through
}
}
} catch (\Exception $e) {
// can throw if storage is missing
}
}
}
class IncorrectFunctionCasing extends \Psalm\Issue\PluginIssue {
}