Skip to content

Commit 51f37a1

Browse files
authored
[Fun] calculate expected types per stage for pipe (#6)
1 parent 99304b6 commit 51f37a1

File tree

7 files changed

+299
-8
lines changed

7 files changed

+299
-8
lines changed

.github/workflows/coding-standards.yml

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
name: "coding standards"
22

3-
on:
3+
on:
44
pull_request: ~
55
push: ~
66

@@ -15,7 +15,7 @@ jobs:
1515
- name: "installing PHP"
1616
uses: "shivammathur/setup-php@v2"
1717
with:
18-
php-version: "7.4"
18+
php-version: "8.1"
1919
ini-values: memory_limit=-1
2020
tools: composer:v2, cs2pr
2121
extensions: bcmath, mbstring, intl, sodium, json
@@ -27,4 +27,4 @@ jobs:
2727
run: "php vendor/bin/phpcs"
2828

2929
- name: "checking coding standards ( php-cs-fixer )"
30-
run: "php vendor/bin/php-cs-fixer fix --dry-run --diff --ansi"
30+
run: "PHP_CS_FIXER_IGNORE_ENV=1 php vendor/bin/php-cs-fixer fix --dry-run --diff --ansi"

.github/workflows/static-analysis.yml

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
name: "static analysis"
22

3-
on:
3+
on:
44
pull_request: ~
55
push: ~
66
schedule:
@@ -17,7 +17,7 @@ jobs:
1717
- name: "installing PHP"
1818
uses: "shivammathur/setup-php@v2"
1919
with:
20-
php-version: "7.4"
20+
php-version: "8.1"
2121
ini-values: memory_limit=-1
2222
tools: composer:v2, cs2pr
2323
extensions: bcmath, mbstring, intl, sodium, json

README.md

+7
Original file line numberDiff line numberDiff line change
@@ -49,6 +49,13 @@ Psalm assumes that `$input` is of type `array<"age"|"location"|"name", array<"ci
4949
If we enable the `php-standard-library/psalm-plugin` plugin, you will get a more specific
5050
and correct type of `array{name: string, age: int, location?: array{city: string, state: string, country: string}}`.
5151

52+
## Compatibility
53+
54+
| PSL | Psalm plugin |
55+
|-----|--------------|
56+
| 2.x | 2.x |
57+
| 1.x | 1.x |
58+
5259
## Sponsors
5360

5461
Thanks to our sponsors and supporters:

composer.json

+5-2
Original file line numberDiff line numberDiff line change
@@ -10,9 +10,12 @@
1010
}
1111
],
1212
"require": {
13-
"php": "^7.4 || ^8.0",
13+
"php": "^8.1",
1414
"vimeo/psalm": "^4.6"
1515
},
16+
"conflict": {
17+
"azjezz/psl": "<2.0"
18+
},
1619
"require-dev": {
1720
"friendsofphp/php-cs-fixer": "^2.18",
1821
"roave/security-advisories": "dev-master",
@@ -52,4 +55,4 @@
5255
}
5356
},
5457
"minimum-stability": "dev"
55-
}
58+
}

psalm.xml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
<?xml version="1.0"?>
2-
<psalm totallyTyped="true" resolveFromConfigFile="true" forbidEcho="true" strictBinaryOperands="true" phpVersion="7.4" allowPhpStormGenerics="true" allowStringToStandInForClass="true" rememberPropertyAssignmentsAfterCall="false" skipChecksOnUnresolvableIncludes="false" checkForThrowsDocblock="true" checkForThrowsInGlobalScope="true" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="https://getpsalm.org/schema/config" xsi:schemaLocation="https://getpsalm.org/schema/config vendor/vimeo/psalm/config.xsd">
2+
<psalm totallyTyped="true" resolveFromConfigFile="true" forbidEcho="true" strictBinaryOperands="true" phpVersion="8.1" allowStringToStandInForClass="true" rememberPropertyAssignmentsAfterCall="false" skipChecksOnUnresolvableIncludes="false" checkForThrowsDocblock="true" checkForThrowsInGlobalScope="true" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="https://getpsalm.org/schema/config" xsi:schemaLocation="https://getpsalm.org/schema/config vendor/vimeo/psalm/config.xsd">
33
<projectFiles>
44
<directory name="src" />
55
<ignoreFiles>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,278 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Psl\Psalm\EventHandler\Fun\Pipe;
6+
7+
use Closure;
8+
use PhpParser\Node\Arg;
9+
use PhpParser\Node\ComplexType;
10+
use PhpParser\Node\Expr;
11+
use PhpParser\Node\FunctionLike;
12+
use PhpParser\Node\Identifier;
13+
use PhpParser\Node\Name;
14+
use PhpParser\Node\Param;
15+
use PhpParser\Node\Stmt\Return_;
16+
use PhpParser\NodeAbstract;
17+
use Psalm\CodeLocation;
18+
use Psalm\Issue\TooFewArguments;
19+
use Psalm\Issue\TooManyArguments;
20+
use Psalm\IssueBuffer;
21+
use Psalm\Plugin\EventHandler\Event\FunctionParamsProviderEvent;
22+
use Psalm\Plugin\EventHandler\Event\FunctionReturnTypeProviderEvent;
23+
use Psalm\Plugin\EventHandler\FunctionParamsProviderInterface;
24+
use Psalm\Plugin\EventHandler\FunctionReturnTypeProviderInterface;
25+
use Psalm\StatementsSource;
26+
use Psalm\Storage\FunctionLikeParameter;
27+
use Psalm\Type;
28+
29+
/**
30+
* @psalm-type Stage = array{0: Type\Union, 1: Type\Union, 2: string}
31+
* @psalm-type StagesOrEmpty = list<Stage>
32+
* @psalm-type Stages = non-empty-list<Stage>
33+
*/
34+
class PipeArgumentsProvider implements FunctionParamsProviderInterface, FunctionReturnTypeProviderInterface
35+
{
36+
/**
37+
* @return array<lowercase-string>
38+
*/
39+
public static function getFunctionIds(): array
40+
{
41+
return [
42+
'psl\fun\pipe'
43+
];
44+
}
45+
46+
/**
47+
* @return list<FunctionLikeParameter>|null
48+
*/
49+
public static function getFunctionParams(FunctionParamsProviderEvent $event): ?array
50+
{
51+
$stages = self::parseStages($event->getStatementsSource(), $event->getCallArgs());
52+
if (!$stages) {
53+
return [];
54+
}
55+
56+
$params = [];
57+
$previousOut = self::pipeInputType($stages);
58+
59+
foreach ($stages as $stage) {
60+
[$_, $currentOut, $paramName] = $stage;
61+
62+
$params[] = self::createFunctionParameter(
63+
'stages',
64+
self::createClosureStage($previousOut, $currentOut, $paramName)
65+
);
66+
67+
$previousOut = $currentOut;
68+
}
69+
70+
return $params;
71+
}
72+
73+
public static function getFunctionReturnType(FunctionReturnTypeProviderEvent $event): ?Type\Union
74+
{
75+
$stages = self::parseStages($event->getStatementsSource(), $event->getCallArgs());
76+
if (!$stages) {
77+
//
78+
// @see https://github.com/vimeo/psalm/issues/7244
79+
// Currently, templated arguments are not being resolved in closures / callables
80+
// For now, we fall back to the built-in types.
81+
82+
// $templated = self::createTemplatedType('T', Type::getMixed(), 'fn-'.$event->getFunctionId());
83+
// return self::createClosureStage($templated, $templated, 'input');
84+
85+
return null;
86+
}
87+
88+
$in = self::pipeInputType($stages);
89+
$out = self::pipeOutputType($stages);
90+
91+
return self::createClosureStage($in, $out, 'input');
92+
}
93+
94+
/**
95+
* @param array<array-key, Arg> $args
96+
*
97+
* @return StagesOrEmpty
98+
*/
99+
private static function parseStages(StatementsSource $source, array $args): array
100+
{
101+
$stages = [];
102+
foreach ($args as $arg) {
103+
$stage = $arg->value;
104+
105+
if (!$stage instanceof FunctionLike) {
106+
// The stage could also be an expression instead of a function-like.
107+
// This plugin currently only supports function-like statements.
108+
// All other input is considered to result in a mixed -> mixed stage
109+
// This way we can still recover if types are known in later stages.
110+
111+
// Expressions currently not covered:
112+
113+
// New_ expression for invokables
114+
// Variable for variables that can point to either FunctionLike or New_
115+
// Assignments during a pipe level: $x = fn () => 123
116+
// `x(...)` results in FuncCall(args: {0: VariadicPlaceholder})
117+
// ...
118+
119+
// Haven't found a way to get the resulting type of an expression in psalm yet.
120+
121+
$stages[] = [Type::getMixed(), Type::getMixed(), 'input'];
122+
continue;
123+
}
124+
125+
$params = $stage->getParams();
126+
$paramName = self::parseNameFromParam($params[0] ?? null);
127+
128+
$in = self::determineValidatedStageInputParam($source, $stage);
129+
$out = self::parseTypeFromASTNode($source, $stage->getReturnType());
130+
131+
$stages[] = [$in, $out, $paramName];
132+
}
133+
134+
return $stages;
135+
}
136+
137+
/**
138+
* This function first validates the parameters of the stage.
139+
* A stage should have exactly one required input parameter.
140+
*
141+
* - If there are no parameters, the input parameter is ignored.
142+
* - If there are too many required parameters, this will result in a runtime exception.
143+
*
144+
* In both situations, we can continue building up the stages
145+
* so that the user has as much analyzer info as possible.
146+
*/
147+
private static function determineValidatedStageInputParam(StatementsSource $source, FunctionLike $stage): Type\Union
148+
{
149+
$params = $stage->getParams();
150+
151+
if (count($params) === 0) {
152+
IssueBuffer::maybeAdd(
153+
new TooFewArguments(
154+
'Pipe stage functions require exactly one input parameter, none given. ' .
155+
'This will ignore the input value.',
156+
new CodeLocation($source, $stage)
157+
),
158+
$source->getSuppressedIssues()
159+
);
160+
}
161+
162+
// The pipe function will crash during runtime when there are more than 1 function parameters required.
163+
// We can still determine the stages Input / Output types at this point.
164+
if (count($params) > 1 && !($params[1] ?? null)?->default) {
165+
IssueBuffer::maybeAdd(
166+
new TooManyArguments(
167+
'Pipe stage functions can only deal with one input parameter.',
168+
new CodeLocation($source, $params[1])
169+
),
170+
$source->getSuppressedIssues()
171+
);
172+
}
173+
174+
$type = $params ? $params[0]->type : null;
175+
176+
return self::parseTypeFromASTNode($source, $type);
177+
}
178+
179+
/**
180+
* This function tries parsing the node type based on psalm's NodeTypeProvider.
181+
* If that one is not able to determine the type, this function will fall back on parsing the AST's node type.
182+
* In case we are not able to determine the type, this function falls back to the $default type.
183+
*/
184+
private static function parseTypeFromASTNode(
185+
StatementsSource $source,
186+
?NodeAbstract $node,
187+
string $default = 'mixed'
188+
): Type\Union {
189+
if (!$node || $node instanceof ComplexType) {
190+
return self::createSimpleType($default);
191+
}
192+
193+
$nodeType = null;
194+
if ($node instanceof Expr || $node instanceof Name || $node instanceof Return_) {
195+
$nodeTypeProvider = $source->getNodeTypeProvider();
196+
$nodeType = $nodeTypeProvider->getType($node);
197+
}
198+
199+
if (!$nodeType && ($node instanceof Name || $node instanceof Identifier)) {
200+
$nodeType = self::createSimpleType($node->toString() ?: $default);
201+
}
202+
203+
return $nodeType ?? self::createSimpleType($default);
204+
}
205+
206+
private static function parseNameFromParam(?Param $param, string $default = 'input'): string
207+
{
208+
if (!$param) {
209+
return $default;
210+
}
211+
212+
$var = $param->var;
213+
if (!$var instanceof Expr\Variable) {
214+
return $default;
215+
}
216+
217+
return is_string($var->name) ? $var->name : $default;
218+
}
219+
220+
/**
221+
* @param Stages $stages
222+
*/
223+
private static function pipeInputType(array $stages): Type\Union
224+
{
225+
$firstStage = array_shift($stages);
226+
[$in, $_, $_] = $firstStage;
227+
228+
return $in;
229+
}
230+
231+
/**
232+
* @param Stages $stages
233+
*/
234+
private static function pipeOutputType(array $stages): Type\Union
235+
{
236+
$lastStage = array_pop($stages);
237+
[$_, $out, $_] = $lastStage;
238+
239+
return $out;
240+
}
241+
242+
private static function createClosureStage(Type\Union $in, Type\Union $out, string $paramName): Type\Union
243+
{
244+
return new Type\Union([
245+
new Type\Atomic\TClosure(
246+
value: Closure::class,
247+
params: [
248+
self::createFunctionParameter($paramName, $in),
249+
],
250+
return_type: $out,
251+
)
252+
]);
253+
}
254+
255+
private static function createFunctionParameter(string $name, Type\Union $type): FunctionLikeParameter
256+
{
257+
return new FunctionLikeParameter(
258+
$name,
259+
false,
260+
$type,
261+
is_optional: false,
262+
is_nullable: false,
263+
is_variadic: false,
264+
);
265+
}
266+
267+
private static function createSimpleType(string $type): Type\Union
268+
{
269+
return new Type\Union([Type\Atomic::create($type)]);
270+
}
271+
272+
private static function createTemplatedType(string $name, Type\Union $baseType, string $definingClass): Type\Union
273+
{
274+
return new Type\Union([
275+
new Type\Atomic\TTemplateParam($name, $baseType, $definingClass)
276+
]);
277+
}
278+
}

src/Plugin.php

+3
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,9 @@ public function __invoke(RegistrationInterface $registration, ?SimpleXMLElement
2929
*/
3030
private function getHooks(): iterable
3131
{
32+
// Psl\Fun hooks
33+
yield EventHandler\Fun\Pipe\PipeArgumentsProvider::class;
34+
3235
// Psl\Iter hooks
3336
yield EventHandler\Iter\First\FunctionReturnTypeProvider::class;
3437
yield EventHandler\Iter\FirstKey\FunctionReturnTypeProvider::class;

0 commit comments

Comments
 (0)