Skip to content

Commit 5909fb2

Browse files
committed
Debugging function - PHPStan\debugScope()
1 parent c50b71f commit 5909fb2

File tree

8 files changed

+157
-2
lines changed

8 files changed

+157
-2
lines changed

composer.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -132,7 +132,7 @@
132132
"src/"
133133
]
134134
},
135-
"files": ["src/dumpType.php", "src/autoloadFunctions.php", "src/Testing/functions.php"]
135+
"files": ["src/debugScope.php", "src/dumpType.php", "src/autoloadFunctions.php", "src/Testing/functions.php"]
136136
},
137137
"autoload-dev": {
138138
"psr-4": {

conf/config.neon

+1
Original file line numberDiff line numberDiff line change
@@ -278,6 +278,7 @@ extensions:
278278
validateExcludePaths: PHPStan\DependencyInjection\ValidateExcludePathsExtension
279279

280280
rules:
281+
- PHPStan\Rules\Debug\DebugScopeRule
281282
- PHPStan\Rules\Debug\DumpTypeRule
282283
- PHPStan\Rules\Debug\FileAssertRule
283284

src/Analyser/MutatingScope.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -5400,7 +5400,7 @@ public function debug(): array
54005400
$descriptions[$key] = $variableTypeHolder->getType()->describe(VerbosityLevel::precise());
54015401
}
54025402
foreach ($this->nativeExpressionTypes as $exprString => $nativeTypeHolder) {
5403-
$key = sprintf('native %s', $exprString);
5403+
$key = sprintf('native %s (%s)', $exprString, $nativeTypeHolder->getCertainty()->describe());
54045404
$descriptions[$key] = $nativeTypeHolder->getType()->describe(VerbosityLevel::precise());
54055405
}
54065406

src/Rules/Debug/DebugScopeRule.php

+66
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,66 @@
1+
<?php declare(strict_types = 1);
2+
3+
namespace PHPStan\Rules\Debug;
4+
5+
use PhpParser\Node;
6+
use PHPStan\Analyser\MutatingScope;
7+
use PHPStan\Analyser\Scope;
8+
use PHPStan\Reflection\ReflectionProvider;
9+
use PHPStan\Rules\Rule;
10+
use PHPStan\Rules\RuleErrorBuilder;
11+
use function count;
12+
use function implode;
13+
use function sprintf;
14+
use function strtolower;
15+
16+
/**
17+
* @implements Rule<Node\Expr\FuncCall>
18+
*/
19+
final class DebugScopeRule implements Rule
20+
{
21+
22+
public function __construct(private ReflectionProvider $reflectionProvider)
23+
{
24+
}
25+
26+
public function getNodeType(): string
27+
{
28+
return Node\Expr\FuncCall::class;
29+
}
30+
31+
public function processNode(Node $node, Scope $scope): array
32+
{
33+
if (!$node->name instanceof Node\Name) {
34+
return [];
35+
}
36+
37+
$functionName = $this->reflectionProvider->resolveFunctionName($node->name, $scope);
38+
if ($functionName === null) {
39+
return [];
40+
}
41+
42+
if (strtolower($functionName) !== 'phpstan\debugscope') {
43+
return [];
44+
}
45+
46+
if (!$scope instanceof MutatingScope) {
47+
return [];
48+
}
49+
50+
$parts = [];
51+
foreach ($scope->debug() as $key => $row) {
52+
$parts[] = sprintf('%s: %s', $key, $row);
53+
}
54+
55+
if (count($parts) === 0) {
56+
$parts[] = 'Scope is empty';
57+
}
58+
59+
return [
60+
RuleErrorBuilder::message(
61+
implode("\n", $parts),
62+
)->nonIgnorable()->identifier('phpstan.debugScope')->build(),
63+
];
64+
}
65+
66+
}

src/Rules/Functions/CallToFunctionStatementWithoutSideEffectsRule.php

+1
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@ final class CallToFunctionStatementWithoutSideEffectsRule implements Rule
2929

3030
public const PHPSTAN_TESTING_FUNCTIONS = [
3131
'PHPStan\\dumpType',
32+
'PHPStan\\debugScope',
3233
'PHPStan\\Testing\\assertType',
3334
'PHPStan\\Testing\\assertNativeType',
3435
'PHPStan\\Testing\\assertVariableCertainty',

src/debugScope.php

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
<?php declare(strict_types = 1);
2+
3+
namespace PHPStan;
4+
5+
/**
6+
* @phpstan-pure
7+
* @return mixed
8+
*
9+
* @throws void
10+
*/
11+
function debugScope() // phpcs:ignore Squiz.Functions.GlobalFunction.Found
12+
{
13+
return null;
14+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
1+
<?php declare(strict_types = 1);
2+
3+
namespace PHPStan\Rules\Debug;
4+
5+
use PHPStan\Rules\Rule;
6+
use PHPStan\Testing\RuleTestCase;
7+
use function implode;
8+
9+
/**
10+
* @extends RuleTestCase<DebugScopeRule>
11+
*/
12+
class DebugScopeRuleTest extends RuleTestCase
13+
{
14+
15+
protected function getRule(): Rule
16+
{
17+
return new DebugScopeRule($this->createReflectionProvider());
18+
}
19+
20+
public function testRuleInPhpStanNamespace(): void
21+
{
22+
$this->analyse([__DIR__ . '/data/debug-scope.php'], [
23+
[
24+
'Scope is empty',
25+
7,
26+
],
27+
[
28+
implode("\n", [
29+
'$a (Yes): int',
30+
'$b (Yes): int',
31+
'$debug (Yes): bool',
32+
'native $a (Yes): int',
33+
'native $b (Yes): int',
34+
'native $debug (Yes): bool',
35+
]),
36+
10,
37+
],
38+
[
39+
implode("\n", [
40+
'$a (Yes): int',
41+
'$b (Yes): int',
42+
'$debug (Yes): bool',
43+
'$c (Maybe): 1',
44+
'native $a (Yes): int',
45+
'native $b (Yes): int',
46+
'native $debug (Yes): bool',
47+
'native $c (Maybe): 1',
48+
'condition about $c #1: if $debug=false then $c is *ERROR* (No)',
49+
'condition about $c #2: if $debug=true then $c is 1 (Yes)',
50+
]),
51+
16,
52+
],
53+
]);
54+
}
55+
56+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
<?php
2+
3+
namespace DebugScope;
4+
5+
use function PHPStan\debugScope;
6+
7+
debugScope();
8+
9+
function (int $a, int $b, bool $debug): void {
10+
debugScope();
11+
12+
if ($debug) {
13+
$c = 1;
14+
}
15+
16+
debugScope();
17+
};

0 commit comments

Comments
 (0)