forked from phpstan/phpstan-src
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathInterpolatedStringHandler.php
More file actions
93 lines (80 loc) · 2.91 KB
/
InterpolatedStringHandler.php
File metadata and controls
93 lines (80 loc) · 2.91 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
<?php declare(strict_types = 1);
namespace PHPStan\Analyser\ExprHandler;
use PhpParser\Node\Expr;
use PhpParser\Node\InterpolatedStringPart;
use PhpParser\Node\Scalar\InterpolatedString;
use PhpParser\Node\Stmt;
use PHPStan\Analyser\ExpressionContext;
use PHPStan\Analyser\ExpressionResult;
use PHPStan\Analyser\ExpressionResultStorage;
use PHPStan\Analyser\ExprHandler;
use PHPStan\Analyser\ExprHandler\Helper\ImplicitToStringCallHelper;
use PHPStan\Analyser\MutatingScope;
use PHPStan\Analyser\NodeScopeResolver;
use PHPStan\DependencyInjection\AutowiredService;
use PHPStan\Reflection\InitializerExprTypeResolver;
use PHPStan\Type\Constant\ConstantStringType;
use PHPStan\Type\Type;
use function array_merge;
/**
* @implements ExprHandler<InterpolatedString>
*/
#[AutowiredService]
final class InterpolatedStringHandler implements ExprHandler
{
public function __construct(
private InitializerExprTypeResolver $initializerExprTypeResolver,
private ImplicitToStringCallHelper $implicitToStringCallHelper,
)
{
}
public function supports(Expr $expr): bool
{
return $expr instanceof InterpolatedString;
}
public function processExpr(NodeScopeResolver $nodeScopeResolver, Stmt $stmt, Expr $expr, MutatingScope $scope, ExpressionResultStorage $storage, callable $nodeCallback, ExpressionContext $context): ExpressionResult
{
$hasYield = false;
$throwPoints = [];
$impurePoints = [];
$isAlwaysTerminating = false;
foreach ($expr->parts as $part) {
if (!$part instanceof Expr) {
continue;
}
$partResult = $nodeScopeResolver->processExprNode($stmt, $part, $scope, $storage, $nodeCallback, $context->enterDeep());
$hasYield = $hasYield || $partResult->hasYield();
$throwPoints = array_merge($throwPoints, $partResult->getThrowPoints());
$impurePoints = array_merge($impurePoints, $partResult->getImpurePoints());
$toStringResult = $this->implicitToStringCallHelper->processImplicitToStringCall($part, $scope);
$throwPoints = array_merge($throwPoints, $toStringResult->getThrowPoints());
$impurePoints = array_merge($impurePoints, $toStringResult->getImpurePoints());
$isAlwaysTerminating = $isAlwaysTerminating || $partResult->isAlwaysTerminating();
$scope = $partResult->getScope();
}
return new ExpressionResult(
$scope,
hasYield: $hasYield,
isAlwaysTerminating: $isAlwaysTerminating,
throwPoints: $throwPoints,
impurePoints: $impurePoints,
);
}
public function resolveType(MutatingScope $scope, Expr $expr): Type
{
$resultType = null;
foreach ($expr->parts as $part) {
if ($part instanceof InterpolatedStringPart) {
$partType = new ConstantStringType($part->value);
} else {
$partType = $scope->getType($part)->toString();
}
if ($resultType === null) {
$resultType = $partType;
continue;
}
$resultType = $this->initializerExprTypeResolver->resolveConcatType($resultType, $partType);
}
return $resultType ?? new ConstantStringType('');
}
}