annotate vendor/psy/psysh/src/Formatter/SignatureFormatter.php @ 13:5fb285c0d0e3

Update Drupal core to 8.4.7 via Composer. Security update; I *think* we've been lucky to get away with this so far, as we don't support self-registration which seems to be used by the so-called "drupalgeddon 2" attack that 8.4.5 was vulnerable to.
author Chris Cannam
date Mon, 23 Apr 2018 09:33:26 +0100
parents
children c2387f117808
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@13 14 use Psy\Reflection\ReflectionConstant;
Chris@13 15 use Psy\Reflection\ReflectionLanguageConstruct;
Chris@13 16 use Psy\Util\Json;
Chris@13 17 use Symfony\Component\Console\Formatter\OutputFormatter;
Chris@13 18
Chris@13 19 /**
Chris@13 20 * An abstract representation of a function, class or property signature.
Chris@13 21 */
Chris@13 22 class SignatureFormatter implements Formatter
Chris@13 23 {
Chris@13 24 /**
Chris@13 25 * Format a signature for the given reflector.
Chris@13 26 *
Chris@13 27 * Defers to subclasses to do the actual formatting.
Chris@13 28 *
Chris@13 29 * @param \Reflector $reflector
Chris@13 30 *
Chris@13 31 * @return string Formatted signature
Chris@13 32 */
Chris@13 33 public static function format(\Reflector $reflector)
Chris@13 34 {
Chris@13 35 switch (true) {
Chris@13 36 case $reflector instanceof \ReflectionFunction:
Chris@13 37 case $reflector instanceof ReflectionLanguageConstruct:
Chris@13 38 return self::formatFunction($reflector);
Chris@13 39
Chris@13 40 // this case also covers \ReflectionObject:
Chris@13 41 case $reflector instanceof \ReflectionClass:
Chris@13 42 return self::formatClass($reflector);
Chris@13 43
Chris@13 44 case $reflector instanceof ReflectionConstant:
Chris@13 45 return self::formatConstant($reflector);
Chris@13 46
Chris@13 47 case $reflector instanceof \ReflectionMethod:
Chris@13 48 return self::formatMethod($reflector);
Chris@13 49
Chris@13 50 case $reflector instanceof \ReflectionProperty:
Chris@13 51 return self::formatProperty($reflector);
Chris@13 52
Chris@13 53 default:
Chris@13 54 throw new \InvalidArgumentException('Unexpected Reflector class: ' . get_class($reflector));
Chris@13 55 }
Chris@13 56 }
Chris@13 57
Chris@13 58 /**
Chris@13 59 * Print the signature name.
Chris@13 60 *
Chris@13 61 * @param \Reflector $reflector
Chris@13 62 *
Chris@13 63 * @return string Formatted name
Chris@13 64 */
Chris@13 65 public static function formatName(\Reflector $reflector)
Chris@13 66 {
Chris@13 67 return $reflector->getName();
Chris@13 68 }
Chris@13 69
Chris@13 70 /**
Chris@13 71 * Print the method, property or class modifiers.
Chris@13 72 *
Chris@13 73 * Technically this should be a trait. Can't wait for 5.4 :)
Chris@13 74 *
Chris@13 75 * @param \Reflector $reflector
Chris@13 76 *
Chris@13 77 * @return string Formatted modifiers
Chris@13 78 */
Chris@13 79 private static function formatModifiers(\Reflector $reflector)
Chris@13 80 {
Chris@13 81 return implode(' ', array_map(function ($modifier) {
Chris@13 82 return sprintf('<keyword>%s</keyword>', $modifier);
Chris@13 83 }, \Reflection::getModifierNames($reflector->getModifiers())));
Chris@13 84 }
Chris@13 85
Chris@13 86 /**
Chris@13 87 * Format a class signature.
Chris@13 88 *
Chris@13 89 * @param \ReflectionClass $reflector
Chris@13 90 *
Chris@13 91 * @return string Formatted signature
Chris@13 92 */
Chris@13 93 private static function formatClass(\ReflectionClass $reflector)
Chris@13 94 {
Chris@13 95 $chunks = [];
Chris@13 96
Chris@13 97 if ($modifiers = self::formatModifiers($reflector)) {
Chris@13 98 $chunks[] = $modifiers;
Chris@13 99 }
Chris@13 100
Chris@13 101 if ($reflector->isTrait()) {
Chris@13 102 $chunks[] = 'trait';
Chris@13 103 } else {
Chris@13 104 $chunks[] = $reflector->isInterface() ? 'interface' : 'class';
Chris@13 105 }
Chris@13 106
Chris@13 107 $chunks[] = sprintf('<class>%s</class>', self::formatName($reflector));
Chris@13 108
Chris@13 109 if ($parent = $reflector->getParentClass()) {
Chris@13 110 $chunks[] = 'extends';
Chris@13 111 $chunks[] = sprintf('<class>%s</class>', $parent->getName());
Chris@13 112 }
Chris@13 113
Chris@13 114 $interfaces = $reflector->getInterfaceNames();
Chris@13 115 if (!empty($interfaces)) {
Chris@13 116 sort($interfaces);
Chris@13 117
Chris@13 118 $chunks[] = 'implements';
Chris@13 119 $chunks[] = implode(', ', array_map(function ($name) {
Chris@13 120 return sprintf('<class>%s</class>', $name);
Chris@13 121 }, $interfaces));
Chris@13 122 }
Chris@13 123
Chris@13 124 return implode(' ', $chunks);
Chris@13 125 }
Chris@13 126
Chris@13 127 /**
Chris@13 128 * Format a constant signature.
Chris@13 129 *
Chris@13 130 * @param ReflectionConstant $reflector
Chris@13 131 *
Chris@13 132 * @return string Formatted signature
Chris@13 133 */
Chris@13 134 private static function formatConstant(ReflectionConstant $reflector)
Chris@13 135 {
Chris@13 136 $value = $reflector->getValue();
Chris@13 137 $style = self::getTypeStyle($value);
Chris@13 138
Chris@13 139 return sprintf(
Chris@13 140 '<keyword>const</keyword> <const>%s</const> = <%s>%s</%s>',
Chris@13 141 self::formatName($reflector),
Chris@13 142 $style,
Chris@13 143 OutputFormatter::escape(Json::encode($value)),
Chris@13 144 $style
Chris@13 145 );
Chris@13 146 }
Chris@13 147
Chris@13 148 /**
Chris@13 149 * Helper for getting output style for a given value's type.
Chris@13 150 *
Chris@13 151 * @param mixed $value
Chris@13 152 *
Chris@13 153 * @return string
Chris@13 154 */
Chris@13 155 private static function getTypeStyle($value)
Chris@13 156 {
Chris@13 157 if (is_int($value) || is_float($value)) {
Chris@13 158 return 'number';
Chris@13 159 } elseif (is_string($value)) {
Chris@13 160 return 'string';
Chris@13 161 } elseif (is_bool($value) || is_null($value)) {
Chris@13 162 return 'bool';
Chris@13 163 } else {
Chris@13 164 return 'strong';
Chris@13 165 }
Chris@13 166 }
Chris@13 167
Chris@13 168 /**
Chris@13 169 * Format a property signature.
Chris@13 170 *
Chris@13 171 * @param \ReflectionProperty $reflector
Chris@13 172 *
Chris@13 173 * @return string Formatted signature
Chris@13 174 */
Chris@13 175 private static function formatProperty(\ReflectionProperty $reflector)
Chris@13 176 {
Chris@13 177 return sprintf(
Chris@13 178 '%s <strong>$%s</strong>',
Chris@13 179 self::formatModifiers($reflector),
Chris@13 180 $reflector->getName()
Chris@13 181 );
Chris@13 182 }
Chris@13 183
Chris@13 184 /**
Chris@13 185 * Format a function signature.
Chris@13 186 *
Chris@13 187 * @param \ReflectionFunction $reflector
Chris@13 188 *
Chris@13 189 * @return string Formatted signature
Chris@13 190 */
Chris@13 191 private static function formatFunction(\ReflectionFunctionAbstract $reflector)
Chris@13 192 {
Chris@13 193 return sprintf(
Chris@13 194 '<keyword>function</keyword> %s<function>%s</function>(%s)',
Chris@13 195 $reflector->returnsReference() ? '&' : '',
Chris@13 196 self::formatName($reflector),
Chris@13 197 implode(', ', self::formatFunctionParams($reflector))
Chris@13 198 );
Chris@13 199 }
Chris@13 200
Chris@13 201 /**
Chris@13 202 * Format a method signature.
Chris@13 203 *
Chris@13 204 * @param \ReflectionMethod $reflector
Chris@13 205 *
Chris@13 206 * @return string Formatted signature
Chris@13 207 */
Chris@13 208 private static function formatMethod(\ReflectionMethod $reflector)
Chris@13 209 {
Chris@13 210 return sprintf(
Chris@13 211 '%s %s',
Chris@13 212 self::formatModifiers($reflector),
Chris@13 213 self::formatFunction($reflector)
Chris@13 214 );
Chris@13 215 }
Chris@13 216
Chris@13 217 /**
Chris@13 218 * Print the function params.
Chris@13 219 *
Chris@13 220 * @param \ReflectionFunctionAbstract $reflector
Chris@13 221 *
Chris@13 222 * @return array
Chris@13 223 */
Chris@13 224 private static function formatFunctionParams(\ReflectionFunctionAbstract $reflector)
Chris@13 225 {
Chris@13 226 $params = [];
Chris@13 227 foreach ($reflector->getParameters() as $param) {
Chris@13 228 $hint = '';
Chris@13 229 try {
Chris@13 230 if ($param->isArray()) {
Chris@13 231 $hint = '<keyword>array</keyword> ';
Chris@13 232 } elseif ($class = $param->getClass()) {
Chris@13 233 $hint = sprintf('<class>%s</class> ', $class->getName());
Chris@13 234 }
Chris@13 235 } catch (\Exception $e) {
Chris@13 236 // sometimes we just don't know...
Chris@13 237 // bad class names, or autoloaded classes that haven't been loaded yet, or whathaveyou.
Chris@13 238 // come to think of it, the only time I've seen this is with the intl extension.
Chris@13 239
Chris@13 240 // Hax: we'll try to extract it :P
Chris@13 241 $chunks = explode('$' . $param->getName(), (string) $param);
Chris@13 242 $chunks = explode(' ', trim($chunks[0]));
Chris@13 243 $guess = end($chunks);
Chris@13 244
Chris@13 245 $hint = sprintf('<urgent>%s</urgent> ', $guess);
Chris@13 246 }
Chris@13 247
Chris@13 248 if ($param->isOptional()) {
Chris@13 249 if (!$param->isDefaultValueAvailable()) {
Chris@13 250 $value = 'unknown';
Chris@13 251 $typeStyle = 'urgent';
Chris@13 252 } else {
Chris@13 253 $value = $param->getDefaultValue();
Chris@13 254 $typeStyle = self::getTypeStyle($value);
Chris@13 255 $value = is_array($value) ? 'array()' : is_null($value) ? 'null' : var_export($value, true);
Chris@13 256 }
Chris@13 257 $default = sprintf(' = <%s>%s</%s>', $typeStyle, OutputFormatter::escape($value), $typeStyle);
Chris@13 258 } else {
Chris@13 259 $default = '';
Chris@13 260 }
Chris@13 261
Chris@13 262 $params[] = sprintf(
Chris@13 263 '%s%s<strong>$%s</strong>%s',
Chris@13 264 $param->isPassedByReference() ? '&' : '',
Chris@13 265 $hint,
Chris@13 266 $param->getName(),
Chris@13 267 $default
Chris@13 268 );
Chris@13 269 }
Chris@13 270
Chris@13 271 return $params;
Chris@13 272 }
Chris@13 273 }