Completed
Push — cs ( 9c9fc7...96045f )
by Akihito
02:05
created

CodeGenMethod::addMethodDocComment()   A

Complexity

Conditions 2
Paths 2

Size

Total Lines 10

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 6
CRAP Score 2

Importance

Changes 0
Metric Value
dl 0
loc 10
ccs 6
cts 6
cp 1
rs 9.9332
c 0
b 0
f 0
cc 2
nc 2
nop 2
crap 2
1
<?php
2
3
declare(strict_types=1);
4
5
namespace Ray\Aop;
6
7
use _HumbugBox90c4dcb919ed\Symfony\Component\Console\Exception\LogicException;
8
use Doctrine\Common\Annotations\AnnotationReader;
9
use PhpParser\Builder\Method;
10
use PhpParser\Builder\Param;
11
use PhpParser\BuilderFactory;
12
use PhpParser\Comment\Doc;
13
use PhpParser\Node;
14
use PhpParser\Node\NullableType;
15
use PhpParser\Node\Stmt\Class_;
16
use PhpParser\Node\Stmt\ClassMethod;
17
use PhpParser\NodeTraverser;
18
use PhpParser\Parser;
19
use PhpParser\PrettyPrinter\Standard;
20
use Ray\Aop\Annotation\AbstractAssisted;
21
use function is_string;
22
23
final class CodeGenMethod
24
{
25
    /**
26
     * @var \PhpParser\Parser
27
     */
28
    private $parser;
29
30
    /**
31
     * @var \PhpParser\BuilderFactory
32
     */
33
    private $factory;
34
35
    /**
36
     * @var \PhpParser\PrettyPrinter\Standard
37
     */
38
    private $printer;
39
40
    private $reader;
41
42
    /**
43
     * @var AbstractAssisted
44
     */
45
    private $assisted;
46
47 33
    public function __construct(
48
        Parser $parser,
49
        BuilderFactory $factory,
50
        Standard $printer
51
    ) {
52 33
        $this->parser = $parser;
53 33
        $this->factory = $factory;
54 33
        $this->printer = $printer;
55 33
        $this->reader = new AnnotationReader;
56 33
    }
57
58 17
    public function getMethods(\ReflectionClass $class, BindInterface $bind) : array
59
    {
60 17
        $bindingMethods = array_keys($bind->getBindings());
61 17
        $stmts = [];
62 17
        $methods = $class->getMethods();
63 17
        foreach ($methods as $method) {
64 16
            $this->assisted = $this->reader->getMethodAnnotation($method, AbstractAssisted::class);
65 16
            $isBindingMethod = in_array($method->name, $bindingMethods, true);
66
            /* @var $method \ReflectionMethod */
67 16
            if ($isBindingMethod && $method->isPublic()) {
68 16
                $stmts[] = $this->getMethod($method);
69
            }
70
        }
71
72 17
        return $stmts;
73
    }
74
75
    /**
76
     * Return method statement
77
     *
78
     * @return \PhpParser\Node\Stmt\ClassMethod
79
     */
80 15
    private function getMethod(\ReflectionMethod $method)
81
    {
82 15
        $methodStmt = $this->factory->method($method->name);
83 15
        $params = $method->getParameters();
84 15
        foreach ($params as $param) {
85 13
            $methodStmt = $this->getMethodStatement($param, $methodStmt);
86
        }
87 15
        $returnType = $method->getReturnType();
88 15
        if ($returnType instanceof \ReflectionType) {
0 ignored issues
show
Bug introduced by
The class ReflectionType does not exist. Did you forget a USE statement, or did you not list all dependencies?

This error could be the result of:

1. Missing dependencies

PHP Analyzer uses your composer.json file (if available) to determine the dependencies of your project and to determine all the available classes and functions. It expects the composer.json to be in the root folder of your repository.

Are you sure this class is defined by one of your dependencies, or did you maybe not list a dependency in either the require or require-dev section?

2. Missing use statement

PHP does not complain about undefined classes in ìnstanceof checks. For example, the following PHP code will work perfectly fine:

if ($x instanceof DoesNotExist) {
    // Do something.
}

If you have not tested against this specific condition, such errors might go unnoticed.

Loading history...
89 5
            $this->setReturnType($returnType, $methodStmt);
90
        }
91 15
        $methodInsideStatements = $this->getMethodInsideStatement($method);
92 15
        $methodStmt->addStmts($methodInsideStatements);
93
94 15
        return $this->addMethodDocComment($methodStmt, $method);
95
    }
96
97
    /**
98
     * Return parameter reflection
99
     */
100 13
    private function getMethodStatement(\ReflectionParameter $param, Method $methodStmt) : Method
101
    {
102
        /* @var $paramStmt Param */
103 13
        $paramStmt = $this->factory->param($param->name);
104
        /* @var $param \ReflectionParameter */
105 13
        $this->setParameterType($param, $paramStmt);
106 13
        $this->setDefault($param, $paramStmt);
107 13
        $methodStmt->addParam($paramStmt);
108
109 13
        return $methodStmt;
110
    }
111
112 15
    private function addMethodDocComment(Method $methodStmt, \ReflectionMethod $method) : ClassMethod
113
    {
114 15
        $node = $methodStmt->getNode();
115 15
        $docComment = $method->getDocComment();
116 15
        if ($docComment) {
117 6
            $node->setAttribute('comments', [new Doc($docComment)]);
118
        }
119
120 15
        return $node;
121
    }
122
123
    /**
124
     * @return \PhpParser\Node[]
125
     */
126 15
    private function getMethodInsideStatement(\ReflectionMethod $method) : array
127
    {
128 15
        $traverser = new NodeTraverser;
129 15
        $traverser->addVisitor(new AopTemplateConverter($method));
130 15
        $stmts = $this->getTemplateMethodNodeStmts();
131
132
        // traverse
133 15
        return $traverser->traverse($stmts);
134
    }
135
136 13
    private function setDefault(\ReflectionParameter $param, Param $paramStmt)
137
    {
138 13
        if ($param->isDefaultValueAvailable()) {
139 3
            $paramStmt->setDefault($param->getDefaultValue());
140
141 3
            return;
142
        }
143 13
        if ($this->assisted instanceof AbstractAssisted && in_array($param->name, $this->assisted->values, true)) {
144 1
            $paramStmt->setDefault(null);
145
        }
146 13
    }
147
148 13
    private function setParameterType(\ReflectionParameter $param, Param $paramStmt)
149
    {
150 13
        $type = $param->getType();
151 13
        if ($type == null) {
152 8
            return;
153
        }
154 7
        if ($param->isVariadic()) {
155 1
            $paramStmt->makeVariadic();
156
        }
157 7
        $paramString = (string) $param;
158 7
        $isNullableType = is_int(strpos($paramString, '<required>')) && is_int(strpos($paramString, 'or NULL'));
159 7
        $destType = $isNullableType ? new NullableType((string) $type) : (string) $type;
160 7
        $paramStmt->setTypeHint($destType);
0 ignored issues
show
Deprecated Code introduced by
The method PhpParser\Builder\Param::setTypeHint() has been deprecated with message: Use setType() instead

This method has been deprecated. The supplier of the class has supplied an explanatory message.

The explanatory message should give you some clue as to whether and when the method will be removed from the class and what other method or class to use instead.

Loading history...
161 7
    }
162
163 5
    private function setReturnType(\ReflectionType $returnType, Method $methodStmt)
164
    {
165 5
        $type = $returnType->allowsNull() ? new NullableType((string) $returnType) : (string) $returnType;
166 5
        $methodStmt->setReturnType($type);
167 5
    }
168
169
    /**
170
     * @return Node[]
171
     */
172 15
    private function getTemplateMethodNodeStmts() : array
173
    {
174 15
        $templateFile = dirname(__DIR__) . '/template/AopTemplate.php';
175 15
        $code = file_get_contents($templateFile);
176 15
        if (! is_string($code)) {
177
            throw new LogicException; // @codeCoverageIgnore
178
        }
179
        /** @var string $code */
180 15
        $node = $this->parser->parse($code)[0];
181 15
        if (! $node instanceof Class_) {
182
            throw new \LogicException; // @codeCoverageIgnore
183
        }
184 15
        $methodNode = $node->getMethods()[0];
185 15
        if ($methodNode->stmts === null) {
186
            throw new \LogicException; // @codeCoverageIgnore
187
        }
188
189 15
        return $methodNode->stmts;
190
    }
191
}
192