mirror of
https://github.com/danog/psalm.git
synced 2025-01-05 20:48:45 +01:00
96 lines
2.6 KiB
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;
|
|
}
|
|
}
|