first commit
This commit is contained in:
83
vendor/sebastian/complexity/src/Calculator.php
vendored
Normal file
83
vendor/sebastian/complexity/src/Calculator.php
vendored
Normal file
@ -0,0 +1,83 @@
|
||||
<?php declare(strict_types=1);
|
||||
/*
|
||||
* This file is part of sebastian/complexity.
|
||||
*
|
||||
* (c) Sebastian Bergmann <sebastian@phpunit.de>
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
namespace SebastianBergmann\Complexity;
|
||||
|
||||
use function assert;
|
||||
use function file_get_contents;
|
||||
use PhpParser\Error;
|
||||
use PhpParser\Node;
|
||||
use PhpParser\NodeTraverser;
|
||||
use PhpParser\NodeVisitor\NameResolver;
|
||||
use PhpParser\NodeVisitor\ParentConnectingVisitor;
|
||||
use PhpParser\ParserFactory;
|
||||
|
||||
final class Calculator
|
||||
{
|
||||
/**
|
||||
* @throws RuntimeException
|
||||
*/
|
||||
public function calculateForSourceFile(string $sourceFile): ComplexityCollection
|
||||
{
|
||||
return $this->calculateForSourceString(file_get_contents($sourceFile));
|
||||
}
|
||||
|
||||
/**
|
||||
* @throws RuntimeException
|
||||
*/
|
||||
public function calculateForSourceString(string $source): ComplexityCollection
|
||||
{
|
||||
try {
|
||||
$nodes = (new ParserFactory)->createForHostVersion()->parse($source);
|
||||
|
||||
assert($nodes !== null);
|
||||
|
||||
return $this->calculateForAbstractSyntaxTree($nodes);
|
||||
|
||||
// @codeCoverageIgnoreStart
|
||||
} catch (Error $error) {
|
||||
throw new RuntimeException(
|
||||
$error->getMessage(),
|
||||
$error->getCode(),
|
||||
$error,
|
||||
);
|
||||
}
|
||||
// @codeCoverageIgnoreEnd
|
||||
}
|
||||
|
||||
/**
|
||||
* @param Node[] $nodes
|
||||
*
|
||||
* @throws RuntimeException
|
||||
*/
|
||||
public function calculateForAbstractSyntaxTree(array $nodes): ComplexityCollection
|
||||
{
|
||||
$traverser = new NodeTraverser;
|
||||
$complexityCalculatingVisitor = new ComplexityCalculatingVisitor(true);
|
||||
|
||||
$traverser->addVisitor(new NameResolver);
|
||||
$traverser->addVisitor(new ParentConnectingVisitor);
|
||||
$traverser->addVisitor($complexityCalculatingVisitor);
|
||||
|
||||
try {
|
||||
/* @noinspection UnusedFunctionResultInspection */
|
||||
$traverser->traverse($nodes);
|
||||
// @codeCoverageIgnoreStart
|
||||
} catch (Error $error) {
|
||||
throw new RuntimeException(
|
||||
$error->getMessage(),
|
||||
$error->getCode(),
|
||||
$error,
|
||||
);
|
||||
}
|
||||
// @codeCoverageIgnoreEnd
|
||||
|
||||
return $complexityCalculatingVisitor->result();
|
||||
}
|
||||
}
|
64
vendor/sebastian/complexity/src/Complexity/Complexity.php
vendored
Normal file
64
vendor/sebastian/complexity/src/Complexity/Complexity.php
vendored
Normal file
@ -0,0 +1,64 @@
|
||||
<?php declare(strict_types=1);
|
||||
/*
|
||||
* This file is part of sebastian/complexity.
|
||||
*
|
||||
* (c) Sebastian Bergmann <sebastian@phpunit.de>
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
namespace SebastianBergmann\Complexity;
|
||||
|
||||
use function str_contains;
|
||||
|
||||
/**
|
||||
* @psalm-immutable
|
||||
*/
|
||||
final class Complexity
|
||||
{
|
||||
/**
|
||||
* @psalm-var non-empty-string
|
||||
*/
|
||||
private readonly string $name;
|
||||
|
||||
/**
|
||||
* @psalm-var positive-int
|
||||
*/
|
||||
private int $cyclomaticComplexity;
|
||||
|
||||
/**
|
||||
* @psalm-param non-empty-string $name
|
||||
* @psalm-param positive-int $cyclomaticComplexity
|
||||
*/
|
||||
public function __construct(string $name, int $cyclomaticComplexity)
|
||||
{
|
||||
$this->name = $name;
|
||||
$this->cyclomaticComplexity = $cyclomaticComplexity;
|
||||
}
|
||||
|
||||
/**
|
||||
* @psalm-return non-empty-string
|
||||
*/
|
||||
public function name(): string
|
||||
{
|
||||
return $this->name;
|
||||
}
|
||||
|
||||
/**
|
||||
* @psalm-return positive-int
|
||||
*/
|
||||
public function cyclomaticComplexity(): int
|
||||
{
|
||||
return $this->cyclomaticComplexity;
|
||||
}
|
||||
|
||||
public function isFunction(): bool
|
||||
{
|
||||
return !$this->isMethod();
|
||||
}
|
||||
|
||||
public function isMethod(): bool
|
||||
{
|
||||
return str_contains($this->name, '::');
|
||||
}
|
||||
}
|
132
vendor/sebastian/complexity/src/Complexity/ComplexityCollection.php
vendored
Normal file
132
vendor/sebastian/complexity/src/Complexity/ComplexityCollection.php
vendored
Normal file
@ -0,0 +1,132 @@
|
||||
<?php declare(strict_types=1);
|
||||
/*
|
||||
* This file is part of sebastian/complexity.
|
||||
*
|
||||
* (c) Sebastian Bergmann <sebastian@phpunit.de>
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
namespace SebastianBergmann\Complexity;
|
||||
|
||||
use function array_filter;
|
||||
use function array_merge;
|
||||
use function array_reverse;
|
||||
use function array_values;
|
||||
use function count;
|
||||
use function usort;
|
||||
use Countable;
|
||||
use IteratorAggregate;
|
||||
|
||||
/**
|
||||
* @psalm-immutable
|
||||
*/
|
||||
final class ComplexityCollection implements Countable, IteratorAggregate
|
||||
{
|
||||
/**
|
||||
* @psalm-var list<Complexity>
|
||||
*/
|
||||
private readonly array $items;
|
||||
|
||||
public static function fromList(Complexity ...$items): self
|
||||
{
|
||||
return new self($items);
|
||||
}
|
||||
|
||||
/**
|
||||
* @psalm-param list<Complexity> $items
|
||||
*/
|
||||
private function __construct(array $items)
|
||||
{
|
||||
$this->items = $items;
|
||||
}
|
||||
|
||||
/**
|
||||
* @psalm-return list<Complexity>
|
||||
*/
|
||||
public function asArray(): array
|
||||
{
|
||||
return $this->items;
|
||||
}
|
||||
|
||||
public function getIterator(): ComplexityCollectionIterator
|
||||
{
|
||||
return new ComplexityCollectionIterator($this);
|
||||
}
|
||||
|
||||
/**
|
||||
* @psalm-return non-negative-int
|
||||
*/
|
||||
public function count(): int
|
||||
{
|
||||
return count($this->items);
|
||||
}
|
||||
|
||||
public function isEmpty(): bool
|
||||
{
|
||||
return empty($this->items);
|
||||
}
|
||||
|
||||
/**
|
||||
* @psalm-return non-negative-int
|
||||
*/
|
||||
public function cyclomaticComplexity(): int
|
||||
{
|
||||
$cyclomaticComplexity = 0;
|
||||
|
||||
foreach ($this as $item) {
|
||||
$cyclomaticComplexity += $item->cyclomaticComplexity();
|
||||
}
|
||||
|
||||
return $cyclomaticComplexity;
|
||||
}
|
||||
|
||||
public function isFunction(): self
|
||||
{
|
||||
return new self(
|
||||
array_values(
|
||||
array_filter(
|
||||
$this->items,
|
||||
static fn (Complexity $complexity): bool => $complexity->isFunction(),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
public function isMethod(): self
|
||||
{
|
||||
return new self(
|
||||
array_values(
|
||||
array_filter(
|
||||
$this->items,
|
||||
static fn (Complexity $complexity): bool => $complexity->isMethod(),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
public function mergeWith(self $other): self
|
||||
{
|
||||
return new self(
|
||||
array_merge(
|
||||
$this->asArray(),
|
||||
$other->asArray(),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
public function sortByDescendingCyclomaticComplexity(): self
|
||||
{
|
||||
$items = $this->items;
|
||||
|
||||
usort(
|
||||
$items,
|
||||
static function (Complexity $a, Complexity $b): int
|
||||
{
|
||||
return $a->cyclomaticComplexity() <=> $b->cyclomaticComplexity();
|
||||
},
|
||||
);
|
||||
|
||||
return new self(array_reverse($items));
|
||||
}
|
||||
}
|
51
vendor/sebastian/complexity/src/Complexity/ComplexityCollectionIterator.php
vendored
Normal file
51
vendor/sebastian/complexity/src/Complexity/ComplexityCollectionIterator.php
vendored
Normal file
@ -0,0 +1,51 @@
|
||||
<?php declare(strict_types=1);
|
||||
/*
|
||||
* This file is part of sebastian/complexity.
|
||||
*
|
||||
* (c) Sebastian Bergmann <sebastian@phpunit.de>
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
namespace SebastianBergmann\Complexity;
|
||||
|
||||
use Iterator;
|
||||
|
||||
final class ComplexityCollectionIterator implements Iterator
|
||||
{
|
||||
/**
|
||||
* @psalm-var list<Complexity>
|
||||
*/
|
||||
private readonly array $items;
|
||||
private int $position = 0;
|
||||
|
||||
public function __construct(ComplexityCollection $items)
|
||||
{
|
||||
$this->items = $items->asArray();
|
||||
}
|
||||
|
||||
public function rewind(): void
|
||||
{
|
||||
$this->position = 0;
|
||||
}
|
||||
|
||||
public function valid(): bool
|
||||
{
|
||||
return isset($this->items[$this->position]);
|
||||
}
|
||||
|
||||
public function key(): int
|
||||
{
|
||||
return $this->position;
|
||||
}
|
||||
|
||||
public function current(): Complexity
|
||||
{
|
||||
return $this->items[$this->position];
|
||||
}
|
||||
|
||||
public function next(): void
|
||||
{
|
||||
$this->position++;
|
||||
}
|
||||
}
|
16
vendor/sebastian/complexity/src/Exception/Exception.php
vendored
Normal file
16
vendor/sebastian/complexity/src/Exception/Exception.php
vendored
Normal file
@ -0,0 +1,16 @@
|
||||
<?php declare(strict_types=1);
|
||||
/*
|
||||
* This file is part of sebastian/complexity.
|
||||
*
|
||||
* (c) Sebastian Bergmann <sebastian@phpunit.de>
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
namespace SebastianBergmann\Complexity;
|
||||
|
||||
use Throwable;
|
||||
|
||||
interface Exception extends Throwable
|
||||
{
|
||||
}
|
14
vendor/sebastian/complexity/src/Exception/RuntimeException.php
vendored
Normal file
14
vendor/sebastian/complexity/src/Exception/RuntimeException.php
vendored
Normal file
@ -0,0 +1,14 @@
|
||||
<?php declare(strict_types=1);
|
||||
/*
|
||||
* This file is part of sebastian/complexity.
|
||||
*
|
||||
* (c) Sebastian Bergmann <sebastian@phpunit.de>
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
namespace SebastianBergmann\Complexity;
|
||||
|
||||
final class RuntimeException extends \RuntimeException implements Exception
|
||||
{
|
||||
}
|
132
vendor/sebastian/complexity/src/Visitor/ComplexityCalculatingVisitor.php
vendored
Normal file
132
vendor/sebastian/complexity/src/Visitor/ComplexityCalculatingVisitor.php
vendored
Normal file
@ -0,0 +1,132 @@
|
||||
<?php declare(strict_types=1);
|
||||
/*
|
||||
* This file is part of sebastian/complexity.
|
||||
*
|
||||
* (c) Sebastian Bergmann <sebastian@phpunit.de>
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
namespace SebastianBergmann\Complexity;
|
||||
|
||||
use function assert;
|
||||
use function is_array;
|
||||
use PhpParser\Node;
|
||||
use PhpParser\Node\Expr\New_;
|
||||
use PhpParser\Node\Name;
|
||||
use PhpParser\Node\Stmt;
|
||||
use PhpParser\Node\Stmt\Class_;
|
||||
use PhpParser\Node\Stmt\ClassMethod;
|
||||
use PhpParser\Node\Stmt\Function_;
|
||||
use PhpParser\Node\Stmt\Interface_;
|
||||
use PhpParser\Node\Stmt\Trait_;
|
||||
use PhpParser\NodeTraverser;
|
||||
use PhpParser\NodeVisitorAbstract;
|
||||
|
||||
final class ComplexityCalculatingVisitor extends NodeVisitorAbstract
|
||||
{
|
||||
/**
|
||||
* @psalm-var list<Complexity>
|
||||
*/
|
||||
private array $result = [];
|
||||
private bool $shortCircuitTraversal;
|
||||
|
||||
public function __construct(bool $shortCircuitTraversal)
|
||||
{
|
||||
$this->shortCircuitTraversal = $shortCircuitTraversal;
|
||||
}
|
||||
|
||||
public function enterNode(Node $node): ?int
|
||||
{
|
||||
if (!$node instanceof ClassMethod && !$node instanceof Function_) {
|
||||
return null;
|
||||
}
|
||||
|
||||
if ($node instanceof ClassMethod) {
|
||||
if ($node->getAttribute('parent') instanceof Interface_) {
|
||||
return null;
|
||||
}
|
||||
|
||||
if ($node->isAbstract()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
$name = $this->classMethodName($node);
|
||||
} else {
|
||||
$name = $this->functionName($node);
|
||||
}
|
||||
|
||||
$statements = $node->getStmts();
|
||||
|
||||
assert(is_array($statements));
|
||||
|
||||
$this->result[] = new Complexity(
|
||||
$name,
|
||||
$this->cyclomaticComplexity($statements),
|
||||
);
|
||||
|
||||
if ($this->shortCircuitTraversal) {
|
||||
return NodeTraverser::DONT_TRAVERSE_CHILDREN;
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
public function result(): ComplexityCollection
|
||||
{
|
||||
return ComplexityCollection::fromList(...$this->result);
|
||||
}
|
||||
|
||||
/**
|
||||
* @param Stmt[] $statements
|
||||
*
|
||||
* @psalm-return positive-int
|
||||
*/
|
||||
private function cyclomaticComplexity(array $statements): int
|
||||
{
|
||||
$traverser = new NodeTraverser;
|
||||
|
||||
$cyclomaticComplexityCalculatingVisitor = new CyclomaticComplexityCalculatingVisitor;
|
||||
|
||||
$traverser->addVisitor($cyclomaticComplexityCalculatingVisitor);
|
||||
|
||||
/* @noinspection UnusedFunctionResultInspection */
|
||||
$traverser->traverse($statements);
|
||||
|
||||
return $cyclomaticComplexityCalculatingVisitor->cyclomaticComplexity();
|
||||
}
|
||||
|
||||
/**
|
||||
* @psalm-return non-empty-string
|
||||
*/
|
||||
private function classMethodName(ClassMethod $node): string
|
||||
{
|
||||
$parent = $node->getAttribute('parent');
|
||||
|
||||
assert($parent instanceof Class_ || $parent instanceof Trait_);
|
||||
|
||||
if ($parent->getAttribute('parent') instanceof New_) {
|
||||
return 'anonymous class';
|
||||
}
|
||||
|
||||
assert(isset($parent->namespacedName));
|
||||
assert($parent->namespacedName instanceof Name);
|
||||
|
||||
return $parent->namespacedName->toString() . '::' . $node->name->toString();
|
||||
}
|
||||
|
||||
/**
|
||||
* @psalm-return non-empty-string
|
||||
*/
|
||||
private function functionName(Function_ $node): string
|
||||
{
|
||||
assert(isset($node->namespacedName));
|
||||
assert($node->namespacedName instanceof Name);
|
||||
|
||||
$functionName = $node->namespacedName->toString();
|
||||
|
||||
assert($functionName !== '');
|
||||
|
||||
return $functionName;
|
||||
}
|
||||
}
|
60
vendor/sebastian/complexity/src/Visitor/CyclomaticComplexityCalculatingVisitor.php
vendored
Normal file
60
vendor/sebastian/complexity/src/Visitor/CyclomaticComplexityCalculatingVisitor.php
vendored
Normal file
@ -0,0 +1,60 @@
|
||||
<?php declare(strict_types=1);
|
||||
/*
|
||||
* This file is part of sebastian/complexity.
|
||||
*
|
||||
* (c) Sebastian Bergmann <sebastian@phpunit.de>
|
||||
*
|
||||
* For the full copyright and license information, please view the LICENSE
|
||||
* file that was distributed with this source code.
|
||||
*/
|
||||
namespace SebastianBergmann\Complexity;
|
||||
|
||||
use PhpParser\Node;
|
||||
use PhpParser\Node\Expr\BinaryOp\BooleanAnd;
|
||||
use PhpParser\Node\Expr\BinaryOp\BooleanOr;
|
||||
use PhpParser\Node\Expr\BinaryOp\LogicalAnd;
|
||||
use PhpParser\Node\Expr\BinaryOp\LogicalOr;
|
||||
use PhpParser\Node\Expr\Ternary;
|
||||
use PhpParser\Node\Stmt\Case_;
|
||||
use PhpParser\Node\Stmt\Catch_;
|
||||
use PhpParser\Node\Stmt\ElseIf_;
|
||||
use PhpParser\Node\Stmt\For_;
|
||||
use PhpParser\Node\Stmt\Foreach_;
|
||||
use PhpParser\Node\Stmt\If_;
|
||||
use PhpParser\Node\Stmt\While_;
|
||||
use PhpParser\NodeVisitorAbstract;
|
||||
|
||||
final class CyclomaticComplexityCalculatingVisitor extends NodeVisitorAbstract
|
||||
{
|
||||
/**
|
||||
* @psalm-var positive-int
|
||||
*/
|
||||
private int $cyclomaticComplexity = 1;
|
||||
|
||||
public function enterNode(Node $node): void
|
||||
{
|
||||
switch ($node::class) {
|
||||
case BooleanAnd::class:
|
||||
case BooleanOr::class:
|
||||
case Case_::class:
|
||||
case Catch_::class:
|
||||
case ElseIf_::class:
|
||||
case For_::class:
|
||||
case Foreach_::class:
|
||||
case If_::class:
|
||||
case LogicalAnd::class:
|
||||
case LogicalOr::class:
|
||||
case Ternary::class:
|
||||
case While_::class:
|
||||
$this->cyclomaticComplexity++;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @psalm-return positive-int
|
||||
*/
|
||||
public function cyclomaticComplexity(): int
|
||||
{
|
||||
return $this->cyclomaticComplexity;
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user