ObjectRouteLoader.php 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  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\Routing\Loader;
  11. use Symfony\Component\Config\Loader\Loader;
  12. use Symfony\Component\Config\Resource\FileResource;
  13. use Symfony\Component\Routing\RouteCollection;
  14. /**
  15. * A route loader that calls a method on an object to load the routes.
  16. *
  17. * @author Ryan Weaver <ryan@knpuniversity.com>
  18. */
  19. abstract class ObjectRouteLoader extends Loader
  20. {
  21. /**
  22. * Returns the object that the method will be called on to load routes.
  23. *
  24. * For example, if your application uses a service container,
  25. * the $id may be a service id.
  26. *
  27. * @param string $id
  28. *
  29. * @return object
  30. */
  31. abstract protected function getServiceObject($id);
  32. /**
  33. * Calls the service that will load the routes.
  34. *
  35. * @param mixed $resource Some value that will resolve to a callable
  36. * @param string|null $type The resource type
  37. *
  38. * @return RouteCollection
  39. */
  40. public function load($resource, $type = null)
  41. {
  42. if (1 === substr_count($resource, ':')) {
  43. $resource = str_replace(':', '::', $resource);
  44. @trigger_error(sprintf('Referencing service route loaders with a single colon is deprecated since Symfony 4.1. Use %s instead.', $resource), E_USER_DEPRECATED);
  45. }
  46. $parts = explode('::', $resource);
  47. if (2 != \count($parts)) {
  48. throw new \InvalidArgumentException(sprintf('Invalid resource "%s" passed to the "service" route loader: use the format "service::method"', $resource));
  49. }
  50. $serviceString = $parts[0];
  51. $method = $parts[1];
  52. $loaderObject = $this->getServiceObject($serviceString);
  53. if (!\is_object($loaderObject)) {
  54. throw new \LogicException(sprintf('%s:getServiceObject() must return an object: %s returned', \get_class($this), \gettype($loaderObject)));
  55. }
  56. if (!\is_callable([$loaderObject, $method])) {
  57. throw new \BadMethodCallException(sprintf('Method "%s" not found on "%s" when importing routing resource "%s"', $method, \get_class($loaderObject), $resource));
  58. }
  59. $routeCollection = $loaderObject->$method($this);
  60. if (!$routeCollection instanceof RouteCollection) {
  61. $type = \is_object($routeCollection) ? \get_class($routeCollection) : \gettype($routeCollection);
  62. throw new \LogicException(sprintf('The %s::%s method must return a RouteCollection: %s returned', \get_class($loaderObject), $method, $type));
  63. }
  64. // make the service file tracked so that if it changes, the cache rebuilds
  65. $this->addClassResource(new \ReflectionClass($loaderObject), $routeCollection);
  66. return $routeCollection;
  67. }
  68. /**
  69. * {@inheritdoc}
  70. */
  71. public function supports($resource, $type = null)
  72. {
  73. return 'service' === $type;
  74. }
  75. private function addClassResource(\ReflectionClass $class, RouteCollection $collection)
  76. {
  77. do {
  78. if (is_file($class->getFileName())) {
  79. $collection->addResource(new FileResource($class->getFileName()));
  80. }
  81. } while ($class = $class->getParentClass());
  82. }
  83. }