2018-10-29 17:15:36 +01:00
|
|
|
<?php
|
2018-11-06 03:57:36 +01:00
|
|
|
namespace Psalm\Example\Plugin\ComposerBased;
|
2018-10-29 17:15:36 +01:00
|
|
|
|
|
|
|
use PhpParser;
|
2018-11-06 03:57:36 +01:00
|
|
|
use Psalm\Codebase;
|
2018-10-29 17:15:36 +01:00
|
|
|
use Psalm\CodeLocation;
|
|
|
|
use Psalm\Context;
|
2018-11-12 16:46:55 +01:00
|
|
|
use Psalm\FileManipulation;
|
2018-10-29 17:15:36 +01:00
|
|
|
use Psalm\IssueBuffer;
|
2019-04-26 00:02:19 +02:00
|
|
|
use Psalm\Issue\ArgumentTypeCoercion;
|
2018-11-12 16:11:08 +01:00
|
|
|
use Psalm\Plugin\Hook\AfterStatementAnalysisInterface;
|
2018-11-06 03:57:36 +01:00
|
|
|
use Psalm\StatementsSource;
|
2018-10-29 17:15:36 +01:00
|
|
|
|
2018-11-12 16:11:08 +01:00
|
|
|
class EchoChecker implements AfterStatementAnalysisInterface
|
2018-10-29 17:15:36 +01:00
|
|
|
{
|
|
|
|
/**
|
2018-11-06 03:57:36 +01:00
|
|
|
* Called after a statement has been checked
|
2018-10-29 17:15:36 +01:00
|
|
|
*
|
|
|
|
* @param FileManipulation[] $file_replacements
|
|
|
|
*
|
|
|
|
* @return null|false
|
|
|
|
*/
|
2018-11-06 03:57:36 +01:00
|
|
|
public static function afterStatementAnalysis(
|
|
|
|
PhpParser\Node\Stmt $stmt,
|
2018-10-29 17:15:36 +01:00
|
|
|
Context $context,
|
2018-11-06 03:57:36 +01:00
|
|
|
StatementsSource $statements_source,
|
|
|
|
Codebase $codebase,
|
2018-10-29 17:15:36 +01:00
|
|
|
array &$file_replacements = []
|
2020-10-12 21:02:52 +02:00
|
|
|
): ?bool {
|
2018-10-29 17:15:36 +01:00
|
|
|
if ($stmt instanceof PhpParser\Node\Stmt\Echo_) {
|
|
|
|
foreach ($stmt->exprs as $expr) {
|
2019-11-25 17:44:54 +01:00
|
|
|
$expr_type = $statements_source->getNodeTypeProvider()->getType($expr);
|
|
|
|
|
|
|
|
if (!$expr_type || $expr_type->hasMixed()) {
|
2018-10-29 17:15:36 +01:00
|
|
|
if (IssueBuffer::accepts(
|
2019-04-26 00:02:19 +02:00
|
|
|
new ArgumentTypeCoercion(
|
2019-11-25 17:44:54 +01:00
|
|
|
'Echo requires an unescaped string, ' . $expr_type . ' provided',
|
2019-04-26 00:02:19 +02:00
|
|
|
new CodeLocation($statements_source, $expr),
|
|
|
|
'echo'
|
2018-10-29 17:15:36 +01:00
|
|
|
),
|
2018-11-06 03:57:36 +01:00
|
|
|
$statements_source->getSuppressedIssues()
|
2018-10-29 17:15:36 +01:00
|
|
|
)) {
|
|
|
|
// keep soldiering on
|
|
|
|
}
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-01-04 18:20:26 +01:00
|
|
|
$types = $expr_type->getAtomicTypes();
|
2018-10-29 17:15:36 +01:00
|
|
|
|
|
|
|
foreach ($types as $type) {
|
|
|
|
if ($type instanceof \Psalm\Type\Atomic\TString
|
|
|
|
&& !$type instanceof \Psalm\Type\Atomic\TLiteralString
|
|
|
|
&& !$type instanceof \Psalm\Type\Atomic\THtmlEscapedString
|
|
|
|
) {
|
|
|
|
if (IssueBuffer::accepts(
|
2019-04-26 00:02:19 +02:00
|
|
|
new ArgumentTypeCoercion(
|
2019-11-25 17:44:54 +01:00
|
|
|
'Echo requires an unescaped string, ' . $expr_type . ' provided',
|
2019-04-26 00:02:19 +02:00
|
|
|
new CodeLocation($statements_source, $expr),
|
|
|
|
'echo'
|
2018-10-29 17:15:36 +01:00
|
|
|
),
|
2018-11-06 03:57:36 +01:00
|
|
|
$statements_source->getSuppressedIssues()
|
2018-10-29 17:15:36 +01:00
|
|
|
)) {
|
|
|
|
// keep soldiering on
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-10-12 21:02:52 +02:00
|
|
|
|
|
|
|
return null;
|
2018-10-29 17:15:36 +01:00
|
|
|
}
|
|
|
|
}
|