annotate vendor/psy/psysh/src/Formatter/SignatureFormatter.php @ 19:fa3358dc1485 tip

Add ndrum files
author Chris Cannam
date Wed, 28 Aug 2019 13:14:47 +0100
parents 129ea1e6d783
children
rev   line source
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 }