1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-10 06:58:41 +01:00
psalm/src/Psalm/Internal/Codebase/DataFlowGraph.php

160 lines
4.2 KiB
PHP
Raw Normal View History

<?php
namespace Psalm\Internal\Codebase;
use Psalm\Internal\DataFlow\DataFlowNode;
2021-06-08 04:55:21 +02:00
use Psalm\Internal\DataFlow\Path;
2021-12-03 21:07:25 +01:00
use function abs;
2021-06-08 04:55:21 +02:00
use function array_keys;
use function array_merge;
2020-06-19 00:48:19 +02:00
use function array_reverse;
2020-11-10 22:19:24 +01:00
use function array_sum;
2021-12-03 21:07:25 +01:00
use function count;
2021-06-08 04:55:21 +02:00
use function strlen;
2021-09-26 23:41:26 +02:00
use function strpos;
2021-06-08 04:55:21 +02:00
use function substr;
abstract class DataFlowGraph
{
2020-06-22 08:10:03 +02:00
/** @var array<string, array<string, Path>> */
protected $forward_edges = [];
abstract public function addNode(DataFlowNode $node): void;
2019-08-14 06:47:57 +02:00
/**
* @param array<string> $added_taints
* @param array<string> $removed_taints
2019-08-14 06:47:57 +02:00
*/
2020-05-22 04:47:58 +02:00
public function addPath(
DataFlowNode $from,
DataFlowNode $to,
2020-06-19 00:48:19 +02:00
string $path_type,
2020-06-26 01:12:30 +02:00
?array $added_taints = null,
?array $removed_taints = null
): void {
2020-05-22 04:47:58 +02:00
$from_id = $from->id;
$to_id = $to->id;
2019-08-14 06:47:57 +02:00
2020-06-19 00:48:19 +02:00
if ($from_id === $to_id) {
return;
}
2020-11-10 18:49:42 +01:00
$length = 0;
if ($from->code_location
&& $to->code_location
&& $from->code_location->file_path === $to->code_location->file_path
) {
$to_line = $to->code_location->raw_line_number;
$from_line = $from->code_location->raw_line_number;
2021-12-03 21:07:25 +01:00
$length = abs($to_line - $from_line);
2020-11-10 18:49:42 +01:00
}
$this->forward_edges[$from_id][$to_id] = new Path($path_type, $length, $added_taints, $removed_taints);
}
2020-08-23 19:52:31 +02:00
/**
* @param array<string> $previous_path_types
*
* @psalm-pure
*/
protected static function shouldIgnoreFetch(
2020-06-25 07:32:57 +02:00
string $path_type,
string $expression_type,
array $previous_path_types
): bool {
$el = strlen($expression_type);
2020-06-25 07:32:57 +02:00
// arraykey-fetch requires a matching arraykey-assignment at the same level
// otherwise the tainting is not valid
2021-09-26 22:51:44 +02:00
if (strpos($path_type, $expression_type . '-fetch-') === 0 || $path_type === 'arraykey-fetch') {
2020-06-25 07:32:57 +02:00
$fetch_nesting = 0;
$previous_path_types = array_reverse($previous_path_types);
foreach ($previous_path_types as $previous_path_type) {
if ($previous_path_type === $expression_type . '-assignment') {
if ($fetch_nesting === 0) {
return false;
}
$fetch_nesting--;
}
2021-09-26 22:51:44 +02:00
if (strpos($previous_path_type, $expression_type . '-fetch') === 0) {
2020-06-25 07:32:57 +02:00
$fetch_nesting++;
}
2021-09-26 22:51:44 +02:00
if (strpos($previous_path_type, $expression_type . '-assignment-') === 0) {
2020-06-25 07:32:57 +02:00
if ($fetch_nesting > 0) {
$fetch_nesting--;
continue;
}
if (substr($previous_path_type, $el + 12) === substr($path_type, $el + 7)) {
return false;
}
return true;
}
}
}
return false;
}
2020-11-10 18:49:42 +01:00
/**
2020-11-27 23:48:39 +01:00
* @return array{int, int, int, float}
2020-11-10 18:49:42 +01:00
*/
public function getEdgeStats(): array
2020-11-10 18:49:42 +01:00
{
$lengths = 0;
$destination_counts = [];
2020-11-27 23:48:39 +01:00
$origin_counts = [];
2020-11-10 18:49:42 +01:00
2020-11-27 23:48:39 +01:00
foreach ($this->forward_edges as $from_id => $destinations) {
foreach ($destinations as $to_id => $path) {
2020-11-10 18:49:42 +01:00
if ($path->length === 0) {
continue;
}
$lengths += $path->length;
2020-11-27 23:48:39 +01:00
if (!isset($destination_counts[$to_id])) {
$destination_counts[$to_id] = 0;
2020-11-10 18:49:42 +01:00
}
2020-11-27 23:48:39 +01:00
$destination_counts[$to_id]++;
$origin_counts[$from_id] = true;
2020-11-10 18:49:42 +01:00
}
}
$count = array_sum($destination_counts);
if (!$count) {
2020-11-27 23:48:39 +01:00
return [0, 0, 0, 0.0];
2020-11-10 18:49:42 +01:00
}
$mean = $lengths / $count;
2021-12-03 21:07:25 +01:00
return [$count, count($origin_counts), count($destination_counts), $mean];
2020-11-10 18:49:42 +01:00
}
/**
* @psalm-return list<list<string>>
*/
public function summarizeEdges(): array
{
$edges = [];
foreach ($this->forward_edges as $source => $destinations) {
$edges[] = array_merge([$source], array_keys($destinations));
}
return $edges;
}
}