Chris@13
|
1 <?php
|
Chris@13
|
2
|
Chris@13
|
3 /*
|
Chris@13
|
4 * This file is part of Psy Shell.
|
Chris@13
|
5 *
|
Chris@13
|
6 * (c) 2012-2018 Justin Hileman
|
Chris@13
|
7 *
|
Chris@13
|
8 * For the full copyright and license information, please view the LICENSE
|
Chris@13
|
9 * file that was distributed with this source code.
|
Chris@13
|
10 */
|
Chris@13
|
11
|
Chris@13
|
12 namespace Psy\Formatter;
|
Chris@13
|
13
|
Chris@16
|
14 use Psy\Reflection\ReflectionClassConstant;
|
Chris@16
|
15 use Psy\Reflection\ReflectionConstant_;
|
Chris@13
|
16 use Psy\Reflection\ReflectionLanguageConstruct;
|
Chris@13
|
17 use Psy\Util\Json;
|
Chris@13
|
18 use Symfony\Component\Console\Formatter\OutputFormatter;
|
Chris@13
|
19
|
Chris@13
|
20 /**
|
Chris@13
|
21 * An abstract representation of a function, class or property signature.
|
Chris@13
|
22 */
|
Chris@13
|
23 class SignatureFormatter implements Formatter
|
Chris@13
|
24 {
|
Chris@13
|
25 /**
|
Chris@13
|
26 * Format a signature for the given reflector.
|
Chris@13
|
27 *
|
Chris@13
|
28 * Defers to subclasses to do the actual formatting.
|
Chris@13
|
29 *
|
Chris@13
|
30 * @param \Reflector $reflector
|
Chris@13
|
31 *
|
Chris@13
|
32 * @return string Formatted signature
|
Chris@13
|
33 */
|
Chris@13
|
34 public static function format(\Reflector $reflector)
|
Chris@13
|
35 {
|
Chris@13
|
36 switch (true) {
|
Chris@13
|
37 case $reflector instanceof \ReflectionFunction:
|
Chris@13
|
38 case $reflector instanceof ReflectionLanguageConstruct:
|
Chris@13
|
39 return self::formatFunction($reflector);
|
Chris@13
|
40
|
Chris@13
|
41 // this case also covers \ReflectionObject:
|
Chris@13
|
42 case $reflector instanceof \ReflectionClass:
|
Chris@13
|
43 return self::formatClass($reflector);
|
Chris@13
|
44
|
Chris@16
|
45 case $reflector instanceof ReflectionClassConstant:
|
Chris@16
|
46 case $reflector instanceof \ReflectionClassConstant:
|
Chris@16
|
47 return self::formatClassConstant($reflector);
|
Chris@13
|
48
|
Chris@13
|
49 case $reflector instanceof \ReflectionMethod:
|
Chris@13
|
50 return self::formatMethod($reflector);
|
Chris@13
|
51
|
Chris@13
|
52 case $reflector instanceof \ReflectionProperty:
|
Chris@13
|
53 return self::formatProperty($reflector);
|
Chris@13
|
54
|
Chris@16
|
55 case $reflector instanceof ReflectionConstant_:
|
Chris@16
|
56 return self::formatConstant($reflector);
|
Chris@16
|
57
|
Chris@13
|
58 default:
|
Chris@17
|
59 throw new \InvalidArgumentException('Unexpected Reflector class: ' . \get_class($reflector));
|
Chris@13
|
60 }
|
Chris@13
|
61 }
|
Chris@13
|
62
|
Chris@13
|
63 /**
|
Chris@13
|
64 * Print the signature name.
|
Chris@13
|
65 *
|
Chris@13
|
66 * @param \Reflector $reflector
|
Chris@13
|
67 *
|
Chris@13
|
68 * @return string Formatted name
|
Chris@13
|
69 */
|
Chris@13
|
70 public static function formatName(\Reflector $reflector)
|
Chris@13
|
71 {
|
Chris@13
|
72 return $reflector->getName();
|
Chris@13
|
73 }
|
Chris@13
|
74
|
Chris@13
|
75 /**
|
Chris@13
|
76 * Print the method, property or class modifiers.
|
Chris@13
|
77 *
|
Chris@13
|
78 * @param \Reflector $reflector
|
Chris@13
|
79 *
|
Chris@13
|
80 * @return string Formatted modifiers
|
Chris@13
|
81 */
|
Chris@13
|
82 private static function formatModifiers(\Reflector $reflector)
|
Chris@13
|
83 {
|
Chris@16
|
84 if ($reflector instanceof \ReflectionClass && $reflector->isTrait()) {
|
Chris@16
|
85 // For some reason, PHP 5.x returns `abstract public` modifiers for
|
Chris@16
|
86 // traits. Let's just ignore that business entirely.
|
Chris@17
|
87 if (\version_compare(PHP_VERSION, '7.0.0', '<')) {
|
Chris@16
|
88 return [];
|
Chris@16
|
89 }
|
Chris@16
|
90 }
|
Chris@16
|
91
|
Chris@17
|
92 return \implode(' ', \array_map(function ($modifier) {
|
Chris@17
|
93 return \sprintf('<keyword>%s</keyword>', $modifier);
|
Chris@13
|
94 }, \Reflection::getModifierNames($reflector->getModifiers())));
|
Chris@13
|
95 }
|
Chris@13
|
96
|
Chris@13
|
97 /**
|
Chris@13
|
98 * Format a class signature.
|
Chris@13
|
99 *
|
Chris@13
|
100 * @param \ReflectionClass $reflector
|
Chris@13
|
101 *
|
Chris@13
|
102 * @return string Formatted signature
|
Chris@13
|
103 */
|
Chris@13
|
104 private static function formatClass(\ReflectionClass $reflector)
|
Chris@13
|
105 {
|
Chris@13
|
106 $chunks = [];
|
Chris@13
|
107
|
Chris@13
|
108 if ($modifiers = self::formatModifiers($reflector)) {
|
Chris@13
|
109 $chunks[] = $modifiers;
|
Chris@13
|
110 }
|
Chris@13
|
111
|
Chris@13
|
112 if ($reflector->isTrait()) {
|
Chris@13
|
113 $chunks[] = 'trait';
|
Chris@13
|
114 } else {
|
Chris@13
|
115 $chunks[] = $reflector->isInterface() ? 'interface' : 'class';
|
Chris@13
|
116 }
|
Chris@13
|
117
|
Chris@17
|
118 $chunks[] = \sprintf('<class>%s</class>', self::formatName($reflector));
|
Chris@13
|
119
|
Chris@13
|
120 if ($parent = $reflector->getParentClass()) {
|
Chris@13
|
121 $chunks[] = 'extends';
|
Chris@17
|
122 $chunks[] = \sprintf('<class>%s</class>', $parent->getName());
|
Chris@13
|
123 }
|
Chris@13
|
124
|
Chris@13
|
125 $interfaces = $reflector->getInterfaceNames();
|
Chris@13
|
126 if (!empty($interfaces)) {
|
Chris@17
|
127 \sort($interfaces);
|
Chris@13
|
128
|
Chris@13
|
129 $chunks[] = 'implements';
|
Chris@17
|
130 $chunks[] = \implode(', ', \array_map(function ($name) {
|
Chris@17
|
131 return \sprintf('<class>%s</class>', $name);
|
Chris@13
|
132 }, $interfaces));
|
Chris@13
|
133 }
|
Chris@13
|
134
|
Chris@17
|
135 return \implode(' ', $chunks);
|
Chris@13
|
136 }
|
Chris@13
|
137
|
Chris@13
|
138 /**
|
Chris@13
|
139 * Format a constant signature.
|
Chris@13
|
140 *
|
Chris@16
|
141 * @param ReflectionClassConstant|\ReflectionClassConstant $reflector
|
Chris@13
|
142 *
|
Chris@13
|
143 * @return string Formatted signature
|
Chris@13
|
144 */
|
Chris@16
|
145 private static function formatClassConstant($reflector)
|
Chris@13
|
146 {
|
Chris@13
|
147 $value = $reflector->getValue();
|
Chris@13
|
148 $style = self::getTypeStyle($value);
|
Chris@13
|
149
|
Chris@17
|
150 return \sprintf(
|
Chris@13
|
151 '<keyword>const</keyword> <const>%s</const> = <%s>%s</%s>',
|
Chris@13
|
152 self::formatName($reflector),
|
Chris@13
|
153 $style,
|
Chris@13
|
154 OutputFormatter::escape(Json::encode($value)),
|
Chris@13
|
155 $style
|
Chris@13
|
156 );
|
Chris@13
|
157 }
|
Chris@13
|
158
|
Chris@13
|
159 /**
|
Chris@16
|
160 * Format a constant signature.
|
Chris@16
|
161 *
|
Chris@16
|
162 * @param ReflectionConstant_ $reflector
|
Chris@16
|
163 *
|
Chris@16
|
164 * @return string Formatted signature
|
Chris@16
|
165 */
|
Chris@16
|
166 private static function formatConstant($reflector)
|
Chris@16
|
167 {
|
Chris@16
|
168 $value = $reflector->getValue();
|
Chris@16
|
169 $style = self::getTypeStyle($value);
|
Chris@16
|
170
|
Chris@17
|
171 return \sprintf(
|
Chris@16
|
172 '<keyword>define</keyword>(<string>%s</string>, <%s>%s</%s>)',
|
Chris@16
|
173 OutputFormatter::escape(Json::encode($reflector->getName())),
|
Chris@16
|
174 $style,
|
Chris@16
|
175 OutputFormatter::escape(Json::encode($value)),
|
Chris@16
|
176 $style
|
Chris@16
|
177 );
|
Chris@16
|
178 }
|
Chris@16
|
179
|
Chris@16
|
180 /**
|
Chris@13
|
181 * Helper for getting output style for a given value's type.
|
Chris@13
|
182 *
|
Chris@13
|
183 * @param mixed $value
|
Chris@13
|
184 *
|
Chris@13
|
185 * @return string
|
Chris@13
|
186 */
|
Chris@13
|
187 private static function getTypeStyle($value)
|
Chris@13
|
188 {
|
Chris@17
|
189 if (\is_int($value) || \is_float($value)) {
|
Chris@13
|
190 return 'number';
|
Chris@17
|
191 } elseif (\is_string($value)) {
|
Chris@13
|
192 return 'string';
|
Chris@17
|
193 } elseif (\is_bool($value) || \is_null($value)) {
|
Chris@13
|
194 return 'bool';
|
Chris@13
|
195 } else {
|
Chris@16
|
196 return 'strong'; // @codeCoverageIgnore
|
Chris@13
|
197 }
|
Chris@13
|
198 }
|
Chris@13
|
199
|
Chris@13
|
200 /**
|
Chris@13
|
201 * Format a property signature.
|
Chris@13
|
202 *
|
Chris@13
|
203 * @param \ReflectionProperty $reflector
|
Chris@13
|
204 *
|
Chris@13
|
205 * @return string Formatted signature
|
Chris@13
|
206 */
|
Chris@13
|
207 private static function formatProperty(\ReflectionProperty $reflector)
|
Chris@13
|
208 {
|
Chris@17
|
209 return \sprintf(
|
Chris@13
|
210 '%s <strong>$%s</strong>',
|
Chris@13
|
211 self::formatModifiers($reflector),
|
Chris@13
|
212 $reflector->getName()
|
Chris@13
|
213 );
|
Chris@13
|
214 }
|
Chris@13
|
215
|
Chris@13
|
216 /**
|
Chris@13
|
217 * Format a function signature.
|
Chris@13
|
218 *
|
Chris@13
|
219 * @param \ReflectionFunction $reflector
|
Chris@13
|
220 *
|
Chris@13
|
221 * @return string Formatted signature
|
Chris@13
|
222 */
|
Chris@13
|
223 private static function formatFunction(\ReflectionFunctionAbstract $reflector)
|
Chris@13
|
224 {
|
Chris@17
|
225 return \sprintf(
|
Chris@13
|
226 '<keyword>function</keyword> %s<function>%s</function>(%s)',
|
Chris@13
|
227 $reflector->returnsReference() ? '&' : '',
|
Chris@13
|
228 self::formatName($reflector),
|
Chris@17
|
229 \implode(', ', self::formatFunctionParams($reflector))
|
Chris@13
|
230 );
|
Chris@13
|
231 }
|
Chris@13
|
232
|
Chris@13
|
233 /**
|
Chris@13
|
234 * Format a method signature.
|
Chris@13
|
235 *
|
Chris@13
|
236 * @param \ReflectionMethod $reflector
|
Chris@13
|
237 *
|
Chris@13
|
238 * @return string Formatted signature
|
Chris@13
|
239 */
|
Chris@13
|
240 private static function formatMethod(\ReflectionMethod $reflector)
|
Chris@13
|
241 {
|
Chris@17
|
242 return \sprintf(
|
Chris@13
|
243 '%s %s',
|
Chris@13
|
244 self::formatModifiers($reflector),
|
Chris@13
|
245 self::formatFunction($reflector)
|
Chris@13
|
246 );
|
Chris@13
|
247 }
|
Chris@13
|
248
|
Chris@13
|
249 /**
|
Chris@13
|
250 * Print the function params.
|
Chris@13
|
251 *
|
Chris@13
|
252 * @param \ReflectionFunctionAbstract $reflector
|
Chris@13
|
253 *
|
Chris@13
|
254 * @return array
|
Chris@13
|
255 */
|
Chris@13
|
256 private static function formatFunctionParams(\ReflectionFunctionAbstract $reflector)
|
Chris@13
|
257 {
|
Chris@13
|
258 $params = [];
|
Chris@13
|
259 foreach ($reflector->getParameters() as $param) {
|
Chris@13
|
260 $hint = '';
|
Chris@13
|
261 try {
|
Chris@13
|
262 if ($param->isArray()) {
|
Chris@13
|
263 $hint = '<keyword>array</keyword> ';
|
Chris@13
|
264 } elseif ($class = $param->getClass()) {
|
Chris@17
|
265 $hint = \sprintf('<class>%s</class> ', $class->getName());
|
Chris@13
|
266 }
|
Chris@13
|
267 } catch (\Exception $e) {
|
Chris@13
|
268 // sometimes we just don't know...
|
Chris@13
|
269 // bad class names, or autoloaded classes that haven't been loaded yet, or whathaveyou.
|
Chris@13
|
270 // come to think of it, the only time I've seen this is with the intl extension.
|
Chris@13
|
271
|
Chris@13
|
272 // Hax: we'll try to extract it :P
|
Chris@16
|
273
|
Chris@16
|
274 // @codeCoverageIgnoreStart
|
Chris@17
|
275 $chunks = \explode('$' . $param->getName(), (string) $param);
|
Chris@17
|
276 $chunks = \explode(' ', \trim($chunks[0]));
|
Chris@17
|
277 $guess = \end($chunks);
|
Chris@13
|
278
|
Chris@17
|
279 $hint = \sprintf('<urgent>%s</urgent> ', $guess);
|
Chris@16
|
280 // @codeCoverageIgnoreEnd
|
Chris@13
|
281 }
|
Chris@13
|
282
|
Chris@13
|
283 if ($param->isOptional()) {
|
Chris@13
|
284 if (!$param->isDefaultValueAvailable()) {
|
Chris@13
|
285 $value = 'unknown';
|
Chris@13
|
286 $typeStyle = 'urgent';
|
Chris@13
|
287 } else {
|
Chris@13
|
288 $value = $param->getDefaultValue();
|
Chris@13
|
289 $typeStyle = self::getTypeStyle($value);
|
Chris@17
|
290 $value = \is_array($value) ? 'array()' : \is_null($value) ? 'null' : \var_export($value, true);
|
Chris@13
|
291 }
|
Chris@17
|
292 $default = \sprintf(' = <%s>%s</%s>', $typeStyle, OutputFormatter::escape($value), $typeStyle);
|
Chris@13
|
293 } else {
|
Chris@13
|
294 $default = '';
|
Chris@13
|
295 }
|
Chris@13
|
296
|
Chris@17
|
297 $params[] = \sprintf(
|
Chris@13
|
298 '%s%s<strong>$%s</strong>%s',
|
Chris@13
|
299 $param->isPassedByReference() ? '&' : '',
|
Chris@13
|
300 $hint,
|
Chris@13
|
301 $param->getName(),
|
Chris@13
|
302 $default
|
Chris@13
|
303 );
|
Chris@13
|
304 }
|
Chris@13
|
305
|
Chris@13
|
306 return $params;
|
Chris@13
|
307 }
|
Chris@13
|
308 }
|