ReflectionClassResource.php 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. <?php
  2. /*
  3. * This file is part of the Symfony package.
  4. *
  5. * (c) Fabien Potencier <fabien@symfony.com>
  6. *
  7. * For the full copyright and license information, please view the LICENSE
  8. * file that was distributed with this source code.
  9. */
  10. namespace Symfony\Component\Config\Resource;
  11. use Symfony\Component\EventDispatcher\EventSubscriberInterface;
  12. use Symfony\Component\Messenger\Handler\MessageSubscriberInterface;
  13. use Symfony\Contracts\Service\ServiceSubscriberInterface;
  14. /**
  15. * @author Nicolas Grekas <p@tchwork.com>
  16. *
  17. * @final
  18. */
  19. class ReflectionClassResource implements SelfCheckingResourceInterface
  20. {
  21. private $files = [];
  22. private $className;
  23. private $classReflector;
  24. private $excludedVendors = [];
  25. private $hash;
  26. public function __construct(\ReflectionClass $classReflector, array $excludedVendors = [])
  27. {
  28. $this->className = $classReflector->name;
  29. $this->classReflector = $classReflector;
  30. $this->excludedVendors = $excludedVendors;
  31. }
  32. /**
  33. * {@inheritdoc}
  34. */
  35. public function isFresh(int $timestamp): bool
  36. {
  37. if (null === $this->hash) {
  38. $this->hash = $this->computeHash();
  39. $this->loadFiles($this->classReflector);
  40. }
  41. foreach ($this->files as $file => $v) {
  42. if (false === $filemtime = @filemtime($file)) {
  43. return false;
  44. }
  45. if ($filemtime > $timestamp) {
  46. return $this->hash === $this->computeHash();
  47. }
  48. }
  49. return true;
  50. }
  51. public function __toString(): string
  52. {
  53. return 'reflection.'.$this->className;
  54. }
  55. /**
  56. * @internal
  57. */
  58. public function __sleep(): array
  59. {
  60. if (null === $this->hash) {
  61. $this->hash = $this->computeHash();
  62. $this->loadFiles($this->classReflector);
  63. }
  64. return ['files', 'className', 'hash'];
  65. }
  66. private function loadFiles(\ReflectionClass $class)
  67. {
  68. foreach ($class->getInterfaces() as $v) {
  69. $this->loadFiles($v);
  70. }
  71. do {
  72. $file = $class->getFileName();
  73. if (false !== $file && is_file($file)) {
  74. foreach ($this->excludedVendors as $vendor) {
  75. if (0 === strpos($file, $vendor) && false !== strpbrk(substr($file, \strlen($vendor), 1), '/'.\DIRECTORY_SEPARATOR)) {
  76. $file = false;
  77. break;
  78. }
  79. }
  80. if ($file) {
  81. $this->files[$file] = null;
  82. }
  83. }
  84. foreach ($class->getTraits() as $v) {
  85. $this->loadFiles($v);
  86. }
  87. } while ($class = $class->getParentClass());
  88. }
  89. private function computeHash(): string
  90. {
  91. if (null === $this->classReflector) {
  92. try {
  93. $this->classReflector = new \ReflectionClass($this->className);
  94. } catch (\ReflectionException $e) {
  95. // the class does not exist anymore
  96. return false;
  97. }
  98. }
  99. $hash = hash_init('md5');
  100. foreach ($this->generateSignature($this->classReflector) as $info) {
  101. hash_update($hash, $info);
  102. }
  103. return hash_final($hash);
  104. }
  105. private function generateSignature(\ReflectionClass $class): iterable
  106. {
  107. yield $class->getDocComment();
  108. yield (int) $class->isFinal();
  109. yield (int) $class->isAbstract();
  110. if ($class->isTrait()) {
  111. yield print_r(class_uses($class->name), true);
  112. } else {
  113. yield print_r(class_parents($class->name), true);
  114. yield print_r(class_implements($class->name), true);
  115. yield print_r($class->getConstants(), true);
  116. }
  117. if (!$class->isInterface()) {
  118. $defaults = $class->getDefaultProperties();
  119. foreach ($class->getProperties(\ReflectionProperty::IS_PUBLIC | \ReflectionProperty::IS_PROTECTED) as $p) {
  120. yield $p->getDocComment();
  121. yield $p->isDefault() ? '<default>' : '';
  122. yield $p->isPublic() ? 'public' : 'protected';
  123. yield $p->isStatic() ? 'static' : '';
  124. yield '$'.$p->name;
  125. yield print_r(isset($defaults[$p->name]) && !\is_object($defaults[$p->name]) ? $defaults[$p->name] : null, true);
  126. }
  127. }
  128. foreach ($class->getMethods(\ReflectionMethod::IS_PUBLIC | \ReflectionMethod::IS_PROTECTED) as $m) {
  129. $defaults = [];
  130. $parametersWithUndefinedConstants = [];
  131. foreach ($m->getParameters() as $p) {
  132. if (!$p->isDefaultValueAvailable()) {
  133. $defaults[$p->name] = null;
  134. continue;
  135. }
  136. if (!$p->isDefaultValueConstant() || \defined($p->getDefaultValueConstantName())) {
  137. $defaults[$p->name] = $p->getDefaultValue();
  138. continue;
  139. }
  140. $defaults[$p->name] = $p->getDefaultValueConstantName();
  141. $parametersWithUndefinedConstants[$p->name] = true;
  142. }
  143. if (!$parametersWithUndefinedConstants) {
  144. yield preg_replace('/^ @@.*/m', '', $m);
  145. } else {
  146. $t = $m->getReturnType();
  147. $stack = [
  148. $m->getDocComment(),
  149. $m->getName(),
  150. $m->isAbstract(),
  151. $m->isFinal(),
  152. $m->isStatic(),
  153. $m->isPublic(),
  154. $m->isPrivate(),
  155. $m->isProtected(),
  156. $m->returnsReference(),
  157. $t instanceof \ReflectionNamedType ? ((string) $t->allowsNull()).$t->getName() : (string) $t,
  158. ];
  159. foreach ($m->getParameters() as $p) {
  160. if (!isset($parametersWithUndefinedConstants[$p->name])) {
  161. $stack[] = (string) $p;
  162. } else {
  163. $t = $p->getType();
  164. $stack[] = $p->isOptional();
  165. $stack[] = $t instanceof \ReflectionNamedType ? ((string) $t->allowsNull()).$t->getName() : (string) $t;
  166. $stack[] = $p->isPassedByReference();
  167. $stack[] = $p->isVariadic();
  168. $stack[] = $p->getName();
  169. }
  170. }
  171. yield implode(',', $stack);
  172. }
  173. yield print_r($defaults, true);
  174. }
  175. if ($class->isAbstract() || $class->isInterface() || $class->isTrait()) {
  176. return;
  177. }
  178. if (interface_exists(EventSubscriberInterface::class, false) && $class->isSubclassOf(EventSubscriberInterface::class)) {
  179. yield EventSubscriberInterface::class;
  180. yield print_r($class->name::getSubscribedEvents(), true);
  181. }
  182. if (interface_exists(MessageSubscriberInterface::class, false) && $class->isSubclassOf(MessageSubscriberInterface::class)) {
  183. yield MessageSubscriberInterface::class;
  184. foreach ($class->name::getHandledMessages() as $key => $value) {
  185. yield $key.print_r($value, true);
  186. }
  187. }
  188. if (interface_exists(ServiceSubscriberInterface::class, false) && $class->isSubclassOf(ServiceSubscriberInterface::class)) {
  189. yield ServiceSubscriberInterface::class;
  190. yield print_r($class->name::getSubscribedServices(), true);
  191. }
  192. }
  193. }