2018-01-14 18:09:40 +01:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Checker\Statements;
|
|
|
|
|
|
|
|
use PhpParser;
|
|
|
|
use Psalm\Checker\StatementsChecker;
|
|
|
|
use Psalm\Checker\TypeChecker;
|
|
|
|
use Psalm\CodeLocation;
|
|
|
|
use Psalm\Context;
|
|
|
|
use Psalm\Issue\InvalidThrow;
|
|
|
|
use Psalm\IssueBuffer;
|
|
|
|
use Psalm\Type\Atomic\TNamedObject;
|
|
|
|
use Psalm\Type\Union;
|
|
|
|
|
|
|
|
class ThrowChecker
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @return false|null
|
|
|
|
*/
|
|
|
|
public static function analyze(
|
|
|
|
StatementsChecker $statements_checker,
|
|
|
|
PhpParser\Node\Stmt\Throw_ $stmt,
|
|
|
|
Context $context
|
|
|
|
) {
|
|
|
|
if (ExpressionChecker::analyze($statements_checker, $stmt->expr, $context) === false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($context->check_classes && isset($stmt->expr->inferredType) && !$stmt->expr->inferredType->isMixed()) {
|
|
|
|
$throw_type = $stmt->expr->inferredType;
|
|
|
|
|
|
|
|
$exception_type = new Union([new TNamedObject('Exception'), new TNamedObject('Throwable')]);
|
|
|
|
|
|
|
|
$file_checker = $statements_checker->getFileChecker();
|
|
|
|
$project_checker = $file_checker->project_checker;
|
|
|
|
|
2018-02-01 06:50:01 +01:00
|
|
|
if (!TypeChecker::isContainedBy($project_checker->codebase, $throw_type, $exception_type)) {
|
2018-01-14 18:09:40 +01:00
|
|
|
if (IssueBuffer::accepts(
|
|
|
|
new InvalidThrow(
|
|
|
|
'Cannot throw ' . $throw_type . ' as it does not extend Exception or implement Throwable',
|
|
|
|
new CodeLocation($file_checker, $stmt)
|
|
|
|
),
|
|
|
|
$statements_checker->getSuppressedIssues()
|
|
|
|
)) {
|
|
|
|
return false;
|
|
|
|
}
|
2018-06-22 07:13:49 +02:00
|
|
|
} elseif ($context->collect_exceptions) {
|
|
|
|
foreach ($throw_type->getTypes() as $throw_atomic_type) {
|
|
|
|
if ($throw_atomic_type instanceof TNamedObject) {
|
|
|
|
$context->possibly_thrown_exceptions[$throw_atomic_type->value] = true;
|
|
|
|
}
|
|
|
|
}
|
2018-01-14 18:09:40 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|