• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

voku / Simple-PHP-Code-Parser / 24287978923

11 Apr 2026 05:41PM UTC coverage: 81.31% (-1.6%) from 82.886%
24287978923

Pull #83

github

web-flow
Merge 29aa9bda8 into 90e1e60d3
Pull Request #83: Fix CI pipeline: phpunit.xml validation warning, php-parser v4 test skip, and comprehensive type-analysis regression coverage

247 of 288 new or added lines in 7 files covered. (85.76%)

30 existing lines in 3 files now uncovered.

1688 of 2076 relevant lines covered (81.31%)

27.68 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

91.96
/src/voku/SimplePhpParser/Model/PHPClass.php
1
<?php
2

3
declare(strict_types=1);
4

5
namespace voku\SimplePhpParser\Model;
6

7
use PhpParser\Comment\Doc;
8
use PhpParser\Node\Stmt\Class_;
9
use ReflectionClass;
10
use voku\SimplePhpParser\Parsers\Helper\DocFactoryProvider;
11
use voku\SimplePhpParser\Parsers\Helper\Utils;
12

13
class PHPClass extends BasePHPClass
14
{
15
    /**
16
     * @phpstan-var class-string
17
     */
18
    public string $name;
19

20
    /**
21
     * @phpstan-var class-string|null
22
     */
23
    public ?string $parentClass = null;
24

25
    /**
26
     * @var string[]
27
     *
28
     * @phpstan-var class-string[]
29
     */
30
    public array $interfaces = [];
31

32
    /**
33
     * @param Class_ $node
34
     * @param null   $dummy
35
     *
36
     * @return $this
37
     */
38
    public function readObjectFromPhpNode($node, $dummy = null): self
39
    {
40
        $this->prepareNode($node);
87✔
41

42
        $this->name = static::getFQN($node);
87✔
43

44
        $this->is_final = $node->isFinal();
87✔
45

46
        $this->is_abstract = $node->isAbstract();
87✔
47

48
        // Keep the guard for cross-version php-parser compatibility when readonly
49
        // helpers are restored or backported differently in downstream installs.
50
        if (\method_exists($node, 'isReadonly')) {
87✔
51
            $this->is_readonly = $node->isReadonly();
87✔
52
        }
53

54
        $this->is_anonymous = $node->isAnonymous();
87✔
55

56
        // Extract PHP 8.0+ attributes
57
        if (!empty($node->attrGroups)) {
87✔
58
            $this->attributes = Utils::extractAttributesFromAstNode($node->attrGroups);
19✔
59
        }
60

61
        // Skip autoloading when the current runtime cannot safely compile newer syntax;
62
        // AST data is still read from the node below.
63
        $canAutoload = self::canAutoloadFromPhpNode($node);
87✔
64
        $classExists = false;
87✔
65
        if ($canAutoload) {
87✔
66
            try {
67
                if (\class_exists($this->name, true)) {
87✔
68
                    $classExists = true;
87✔
69
                }
70
            } catch (\Throwable $e) {
3✔
71
                // nothing
72
            }
73
        }
74
        if ($classExists) {
87✔
75
            $reflectionClass = Utils::createClassReflectionInstance($this->name);
54✔
76
            $this->readObjectFromReflection($reflectionClass);
54✔
77
        }
78

79
        $this->collectTags($node);
87✔
80

81
        if (!empty($node->extends)) {
87✔
82
            $classExtended = $node->extends->toString();
26✔
83
            /** @noinspection PhpSillyAssignmentInspection - hack for phpstan */
84
            /** @var class-string $classExtended */
85
            $classExtended = $classExtended;
26✔
86
            $this->parentClass = $classExtended;
26✔
87
        }
88

89
        $docComment = $node->getDocComment();
87✔
90
        if ($docComment) {
87✔
91
            $this->readPhpDocProperties($docComment->getText());
61✔
92
        }
93

94
        foreach ($node->getProperties() as $property) {
87✔
95
            $propertyNameTmp = $this->getConstantFQN($property, $property->props[0]->name->name);
48✔
96

97
            if (isset($this->properties[$propertyNameTmp])) {
48✔
98
                $this->properties[$propertyNameTmp] = $this->properties[$propertyNameTmp]->readObjectFromPhpNode($property, $this->name);
42✔
99
            } else {
100
                $this->properties[$propertyNameTmp] = (new PHPProperty($this->parserContainer))->readObjectFromPhpNode($property, $this->name);
6✔
101
            }
102

103
            if ($this->is_readonly) {
48✔
104
                $this->properties[$propertyNameTmp]->is_readonly = true;
4✔
105
            }
106
        }
107

108
        foreach ($node->getMethods() as $method) {
87✔
109
            $methodNameTmp = $method->name->name;
73✔
110

111
            if (isset($this->methods[$methodNameTmp])) {
73✔
112
                $this->methods[$methodNameTmp] = $this->methods[$methodNameTmp]->readObjectFromPhpNode($method, $this->name);
48✔
113
            } else {
114
                $this->methods[$methodNameTmp] = (new PHPMethod($this->parserContainer))->readObjectFromPhpNode($method, $this->name);
29✔
115
            }
116

117
            if (!$this->methods[$methodNameTmp]->file) {
73✔
118
                $this->methods[$methodNameTmp]->file = $this->file;
29✔
119
            }
120
        }
121

122
        $this->addPromotedPropertiesFromConstructor($node);
87✔
123

124
        if (!empty($node->implements)) {
87✔
125
            foreach ($node->implements as $interfaceObject) {
16✔
126
                $interfaceFQN = $interfaceObject->toString();
16✔
127
                /** @noinspection PhpSillyAssignmentInspection - hack for phpstan */
128
                /** @var class-string $interfaceFQN */
129
                $interfaceFQN = $interfaceFQN;
16✔
130
                $this->interfaces[$interfaceFQN] = $interfaceFQN;
16✔
131
            }
132
        }
133

134
        return $this;
87✔
135
    }
136

137
    /**
138
     * @param ReflectionClass<object> $clazz
139
     *
140
     * @return $this
141
     */
142
    public function readObjectFromReflection($clazz): self
143
    {
144
        $this->name = $clazz->getName();
54✔
145

146
        if (!$this->line) {
54✔
147
            $lineTmp = $clazz->getStartLine();
22✔
148
            if ($lineTmp !== false) {
22✔
149
                $this->line = $lineTmp;
14✔
150
            }
151
        }
152

153
        $file = $clazz->getFileName();
54✔
154
        if ($file) {
54✔
155
            $this->file = $file;
54✔
156
        }
157

158
        $this->is_final = $clazz->isFinal();
54✔
159

160
        $this->is_abstract = $clazz->isAbstract();
54✔
161

162
        if (method_exists($clazz, 'isReadOnly')) {
54✔
163
            $this->is_readonly = $clazz->isReadOnly();
54✔
164
        }
165

166
        $this->is_anonymous = $clazz->isAnonymous();
54✔
167

168
        $this->is_cloneable = $clazz->isCloneable();
54✔
169

170
        $this->is_instantiable = $clazz->isInstantiable();
54✔
171

172
        $this->is_iterable = $clazz->isIterable();
54✔
173

174
        // Extract PHP 8.0+ attributes
175
        $this->attributes = Utils::extractAttributesFromReflection($clazz);
54✔
176

177
        $parent = $clazz->getParentClass();
54✔
178
        if ($parent) {
54✔
179
            $this->parentClass = $parent->getName();
22✔
180

181
            $classExists = false;
22✔
182
            try {
183
                if (
184
                    !$this->parserContainer->getClass($this->parentClass)
22✔
185
                    &&
186
                    \class_exists($this->parentClass, true)
22✔
187
                ) {
188
                    $classExists = true;
22✔
189
                }
190
            } catch (\Throwable $e) {
×
191
                // nothing
192
            }
193
            if ($classExists) {
22✔
194
                $reflectionClass = Utils::createClassReflectionInstance($this->parentClass);
22✔
195
                $class = (new self($this->parserContainer))->readObjectFromReflection($reflectionClass);
22✔
196
                $this->parserContainer->addClass($class);
22✔
197
            }
198
        }
199

200
        foreach ($clazz->getProperties() as $property) {
54✔
201
            $propertyPhp = (new PHPProperty($this->parserContainer))->readObjectFromReflection($property);
50✔
202
            $this->properties[$propertyPhp->name] = $propertyPhp;
50✔
203

204
            if ($this->is_readonly) {
50✔
205
                $this->properties[$propertyPhp->name]->is_readonly = true;
4✔
206
            }
207
        }
208

209
        foreach ($clazz->getInterfaceNames() as $interfaceName) {
54✔
210
            /** @noinspection PhpSillyAssignmentInspection - hack for phpstan */
211
            /** @var class-string $interfaceName */
212
            $interfaceName = $interfaceName;
24✔
213
            $this->interfaces[$interfaceName] = $interfaceName;
24✔
214
        }
215

216
        foreach ($clazz->getMethods() as $method) {
54✔
217
            $methodNameTmp = $method->getName();
50✔
218

219
            $this->methods[$methodNameTmp] = (new PHPMethod($this->parserContainer))->readObjectFromReflection($method);
50✔
220

221
            if (!$this->methods[$methodNameTmp]->file) {
50✔
222
                $this->methods[$methodNameTmp]->file = $this->file;
×
223
            }
224
        }
225

226
        foreach ($clazz->getReflectionConstants() as $constant) {
54✔
227
            $constantNameTmp = $constant->getName();
28✔
228

229
            $this->constants[$constantNameTmp] = (new PHPConst($this->parserContainer))->readObjectFromReflection($constant);
28✔
230

231
            if (!$this->constants[$constantNameTmp]->file) {
28✔
232
                $this->constants[$constantNameTmp]->file = $this->file;
×
233
            }
234
        }
235

236
        return $this;
54✔
237
    }
238

239
    /**
240
     * @param string[] $access
241
     * @param bool     $skipMethodsWithLeadingUnderscore
242
     *
243
     * @return array
244
     *
245
     * @psalm-return array<string, array{
246
     *     type: null|string,
247
     *     typeFromPhpDocMaybeWithComment: null|string,
248
     *     typeFromPhpDoc: null|string,
249
     *     typeFromPhpDocSimple: null|string,
250
     *     typeFromPhpDocExtended: null|string,
251
     *     typeFromDefaultValue: null|string
252
     * }>
253
     */
254
    public function getPropertiesInfo(
255
        array $access = ['public', 'protected', 'private'],
256
        bool $skipMethodsWithLeadingUnderscore = false
257
    ): array {
258
        // init
259
        $allInfo = [];
4✔
260

261
        foreach ($this->properties as $property) {
4✔
262
            if (!\in_array($property->access, $access, true)) {
4✔
263
                continue;
×
264
            }
265

266
            if ($skipMethodsWithLeadingUnderscore && \strpos($property->name, '_') === 0) {
4✔
267
                continue;
×
268
            }
269

270
            $types = [];
4✔
271
            $types['type'] = $property->type;
4✔
272
            $types['typeFromPhpDocMaybeWithComment'] = $property->typeFromPhpDocMaybeWithComment;
4✔
273
            $types['typeFromPhpDoc'] = $property->typeFromPhpDoc;
4✔
274
            $types['typeFromPhpDocSimple'] = $property->typeFromPhpDocSimple;
4✔
275
            $types['typeFromPhpDocExtended'] = $property->typeFromPhpDocExtended;
4✔
276
            $types['typeFromDefaultValue'] = $property->typeFromDefaultValue;
4✔
277

278
            $allInfo[$property->name] = $types;
4✔
279
        }
280

281
        return $allInfo;
4✔
282
    }
283

284
    /**
285
     * @param string[] $access
286
     * @param bool     $skipDeprecatedMethods
287
     * @param bool     $skipMethodsWithLeadingUnderscore
288
     *
289
     * @return array<mixed>
290
     *
291
     * @psalm-return array<string, array{
292
     *     fullDescription: string,
293
     *     line: null|int,
294
     *     file: null|string,
295
     *     error: string,
296
     *     is_deprecated: bool,
297
     *     is_static: null|bool,
298
     *     is_meta: bool,
299
     *     is_internal: bool,
300
     *     is_removed: bool,
301
     *     paramsTypes: array<string,
302
     *         array{
303
     *              type?: null|string,
304
     *              typeFromPhpDoc?: null|string,
305
     *              typeFromPhpDocExtended?: null|string,
306
     *              typeFromPhpDocSimple?: null|string,
307
     *              typeFromPhpDocMaybeWithComment?: null|string,
308
     *              typeFromDefaultValue?: null|string
309
     *         }
310
     *     >,
311
     *     returnTypes: array{
312
     *         type: null|string,
313
     *         typeFromPhpDoc: null|string,
314
     *         typeFromPhpDocExtended: null|string,
315
     *         typeFromPhpDocSimple: null|string,
316
     *         typeFromPhpDocMaybeWithComment: null|string
317
     *     },
318
     *     paramsPhpDocRaw: array<string, null|string>,
319
     *     returnPhpDocRaw: null|string
320
     * }>
321
     */
322
    public function getMethodsInfo(
323
        array $access = ['public', 'protected', 'private'],
324
        bool $skipDeprecatedMethods = false,
325
        bool $skipMethodsWithLeadingUnderscore = false
326
    ): array {
327
        // init
328
        $allInfo = [];
8✔
329

330
        foreach ($this->methods as $method) {
8✔
331
            if (!\in_array($method->access, $access, true)) {
8✔
332
                continue;
×
333
            }
334

335
            if ($skipDeprecatedMethods && $method->hasDeprecatedTag) {
8✔
336
                continue;
×
337
            }
338

339
            if ($skipMethodsWithLeadingUnderscore && \strpos($method->name, '_') === 0) {
8✔
340
                continue;
×
341
            }
342

343
            $paramsTypes = [];
8✔
344
            foreach ($method->parameters as $tagParam) {
8✔
345
                $paramsTypes[$tagParam->name]['type'] = $tagParam->type;
8✔
346
                $paramsTypes[$tagParam->name]['typeFromPhpDocMaybeWithComment'] = $tagParam->typeFromPhpDocMaybeWithComment;
8✔
347
                $paramsTypes[$tagParam->name]['typeFromPhpDoc'] = $tagParam->typeFromPhpDoc;
8✔
348
                $paramsTypes[$tagParam->name]['typeFromPhpDocSimple'] = $tagParam->typeFromPhpDocSimple;
8✔
349
                $paramsTypes[$tagParam->name]['typeFromPhpDocExtended'] = $tagParam->typeFromPhpDocExtended;
8✔
350
                $paramsTypes[$tagParam->name]['typeFromDefaultValue'] = $tagParam->typeFromDefaultValue;
8✔
351
            }
352

353
            $returnTypes = [];
8✔
354
            $returnTypes['type'] = $method->returnType;
8✔
355
            $returnTypes['typeFromPhpDocMaybeWithComment'] = $method->returnTypeFromPhpDocMaybeWithComment;
8✔
356
            $returnTypes['typeFromPhpDoc'] = $method->returnTypeFromPhpDoc;
8✔
357
            $returnTypes['typeFromPhpDocSimple'] = $method->returnTypeFromPhpDocSimple;
8✔
358
            $returnTypes['typeFromPhpDocExtended'] = $method->returnTypeFromPhpDocExtended;
8✔
359

360
            $paramsPhpDocRaw = [];
8✔
361
            foreach ($method->parameters as $tagParam) {
8✔
362
                $paramsPhpDocRaw[$tagParam->name] = $tagParam->phpDocRaw;
8✔
363
            }
364

365
            $infoTmp = [];
8✔
366
            $infoTmp['fullDescription'] = \trim($method->summary . "\n\n" . $method->description);
8✔
367
            $infoTmp['paramsTypes'] = $paramsTypes;
8✔
368
            $infoTmp['returnTypes'] = $returnTypes;
8✔
369
            $infoTmp['paramsPhpDocRaw'] = $paramsPhpDocRaw;
8✔
370
            $infoTmp['returnPhpDocRaw'] = $method->returnPhpDocRaw;
8✔
371
            $infoTmp['line'] = $method->line ?? $this->line;
8✔
372
            $infoTmp['file'] = $method->file ?? $this->file;
8✔
373
            $infoTmp['error'] = \implode("\n", $method->parseError);
8✔
374
            foreach ($method->parameters as $parameter) {
8✔
375
                $infoTmp['error'] .= ($infoTmp['error'] ? "\n" : '') . \implode("\n", $parameter->parseError);
8✔
376
            }
377
            $infoTmp['is_deprecated'] = $method->hasDeprecatedTag;
8✔
378
            $infoTmp['is_static'] = $method->is_static;
8✔
379
            $infoTmp['is_meta'] = $method->hasMetaTag;
8✔
380
            $infoTmp['is_internal'] = $method->hasInternalTag;
8✔
381
            $infoTmp['is_removed'] = $method->hasRemovedTag;
8✔
382

383
            $allInfo[$method->name] = $infoTmp;
8✔
384
        }
385

386
        \asort($allInfo);
8✔
387

388
        return $allInfo;
8✔
389
    }
390

391
    /**
392
     * @param Doc|string $doc
393
     */
394
    private function readPhpDocProperties($doc): void
395
    {
396
        if ($doc instanceof Doc) {
61✔
397
            $docComment = $doc->getText();
×
398
        } else {
399
            $docComment = $doc;
61✔
400
        }
401
        if ($docComment === '') {
61✔
402
            return;
×
403
        }
404

405
        try {
406
            $phpDoc = DocFactoryProvider::getDocFactory()->create($docComment);
61✔
407

408
            $parsedPropertyTags = $phpDoc->getTagsByName('property')
61✔
409
                               + $phpDoc->getTagsByName('property-read')
61✔
410
                               + $phpDoc->getTagsByName('property-write');
61✔
411

412
            if (!empty($parsedPropertyTags)) {
61✔
413
                foreach ($parsedPropertyTags as $parsedPropertyTag) {
61✔
414
                    if (
415
                        $parsedPropertyTag instanceof \phpDocumentor\Reflection\DocBlock\Tags\PropertyRead
16✔
416
                        ||
417
                        $parsedPropertyTag instanceof \phpDocumentor\Reflection\DocBlock\Tags\PropertyWrite
16✔
418
                        ||
419
                        $parsedPropertyTag instanceof \phpDocumentor\Reflection\DocBlock\Tags\Property
16✔
420
                    ) {
421
                        $propertyPhp = new PHPProperty($this->parserContainer);
16✔
422

423
                        $nameTmp = $parsedPropertyTag->getVariableName();
16✔
424
                        if (!$nameTmp) {
16✔
425
                            continue;
×
426
                        }
427

428
                        $propertyPhp->name = $nameTmp;
16✔
429

430
                        $propertyPhp->access = 'public';
16✔
431

432
                        $type = $parsedPropertyTag->getType();
16✔
433

434
                        $propertyPhp->typeFromPhpDoc = Utils::normalizePhpType($type . '');
16✔
435

436
                        $typeFromPhpDocMaybeWithCommentTmp = \trim((string) $parsedPropertyTag);
16✔
437
                        if (
438
                            $typeFromPhpDocMaybeWithCommentTmp
16✔
439
                            &&
440
                            \strpos($typeFromPhpDocMaybeWithCommentTmp, '$') !== 0
16✔
441
                        ) {
442
                            $propertyPhp->typeFromPhpDocMaybeWithComment = $typeFromPhpDocMaybeWithCommentTmp;
16✔
443
                        }
444

445
                        $typeTmp = Utils::parseDocTypeObject($type);
16✔
446
                        if ($typeTmp !== '') {
16✔
447
                            $propertyPhp->typeFromPhpDocSimple = $typeTmp;
16✔
448
                        }
449

450
                        if ($propertyPhp->typeFromPhpDoc) {
16✔
451
                            $propertyPhp->typeFromPhpDocExtended = Utils::modernPhpdoc($propertyPhp->typeFromPhpDoc);
16✔
452
                        }
453

454
                        $this->properties[$propertyPhp->name] = $propertyPhp;
16✔
455
                    }
456
                }
457
            }
458
        } catch (\Exception $e) {
×
459
            $tmpErrorMessage = ($this->name ?: '?') . ':' . ($this->line ?? '?') . ' | ' . \print_r($e->getMessage(), true);
×
460
            $this->parseError[\md5($tmpErrorMessage)] = $tmpErrorMessage;
×
461
        }
462
    }
463

464
    private function addPromotedPropertiesFromConstructor(Class_ $node): void
465
    {
466
        $method = $node->getMethod('__construct');
87✔
467
        if ($method === null) {
87✔
468
            return;
76✔
469
        }
470

471
        foreach ($method->params as $parameter) {
27✔
472
            if (!self::isPromotedParameter($parameter)) {
27✔
473
                continue;
8✔
474
            }
475

476
            $parameterVar = $parameter->var;
23✔
477
            if (
478
                !($parameterVar instanceof \PhpParser\Node\Expr\Variable)
23✔
479
                || !\is_string($parameterVar->name)
23✔
480
            ) {
NEW
481
                continue;
×
482
            }
483

484
            $promotedProperty = (new PHPProperty($this->parserContainer))
23✔
485
                ->readObjectFromPromotedParam($parameter, $this->name);
23✔
486

487
            $propertyName = $parameterVar->name;
23✔
488
            $existingProperty = $this->properties[$propertyName] ?? null;
23✔
489
            if ($existingProperty !== null) {
23✔
490
                $this->mergePromotedPropertyData($existingProperty, $promotedProperty, $parameter);
18✔
491

492
                continue;
18✔
493
            }
494

495
            $this->properties[$propertyName] = $promotedProperty;
8✔
496
        }
497
    }
498

499
    private function mergePromotedPropertyData(
500
        PHPProperty $existingProperty,
501
        PHPProperty $promotedProperty,
502
        \PhpParser\Node\Param $parameter
503
    ): void {
504
        if ($promotedProperty->access !== '') {
18✔
505
            $existingProperty->access = $promotedProperty->access;
18✔
506
        }
507

508
        if ($existingProperty->type === null && $promotedProperty->type !== null) {
18✔
509
            $existingProperty->type = $promotedProperty->type;
2✔
510
        }
511

512
        if ($existingProperty->is_readonly === null && $promotedProperty->is_readonly !== null) {
18✔
513
            $existingProperty->is_readonly = $promotedProperty->is_readonly;
2✔
514
        }
515

516
        if ($existingProperty->is_final === null && $promotedProperty->is_final !== null) {
18✔
517
            $existingProperty->is_final = $promotedProperty->is_final;
2✔
518
        }
519

520
        if ($existingProperty->access_set === '' && $promotedProperty->access_set !== '') {
18✔
NEW
521
            $existingProperty->access_set = $promotedProperty->access_set;
×
522
        }
523

524
        if ($existingProperty->hooks === [] && $promotedProperty->hooks !== []) {
18✔
NEW
525
            $existingProperty->hooks = $promotedProperty->hooks;
×
526
        }
527

528
        if ($existingProperty->attributes === [] && $promotedProperty->attributes !== []) {
18✔
NEW
529
            $existingProperty->attributes = $promotedProperty->attributes;
×
530
        }
531

532
        if ($parameter->default !== null && $promotedProperty->typeFromDefaultValue !== null) {
18✔
533
            $existingProperty->defaultValue = $promotedProperty->defaultValue;
14✔
534
            $existingProperty->typeFromDefaultValue = $promotedProperty->typeFromDefaultValue;
14✔
535
        }
536
    }
537
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc