Passed
Pull Request — master (#69)
by Dave
02:17
created

PhpstanJsonResultsParser::convertFromArray()   A

Complexity

Conditions 6
Paths 13

Size

Total Lines 34
Code Lines 19

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 6
eloc 19
c 0
b 0
f 0
nc 13
nop 2
dl 0
loc 34
rs 9.0111
1
<?php
2
3
/**
4
 * Static Analysis Results Baseliner (sarb).
5
 *
6
 * (c) Dave Liddament
7
 *
8
 * For the full copyright and licence information please view the LICENSE file distributed with this source code.
9
 */
10
11
declare(strict_types=1);
12
13
namespace DaveLiddament\StaticAnalysisResultsBaseliner\Plugins\ResultsParsers\PhpstanJsonResultsParser;
14
15
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\Common\AbsoluteFileName;
16
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\Common\InvalidPathException;
17
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\Common\LineNumber;
18
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\Common\Location;
19
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\Common\ProjectRoot;
20
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\Common\Type;
21
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\File\InvalidFileFormatException;
22
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\ResultsParser\AnalysisResult;
23
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\ResultsParser\AnalysisResults;
24
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\ResultsParser\AnalysisResultsBuilder;
25
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\ResultsParser\Identifier;
26
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\ResultsParser\ResultsParser;
27
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\Utils\ArrayParseException;
28
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\Utils\ArrayUtils;
29
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\Utils\FqcnRemover;
30
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\Utils\JsonParseException;
31
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\Utils\JsonUtils;
32
use DaveLiddament\StaticAnalysisResultsBaseliner\Domain\Utils\ParseAtLocationException;
33
34
/**
35
 * Handles PHPStan's JSON output.
36
 *
37
 * NOTE: SARB only deals with errors that are attached to a particular line in a file.
38
 * PHPStan can report general errors (not specific to file). These are ignored by SARB.
39
 */
40
class PhpstanJsonResultsParser implements ResultsParser
41
{
42
    private const LINE = 'line';
43
    private const TYPE = 'message';
44
    private const FILES = 'files';
45
    private const MESSAGES = 'messages';
46
    private const MESSAGE = 'message';
47
    private const ABSOLUTE_FILE_PATH = 'absoluteFilePath';
48
49
    /**
50
     * @var FqcnRemover
51
     */
52
    private $fqcnRemover;
53
54
    /**
55
     * PhpstanJsonResultsParser constructor.
56
     */
57
    public function __construct(FqcnRemover $fqcnRemover)
58
    {
59
        $this->fqcnRemover = $fqcnRemover;
60
    }
61
62
    /**
63
     * {@inheritdoc}
64
     */
65
    public function convertFromString(string $resultsAsString, ProjectRoot $projectRoot): AnalysisResults
66
    {
67
        try {
68
            $asArray = JsonUtils::toArray($resultsAsString);
69
        } catch (JsonParseException $e) {
70
            throw new InvalidFileFormatException('Not a valid JSON format');
71
        }
72
73
        return $this->convertFromArray($asArray, $projectRoot);
74
    }
75
76
    /**
77
     * {@inheritdoc}
78
     */
79
    public function getIdentifier(): Identifier
80
    {
81
        return new PhpstanJsonIdentifier();
82
    }
83
84
    /**
85
     * {@inheritdoc}
86
     */
87
    public function showTypeGuessingWarning(): bool
88
    {
89
        return true;
90
    }
91
92
    /**
93
     * Converts from an array.
94
     *
95
     * @psalm-param array<mixed> $analysisResultsAsArray
96
     *
97
     * @throws ParseAtLocationException
98
     */
99
    private function convertFromArray(array $analysisResultsAsArray, ProjectRoot $projectRoot): AnalysisResults
100
    {
101
        $analysisResultsBuilder = new AnalysisResultsBuilder();
102
103
        try {
104
            $filesErrors = ArrayUtils::getArrayValue($analysisResultsAsArray, self::FILES);
105
        } catch (ArrayParseException $e) {
106
            throw ParseAtLocationException::issueParsing($e, 'Root node');
107
        }
108
109
        /** @psalm-suppress MixedAssignment */
110
        foreach ($filesErrors as $absoluteFileNameAsString => $fileErrors) {
111
            try {
112
                if (!is_string($absoluteFileNameAsString)) {
113
                    throw new ArrayParseException('Expected filename to be of type string');
114
                }
115
116
                ArrayUtils::assertArray($fileErrors);
117
118
                $absoluteFileName = new AbsoluteFileName($absoluteFileNameAsString);
119
120
                $messages = ArrayUtils::getArrayValue($fileErrors, self::MESSAGES);
121
122
                foreach ($messages as $message) {
123
                    ArrayUtils::assertArray($message);
124
                    $analysisResult = $this->convertAnalysisResultFromArray($message, $absoluteFileName, $projectRoot);
125
                    $analysisResultsBuilder->addAnalysisResult($analysisResult);
126
                }
127
            } catch (ArrayParseException | JsonParseException | InvalidPathException $e) {
128
                throw ParseAtLocationException::issueParsing($e, "Result [$absoluteFileNameAsString]");
129
            }
130
        }
131
132
        return $analysisResultsBuilder->build();
133
    }
134
135
    /**
136
     * @psalm-param array<mixed> $analysisResultAsArray
137
     *
138
     * @throws ArrayParseException
139
     * @throws JsonParseException
140
     */
141
    private function convertAnalysisResultFromArray(
142
        array $analysisResultAsArray,
143
        AbsoluteFileName $absoluteFileName,
144
        ProjectRoot $projectRoot
145
    ): AnalysisResult {
146
        $lineAsInt = ArrayUtils::getIntOrNullValue($analysisResultAsArray, self::LINE);
147
148
        // PHPStan sometimes reports errors not assigned to any line number. In this case give the line number as 0
149
        if (null === $lineAsInt) {
150
            $lineAsInt = 0;
151
        }
152
153
        $rawType = ArrayUtils::getStringValue($analysisResultAsArray, self::TYPE);
154
        $type = $this->fqcnRemover->removeRqcn($rawType);
155
156
        $location = Location::fromAbsoluteFileName(
157
            $absoluteFileName,
158
            $projectRoot,
159
            new LineNumber($lineAsInt)
160
        );
161
162
        return new AnalysisResult(
163
            $location,
164
            new Type($type),
165
            $rawType,
166
            JsonUtils::toString($analysisResultAsArray)
167
        );
168
    }
169
}
170