1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-05 20:48:45 +01:00
psalm/src/Psalm/Internal/Codebase/DataFlowGraph.php

96 lines
2.6 KiB
PHP

<?php
namespace Psalm\Internal\Codebase;
use Psalm\CodeLocation;
use Psalm\Internal\DataFlow\Path;
use Psalm\Internal\DataFlow\TaintSink;
use Psalm\Internal\DataFlow\TaintSource;
use Psalm\Internal\DataFlow\DataFlowNode;
use Psalm\IssueBuffer;
use Psalm\Issue\TaintedInput;
use function array_merge;
use function count;
use function implode;
use function substr;
use function strlen;
use function array_intersect;
use function array_reverse;
abstract class DataFlowGraph
{
/** @var array<string, array<string, Path>> */
protected $forward_edges = [];
abstract public function addNode(DataFlowNode $node) : void;
/**
* @param array<string> $added_taints
* @param array<string> $removed_taints
*/
public function addPath(
DataFlowNode $from,
DataFlowNode $to,
string $path_type,
?array $added_taints = null,
?array $removed_taints = null
) : void {
$from_id = $from->id;
$to_id = $to->id;
if ($from_id === $to_id) {
return;
}
$this->forward_edges[$from_id][$to_id] = new Path($path_type, $added_taints, $removed_taints);
}
/**
* @param array<string> $previous_path_types
*
* @psalm-pure
*/
protected static function shouldIgnoreFetch(
string $path_type,
string $expression_type,
array $previous_path_types
) : bool {
$el = \strlen($expression_type);
if (substr($path_type, 0, $el + 7) === $expression_type . '-fetch-') {
$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--;
}
if (substr($previous_path_type, 0, $el + 6) === $expression_type . '-fetch') {
$fetch_nesting++;
}
if (substr($previous_path_type, 0, $el + 12) === $expression_type . '-assignment-') {
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;
}
}