From a6c393effcb79907c9e645bc877ee197abacda1f Mon Sep 17 00:00:00 2001 From: Dries <dries@buytaert.net> Date: Fri, 4 Jan 2013 11:18:00 -0500 Subject: [PATCH] Issue #1874530 by Crell: Add Symfony CMF Routing component. --- core/composer.json | 3 +- core/composer.lock | 55 +- core/vendor/autoload.php | 2 +- core/vendor/composer/autoload_namespaces.php | 1 + core/vendor/composer/autoload_real.php | 6 +- core/vendor/composer/installed.json | 51 ++ .../Symfony/Cmf/Component/Routing/.travis.yml | 15 + .../Cmf/Component/Routing/ChainRouter.php | 269 ++++++++ .../Routing/ChainedRouterInterface.php | 25 + .../Routing/ContentAwareGenerator.php | 236 +++++++ .../Routing/ContentRepositoryInterface.php | 27 + .../Cmf/Component/Routing/DynamicRouter.php | 297 +++++++++ .../Routing/Enhancer/FieldByClassEnhancer.php | 78 +++ .../Routing/Enhancer/FieldMapEnhancer.php | 61 ++ .../Enhancer/FieldPresenceEnhancer.php | 60 ++ .../Routing/Enhancer/RouteContentEnhancer.php | 70 ++ .../Enhancer/RouteEnhancerInterface.php | 27 + .../Symfony/Cmf/Component/Routing/LICENSE | 23 + .../NestedMatcher/ConfigurableUrlMatcher.php | 66 ++ .../NestedMatcher/FinalMatcherInterface.php | 29 + .../Routing/NestedMatcher/NestedMatcher.php | 168 +++++ .../NestedMatcher/RouteFilterInterface.php | 36 + .../Routing/NestedMatcher/UrlMatcher.php | 46 ++ .../Routing/ProviderBasedGenerator.php | 67 ++ .../Symfony/Cmf/Component/Routing/README.md | 7 + .../Routing/RedirectRouteInterface.php | 76 +++ .../Component/Routing/RouteAwareInterface.php | 22 + .../Routing/RouteObjectInterface.php | 70 ++ .../Routing/RouteProviderInterface.php | 73 +++ .../Routing/Test/CmfUnitTestCase.php | 16 + .../Enhancer/FieldByClassEnhancerTest.php | 62 ++ .../Tests/Enhancer/FieldMapEnhancerTest.php | 61 ++ .../Enhancer/FieldPresenceEnhancerTest.php | 51 ++ .../Enhancer/RouteContentEnhancerTest.php | 78 +++ .../Routing/Tests/Enhancer/RouteObject.php | 18 + .../ConfigurableUrlMatcherTest.php | 129 ++++ .../Tests/NestedMatcher/NestedMatcherTest.php | 134 ++++ .../Tests/NestedMatcher/UrlMatcherTest.php | 148 +++++ .../Routing/Tests/Routing/ChainRouterTest.php | 617 ++++++++++++++++++ .../Routing/ContentAwareGeneratorTest.php | 285 ++++++++ .../Tests/Routing/DynamicRouterTest.php | 241 +++++++ .../Routing/ProviderBasedGeneratorTest.php | 94 +++ .../Routing/Tests/Routing/RouteMock.php | 39 ++ .../Cmf/Component/Routing/Tests/bootstrap.php | 19 + .../Cmf/Component/Routing/composer.json | 29 + .../Cmf/Component/Routing/phpunit.xml.dist | 26 + 46 files changed, 4006 insertions(+), 7 deletions(-) create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/.travis.yml create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ChainRouter.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ChainedRouterInterface.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ContentAwareGenerator.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ContentRepositoryInterface.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/DynamicRouter.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/FieldByClassEnhancer.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/FieldMapEnhancer.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/FieldPresenceEnhancer.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/RouteContentEnhancer.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/RouteEnhancerInterface.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/LICENSE create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/ConfigurableUrlMatcher.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/FinalMatcherInterface.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/NestedMatcher.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/RouteFilterInterface.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/UrlMatcher.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ProviderBasedGenerator.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/README.md create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RedirectRouteInterface.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RouteAwareInterface.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RouteObjectInterface.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RouteProviderInterface.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Test/CmfUnitTestCase.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/FieldByClassEnhancerTest.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/FieldMapEnhancerTest.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/FieldPresenceEnhancerTest.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/RouteContentEnhancerTest.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/RouteObject.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/NestedMatcher/ConfigurableUrlMatcherTest.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/NestedMatcher/NestedMatcherTest.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/NestedMatcher/UrlMatcherTest.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/ChainRouterTest.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/ContentAwareGeneratorTest.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/DynamicRouterTest.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/ProviderBasedGeneratorTest.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/RouteMock.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/bootstrap.php create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/composer.json create mode 100644 core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/phpunit.xml.dist diff --git a/core/composer.json b/core/composer.json index 87e60b4647d7..26030b061ede 100644 --- a/core/composer.json +++ b/core/composer.json @@ -14,7 +14,8 @@ "twig/twig": "1.*@stable", "doctrine/common": "2.3.*@stable", "guzzle/http": "*", - "kriswallsmith/assetic": "1.1.*@alpha" + "kriswallsmith/assetic": "1.1.*@alpha", + "symfony-cmf/routing": "1.0.*@dev" }, "minimum-stability": "dev" } diff --git a/core/composer.lock b/core/composer.lock index 1894ab38cfb0..d8e860821325 100644 --- a/core/composer.lock +++ b/core/composer.lock @@ -1,5 +1,5 @@ { - "hash": "5d17aee0bd24c24563c2c864600fc5bd", + "hash": "27b5fb7194e0d492c69d372d8ba17b2b", "packages": [ { "name": "doctrine/common", @@ -309,6 +309,56 @@ "minification" ] }, + { + "name": "symfony-cmf/routing", + "version": "dev-master", + "target-dir": "Symfony/Cmf/Component/Routing", + "source": { + "type": "git", + "url": "https://github.com/symfony-cmf/Routing", + "reference": "1.0.0-alpha3" + }, + "dist": { + "type": "zip", + "url": "https://github.com/symfony-cmf/Routing/archive/1.0.0-alpha3.zip", + "reference": "1.0.0-alpha3", + "shasum": "" + }, + "require": { + "php": ">=5.3.2", + "symfony/routing": ">=2.1,<2.3-dev", + "symfony/http-kernel": ">=2.1,<2.3-dev" + }, + "time": "2012-12-16 17:52:57", + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0-dev" + } + }, + "installation-source": "source", + "autoload": { + "psr-0": { + "Symfony\\Cmf\\Component\\Routing": "" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Symfony CMF Community", + "homepage": "https://github.com/symfony-cmf/Routing/contributors" + } + ], + "description": "Extends the Symfony2 routing component for dynamic routes and chaining several routers", + "homepage": "http://cmf.symfony.com", + "keywords": [ + "database", + "routing" + ] + }, { "name": "symfony/class-loader", "version": "dev-master", @@ -848,6 +898,7 @@ "stability-flags": { "twig/twig": 0, "doctrine/common": 0, - "kriswallsmith/assetic": 15 + "kriswallsmith/assetic": 15, + "symfony-cmf/routing": 20 } } diff --git a/core/vendor/autoload.php b/core/vendor/autoload.php index daa7ee7447bc..7bc91232cbcd 100644 --- a/core/vendor/autoload.php +++ b/core/vendor/autoload.php @@ -4,4 +4,4 @@ require_once __DIR__ . '/composer' . '/autoload_real.php'; -return ComposerAutoloaderInit295209ab8f7c3b45c210d28fa6db3592::getLoader(); +return ComposerAutoloaderInit23a41d5f637bb8e297fc063ef4ab931a::getLoader(); diff --git a/core/vendor/composer/autoload_namespaces.php b/core/vendor/composer/autoload_namespaces.php index 907240136d62..ecb4b198634e 100644 --- a/core/vendor/composer/autoload_namespaces.php +++ b/core/vendor/composer/autoload_namespaces.php @@ -16,6 +16,7 @@ 'Symfony\\Component\\EventDispatcher\\' => $vendorDir . '/symfony/event-dispatcher/', 'Symfony\\Component\\DependencyInjection\\' => $vendorDir . '/symfony/dependency-injection/', 'Symfony\\Component\\ClassLoader\\' => $vendorDir . '/symfony/class-loader/', + 'Symfony\\Cmf\\Component\\Routing' => $vendorDir . '/symfony-cmf/routing/', 'Guzzle\\Stream' => $vendorDir . '/guzzle/stream/', 'Guzzle\\Parser' => $vendorDir . '/guzzle/parser/', 'Guzzle\\Http' => $vendorDir . '/guzzle/http/', diff --git a/core/vendor/composer/autoload_real.php b/core/vendor/composer/autoload_real.php index dbe463c780bc..9634e0b11d73 100644 --- a/core/vendor/composer/autoload_real.php +++ b/core/vendor/composer/autoload_real.php @@ -2,7 +2,7 @@ // autoload_real.php generated by Composer -class ComposerAutoloaderInit295209ab8f7c3b45c210d28fa6db3592 +class ComposerAutoloaderInit23a41d5f637bb8e297fc063ef4ab931a { private static $loader; @@ -19,9 +19,9 @@ public static function getLoader() return static::$loader; } - spl_autoload_register(array('ComposerAutoloaderInit295209ab8f7c3b45c210d28fa6db3592', 'loadClassLoader')); + spl_autoload_register(array('ComposerAutoloaderInit23a41d5f637bb8e297fc063ef4ab931a', 'loadClassLoader')); static::$loader = $loader = new \Composer\Autoload\ClassLoader(); - spl_autoload_unregister(array('ComposerAutoloaderInit295209ab8f7c3b45c210d28fa6db3592', 'loadClassLoader')); + spl_autoload_unregister(array('ComposerAutoloaderInit23a41d5f637bb8e297fc063ef4ab931a', 'loadClassLoader')); $vendorDir = dirname(__DIR__); $baseDir = dirname($vendorDir); diff --git a/core/vendor/composer/installed.json b/core/vendor/composer/installed.json index aaa926b36ccb..9dd2912b4cdc 100644 --- a/core/vendor/composer/installed.json +++ b/core/vendor/composer/installed.json @@ -852,5 +852,56 @@ "client", "Guzzle" ] + }, + { + "name": "symfony-cmf/routing", + "version": "dev-master", + "version_normalized": "9999999-dev", + "target-dir": "Symfony/Cmf/Component/Routing", + "source": { + "type": "git", + "url": "https://github.com/symfony-cmf/Routing", + "reference": "1.0.0-alpha3" + }, + "dist": { + "type": "zip", + "url": "https://github.com/symfony-cmf/Routing/archive/1.0.0-alpha3.zip", + "reference": "1.0.0-alpha3", + "shasum": "" + }, + "require": { + "php": ">=5.3.2", + "symfony/routing": ">=2.1,<2.3-dev", + "symfony/http-kernel": ">=2.1,<2.3-dev" + }, + "time": "2012-12-16 17:52:57", + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0-dev" + } + }, + "installation-source": "source", + "autoload": { + "psr-0": { + "Symfony\\Cmf\\Component\\Routing": "" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Symfony CMF Community", + "homepage": "https://github.com/symfony-cmf/Routing/contributors" + } + ], + "description": "Extends the Symfony2 routing component for dynamic routes and chaining several routers", + "homepage": "http://cmf.symfony.com", + "keywords": [ + "database", + "routing" + ] } ] diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/.travis.yml b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/.travis.yml new file mode 100644 index 000000000000..8bae46cfb33f --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/.travis.yml @@ -0,0 +1,15 @@ +language: php + +env: + - SYMFONY_VERSION=2.1.* + - SYMFONY_VERSION=dev-master + +before_script: + - composer require symfony/routing:${SYMFONY_VERSION} + - composer install --dev + +script: phpunit --coverage-text + +notifications: + irc: "irc.freenode.org#symfony-cmf" + email: "symfony-cmf-devs@googlegroups.com" diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ChainRouter.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ChainRouter.php new file mode 100644 index 000000000000..6160fc3cc8b6 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ChainRouter.php @@ -0,0 +1,269 @@ +<?php + +namespace Symfony\Cmf\Component\Routing; + +use Symfony\Component\Routing\RouterInterface; +use Symfony\Component\Routing\Matcher\RequestMatcherInterface; +use Symfony\Component\Routing\RequestContext; +use Symfony\Component\Routing\RequestContextAwareInterface; +use Symfony\Component\Routing\Exception\ResourceNotFoundException; +use Symfony\Component\Routing\Exception\RouteNotFoundException; +use Symfony\Component\Routing\Exception\MethodNotAllowedException; +use Symfony\Component\Routing\RouteCollection; +use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\HttpKernel\CacheWarmer\WarmableInterface; +use Symfony\Component\HttpKernel\Log\LoggerInterface; + +/** + * ChainRouter + * + * Allows access to a lot of different routers. + * + * @author Henrik Bjornskov <henrik@bjrnskov.dk> + * @author Magnus Nordlander <magnus@e-butik.se> + */ +class ChainRouter implements RouterInterface, RequestMatcherInterface, WarmableInterface +{ + /** + * @var \Symfony\Component\Routing\RequestContext + */ + private $context; + + /** + * @var Symfony\Component\Routing\RouterInterface[] + */ + private $routers = array(); + + /** + * @var \Symfony\Component\Routing\RouterInterface[] Array of routers, sorted by priority + */ + private $sortedRouters; + + /** + * @var \Symfony\Component\Routing\RouteCollection + */ + private $routeCollection; + + /** + * @var null|\Symfony\Component\HttpKernel\Log\LoggerInterface + */ + protected $logger; + + /** + * @param LoggerInterface $logger + */ + public function __construct(LoggerInterface $logger = null) + { + $this->logger = $logger; + } + + /** + * @return RequestContext + */ + public function getContext() + { + return $this->context; + } + + /** + * Add a Router to the index + * + * @param RouterInterface $router The router instance + * @param integer $priority The priority + */ + public function add(RouterInterface $router, $priority = 0) + { + if (empty($this->routers[$priority])) { + $this->routers[$priority] = array(); + } + + $this->routers[$priority][] = $router; + $this->sortedRouters = array(); + } + + /** + * Sorts the routers and flattens them. + * + * @return array + */ + public function all() + { + if (empty($this->sortedRouters)) { + $this->sortedRouters = $this->sortRouters(); + + // setContext() is done here instead of in add() to avoid fatal errors when clearing and warming up caches + // See https://github.com/symfony-cmf/Routing/pull/18 + $context = $this->getContext(); + if (null !== $context) { + foreach ($this->sortedRouters as $router) { + if ($router instanceof RequestContextAwareInterface) { + $router->setContext($context); + } + } + } + } + + return $this->sortedRouters; + } + + /** + * Sort routers by priority. + * The highest priority number is the highest priority (reverse sorting) + * + * @return RouterInterface[] + */ + protected function sortRouters() + { + $sortedRouters = array(); + krsort($this->routers); + + foreach ($this->routers as $routers) { + $sortedRouters = array_merge($sortedRouters, $routers); + } + + return $sortedRouters; + } + + /** + * {@inheritdoc} + * + * Loops through all routes and tries to match the passed url. + * + * Note: You should use matchRequest if you can. + */ + public function match($url) + { + $methodNotAllowed = null; + + /** @var $router ChainedRouterInterface */ + foreach ($this->all() as $router) { + try { + return $router->match($url); + } catch (ResourceNotFoundException $e) { + if ($this->logger) { + $this->logger->info('Router '.get_class($router).' was not able to match, message "'.$e->getMessage().'"'); + } + // Needs special care + } catch (MethodNotAllowedException $e) { + if ($this->logger) { + $this->logger->info('Router '.get_class($router).' throws MethodNotAllowedException with message "'.$e->getMessage().'"'); + } + $methodNotAllowed = $e; + } + } + + throw $methodNotAllowed ?: new ResourceNotFoundException("None of the routers in the chain matched '$url'"); + } + + /** + * {@inheritdoc} + * + * Loops through all routes and tries to match the passed request. + */ + public function matchRequest(Request $request) + { + $methodNotAllowed = null; + + foreach ($this->all() as $router) { + try { + // the request/url match logic is the same as in Symfony/Component/HttpKernel/EventListener/RouterListener.php + // matching requests is more powerful than matching URLs only, so try that first + if ($router instanceof RequestMatcherInterface) { + return $router->matchRequest($request); + } + return $router->match($request->getPathInfo()); + } catch (ResourceNotFoundException $e) { + if ($this->logger) { + $this->logger->info('Router '.get_class($router).' was not able to match, message "'.$e->getMessage().'"'); + } + // Needs special care + } catch (MethodNotAllowedException $e) { + if ($this->logger) { + $this->logger->info('Router '.get_class($router).' throws MethodNotAllowedException with message "'.$e->getMessage().'"'); + } + $methodNotAllowed = $e; + } + } + + throw $methodNotAllowed ?: new ResourceNotFoundException("None of the routers in the chain matched this request"); + } + + /** + * {@inheritdoc} + * + * Loops through all registered routers and returns a router if one is found. + * It will always return the first route generated. + */ + public function generate($name, $parameters = array(), $absolute = false) + { + /** @var $router ChainedRouterInterface */ + foreach ($this->all() as $router) { + + // if $name and $router does not implement ChainedRouterInterface and $name is not a string, continue + // if $name and $router does not implement ChainedRouterInterface and $name is string but does not match a default Symfony2 route name, continue + if ($name && !$router instanceof ChainedRouterInterface) { + if (!is_string($name) || !preg_match('/^[a-z0-9A-Z_.]+$/', $name)) { + continue; + } + } + + // If $router implements ChainedRouterInterface but doesn't support this route name, continue + if ($router instanceof ChainedRouterInterface && !$router->supports($name)) { + continue; + } + + try { + return $router->generate($name, $parameters, $absolute); + } catch (RouteNotFoundException $e) { + if ($this->logger) { + $this->logger->info($e->getMessage()); + } + } + } + + throw new RouteNotFoundException(sprintf('None of the chained routers were able to generate route "%s".', $name)); + } + + /** + * {@inheritdoc} + */ + public function setContext(RequestContext $context) + { + foreach ($this->all() as $router) { + if ($router instanceof RequestContextAwareInterface) { + $router->setContext($context); + } + } + + $this->context = $context; + } + + /** + * {@inheritdoc} + * + * check for each contained router if it can warmup + */ + public function warmUp($cacheDir) + { + foreach ($this->all() as $router) { + if ($router instanceof WarmableInterface) { + $router->warmUp($cacheDir); + } + } + } + + /** + * {@inheritdoc} + */ + public function getRouteCollection() + { + if (!$this->routeCollection instanceof RouteCollection) { + $this->routeCollection = new RouteCollection(); + foreach ($this->all() as $router) { + $this->routeCollection->addCollection($router->getRouteCollection()); + } + } + + return $this->routeCollection; + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ChainedRouterInterface.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ChainedRouterInterface.php new file mode 100644 index 000000000000..a4247aa03863 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ChainedRouterInterface.php @@ -0,0 +1,25 @@ +<?php + +namespace Symfony\Cmf\Component\Routing; + +use Symfony\Component\Routing\RouterInterface; + +/** + * Use this interface on custom routers that can handle non-string route + * "names". + */ +interface ChainedRouterInterface extends RouterInterface +{ + /** + * Whether the router supports the thing in $name to generate a route. + * + * This check does not need to look if the specific instance can be + * resolved to a route, only whether the router can generate routes from + * objects of this class. + + * @param mixed $name The route name or route object + * + * @return bool + */ + public function supports($name); +} \ No newline at end of file diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ContentAwareGenerator.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ContentAwareGenerator.php new file mode 100644 index 000000000000..aff0c0026cde --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ContentAwareGenerator.php @@ -0,0 +1,236 @@ +<?php + +namespace Symfony\Cmf\Component\Routing; + +use Symfony\Component\Routing\Route as SymfonyRoute; +use Symfony\Component\Routing\RequestContext; +use Symfony\Component\Routing\Exception\RouteNotFoundException; +use Symfony\Component\Routing\RouteCollection; +use Symfony\Component\Routing\Generator\UrlGeneratorInterface; +use Symfony\Component\Routing\Generator\UrlGenerator; + +use Symfony\Cmf\Component\Routing\RouteProviderInterface; + +/** + * A generator that tries to generate routes from object, route names or + * content objects or names. + * + * @author Philippo de Santis + * @author David Buchmann + * @author Uwe Jäger + */ +class ContentAwareGenerator extends ProviderBasedGenerator +{ + /** + * The content repository used to find content by it's id + * This can be used to specify a parameter content_id when generating urls + * + * This is optional and might not be initialized. + * + * @var ContentRepositoryInterface + */ + protected $contentRepository; + + /** + * Set an optional content repository to find content by ids + * + * @param ContentRepositoryInterface $contentRepository + */ + public function setContentRepository(ContentRepositoryInterface $contentRepository) + { + $this->contentRepository = $contentRepository; + } + + /** + * {@inheritDoc} + * + * @param string $name ignored + * @param array $parameters must either contain the field 'route' with a + * RouteObjectInterface or the field 'content' with the document + * instance to get the route for (implementing RouteAwareInterface) + * + * @throws RouteNotFoundException If there is no such route in the database + */ + public function generate($name, $parameters = array(), $absolute = false) + { + if ($name instanceof SymfonyRoute) { + $route = $this->getBestLocaleRoute($name, $parameters); + } elseif (is_string($name) && $name) { + $route = $this->getRouteByName($name, $parameters); + } else { + $route = $this->getRouteByContent($name, $parameters); + } + + if (! $route instanceof SymfonyRoute) { + $hint = is_object($route) ? get_class($route) : gettype($route); + throw new RouteNotFoundException('Route of this document is not an instance of Symfony\Component\Routing\Route but: '.$hint); + } + + return parent::generate($route, $parameters, $absolute); + } + + /** + * Get the route by a string name + * + * @param string $route + * @param array $parameters + * + * @return SymfonyRoute + * + * @throws RouteNotFoundException if there is no route found for the provided name + */ + protected function getRouteByName($name, array $parameters) + { + $route = $this->provider->getRouteByName($name, $parameters); + if (empty($route)) { + throw new RouteNotFoundException('No route found for name: ' . $name); + } + + return $this->getBestLocaleRoute($route, $parameters); + } + + /** + * Determine if there is a route with matching locale associated with the + * given route via associated content. + * + * @param SymfonyRoute $route + * @param array $parameters + * + * @return SymfonyRoute either the passed route or an alternative with better locale + */ + protected function getBestLocaleRoute(SymfonyRoute $route, $parameters) + { + if (! $route instanceof RouteObjectInterface) { + // this route has no content, we can't get the alternatives + return $route; + } + $locale = $this->getLocale($parameters); + if (! $this->checkLocaleRequirement($route, $locale)) { + $content = $route->getRouteContent(); + if ($content instanceof RouteAwareInterface) { + $routes = $content->getRoutes(); + $contentRoute = $this->getRouteByLocale($routes, $locale); + if ($contentRoute) { + return $contentRoute; + } + } + } + + return $route; + } + + /** + * Get the route based on the content field in parameters + * + * Called in generate when there is no route given in the parameters. + * + * If there is more than one route for the content, tries to find the + * first one that matches the _locale (provided in $parameters or otherwise + * defaulting to the request locale). + * + * If none is found, falls back to just return the first route. + * + * @param mixed $name + * @param array $parameters which should contain a content field containing a RouteAwareInterface object + * + * @return SymfonyRoute the route instance + * + * @throws RouteNotFoundException if there is no content field in the + * parameters or its not possible to build a route from that object + */ + protected function getRouteByContent($name, &$parameters) + { + if ($name instanceof RouteAwareInterface) { + $content = $name; + } elseif (isset($parameters['content_id']) && null !== $this->contentRepository) { + $content = $this->contentRepository->findById($parameters['content_id']); + } elseif (isset($parameters['content'])) { + $content = $parameters['content']; + } + + unset($parameters['content'], $parameters['content_id']); + + if (empty($content)) { + throw new RouteNotFoundException('Neither the route name, nor a parameter "content" or "content_id" could be resolved to an content instance'); + } + + if (!$content instanceof RouteAwareInterface) { + $hint = is_object($content) ? get_class($content) : gettype($content); + throw new RouteNotFoundException('The content does not implement RouteAwareInterface: ' . $hint); + } + + $routes = $content->getRoutes(); + if (empty($routes)) { + $hint = method_exists($content, 'getPath') ? $content->getPath() : get_class($content); + throw new RouteNotFoundException('Document has no route: ' . $hint); + } + + $route = $this->getRouteByLocale($routes, $this->getLocale($parameters)); + if ($route) { + return $route; + } + + // if none matched, continue and randomly return the first one + return reset($routes); + } + + /** + * @param RouteCollection $routes + * @param string $locale + * + * @return bool|SymfonyRoute false if no route requirement matches the provided locale + */ + protected function getRouteByLocale($routes, $locale) + { + foreach ($routes as $route) { + if (! $route instanceof SymfonyRoute) { + continue; + } + + if ($this->checkLocaleRequirement($route, $locale)) { + return $route; + } + } + + return false; + } + + /** + * @param SymfonyRoute $route + * @param string $locale + * + * @return bool TRUE if there is either no _locale, no _locale requirement or if the two match + */ + private function checkLocaleRequirement(SymfonyRoute $route, $locale) + { + return empty($locale) + || !$route->getRequirement('_locale') + || preg_match('/'.$route->getRequirement('_locale').'/', $locale) + ; + } + + /** + * Determine the locale to be used with this request + * + * @param array $parameters the parameters determined by the route + * + * @return string|null the locale following of the parameters or any other + * information the router has available. + */ + protected function getLocale($parameters) + { + if (isset($parameters['_locale'])) { + return $parameters['_locale']; + } + + return null; + } + + /** + * We additionally support empty name and data in parameters and RouteAware content + */ + public function supports($name) + { + return ! $name || parent::supports($name) || $name instanceof RouteAwareInterface; + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ContentRepositoryInterface.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ContentRepositoryInterface.php new file mode 100644 index 000000000000..415cd7d95d29 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ContentRepositoryInterface.php @@ -0,0 +1,27 @@ +<?php + +namespace Symfony\Cmf\Component\Routing; + +/** + * Interface used by the DynamicRouter to retrieve content by it's id when + * generating routes from content-id. + * + * This can be easily implemented using i.e. the Doctrine PHPCR-ODM + * DocumentManager. + * + * @author Uwe Jäger + */ +interface ContentRepositoryInterface +{ + /** + * Return a content object by it's id or null if there is none. + * + * If the returned content implements RouteAwareInterface, it will be used + * to get the route from it to generate an URL. + * + * @param string $id id of the content object + * + * @return object A content that matches this id. + */ + public function findById($id); +} \ No newline at end of file diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/DynamicRouter.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/DynamicRouter.php new file mode 100644 index 000000000000..44407e05fe2d --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/DynamicRouter.php @@ -0,0 +1,297 @@ +<?php + +namespace Symfony\Cmf\Component\Routing; + +use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\Routing\RequestContext; +use Symfony\Component\Routing\RouteCollection; +use Symfony\Component\Routing\RouterInterface; +use Symfony\Component\Routing\Generator\UrlGeneratorInterface; +use Symfony\Component\Routing\Matcher\RequestMatcherInterface; +use Symfony\Component\Routing\Matcher\UrlMatcherInterface; +use Symfony\Component\Routing\RequestContextAwareInterface; +use Symfony\Component\Routing\Exception\ResourceNotFoundException; + +use Symfony\Cmf\Component\Routing\Enhancer\RouteEnhancerInterface; + +/** + * A flexible router accepting matcher and generator through injection and + * using the RouteEnhancer concept to generate additional data on the routes. + * + * @author Crell + * @author David Buchmann + */ +class DynamicRouter implements RouterInterface, RequestMatcherInterface, ChainedRouterInterface +{ + /** + * @var RequestMatcherInterface|UrlMatcherInterface + */ + protected $matcher; + + /** + * @var UrlGeneratorInterface + */ + protected $generator; + + /** + * @var RouteEnhancerInterface[] + */ + protected $enhancers = array(); + + /** + * Cached sorted list of enhancers + * + * @var RouteEnhancerInterface[] + */ + protected $sortedEnhancers = array(); + + /** + * The regexp pattern that needs to be matched before a dynamic lookup is made + * + * @var string + */ + protected $uriFilterRegexp; + + /** + * @var RequestContext + */ + protected $context; + + /** + * @param RequestContext $context + * @param RequestMatcherInterface|UrlMatcherInterface $matcher + * @param UrlGeneratorInterface $generator + * @param string $uriFilterRegexp + */ + public function __construct(RequestContext $context, $matcher, UrlGeneratorInterface $generator, $uriFilterRegexp = '') + { + $this->context = $context; + if (! $matcher instanceof RequestMatcherInterface && ! $matcher instanceof UrlMatcherInterface) { + throw new \InvalidArgumentException('Invalid $matcher'); + } + $this->matcher = $matcher; + $this->generator = $generator; + $this->uriFilterRegexp = $uriFilterRegexp; + + $this->generator->setContext($context); + } + + /** + * Not implemented. + */ + public function getRouteCollection() + { + return new RouteCollection(); + } + + /** + * @return RequestMatcherInterface|UrlMatcherInterface + */ + public function getMatcher() + { + // we may not set the context in DynamicRouter::setContext as this would lead to symfony cache warmup problems + // a request matcher does not need the request context separately as it can get it from the request. + if ($this->matcher instanceof RequestContextAwareInterface) { + $this->matcher->setContext($this->getContext()); + } + + return $this->matcher; + } + + /** + * @return UrlGeneratorInterface + */ + public function getGenerator() + { + $this->generator->setContext($this->getContext()); + + return $this->generator; + } + + /** + * Generates a URL from the given parameters. + * + * If the generator is not able to generate the url, it must throw the RouteNotFoundException + * as documented below. + * + * @param string $name The name of the route + * @param mixed $parameters An array of parameters + * @param Boolean $absolute Whether to generate an absolute URL + * + * @return string The generated URL + * + * @throws RouteNotFoundException if route doesn't exist + * + * @api + */ + public function generate($name, $parameters = array(), $absolute = false) + { + return $this->getGenerator()->generate($name, $parameters, $absolute); + } + + /** + * Support any string as route name + * + * {@inheritDoc} + */ + public function supports($name) + { + // TODO: check $this->generator instanceof VersatileGeneratorInterface + return $this->generator->supports($name); + } + + /** + * Tries to match a URL path with a set of routes. + * + * If the matcher can not find information, it must throw one of the + * exceptions documented below. + * + * @param string $pathinfo The path info to be parsed (raw format, i.e. not urldecoded) + * + * @return array An array of parameters + * + * @throws ResourceNotFoundException If the resource could not be found + * @throws MethodNotAllowedException If the resource was found but the request method is not allowed + * + * @api + */ + public function match($pathinfo) + { + if (! empty($this->uriFilterRegexp) && ! preg_match($this->uriFilterRegexp, $pathinfo)) { + throw new ResourceNotFoundException("$pathinfo does not match the '{$this->uriFilterRegexp}' pattern"); + } + + $matcher = $this->getMatcher(); + if (! $matcher instanceof UrlMatcherInterface) { + throw new \InvalidArgumentException('Wrong matcher type, you need to call matchRequest'); + } + + $defaults = $matcher->match($pathinfo); + + return $this->applyRouteEnhancers($defaults, Request::create($pathinfo)); + } + + /** + * Tries to match a request with a set of routes and returns the array of + * information for that route. + * + * If the matcher can not find information, it must throw one of the + * exceptions documented below. + * + * @param Request $request The request to match + * + * @return array An array of parameters + * + * @throws ResourceNotFoundException If no matching resource could be found + * @throws MethodNotAllowedException If a matching resource was found but the request method is not allowed + */ + public function matchRequest(Request $request) + { + if (! empty($this->uriFilterRegexp) && ! preg_match($this->uriFilterRegexp, $request->getPathInfo())) { + throw new ResourceNotFoundException("{$request->getPathInfo()} does not match the '{$this->uriFilterRegexp}' pattern"); + } + + $matcher = $this->getMatcher(); + if ($matcher instanceof UrlMatcherInterface) { + return $this->match($request->getPathInfo()); + } + + $defaults = $matcher->matchRequest($request); + + return $this->applyRouteEnhancers($defaults, $request); + } + + /** + * Apply the route enhancers to the defaults, according to priorities + * + * @param array $defaults + * @param Request $request + * @return array + */ + protected function applyRouteEnhancers($defaults, Request $request) + { + foreach ($this->getRouteEnhancers() as $enhancer) { + $defaults = $enhancer->enhance($defaults, $request); + } + return $defaults; + } + + /** + * Add route enhancers to the router to let them generate information on + * matched routes. + * + * The order of the enhancers is determined by the priority, the higher the + * value, the earlier the enhancer is run. + * + * @param RouteEnhancerInterface $enhancer + * @param int $priority + */ + public function addRouteEnhancer(RouteEnhancerInterface $enhancer, $priority = 0) + { + if (empty($this->enhancers[$priority])) { + $this->enhancers[$priority] = array(); + } + + $this->enhancers[$priority][] = $enhancer; + $this->sortedEnhancers = array(); + + return $this; + } + + /** + * Sorts the enhancers and flattens them. + * + * @return RouteEnhancerInterface[] the enhancers ordered by priority + */ + public function getRouteEnhancers() + { + if (empty($this->sortedEnhancers)) { + $this->sortedEnhancers = $this->sortRouteEnhancers(); + } + + return $this->sortedEnhancers; + } + + /** + * Sort enhancers by priority. + * + * The highest priority number is the highest priority (reverse sorting). + * + * @return RouteEnhancerInterface[] the sorted enhancers + */ + protected function sortRouteEnhancers() + { + $sortedEnhancers = array(); + krsort($this->enhancers); + + foreach ($this->enhancers as $enhancers) { + $sortedEnhancers = array_merge($sortedEnhancers, $enhancers); + } + + return $sortedEnhancers; + } + + /** + * Sets the request context. + * + * @param RequestContext $context The context + * + * @api + */ + public function setContext(RequestContext $context) + { + $this->context = $context; + } + + /** + * Gets the request context. + * + * @return RequestContext The context + * + * @api + */ + public function getContext() + { + return $this->context; + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/FieldByClassEnhancer.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/FieldByClassEnhancer.php new file mode 100644 index 000000000000..23bc714ccb68 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/FieldByClassEnhancer.php @@ -0,0 +1,78 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Enhancer; + +use Symfony\Component\HttpFoundation\Request; + +use Symfony\Cmf\Component\Routing\RouteObjectInterface; + +/** + * This enhancer sets a field if not yet existing from the class of an object + * in another field. + * + * The comparison is done with instanceof to support proxy classes and such. + * + * Only works with RouteObjectInterface routes that can return a referenced + * content. + * + * @author David Buchmann + */ +class FieldByClassEnhancer implements RouteEnhancerInterface +{ + /** + * @var string field for the source class + */ + protected $source; + /** + * @var string field to write hashmap lookup result into + */ + protected $target; + /** + * @var array containing the mapping between a class name and the target value + */ + protected $map; + + /** + * @param string $source the field name of the class + * @param string $target the field name to set from the map + * @param array $map the map of class names to field values + */ + public function __construct($source, $target, $map) + { + $this->source = $source; + $this->target = $target; + $this->map = $map; + } + + /** + * If the source field is instance of one of the entries in the map, + * target is set to the value of that map entry. + * + * {@inheritDoc} + */ + public function enhance(array $defaults, Request $request) + { + if (isset($defaults[$this->target])) { + // no need to do anything + return $defaults; + } + + if (! isset($defaults[$this->source])) { + return $defaults; + } + + // we need to loop over the array and do instanceof in case the content + // class extends the specified class + // i.e. phpcr-odm generates proxy class for the content. + foreach ($this->map as $class => $value) { + if ($defaults[$this->source] instanceof $class) { + // found a matching entry in the map + $defaults[$this->target] = $value; + + return $defaults; + } + } + + return $defaults; + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/FieldMapEnhancer.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/FieldMapEnhancer.php new file mode 100644 index 000000000000..53f6c0edb02d --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/FieldMapEnhancer.php @@ -0,0 +1,61 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Enhancer; + +use Symfony\Component\HttpFoundation\Request; + +/** + * This enhancer can fill one field with the result of a hashmap lookup of + * another field. If the target field is already set, it does nothing. + * + * @author David Buchmann + */ +class FieldMapEnhancer implements RouteEnhancerInterface +{ + /** + * @var string field for key in hashmap lookup + */ + protected $source; + /** + * @var string field to write hashmap lookup result into + */ + protected $target; + /** + * @var array containing the mapping between the source field value and target field value + */ + protected $hashmap; + + /** + * @param string $source the field to read + * @param string $target the field to write the result of the lookup into + * @param array $hashmap for looking up value from source and get value for target + */ + public function __construct($source, $target, array $hashmap) + { + $this->source = $source; + $this->target = $target; + $this->hashmap = $hashmap; + } + + /** + * If the target field is not set but the source field is, map the field + * + * {@inheritDoc} + */ + public function enhance(array $defaults, Request $request) + { + if (isset($defaults[$this->target])) { + return $defaults; + } + if (! isset($defaults[$this->source])) { + return $defaults; + } + if (! isset($this->hashmap[$defaults[$this->source]])) { + return $defaults; + } + + $defaults[$this->target] = $this->hashmap[$defaults[$this->source]]; + + return $defaults; + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/FieldPresenceEnhancer.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/FieldPresenceEnhancer.php new file mode 100644 index 000000000000..9eb3a2a1c131 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/FieldPresenceEnhancer.php @@ -0,0 +1,60 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Enhancer; + +use Symfony\Component\HttpFoundation\Request; + +/** + * This enhancer can set a field to a fixed value if an other field is present. + * + * @author David Buchmann + */ +class FieldPresenceEnhancer implements RouteEnhancerInterface +{ + /** + * @var string field for the source class + */ + protected $source; + /** + * @var string field to write hashmap lookup result into + */ + protected $target; + /** + * value to set the target field to + * + * @var string + */ + private $value; + + /** + * @param string $source the field name of the class + * @param string $target the field name to set from the map + * @param string $value value to set target field to if source field exists + */ + public function __construct($source, $target, $value) + { + $this->source = $source; + $this->target = $target; + $this->value = $value; + } + + /** + * {@inheritDoc} + */ + public function enhance(array $defaults, Request $request) + { + if (isset($defaults[$this->target])) { + // no need to do anything + return $defaults; + } + + if (! isset($defaults[$this->source])) { + return $defaults; + } + + $defaults[$this->target] = $this->value; + + return $defaults; + } + +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/RouteContentEnhancer.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/RouteContentEnhancer.php new file mode 100644 index 000000000000..39e54d878720 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/RouteContentEnhancer.php @@ -0,0 +1,70 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Enhancer; + +use Symfony\Component\Routing\Route; +use Symfony\Component\HttpFoundation\Request; + +use Symfony\Cmf\Component\Routing\RouteObjectInterface; + +/** + * This enhancer sets the content to target field if the route provides content + * + * Only works with RouteObjectInterface routes that can return a referenced + * content. + * + * @author David Buchmann + */ +class RouteContentEnhancer implements RouteEnhancerInterface +{ + /** + * @var string field for the route class + */ + protected $routefield; + /** + * @var string field to write hashmap lookup result into + */ + protected $target; + + /** + * @param string $routefield the field name of the route class + * @param string $target the field name to set from the map + * @param array $hashmap the map of class names to field values + */ + public function __construct($routefield, $target) + { + $this->routefield = $routefield; + $this->target = $target; + } + + /** + * If the route has a non-null content and if that content class is in the + * injected map, returns that controller. + * + * {@inheritDoc} + */ + public function enhance(array $defaults, Request $request) + { + if (isset($defaults[$this->target])) { + // no need to do anything + return $defaults; + } + + if (! isset($defaults[$this->routefield]) + || ! $defaults[$this->routefield] instanceof RouteObjectInterface + ) { + // we can't determine the content + return $defaults; + } + $route = $defaults[$this->routefield]; + + $content = $route->getRouteContent(); + if (! $content) { + // we have no content + return $defaults; + } + $defaults[$this->target] = $content; + + return $defaults; + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/RouteEnhancerInterface.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/RouteEnhancerInterface.php new file mode 100644 index 000000000000..5932c71609eb --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Enhancer/RouteEnhancerInterface.php @@ -0,0 +1,27 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Enhancer; + +use Symfony\Component\Routing\Route; +use Symfony\Component\HttpFoundation\Request; + +/** + * A route enhancer can change the values in the route data arrays + * + * This is useful to provide information to the rest of the routing system + * that can be inferred from other parameters rather than hardcode that + * information in every route. + * + * @author David Buchmann + */ +interface RouteEnhancerInterface +{ + /** + * Update the defaults based on its own data and the request. + * + * @param array $defaults the getRouteDefaults array + * + * @return array the modified defaults. Each enhancer MUST return the $defaults but may add or remove values + */ + public function enhance(array $defaults, Request $request); +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/LICENSE b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/LICENSE new file mode 100644 index 000000000000..797b3965fad0 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/LICENSE @@ -0,0 +1,23 @@ +Routing + + The MIT License + + Copyright (c) 2011-2012 Symfony2 CMF + + Permission is hereby granted, free of charge, to any person obtaining a copy + of this software and associated documentation files (the "Software"), to deal + in the Software without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + copies of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be included in + all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + THE SOFTWARE. diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/ConfigurableUrlMatcher.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/ConfigurableUrlMatcher.php new file mode 100644 index 000000000000..e1e17b58d79c --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/ConfigurableUrlMatcher.php @@ -0,0 +1,66 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\NestedMatcher; + +use Symfony\Component\Routing\Route; +use Symfony\Component\Routing\RequestContext; +use Symfony\Component\Routing\RouteCollection; +use Symfony\Component\HttpFoundation\Request; + +use Symfony\Cmf\Component\Routing\RouteObjectInterface; +use Symfony\Cmf\Component\Routing\NestedMatcher\FinalMatcherInterface; + +/** + * A final matcher that can proxy any matcher having the right constructor + * signature, the same way the symfony core Router class does. + * + * @author DavidBuchmann + */ +class ConfigurableUrlMatcher implements FinalMatcherInterface +{ + private $matcherClass; + + public function __construct($matcherClass = 'Symfony\\Component\\Routing\\Matcher\\UrlMatcher') + { + $this->matcherClass = $matcherClass; + } + + /** + * {@inheritdoc} + */ + public function finalMatch(RouteCollection $collection, Request $request) + { + $context = new RequestContext(); + $context->fromRequest($request); + $matcher = $this->getMatcher($collection, $context); + $attributes = $matcher->match($request->getPathInfo()); + + // cleanup route attributes + if (! isset($attributes['_route']) || ! $attributes['_route'] instanceof Route) { + $name = $attributes['_route']; + $route = $collection->get($attributes['_route']); + $attributes['_route'] = $route; + + if ($route instanceof RouteObjectInterface && is_string($route->getRouteKey())) { + $name = $route->getRouteKey(); + } + + if (empty($attributes['_route_name']) && is_string($name)) { + $attributes['_route_name'] = $name; + } + } + + return $attributes; + } + + /** + * @param RouteCollection $collection the route collection to match + * @param RequestContext $context the context to match in + * + * @return \Symfony\Component\Routing\Matcher\UrlMatcherInterface + */ + protected function getMatcher(RouteCollection $collection, RequestContext $context) + { + return new $this->matcherClass($collection, $context); + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/FinalMatcherInterface.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/FinalMatcherInterface.php new file mode 100644 index 000000000000..b48859c7e639 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/FinalMatcherInterface.php @@ -0,0 +1,29 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\NestedMatcher; + +use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\Routing\RouteCollection; +use Symfony\Component\Routing\Exception\ResourceNotFoundException; + +/** + * A FinalMatcher returns only one route from a collection of candidate routes. + * + * @author Larry Garfield + * @author David Buchmann + */ +interface FinalMatcherInterface +{ + /** + * Matches a request against a route collection and returns exactly one result. + * + * @param RouteCollection $collection The collection against which to match. + * @param Request $request The request to match. + * + * @return array An array of parameters + * + * @throws ResourceNotFoundException if none of the routes in $collection + * matches $request + */ + public function finalMatch(RouteCollection $collection, Request $request); +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/NestedMatcher.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/NestedMatcher.php new file mode 100644 index 000000000000..9d31d8116e9f --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/NestedMatcher.php @@ -0,0 +1,168 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\NestedMatcher; + +use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\Routing\Exception\ResourceNotFoundException; +use Symfony\Component\Routing\Matcher\RequestMatcherInterface; +use Symfony\Component\Routing\Route; +use Symfony\Cmf\Component\Routing\RouteProviderInterface; + +/** + * A more flexible approach to matching. The route collection to match against + * can be dynamically determined based on the request and users can inject + * their own filters or use a custom final matching strategy. + * + * The nested matcher splits matching into three configurable steps: + * + * 1) Get potential matches from a RouteProviderInterface + * 2) Apply any RouteFilterInterface to reduce the route collection + * 3) Have FinalMatcherInterface select the best match of the remaining routes + * + * @author Larry Garfield + * @author David Buchmann + */ +class NestedMatcher implements RequestMatcherInterface +{ + /** + * The route provider responsible for the first-pass match. + * + * @var RouteProviderInterface + */ + protected $routeProvider; + + /** + * The final matcher. + * + * @var FinalMatcherInterface + */ + protected $finalMatcher; + + /** + * An array of RouteFilterInterface objects. + * + * @var RouteFilterInterface[] + */ + protected $filters = array(); + + /** + * Array of RouteFilterInterface objects, sorted. + * + * @var RouteFilterInterface[] + */ + protected $sortedFilters = array(); + + /** + * Constructs a new NestedMatcher + * + * @param RouteProviderInterface $provider The Route Provider this matcher should use. + */ + public function __construct(RouteProviderInterface $provider) + { + $this->routeProvider = $provider; + } + + /** + * Sets the route provider for the matching plan. + * + * @param RouteProviderInterface $provider A route provider. It is responsible for its own configuration. + * + * @return NestedMatcher this object to have a fluent interface + */ + public function setRouteProvider(RouteProviderInterface $provider) + { + $this->routeProvider = $provider; + + return $this; + } + + /** + * Adds a partial matcher to the matching plan. + * + * Partial matchers will be run in the order in which they are added. + * + * @param RouteFilterInterface $filter + * @param int $priority (optional) The priority of the filter. Higher number filters will be used first. Default to 0. + * + * @return NestedMatcher this object to have a fluent interface + */ + public function addRouteFilter(RouteFilterInterface $filter, $priority = 0) + { + if (empty($this->filters[$priority])) { + $this->filters[$priority] = array(); + } + + $this->filters[$priority][] = $filter; + $this->sortedFilters = array(); + + return $this; + } + + /** + * Sets the final matcher for the matching plan. + * + * @param FinalMatcherInterface $final The final matcher that will have to pick the route that will be used. + * + * @return NestedMatcher this object to have a fluent interface + */ + public function setFinalMatcher(FinalMatcherInterface $final) + { + $this->finalMatcher = $final; + + return $this; + } + + /** + * {@inheritdoc} + */ + public function matchRequest(Request $request) + { + $collection = $this->routeProvider->getRouteCollectionForRequest($request); + if (!count($collection)) { + throw new ResourceNotFoundException(); + } + + // Route Filters are expected to throw an exception themselves if they + // end up filtering the list down to 0. + foreach ($this->getRouteFilters() as $filter) { + $collection = $filter->filter($collection, $request); + } + + $attributes = $this->finalMatcher->finalMatch($collection, $request); + + return $attributes; + } + + /** + * Sorts the filters and flattens them. + * + * @return RouteFilterInterface[] the filters ordered by priority + */ + public function getRouteFilters() + { + if (empty($this->sortedFilters)) { + $this->sortedFilters = $this->sortFilters(); + } + + return $this->sortedFilters; + } + + /** + * Sort filters by priority. + * + * The highest priority number is the highest priority (reverse sorting). + * + * @return RouteFilterInterface[] the sorted filters + */ + protected function sortFilters() + { + $sortedFilters = array(); + krsort($this->filters); + + foreach ($this->filters as $filters) { + $sortedFilters = array_merge($sortedFilters, $filters); + } + + return $sortedFilters; + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/RouteFilterInterface.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/RouteFilterInterface.php new file mode 100644 index 000000000000..7fe4ca9bb6e4 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/RouteFilterInterface.php @@ -0,0 +1,36 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\NestedMatcher; + +use Symfony\Component\Routing\RouteCollection; +use Symfony\Component\Routing\Exception\ResourceNotFoundException; + +use Symfony\Component\HttpFoundation\Request; + +/** + * A RouteFilter takes a RouteCollection and returns a filtered subset. + * + * It is not implemented as a filter iterator because we want to allow + * router filters to handle their own empty-case handling, usually by throwing + * an appropriate exception if no routes match the object's rules. + * + * @author Larry Garfield + * @author David Buchmann + */ +interface RouteFilterInterface +{ + /** + * Filters the route collection against a request and returns all matching + * routes. + * + * @param RouteCollection $collection The collection against which to match. + * @param Request $request A Request object against which to match. + * + * @return RouteCollection A non-empty RouteCollection of matched routes. + * + * @throws ResourceNotFoundException if none of the routes in $collection + * matches $request. This is a performance optimization to not continue + * the match process when a match will no longer be possible. + */ + public function filter(RouteCollection $collection, Request $request); +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/UrlMatcher.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/UrlMatcher.php new file mode 100644 index 000000000000..09bb2c474442 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/NestedMatcher/UrlMatcher.php @@ -0,0 +1,46 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\NestedMatcher; + +use Symfony\Component\Routing\Route; +use Symfony\Component\Routing\RouteCollection; +use Symfony\Component\Routing\Matcher\UrlMatcher as SymfonyUrlMatcher; +use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\Routing\RequestContext; + +use Symfony\Cmf\Component\Routing\RouteObjectInterface; + +/** + * Extended UrlMatcher to provide an additional interface and enhanced features. + * + * This class requires Symfony 2.2 for a refactoring done to the symfony UrlMatcher + * + * @author Larry Garfield + */ +class UrlMatcher extends SymfonyUrlMatcher implements FinalMatcherInterface +{ + /** + * {@inheritdoc} + */ + public function finalMatch(RouteCollection $collection, Request $request) + { + $this->routes = $collection; + $context = new RequestContext(); + $context->fromRequest($request); + $this->setContext($context); + return $this->match($request->getPathInfo()); + } + + /** + * {@inheritdoc} + */ + protected function getAttributes(Route $route, $name, array $attributes) + { + if ($route instanceof RouteObjectInterface && is_string($route->getRouteKey())) { + $name = $route->getRouteKey(); + } + $attributes['_route_name'] = $name; + $attributes['_route'] = $route; + return $this->mergeDefaults($attributes, $route->getDefaults()); + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ProviderBasedGenerator.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ProviderBasedGenerator.php new file mode 100644 index 000000000000..1c353f83589d --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/ProviderBasedGenerator.php @@ -0,0 +1,67 @@ +<?php + +namespace Symfony\Cmf\Component\Routing; + +use Symfony\Component\Routing\Route as SymfonyRoute; +use Symfony\Component\Routing\Exception\RouteNotFoundException; + +use Symfony\Component\Routing\Generator\UrlGenerator; +use Symfony\Component\HttpFoundation\Request; +use Symfony\Component\HttpKernel\Log\LoggerInterface; + +use Symfony\Cmf\Component\Routing\RouteProviderInterface; + +/** + * A Generator that uses a RouteProvider rather than a RouteCollection + * + * @author Larry Garfield + */ +class ProviderBasedGenerator extends UrlGenerator +{ + /** + * The route provider for this generator. + * + * @var RouteProviderInterface + */ + protected $provider; + + public function __construct(RouteProviderInterface $provider, LoggerInterface $logger = null) + { + $this->provider = $provider; + $this->logger = $logger; + } + + /** + * {@inheritDoc} + */ + public function generate($name, $parameters = array(), $absolute = false) + { + if ($name instanceof SymfonyRoute) { + $route = $name; + } elseif (null === $route = $this->provider->getRouteByName($name, $parameters)) { + throw new RouteNotFoundException(sprintf('Route "%s" does not exist.', $name)); + } + + // the Route has a cache of its own and is not recompiled as long as it does not get modified + $compiledRoute = $route->compile(); + + // handle symfony 2.1 and 2.2 + // getHostnameTokens exists only since 2.2 + $hostnameTokens = null; + if (method_exists($compiledRoute, 'getHostnameTokens')) { + $hostnameTokens = $compiledRoute->getHostnameTokens(); + } + + return $this->doGenerate($compiledRoute->getVariables(), $route->getDefaults(), $route->getRequirements(), $compiledRoute->getTokens(), $parameters, $name, $absolute, $hostnameTokens); + } + + /** + * Support a route object and any string as route name + * + * {@inheritDoc} + */ + public function supports($name) + { + return is_string($name) || $name instanceof SymfonyRoute; + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/README.md b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/README.md new file mode 100644 index 000000000000..607fb7234b24 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/README.md @@ -0,0 +1,7 @@ +# Symfony CMF Routing Component [](http://travis-ci.org/symfony-cmf/Routing) + +This library extends the Symfony2 Routing component. Even though it has Symfony +in its name, it does not need the full Symfony2 framework and can be used in +standalone projects. + +http://symfony.com/doc/master/cmf/components/routing.html diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RedirectRouteInterface.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RedirectRouteInterface.php new file mode 100644 index 000000000000..c4ac783fd915 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RedirectRouteInterface.php @@ -0,0 +1,76 @@ +<?php + +namespace Symfony\Cmf\Component\Routing; + +/** + * Document for redirection entries with the RedirectController. + * + * Defines additional methods needed by the RedirectController to redirect + * based on the route. + * + * This document may define (in order of precedence - the others can be empty): + * + * - uri: an absolute uri + * - routeName and routeParameters: to be used with the standard symfony router + * or a route entry in the routeParameters for the DynamicRouter. Precedency + * between these is determined by the order of the routers in the chain + * router. + * + * With standard Symfony routing, you can just use uri / routeName and a + * hashmap of parameters. + * + * For the dynamic router, you can return a RouteInterface instance in the + * field 'route' of the parameters. + * + * Note: getRedirectContent must return the redirect route itself for the + * integration with DynamicRouter to work. + * + * @author David Buchmann <david@liip.ch> + */ +interface RedirectRouteInterface extends RouteObjectInterface +{ + /** + * Get the absolute uri to redirect to external domains. + * + * If this is non-empty, the other methods won't be used. + * + * @return string target absolute uri + */ + public function getUri(); + + /** + * Get the target route document this route redirects to. + * + * If non-null, it is added as route into the parameters, which will lead + * to have the generate call issued by the RedirectController to have + * the target route in the parameters. + * + * @return RouteObjectInterface the route this redirection points to + */ + public function getRouteTarget(); + + /** + * Get the name of the target route for working with the symfony standard + * router. + * + * @return string target route name + */ + public function getRouteName(); + + /** + * Whether this should be a permanent or temporary redirect + * + * @return boolean + */ + public function isPermanent(); + + /** + * Get the parameters for the target route router::generate() + * + * Note that for the DynamicRouter, you return the target route + * document as field 'route' of the hashmap. + * + * @return array Information to build the route + */ + public function getParameters(); +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RouteAwareInterface.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RouteAwareInterface.php new file mode 100644 index 000000000000..59983c6a20b0 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RouteAwareInterface.php @@ -0,0 +1,22 @@ +<?php + +namespace Symfony\Cmf\Component\Routing; + +/** + * Interface to be implemented by content that wants to be compatible with the + * DynamicRouter + */ +interface RouteAwareInterface +{ + /** + * Get the routes that point to this content. + * + * Note: For PHPCR, as explained in RouteObjectInterface the route must use + * the routeContent field to store the reference to the content so you can + * get the routes with Referrers(filter="routeContent") + * + * @return \Symfony\Component\Routing\Route[] Route instances that point to this content + */ + public function getRoutes(); +} + diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RouteObjectInterface.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RouteObjectInterface.php new file mode 100644 index 000000000000..496e3cc60308 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RouteObjectInterface.php @@ -0,0 +1,70 @@ +<?php + +namespace Symfony\Cmf\Component\Routing; + +/** + * Classes for entries in the routing table may implement this interface in + * addition to extending Symfony\Component\Routing\Route. + * + * If they do, the DynamicRouter will request the route content and put it into + * the RouteObjectInterface::CONTENT_OBJECT field. The DynamicRouter will also + * request getRouteKey and this will be used instead of the symfony core compatible + * route name and can contain any characters. + * + * Some fields in defaults have a special meaning in the getDefaults(). In addition + * to the constants defined in this class, _locale and _controller are also used. + */ +interface RouteObjectInterface +{ + /** + * Constant for the field that is given to the ControllerAliasMapper. + * The value must be configured in the controllers_by_alias mapping. + * + * This is ignored if a _controller default value is provided as well + */ + const CONTROLLER_ALIAS = '_controller_alias'; + + /** + * Field name for an explicit controller name to be used with this route + */ + const CONTROLLER_NAME = '_controller'; + + /** + * Field name for an explicit template to be used with this route. + * i.e. SymfonyCmfContentBundle:StaticContent:index.html.twig + */ + const TEMPLATE_NAME = '_template'; + + /** + * Field name for the content of the current route, if any. + */ + const CONTENT_OBJECT = '_content'; + + /** + * Get the content document this route entry stands for. If non-null, + * the ControllerClassMapper uses it to identify a controller and + * the content is passed to the controller. + * + * If there is no specific content for this url (i.e. its an "application" + * page), may return null. + * + * To interoperate with the standard Symfony\Cmf\Bundle\ContentBundle\Document\StaticContent + * the instance MUST store the property in the field <code>routeContent</code> + * to have referrer resolution work. + * + * @return object the document or entity this route entry points to + */ + public function getRouteContent(); + + /** + * Get the route key. + * + * This key will be used as route name instead of the symfony core compatible + * route name and can contain any characters. + * + * Return null if you want to use the default key. + * + * @return string the route name + */ + public function getRouteKey(); +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RouteProviderInterface.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RouteProviderInterface.php new file mode 100644 index 000000000000..de9b64636484 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/RouteProviderInterface.php @@ -0,0 +1,73 @@ +<?php + +namespace Symfony\Cmf\Component\Routing; + +use Symfony\Component\HttpFoundation\Request; + +/** + * Interface for the route provider the DynamicRouter is using. + * + * Typically this could be a doctrine orm or odm repository, but you can + * implement something else if you need to. + */ +interface RouteProviderInterface +{ + /** + * Finds routes that may potentially match the request. + * + * This may return a mixed list of class instances, but all routes returned + * must extend the core symfony route. The classes may also implement + * RouteObjectInterface to link to a content document. + * + * This method may not throw an exception based on implementation specific + * restrictions on the url. That case is considered a not found - returning + * an empty array. Exceptions are only used to abort the whole request in + * case something is seriously broken, like the storage backend being down. + * + * Note that implementations may not implement an optimal matching + * algorithm, simply a reasonable first pass. That allows for potentially + * very large route sets to be filtered down to likely candidates, which + * may then be filtered in memory more completely. + * + * @param Request $request A request against which to match. + * + * @return \Symfony\Component\Routing\RouteCollection with all urls that + * could potentially match $request. Empty collection if nothing can + * match. + */ + public function getRouteCollectionForRequest(Request $request); + + /** + * Find the route using the provided route name (and parameters) + * + * @param string $name the route name to fetch + * @param array $parameters the parameters as they are passed to the + * UrlGeneratorInterface::generate call + * + * @return \Symfony\Component\Routing\Route + * + * @throws \Symfony\Component\Routing\Exception\RouteNotFoundException if + * there is no route with that name in this repository + */ + public function getRouteByName($name, $parameters = array()); + + /** + * Find many routes by their names using the provided list of names + * + * Note that this method may not throw an exception if some of the routes + * are not found. It will just return the list of those routes it found. + * + * This method exists in order to allow performance optimizations. The + * simple implementation could be to just repeatedly call + * $this->getRouteByName() + * + * @param array $names the list of names to retrieve + * @param array $parameters the parameters as they are passed to the + * UrlGeneratorInterface::generate call. (Only one array, not one for + * each entry in $names. + * + * @return \Symfony\Component\Routing\Route[] iterable thing with the keys + * the names of the $names argument. + */ + public function getRoutesByNames($names, $parameters = array()); +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Test/CmfUnitTestCase.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Test/CmfUnitTestCase.php new file mode 100644 index 000000000000..789823876d48 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Test/CmfUnitTestCase.php @@ -0,0 +1,16 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Test; + +class CmfUnitTestCase extends \PHPUnit_Framework_TestCase +{ + + protected function buildMock($class, array $methods = array()) + { + return $this->getMockBuilder($class) + ->disableOriginalConstructor() + ->setMethods($methods) + ->getMock(); + } + +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/FieldByClassEnhancerTest.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/FieldByClassEnhancerTest.php new file mode 100644 index 000000000000..d7eb0a86529d --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/FieldByClassEnhancerTest.php @@ -0,0 +1,62 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Tests\Enhancer; + +use Symfony\Component\HttpFoundation\Request; + +use Symfony\Cmf\Component\Routing\Test\CmfUnitTestCase; +use Symfony\Cmf\Component\Routing\Enhancer\FieldByClassEnhancer; + +class FieldByClassEnhancerTest extends CmfUnitTestCase +{ + private $request; + /** + * @var FieldByClassEnhancer + */ + private $mapper; + private $document; + + public function setUp() + { + $this->document = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\Tests\\Enhancer\\RouteObject'); + + $mapping = array('Symfony\\Cmf\\Component\\Routing\\Tests\\Enhancer\\RouteObject' + => 'symfony_cmf_content.controller:indexAction'); + + $this->mapper = new FieldByClassEnhancer('_content', '_controller', $mapping); + + $this->request = Request::create('/test'); + } + + public function testClassFoundInMapping() + { + // this is the mock, thus a child class to make sure we properly check with instanceof + $defaults = array('_content' => $this->document); + $expected = array( + '_content' => $this->document, + '_controller' => 'symfony_cmf_content.controller:indexAction', + ); + $this->assertEquals($expected, $this->mapper->enhance($defaults, $this->request)); + } + + public function testFieldAlreadyThere() + { + $defaults = array( + '_content' => $this->document, + '_controller' => 'custom.controller:indexAction', + ); + $this->assertEquals($defaults, $this->mapper->enhance($defaults, $this->request)); + } + + public function testClassNotFoundInMapping() + { + $defaults = array('_content' => $this); + $this->assertEquals($defaults, $this->mapper->enhance($defaults, $this->request)); + } + + public function testNoClass() + { + $defaults = array('foo' => 'bar'); + $this->assertEquals($defaults, $this->mapper->enhance($defaults, $this->request)); + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/FieldMapEnhancerTest.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/FieldMapEnhancerTest.php new file mode 100644 index 000000000000..982467a3add9 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/FieldMapEnhancerTest.php @@ -0,0 +1,61 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Tests\Mapper; + +use Symfony\Component\HttpFoundation\Request; + +use Symfony\Cmf\Component\Routing\Test\CmfUnitTestCase; +use Symfony\Cmf\Component\Routing\Enhancer\FieldMapEnhancer; + +class FieldMapEnhancerTest extends CmfUnitTestCase +{ + /** + * @var Request + */ + private $request; + + /** + * @var FieldMapEnhancer + */ + private $enhancer; + + public function setUp() + { + $this->request = Request::create('/test'); + $mapping = array('static_pages' => 'symfony_cmf_content.controller:indexAction'); + + $this->enhancer = new FieldMapEnhancer('type', '_controller', $mapping); + } + + public function testFieldFoundInMapping() + { + $defaults = array('type' => 'static_pages'); + $expected = array( + 'type' => 'static_pages', + '_controller' => 'symfony_cmf_content.controller:indexAction', + ); + $this->assertEquals($expected, $this->enhancer->enhance($defaults, $this->request)); + } + + public function testFieldAlreadyThere() + { + $defaults = array( + 'type' => 'static_pages', + '_controller' => 'custom.controller:indexAction', + ); + $this->assertEquals($defaults, $this->enhancer->enhance($defaults, $this->request)); + } + + + public function testNoType() + { + $defaults = array(); + $this->assertEquals(array(), $this->enhancer->enhance($defaults, $this->request)); + } + + public function testNotFoundInMapping() + { + $defaults = array('type' => 'unknown_route'); + $this->assertEquals($defaults, $this->enhancer->enhance($defaults, $this->request)); + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/FieldPresenceEnhancerTest.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/FieldPresenceEnhancerTest.php new file mode 100644 index 000000000000..78c3e100dc8b --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/FieldPresenceEnhancerTest.php @@ -0,0 +1,51 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Tests\Enhancer; + +use Symfony\Component\HttpFoundation\Request; + +use Symfony\Cmf\Component\Routing\RouteObjectInterface; +use Symfony\Cmf\Component\Routing\Enhancer\FieldPresenceEnhancer; + +use Symfony\Cmf\Component\Routing\Test\CmfUnitTestCase; + +class FieldPresenceEnhancerTest extends CmfUnitTestCase +{ + /** + * @var FieldPresenceEnhancer + */ + private $mapper; + private $request; + + public function setUp() + { + $this->mapper = new FieldPresenceEnhancer('_template', '_controller', 'symfony_cmf_content.controller:indexAction'); + + $this->request = Request::create('/test'); + } + + public function testHasTemplate() + { + $defaults = array('_template' => 'Bundle:Topic:template.html.twig'); + $expected = array( + '_template' => 'Bundle:Topic:template.html.twig', + '_controller' => 'symfony_cmf_content.controller:indexAction', + ); + $this->assertEquals($expected, $this->mapper->enhance($defaults, $this->request)); + } + + public function testFieldAlreadyThere() + { + $defaults = array( + '_template' => 'Bundle:Topic:template.html.twig', + '_controller' => 'custom.controller:indexAction', + ); + $this->assertEquals($defaults, $this->mapper->enhance($defaults, $this->request)); + } + + public function testHasNoTemplate() + { + $defaults = array('foo' => 'bar'); + $this->assertEquals($defaults, $this->mapper->enhance($defaults, $this->request)); + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/RouteContentEnhancerTest.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/RouteContentEnhancerTest.php new file mode 100644 index 000000000000..df9008a2413e --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/RouteContentEnhancerTest.php @@ -0,0 +1,78 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Tests\Enhancer; + +use Symfony\Component\HttpFoundation\Request; + +use Symfony\Cmf\Component\Routing\Enhancer\RouteContentEnhancer; + +use Symfony\Cmf\Component\Routing\Test\CmfUnitTestCase; + +class RouteContentEnhancerTest extends CmfUnitTestCase +{ + /** + * @var RouteContentEnhancer + */ + private $mapper; + private $document; + private $request; + + public function setUp() + { + $this->document = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\Tests\\Enhancer\\RouteObject', + array('getRouteContent', 'getRouteDefaults', 'getUrl')); + + $this->mapper = new RouteContentEnhancer('_route', '_content'); + + $this->request = Request::create('/test'); + } + + public function testContent() + { + $targetDocument = new TargetDocument(); + $this->document->expects($this->once()) + ->method('getRouteContent') + ->will($this->returnValue($targetDocument)); + + $defaults = array('_route' => $this->document); + $expected = array('_route' => $this->document, '_content' => $targetDocument); + + $this->assertEquals($expected, $this->mapper->enhance($defaults, $this->request)); + } + + + public function testFieldAlreadyThere() + { + $this->document->expects($this->never()) + ->method('getRouteContent') + ; + + $defaults = array('_route' => $this->document, '_content' => 'foo'); + + $this->assertEquals($defaults, $this->mapper->enhance($defaults, $this->request)); + } + + public function testNoContent() + { + $this->document->expects($this->once()) + ->method('getRouteContent') + ->will($this->returnValue(null)); + + $defaults = array('_route' => $this->document); + $this->assertEquals($defaults, $this->mapper->enhance($defaults, $this->request)); + } + + public function testNoCmfRoute() + { + $defaults = array('_route' => $this->buildMock('Symfony\\Component\\Routing\\Route')); + $this->assertEquals($defaults, $this->mapper->enhance($defaults, $this->request)); + } +} + +class TargetDocument +{ +} + +class UnknownDocument +{ +} \ No newline at end of file diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/RouteObject.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/RouteObject.php new file mode 100644 index 000000000000..61093398bf02 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Enhancer/RouteObject.php @@ -0,0 +1,18 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Tests\Enhancer; + +use Symfony\Component\Routing\Route; +use Symfony\Cmf\Component\Routing\RouteObjectInterface; + +/** + * Empty abstract class to be able to mock an object that both extends Route + * and implements RouteObjectInterface + */ +abstract class RouteObject extends Route implements RouteObjectInterface +{ + public function getRouteKey() + { + return null; + } +} \ No newline at end of file diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/NestedMatcher/ConfigurableUrlMatcherTest.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/NestedMatcher/ConfigurableUrlMatcherTest.php new file mode 100644 index 000000000000..18bcae787d1a --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/NestedMatcher/ConfigurableUrlMatcherTest.php @@ -0,0 +1,129 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Tests\NestedMatcher; + +use Symfony\Component\HttpFoundation\Request; + +use Symfony\Component\Routing\RouteCollection; +use Symfony\Component\Routing\Route; +use Symfony\Cmf\Component\Routing\NestedMatcher\ConfigurableUrlMatcher; + +use Symfony\Cmf\Component\Routing\RouteObjectInterface; + +use Symfony\Cmf\Component\Routing\Test\CmfUnitTestCase; + +class ConfigurableUrlMatcherTest extends CmfUnitTestCase +{ + protected $routeDocument; + protected $routeCompiled; + protected $matcher; + protected $context; + protected $request; + + protected $url = '/foo/bar'; + + public function setUp() + { + $this->routeDocument = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\Tests\\Routing\\RouteMock', array('getDefaults', 'getRouteKey', 'compile')); + $this->routeCompiled = $this->buildMock('Symfony\\Component\\Routing\\CompiledRoute'); + + $this->context = $this->buildMock('Symfony\\Component\\Routing\\RequestContext'); + $this->request = Request::create($this->url); + + $this->matcher = new ConfigurableUrlMatcher(); + } + + public function testMatch() + { + $this->routeCompiled->expects($this->atLeastOnce()) + ->method('getStaticPrefix') + ->will($this->returnValue($this->url)) + ; + $this->routeCompiled->expects($this->atLeastOnce()) + ->method('getRegex') + ->will($this->returnValue('#'.str_replace('/', '\\/', $this->url).'#')) + ; + $this->routeDocument->expects($this->atLeastOnce()) + ->method('compile') + ->will($this->returnValue($this->routeCompiled)) + ; + $this->routeDocument->expects($this->atLeastOnce()) + ->method('getDefaults') + ->will($this->returnValue(array('foo' => 'bar'))) + ; + + $mockCompiled = $this->buildMock('Symfony\\Component\\Routing\\CompiledRoute'); + $mockCompiled->expects($this->any()) + ->method('getStaticPrefix') + ->will($this->returnValue('/no/match')) + ; + $mockRoute = $this->getMockBuilder('Symfony\\Component\\Routing\\Route')->disableOriginalConstructor()->getMock(); + $mockRoute->expects($this->any()) + ->method('compile') + ->will($this->returnValue($mockCompiled)) + ; + $routeCollection = new RouteCollection(); + $routeCollection->add('some', $mockRoute); + $routeCollection->add('_company_more', $this->routeDocument); + $routeCollection->add('other', $mockRoute); + + $results = $this->matcher->finalMatch($routeCollection, $this->request); + + $expected = array( + '_route_name' => '_company_more', + '_route' => $this->routeDocument, + 'foo' => 'bar', + ); + + $this->assertEquals($expected, $results); + } + + public function testMatchNoRouteObject() + { + $this->routeCompiled->expects($this->atLeastOnce()) + ->method('getStaticPrefix') + ->will($this->returnValue($this->url)) + ; + $this->routeCompiled->expects($this->atLeastOnce()) + ->method('getRegex') + ->will($this->returnValue('#'.str_replace('/', '\\/', $this->url).'#')) + ; + $this->routeDocument = $this->getMockBuilder('Symfony\\Component\\Routing\\Route')->disableOriginalConstructor()->getMock(); + $this->routeDocument->expects($this->atLeastOnce()) + ->method('compile') + ->will($this->returnValue($this->routeCompiled)) + ; + $this->routeDocument->expects($this->never()) + ->method('getRouteKey') + ; + $this->routeDocument->expects($this->atLeastOnce()) + ->method('getDefaults') + ->will($this->returnValue(array('foo' => 'bar'))) + ; + + $mockCompiled = $this->buildMock('Symfony\\Component\\Routing\\CompiledRoute'); + $mockCompiled->expects($this->any()) + ->method('getStaticPrefix') + ->will($this->returnValue('/no/match')) + ; + $mockRoute = $this->getMockBuilder('Symfony\\Component\\Routing\\Route')->disableOriginalConstructor()->getMock(); + $mockRoute->expects($this->any()) + ->method('compile') + ->will($this->returnValue($mockCompiled)) + ; + $routeCollection = new RouteCollection(); + $routeCollection->add('some', $mockRoute); + $routeCollection->add('_company_more', $this->routeDocument); + $routeCollection->add('other', $mockRoute); + + $results = $this->matcher->finalMatch($routeCollection, $this->request); + + $expected = array( + '_route_name' => '_company_more', + '_route' => $this->routeDocument, + 'foo' => 'bar', + ); + + $this->assertEquals($expected, $results); + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/NestedMatcher/NestedMatcherTest.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/NestedMatcher/NestedMatcherTest.php new file mode 100644 index 000000000000..2bcee0e677fd --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/NestedMatcher/NestedMatcherTest.php @@ -0,0 +1,134 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Tests\NestedMatcher; + +use Symfony\Component\HttpFoundation\Request; + +use Symfony\Component\Routing\RouteCollection; +use Symfony\Component\Routing\Route; +use Symfony\Component\Routing\Exception\ResourceNotFoundException; + +use Symfony\Cmf\Component\Routing\NestedMatcher\NestedMatcher; + +use Symfony\Cmf\Component\Routing\Test\CmfUnitTestCase; + +class NestedMatcherTest extends CmfUnitTestCase +{ + private $provider; + private $routeFilter1; + private $routeFilter2; + private $finalMatcher; + + public function setUp() + { + $this->provider = $this->buildMock("Symfony\\Cmf\\Component\\Routing\\RouteProviderInterface"); + $this->routeFilter1 = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\NestedMatcher\\RouteFilterInterface'); + $this->routeFilter2 = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\NestedMatcher\\RouteFilterInterface'); + $this->finalMatcher = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\NestedMatcher\\FinalMatcherInterface'); + } + + public function testNestedMatcher() + { + $request = Request::create('/path/one'); + $routeCollection = new RouteCollection(); + $route = $this->getMockBuilder('Symfony\\Component\\Routing\\Route')->disableOriginalConstructor()->getMock(); + $routeCollection->add('route', $route); + + $this->provider->expects($this->once()) + ->method('getRouteCollectionForRequest') + ->with($request) + ->will($this->returnValue($routeCollection)) + ; + $this->routeFilter1->expects($this->once()) + ->method('filter') + ->with($routeCollection, $request) + ->will($this->returnValue($routeCollection)) + ; + $this->routeFilter2->expects($this->once()) + ->method('filter') + ->with($routeCollection, $request) + ->will($this->returnValue($routeCollection)) + ; + $this->finalMatcher->expects($this->once()) + ->method('finalMatch') + ->with($routeCollection, $request) + ->will($this->returnValue(array('foo' => 'bar'))) + ; + + $matcher = new NestedMatcher($this->provider); + $matcher->addRouteFilter($this->routeFilter1); + $matcher->addRouteFilter($this->routeFilter2); + $matcher->setFinalMatcher($this->finalMatcher); + + $attributes = $matcher->matchRequest($request); + + $this->assertEquals(array('foo' => 'bar'), $attributes); + } + + /** + * Test priorities and exception handling + */ + public function testNestedMatcherPriority() + { + $request = Request::create('/path/one'); + $routeCollection = new RouteCollection(); + $route = $this->getMockBuilder('Symfony\\Component\\Routing\\Route')->disableOriginalConstructor()->getMock(); + $routeCollection->add('route', $route); + + $wrongProvider = $this->buildMock("Symfony\\Cmf\\Component\\Routing\\RouteProviderInterface"); + $wrongProvider->expects($this->never()) + ->method('getRouteCollectionForRequest') + ; + $this->provider->expects($this->once()) + ->method('getRouteCollectionForRequest') + ->with($request) + ->will($this->returnValue($routeCollection)) + ; + $this->routeFilter1->expects($this->once()) + ->method('filter') + ->with($routeCollection, $request) + ->will($this->throwException(new ResourceNotFoundException())) + ; + $this->routeFilter2->expects($this->never()) + ->method('filter') + ; + $this->finalMatcher->expects($this->never()) + ->method('finalMatch') + ; + + $matcher = new NestedMatcher($wrongProvider); + $matcher->setRouteProvider($this->provider); + $matcher->addRouteFilter($this->routeFilter2, 10); + $matcher->addRouteFilter($this->routeFilter1, 20); + $matcher->setFinalMatcher($this->finalMatcher); + + try { + $matcher->matchRequest($request); + fail('nested matcher is eating exception'); + } catch(ResourceNotFoundException $e) + { + // expected + } + } + + public function testProviderNoMatch() + { + $request = Request::create('/path/one'); + $routeCollection = new RouteCollection(); + $this->provider->expects($this->once()) + ->method('getRouteCollectionForRequest') + ->with($request) + ->will($this->returnValue($routeCollection)) + ; + $this->finalMatcher->expects($this->never()) + ->method('finalMatch') + ; + + $matcher = new NestedMatcher($this->provider); + $matcher->setFinalMatcher($this->finalMatcher); + + $this->setExpectedException('Symfony\\Component\\Routing\\Exception\\ResourceNotFoundException'); + $matcher->matchRequest($request); + } + +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/NestedMatcher/UrlMatcherTest.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/NestedMatcher/UrlMatcherTest.php new file mode 100644 index 000000000000..08a1dec3a2e6 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/NestedMatcher/UrlMatcherTest.php @@ -0,0 +1,148 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Tests\NestedMatcher; + +use Symfony\Component\HttpFoundation\Request; + +use Symfony\Component\Routing\RouteCollection; +use Symfony\Component\Routing\Route; +use Symfony\Cmf\Component\Routing\NestedMatcher\UrlMatcher; + +use Symfony\Cmf\Component\Routing\RouteObjectInterface; + +use Symfony\Cmf\Component\Routing\Test\CmfUnitTestCase; + +class UrlMatcherTest extends CmfUnitTestCase +{ + protected $routeDocument; + protected $routeCompiled; + protected $matcher; + protected $context; + protected $request; + + protected $url = '/foo/bar'; + + public function setUp() + { + $reflection = new \ReflectionClass('Symfony\\Component\\Routing\\Matcher\\UrlMatcher'); + if (! $reflection->hasMethod('getAttributes')) { + $this->markTestSkipped('This only works with symfony 2.2'); + } + + $this->routeDocument = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\Tests\\Routing\\RouteMock', array('getDefaults', 'getRouteKey', 'compile')); + $this->routeCompiled = $this->buildMock('Symfony\\Component\\Routing\\CompiledRoute'); + + $this->context = $this->buildMock('Symfony\\Component\\Routing\\RequestContext'); + $this->request = Request::create($this->url); + + $this->matcher = new UrlMatcher(new RouteCollection(), $this->context); + } + + public function testMatchRouteKey() + { + $this->doTestMatchRouteKey($this->url); + } + + public function testMatchNoKey() + { + $this->doTestMatchRouteKey(null); + } + + public function doTestMatchRouteKey($routeKey) + { + $this->routeCompiled->expects($this->atLeastOnce()) + ->method('getStaticPrefix') + ->will($this->returnValue($this->url)) + ; + $this->routeCompiled->expects($this->atLeastOnce()) + ->method('getRegex') + ->will($this->returnValue('#'.str_replace('/', '\\/', $this->url).'#')) + ; + $this->routeDocument->expects($this->atLeastOnce()) + ->method('compile') + ->will($this->returnValue($this->routeCompiled)) + ; + $this->routeDocument->expects($this->atLeastOnce()) + ->method('getRouteKey') + ->will($this->returnValue($routeKey)) + ; + $this->routeDocument->expects($this->atLeastOnce()) + ->method('getDefaults') + ->will($this->returnValue(array('foo' => 'bar'))) + ; + + $mockCompiled = $this->buildMock('Symfony\\Component\\Routing\\CompiledRoute'); + $mockCompiled->expects($this->any()) + ->method('getStaticPrefix') + ->will($this->returnValue('/no/match')) + ; + $mockRoute = $this->getMockBuilder('Symfony\\Component\\Routing\\Route')->disableOriginalConstructor()->getMock(); + $mockRoute->expects($this->any()) + ->method('compile') + ->will($this->returnValue($mockCompiled)) + ; + $routeCollection = new RouteCollection(); + $routeCollection->add('some', $mockRoute); + $routeCollection->add('_company_more', $this->routeDocument); + $routeCollection->add('other', $mockRoute); + + $results = $this->matcher->finalMatch($routeCollection, $this->request); + + $expected = array( + '_route_name' => ($routeKey) ? $routeKey : '_company_more', + '_route' => $this->routeDocument, + 'foo' => 'bar', + ); + + $this->assertEquals($expected, $results); + } + + public function testMatchNoRouteObject() + { + $this->routeCompiled->expects($this->atLeastOnce()) + ->method('getStaticPrefix') + ->will($this->returnValue($this->url)) + ; + $this->routeCompiled->expects($this->atLeastOnce()) + ->method('getRegex') + ->will($this->returnValue('#'.str_replace('/', '\\/', $this->url).'#')) + ; + $this->routeDocument = $this->getMockBuilder('Symfony\\Component\\Routing\\Route')->disableOriginalConstructor()->getMock(); + $this->routeDocument->expects($this->atLeastOnce()) + ->method('compile') + ->will($this->returnValue($this->routeCompiled)) + ; + $this->routeDocument->expects($this->never()) + ->method('getRouteKey') + ; + $this->routeDocument->expects($this->atLeastOnce()) + ->method('getDefaults') + ->will($this->returnValue(array('foo' => 'bar'))) + ; + + $mockCompiled = $this->buildMock('Symfony\\Component\\Routing\\CompiledRoute'); + $mockCompiled->expects($this->any()) + ->method('getStaticPrefix') + ->will($this->returnValue('/no/match')) + ; + $mockRoute = $this->getMockBuilder('Symfony\\Component\\Routing\\Route')->disableOriginalConstructor()->getMock(); + $mockRoute->expects($this->any()) + ->method('compile') + ->will($this->returnValue($mockCompiled)) + ; + $routeCollection = new RouteCollection(); + $routeCollection->add('some', $mockRoute); + $routeCollection->add('_company_more', $this->routeDocument); + $routeCollection->add('other', $mockRoute); + + $results = $this->matcher->finalMatch($routeCollection, $this->request); + + $expected = array( + '_route_name' => '_company_more', + '_route' => $this->routeDocument, + 'foo' => 'bar', + ); + + $this->assertEquals($expected, $results); + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/ChainRouterTest.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/ChainRouterTest.php new file mode 100644 index 000000000000..d76c1f95170e --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/ChainRouterTest.php @@ -0,0 +1,617 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Tests\Routing; + +use Symfony\Component\Routing\RouteCollection; +use Symfony\Component\HttpFoundation\Request; + +use Symfony\Cmf\Component\Routing\ChainRouter; +use Symfony\Cmf\Component\Routing\Test\CmfUnitTestCase; + +class ChainRouterTest extends CmfUnitTestCase +{ + public function setUp() + { + $this->router = new ChainRouter($this->getMock('Symfony\Component\HttpKernel\Log\LoggerInterface')); + $this->context = $this->getMock('Symfony\\Component\\Routing\\RequestContext'); + } + + public function testPriority() + { + $this->assertEquals(array(), $this->router->all()); + + list($low, $high) = $this->createRouterMocks(); + + $this->router->add($low, 10); + $this->router->add($high, 100); + + $this->assertEquals(array( + $high, + $low, + ), $this->router->all()); + } + + /** + * Routers are supposed to be sorted only once. + * This test will check that by trying to get all routers several times. + * + * @covers \Symfony\Cmf\Component\Routing\ChainRouter::sortRouters + * @covers \Symfony\Cmf\Component\Routing\ChainRouter::all + */ + public function testSortRouters() + { + list($low, $medium, $high) = $this->createRouterMocks(); + // We're using a mock here and not $this->router because we need to ensure that the sorting operation is done only once. + $router = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\ChainRouter', array('sortRouters')); + $router + ->expects($this->once()) + ->method('sortRouters') + ->will( + $this->returnValue( + array($high, $medium, $low) + ) + ) + ; + + $router->add($low, 10); + $router->add($medium, 50); + $router->add($high, 100); + $expectedSortedRouters = array($high, $medium, $low); + // Let's get all routers 5 times, we should only sort once. + for ($i = 0; $i < 5; ++$i) { + $this->assertSame($expectedSortedRouters, $router->all()); + } + } + + /** + * This test ensures that if a router is being added on the fly, the sorting is reset. + * + * @covers \Symfony\Cmf\Component\Routing\ChainRouter::sortRouters + * @covers \Symfony\Cmf\Component\Routing\ChainRouter::all + * @covers \Symfony\Cmf\Component\Routing\ChainRouter::add + */ + public function testReSortRouters() + { + list($low, $medium, $high) = $this->createRouterMocks(); + $highest = clone $high; + // We're using a mock here and not $this->router because we need to ensure that the sorting operation is done only once. + $router = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\ChainRouter', array('sortRouters')); + $router + ->expects($this->at(0)) + ->method('sortRouters') + ->will( + $this->returnValue( + array($high, $medium, $low) + ) + ) + ; + // The second time sortRouters() is called, we're supposed to get the newly added router ($highest) + $router + ->expects($this->at(1)) + ->method('sortRouters') + ->will( + $this->returnValue( + array($highest, $high, $medium, $low) + ) + ) + ; + + $router->add($low, 10); + $router->add($medium, 50); + $router->add($high, 100); + $this->assertSame(array($high, $medium, $low), $router->all()); + + // Now adding another router on the fly, sorting must have been reset + $router->add($highest, 101); + $this->assertSame(array($highest, $high, $medium, $low), $router->all()); + } + + /** + * context must be propagated to chained routers and be stored locally + */ + public function testContext() + { + list($low, $high) = $this->createRouterMocks(); + + $low + ->expects($this->once()) + ->method('setContext') + ->with($this->context) + ; + + $high + ->expects($this->once()) + ->method('setContext') + ->with($this->context) + ; + + + $this->router->add($low, 10); + $this->router->add($high, 100); + + $this->router->setContext($this->context); + $this->assertSame($this->context, $this->router->getContext()); + } + + /** + * context must be propagated also when routers are added after context is set + */ + public function testContextOrder() + { + list($low, $high) = $this->createRouterMocks(); + + $low + ->expects($this->once()) + ->method('setContext') + ->with($this->context) + ; + + $high + ->expects($this->once()) + ->method('setContext') + ->with($this->context) + ; + + $this->router->setContext($this->context); + + $this->router->add($low, 10); + $this->router->add($high, 100); + + $this->router->all(); + + $this->assertSame($this->context, $this->router->getContext()); + } + + /** + * The first usable match is used, no further routers are queried once a match is found + */ + public function testMatch() + { + $url = '/test'; + list($lower, $low, $high) = $this->createRouterMocks(); + + $high + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->throwException(new \Symfony\Component\Routing\Exception\ResourceNotFoundException)) + ; + $low + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->returnValue(array('test'))) + ; + $lower + ->expects($this->never()) + ->method('match'); + $this->router->add($lower, 5); + $this->router->add($low, 10); + $this->router->add($high, 100); + + $result = $this->router->match('/test'); + $this->assertEquals(array('test'), $result); + } + + /** + * The first usable match is used, no further routers are queried once a match is found + */ + public function testMatchRequest() + { + $url = '/test'; + list($lower, $low, $high) = $this->createRouterMocks(); + + $highest = $this->getMock('Symfony\\Cmf\\Component\\Routing\\Tests\\Routing\\RequestMatcher'); + + $request = Request::create('/test'); + + $highest + ->expects($this->once()) + ->method('matchRequest') + ->will($this->throwException(new \Symfony\Component\Routing\Exception\ResourceNotFoundException)) + ; + $high + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->throwException(new \Symfony\Component\Routing\Exception\ResourceNotFoundException)) + ; + $low + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->returnValue(array('test'))) + ; + $lower + ->expects($this->never()) + ->method('match') + ; + + $this->router->add($lower, 5); + $this->router->add($low, 10); + $this->router->add($high, 100); + $this->router->add($highest, 200); + + $result = $this->router->matchRequest($request); + $this->assertEquals(array('test'), $result); + } + + /** + * If there is a method not allowed but another router matches, that one is used + */ + public function testMatchAndNotAllowed() + { + $url = '/test'; + list($low, $high) = $this->createRouterMocks(); + + $high + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->throwException(new \Symfony\Component\Routing\Exception\MethodNotAllowedException(array()))) + ; + $low + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->returnValue(array('test'))) + ; + $this->router->add($low, 10); + $this->router->add($high, 100); + + $result = $this->router->match('/test'); + $this->assertEquals(array('test'), $result); + } + + /** + * If there is a method not allowed but another router matches, that one is used + */ + public function testMatchRequestAndNotAllowed() + { + $url = '/test'; + list($low, $high) = $this->createRouterMocks(); + + $high + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->throwException(new \Symfony\Component\Routing\Exception\MethodNotAllowedException(array()))) + ; + $low + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->returnValue(array('test'))) + ; + $this->router->add($low, 10); + $this->router->add($high, 100); + + $result = $this->router->matchRequest(Request::create('/test')); + $this->assertEquals(array('test'), $result); + } + + /** + * @expectedException \Symfony\Component\Routing\Exception\ResourceNotFoundException + */ + public function testMatchNotFound() + { + $url = '/test'; + list($low, $high) = $this->createRouterMocks(); + + $high + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->throwException(new \Symfony\Component\Routing\Exception\ResourceNotFoundException)) + ; + $low + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->throwException(new \Symfony\Component\Routing\Exception\ResourceNotFoundException)) + ; + $this->router->add($low, 10); + $this->router->add($high, 100); + + $this->router->match('/test'); + } + + /** + * @expectedException \Symfony\Component\Routing\Exception\ResourceNotFoundException + */ + public function testMatchRequestNotFound() + { + $url = '/test'; + list($low, $high) = $this->createRouterMocks(); + + $high + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->throwException(new \Symfony\Component\Routing\Exception\ResourceNotFoundException)) + ; + $low + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->throwException(new \Symfony\Component\Routing\Exception\ResourceNotFoundException)) + ; + $this->router->add($low, 10); + $this->router->add($high, 100); + + $this->router->matchRequest(Request::create('/test')); + } + + /** + * If any of the routers throws a not allowed exception and no other matches, we need to see this + * + * @expectedException \Symfony\Component\Routing\Exception\MethodNotAllowedException + */ + public function testMatchMethodNotAllowed() + { + $url = '/test'; + list($low, $high) = $this->createRouterMocks(); + + $high + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->throwException(new \Symfony\Component\Routing\Exception\MethodNotAllowedException(array()))) + ; + $low + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->throwException(new \Symfony\Component\Routing\Exception\ResourceNotFoundException)) + ; + $this->router->add($low, 10); + $this->router->add($high, 100); + + $this->router->match('/test'); + } + + /** + * If any of the routers throws a not allowed exception and no other matches, we need to see this + * + * @expectedException \Symfony\Component\Routing\Exception\MethodNotAllowedException + */ + public function testMatchRequestMethodNotAllowed() + { + $url = '/test'; + list($low, $high) = $this->createRouterMocks(); + + $high + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->throwException(new \Symfony\Component\Routing\Exception\MethodNotAllowedException(array()))) + ; + $low + ->expects($this->once()) + ->method('match') + ->with($url) + ->will($this->throwException(new \Symfony\Component\Routing\Exception\ResourceNotFoundException)) + ; + $this->router->add($low, 10); + $this->router->add($high, 100); + + $this->router->matchRequest(Request::create('/test')); + } + + public function testGenerate() + { + $url = '/test'; + $name = 'test'; + $parameters = array('test' => 'value'); + list($lower, $low, $high) = $this->createRouterMocks(); + + $high + ->expects($this->once()) + ->method('generate') + ->with($name, $parameters, false) + ->will($this->throwException(new \Symfony\Component\Routing\Exception\RouteNotFoundException())) + ; + $low + ->expects($this->once()) + ->method('generate') + ->with($name, $parameters, false) + ->will($this->returnValue($url)) + ; + $lower + ->expects($this->never()) + ->method('generate') + ; + + $this->router->add($low, 10); + $this->router->add($high, 100); + + $result = $this->router->generate($name, $parameters); + $this->assertEquals($url, $result); + } + + /** + * @expectedException \Symfony\Component\Routing\Exception\RouteNotFoundException + */ + public function testGenerateNotFound() + { + $url = '/test'; + $name = 'test'; + $parameters = array('test' => 'value'); + list($low, $high) = $this->createRouterMocks(); + + $high + ->expects($this->once()) + ->method('generate') + ->with($name, $parameters, false) + ->will($this->throwException(new \Symfony\Component\Routing\Exception\RouteNotFoundException())) + ; + $low->expects($this->once()) + ->method('generate') + ->with($name, $parameters, false) + ->will($this->throwException(new \Symfony\Component\Routing\Exception\RouteNotFoundException())) + ; + $this->router->add($low, 10); + $this->router->add($high, 100); + + $result = $this->router->generate($name, $parameters); + $this->assertEquals($url, $result); + } + + public function testGenerateObjectName() + { + $name = new \stdClass(); + $parameters = array('test' => 'value'); + + $defaultRouter = $this->getMock('Symfony\\Component\\Routing\\RouterInterface'); + $chainedRouter = $this->getMock('Symfony\\Cmf\\Component\\Routing\\ChainedRouterInterface'); + + $defaultRouter + ->expects($this->never()) + ->method('generate') + ; + $chainedRouter + ->expects($this->once()) + ->method('supports') + ->will($this->returnValue(true)) + ; + $chainedRouter + ->expects($this->once()) + ->method('generate') + ->with($name, $parameters, false) + ->will($this->returnValue($name)) + ; + + $this->router->add($defaultRouter, 200); + $this->router->add($chainedRouter, 100); + + $result = $this->router->generate($name, $parameters); + $this->assertEquals($name, $result); + } + + public function testGenerateNonDefaultStringName() + { + $name = '/test/this'; + $parameters = array('test' => 'value'); + + $defaultRouter = $this->getMock('Symfony\\Component\\Routing\\RouterInterface'); + $chainedRouter = $this->getMock('Symfony\\Cmf\\Component\\Routing\\ChainedRouterInterface'); + + $defaultRouter + ->expects($this->never()) + ->method('generate') + ; + $chainedRouter + ->expects($this->once()) + ->method('supports') + ->will($this->returnValue(true)) + ; + $chainedRouter + ->expects($this->once()) + ->method('generate') + ->with($name, $parameters, false) + ->will($this->returnValue($name)) + ; + + $this->router->add($defaultRouter, 200); + $this->router->add($chainedRouter, 100); + + $result = $this->router->generate($name, $parameters); + $this->assertEquals($name, $result); + } + + public function testWarmup() + { + $dir = 'test_dir'; + list($low) = $this->createRouterMocks(); + + $low + ->expects($this->never()) + ->method('warmUp') + ; + $high = $this->getMock('Symfony\\Cmf\\Component\\Routing\\Tests\\Routing\\WarmableRouterMock'); + $high + ->expects($this->once()) + ->method('warmUp') + ->with($dir) + ; + + $this->router->add($low, 10); + $this->router->add($high, 100); + + $this->router->warmUp($dir); + } + + public function testRouteCollection() + { + list($low, $high) = $this->createRouterMocks(); + $lowcol = new RouteCollection(); + $lowcol->add('low', $this->buildMock('Symfony\\Component\\Routing\\Route')); + $highcol = new RouteCollection(); + $highcol->add('high', $this->buildMock('Symfony\\Component\\Routing\\Route')); + + $low + ->expects($this->once()) + ->method('getRouteCollection') + ->will($this->returnValue($lowcol)) + ; + $high + ->expects($this->once()) + ->method('getRouteCollection') + ->will($this->returnValue($highcol)) + ; + + $this->router->add($low, 10); + $this->router->add($high, 100); + + $collection = $this->router->getRouteCollection(); + $this->assertInstanceOf('Symfony\\Component\\Routing\\RouteCollection', $collection); + + $names = array(); + foreach ($collection->all() as $name => $route) { + $this->assertInstanceOf('Symfony\\Component\\Routing\\Route', $route); + $names[] = $name; + } + $this->assertEquals(array('high', 'low'), $names); + } + + /** + * @expectedException \Symfony\Component\Routing\Exception\RouteNotFoundException + */ + public function testSupport() + { + + $router = $this->getMock('Symfony\Cmf\Component\Routing\ChainedRouterInterface'); + $router + ->expects($this->once()) + ->method('supports') + ->will($this->returnValue(false)) + ; + + $router + ->expects($this->never()) + ->method('generate') + ->will($this->returnValue(false)) + ; + + $this->router->add($router); + + $this->router->generate('foobar'); + } + + protected function createRouterMocks() + { + return array( + $this->getMock('Symfony\\Component\\Routing\\RouterInterface'), + $this->getMock('Symfony\\Component\\Routing\\RouterInterface'), + $this->getMock('Symfony\\Component\\Routing\\RouterInterface'), + ); + } +} + +abstract class WarmableRouterMock implements \Symfony\Component\Routing\RouterInterface, \Symfony\Component\HttpKernel\CacheWarmer\WarmableInterface +{ +} + +abstract class RequestMatcher implements \Symfony\Component\Routing\RouterInterface, \Symfony\Component\Routing\Matcher\RequestMatcherInterface +{ +} \ No newline at end of file diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/ContentAwareGeneratorTest.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/ContentAwareGeneratorTest.php new file mode 100644 index 000000000000..6ddf3c71075c --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/ContentAwareGeneratorTest.php @@ -0,0 +1,285 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Tests\Routing; + +use Symfony\Component\Routing\Route as SymfonyRoute; + +use Symfony\Cmf\Component\Routing\ContentAwareGenerator; +use Symfony\Cmf\Component\Routing\Test\CmfUnitTestCase; + +class ContentAwareGeneratorTest extends CmfUnitTestCase +{ + protected $contentDocument; + protected $routeDocument; + protected $routeCompiled; + protected $provider; + + protected $generator; + protected $context; + + public function setUp() + { + $this->contentDocument = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\RouteAwareInterface'); + $this->routeDocument = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\Tests\\Routing\\RouteMock', array('getDefaults', 'compile')); + $this->routeCompiled = $this->buildMock('Symfony\\Component\\Routing\\CompiledRoute'); + $this->provider = $this->buildMock("Symfony\\Cmf\\Component\\Routing\\RouteProviderInterface"); + $this->context = $this->buildMock('Symfony\\Component\\Routing\\RequestContext'); + + $this->generator = new TestableContentAwareGenerator($this->provider); + } + + + public function testGenerateFromContent() + { + $this->provider->expects($this->never()) + ->method('getRouteByName') + ; + $this->contentDocument->expects($this->once()) + ->method('getRoutes') + ->will($this->returnValue(array($this->routeDocument))) + ; + $this->routeDocument->expects($this->once()) + ->method('compile') + ->will($this->returnValue($this->routeCompiled)) + ; + + $this->assertEquals('result_url', $this->generator->generate($this->contentDocument)); + } + + public function testGenerateFromContentId() + { + $this->provider->expects($this->never()) + ->method('getRouteByName') + ; + + $contentRepository = $this->buildMock("Symfony\\Cmf\\Component\\Routing\\ContentRepositoryInterface", array('findById')); + $contentRepository->expects($this->once()) + ->method('findById') + ->with('/content/id') + ->will($this->returnValue($this->contentDocument)) + ; + $this->generator->setContentRepository($contentRepository); + + $this->contentDocument->expects($this->once()) + ->method('getRoutes') + ->will($this->returnValue(array($this->routeDocument))) + ; + + $this->routeDocument->expects($this->once()) + ->method('compile') + ->will($this->returnValue($this->routeCompiled)) + ; + + $this->assertEquals('result_url', $this->generator->generate('', array('content_id' => '/content/id'))); + } + + public function testGenerateEmptyRouteString() + { + $this->provider->expects($this->never()) + ->method('getRouteByName') + ; + + $this->contentDocument->expects($this->once()) + ->method('getRoutes') + ->will($this->returnValue(array($this->routeDocument))) + ; + + $this->routeDocument->expects($this->once()) + ->method('compile') + ->will($this->returnValue($this->routeCompiled)) + ; + + $this->assertEquals('result_url', $this->generator->generate('', array('content'=>$this->contentDocument))); + } + + public function testGenerateRouteMultilang() + { + $route_en = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\Tests\\Routing\\RouteMock', array('getDefaults', 'compile', 'getRouteContent')); + $route_en->setLocale('en'); + $route_de = $this->routeDocument; + $route_de->setLocale('de'); + + $this->contentDocument->expects($this->once()) + ->method('getRoutes') + ->will($this->returnValue(array($route_en, $route_de))) + ; + $route_en->expects($this->once()) + ->method('getRouteContent') + ->will($this->returnValue($this->contentDocument)) + ; + $route_en->expects($this->never()) + ->method('compile') + ; + $route_de->expects($this->once()) + ->method('compile') + ->will($this->returnValue($this->routeCompiled)) + ; + + $this->assertEquals('result_url', $this->generator->generate($route_en, array('_locale' => 'de'))); + } + + + public function testGenerateRouteMultilangNomatch() + { + $route_en = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\Tests\\Routing\\RouteMock', array('getDefaults', 'compile', 'getRouteContent')); + $route_en->setLocale('en'); + $route_de = $this->routeDocument; + $route_de->setLocale('de'); + + $this->contentDocument->expects($this->once()) + ->method('getRoutes') + ->will($this->returnValue(array($route_en, $route_de))) + ; + $route_en->expects($this->once()) + ->method('getRouteContent') + ->will($this->returnValue($this->contentDocument)) + ; + $route_en->expects($this->once()) + ->method('compile') + ->will($this->returnValue($this->routeCompiled)) + ; + $route_de->expects($this->never()) + ->method('compile') + ; + + $this->assertEquals('result_url', $this->generator->generate($route_en, array('_locale' => 'fr'))); + } + + public function testGenerateNoncmfRouteMultilang() + { + $route_en = $this->buildMock('Symfony\\Component\\Routing\\Route', array('getDefaults', 'compile', 'getRouteContent')); + + $route_en->expects($this->once()) + ->method('compile') + ->will($this->returnValue($this->routeCompiled)) + ; + + $this->assertEquals('result_url', $this->generator->generate($route_en, array('_locale' => 'de'))); + } + + public function testGenerateRoutenameMultilang() + { + $name = 'foo/bar'; + $route_en = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\Tests\\Routing\\RouteMock', array('getDefaults', 'compile', 'getRouteContent')); + $route_en->setLocale('en'); + $route_de = $this->routeDocument; + $route_de->setLocale('de'); + + $this->provider->expects($this->once()) + ->method('getRouteByName') + ->with($name) + ->will($this->returnValue($route_en)) + ; + $this->contentDocument->expects($this->once()) + ->method('getRoutes') + ->will($this->returnValue(array($route_en, $route_de))) + ; + $route_en->expects($this->once()) + ->method('getRouteContent') + ->will($this->returnValue($this->contentDocument)) + ; + $route_en->expects($this->never()) + ->method('compile') + ; + $route_de->expects($this->once()) + ->method('compile') + ->will($this->returnValue($this->routeCompiled)) + ; + + $this->assertEquals('result_url', $this->generator->generate($name, array('_locale' => 'de'))); + } + + /** + * @expectedException \Symfony\Component\Routing\Exception\RouteNotFoundException + */ + public function testGenerateRoutenameMultilangNotFound() + { + $name = 'foo/bar'; + + $this->provider->expects($this->once()) + ->method('getRouteByName') + ->with($name) + ->will($this->returnValue(null)) + ; + + $this->generator->generate($name, array('_locale' => 'de')); + } + + public function testGenerateDocumentMultilang() + { + $route_en = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\Tests\\Routing\\RouteMock', array('getDefaults', 'compile')); + $route_en->setLocale('en'); + $route_de = $this->routeDocument; + $route_de->setLocale('de'); + + $this->contentDocument->expects($this->once()) + ->method('getRoutes') + ->will($this->returnValue(array($route_en, $route_de))) + ; + $route_en->expects($this->never()) + ->method('compile') + ; + $route_de->expects($this->once()) + ->method('compile') + ->will($this->returnValue($this->routeCompiled)) + ; + + $this->assertEquals('result_url', $this->generator->generate('', array('content'=>$this->contentDocument, '_locale' => 'de'))); + } + + /** + * @expectedException Symfony\Component\Routing\Exception\RouteNotFoundException + */ + public function testGenerateNoContent() + { + $this->generator->generate('', array()); + } + /** + * @expectedException Symfony\Component\Routing\Exception\RouteNotFoundException + */ + public function testGenerateInvalidContent() + { + $this->generator->generate('', array('content' => $this)); + } + /** + * @expectedException Symfony\Component\Routing\Exception\RouteNotFoundException + */ + public function testGenerateNoRoutes() + { + $this->contentDocument->expects($this->once()) + ->method('getRoutes') + ->will($this->returnValue(array())); + + $this->generator->generate('', array('content'=>$this->contentDocument)); + } + /** + * @expectedException Symfony\Component\Routing\Exception\RouteNotFoundException + */ + public function testGenerateInvalidRoute() + { + $this->contentDocument->expects($this->once()) + ->method('getRoutes') + ->will($this->returnValue(array($this))); + + $this->generator->generate('', array('content'=>$this->contentDocument)); + } + + public function testSupports() + { + $this->assertTrue($this->generator->supports('')); + $this->assertTrue($this->generator->supports(null)); + $this->assertTrue($this->generator->supports($this->contentDocument)); + $this->assertFalse($this->generator->supports($this)); + } +} + +/** + * Overwrite doGenerate to reduce amount of mocking needed + */ +class TestableContentAwareGenerator extends ContentAwareGenerator +{ + protected function doGenerate($variables, $defaults, $requirements, $tokens, $parameters, $name, $absolute, $hostnameTokens = null) + { + return 'result_url'; + } +} \ No newline at end of file diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/DynamicRouterTest.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/DynamicRouterTest.php new file mode 100644 index 000000000000..af790a88d913 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/DynamicRouterTest.php @@ -0,0 +1,241 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Tests\Routing; + +use Symfony\Component\HttpFoundation\Request; + +use Symfony\Component\Routing\RouteCollection; +use Symfony\Component\Routing\Route; + +use Symfony\Cmf\Component\Routing\RouteObjectInterface; +use Symfony\Cmf\Component\Routing\DynamicRouter; + +use Symfony\Cmf\Component\Routing\Test\CmfUnitTestCase; + +class DynamicRouterTest extends CmfUnitTestCase +{ + protected $routeDocument; + protected $matcher; + protected $generator; + protected $enhancer; + protected $router; + protected $context; + protected $request; + + protected $url = '/foo/bar'; + + public function setUp() + { + $this->routeDocument = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\Tests\\Routing\\RouteMock', array('getDefaults')); + + $this->matcher = $this->buildMock('Symfony\\Component\\Routing\\Matcher\\UrlMatcherInterface'); + $this->generator = $this->buildMock('Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface', array('supports', 'generate', 'setContext', 'getContext')); + $this->enhancer = $this->buildMock('Symfony\\Cmf\\Component\\Routing\\Enhancer\\RouteEnhancerInterface', array('enhance')); + + $this->context = $this->buildMock('Symfony\\Component\\Routing\\RequestContext'); + $this->request = Request::create($this->url); + + $this->router = new DynamicRouter($this->context, $this->matcher, $this->generator); + $this->router->addRouteEnhancer($this->enhancer); + } + + /** + * rather trivial, but we want 100% coverage + */ + public function testContext() + { + $this->router->setContext($this->context); + $this->assertSame($this->context, $this->router->getContext()); + } + + public function testRouteCollection() + { + $collection = $this->router->getRouteCollection(); + $this->assertInstanceOf('Symfony\\Component\\Routing\\RouteCollection', $collection); + // TODO: once this is implemented, check content of collection + } + + + /// generator tests /// + + public function testGetGenerator() + { + $this->generator->expects($this->once()) + ->method('setContext') + ->with($this->equalTo($this->context)); + + $generator = $this->router->getGenerator(); + $this->assertInstanceOf('Symfony\Component\Routing\Generator\UrlGeneratorInterface', $generator); + $this->assertSame($this->generator, $generator); + } + + public function testGenerate() + { + $name = 'my_route_name'; + $parameters = array('foo' => 'bar'); + $absolute = true; + + $this->generator->expects($this->once()) + ->method('generate') + ->with($name, $parameters, $absolute) + ->will($this->returnValue('http://test')) + ; + + $url = $this->router->generate($name, $parameters, $absolute); + $this->assertEquals('http://test', $url); + } + + public function testSupports() + { + $name = 'foo/bar'; + $this->generator->expects($this->once()) + ->method('supports') + ->with($this->equalTo($name)) + ->will($this->returnValue(true)) + ; + + $this->assertTrue($this->router->supports($name)); + } + + /// match tests /// + + public function testGetMatcher() + { + $this->matcher->expects($this->once()) + ->method('setContext') + ->with($this->equalTo($this->context)); + + $matcher = $this->router->getMatcher(); + $this->assertInstanceOf('Symfony\\Component\\Routing\\Matcher\\UrlMatcherInterface', $matcher); + $this->assertSame($this->matcher, $matcher); + } + + public function testMatchUrl() + { + $routeDefaults = array('foo' => 'bar'); + $this->matcher->expects($this->once()) + ->method('match') + ->with($this->url) + ->will($this->returnValue($routeDefaults)) + ; + + $expected = array('this' => 'that'); + $this->enhancer->expects($this->once()) + ->method('enhance') + ->with($this->equalTo($routeDefaults), $this->equalTo($this->request)) + ->will($this->returnValue($expected)) + ; + + $results = $this->router->match($this->url); + + $this->assertEquals($expected, $results); + } + + + public function testMatchRequestWithUrlMatcher() + { + $routeDefaults = array('foo' => 'bar'); + + $this->matcher->expects($this->once()) + ->method('match') + ->with($this->url) + ->will($this->returnValue($routeDefaults)) + ; + + $expected = array('this' => 'that'); + $this->enhancer->expects($this->once()) + ->method('enhance') + // somehow request object gets confused, check on instance only + ->with($this->equalTo($routeDefaults), $this->isInstanceOf('Symfony\\Component\\HttpFoundation\\Request')) + ->will($this->returnValue($expected)) + ; + + $results = $this->router->matchRequest($this->request); + + $this->assertEquals($expected, $results); + } + + public function testMatchRequest() + { + $routeDefaults = array('foo' => 'bar'); + + $matcher = $this->buildMock('Symfony\\Component\\Routing\\Matcher\\RequestMatcherInterface', array('matchRequest', 'setContext', 'getContext')); + $router = new DynamicRouter($this->context, $matcher, $this->generator); + + $matcher->expects($this->once()) + ->method('matchRequest') + ->with($this->request) + ->will($this->returnValue($routeDefaults)) + ; + + $expected = array('this' => 'that'); + $this->enhancer->expects($this->once()) + ->method('enhance') + ->with($this->equalTo($routeDefaults), $this->equalTo($this->request)) // TODO: why do we not get the right thing? + ->will($this->returnValue($expected)) + ; + + $router->addRouteEnhancer($this->enhancer); + + $this->assertEquals($expected, $router->matchRequest($this->request)); + } + + /** + * @expectedException \Symfony\Component\Routing\Exception\ResourceNotFoundException + */ + public function testMatchFilter() + { + $router = new DynamicRouter($this->context, $this->matcher, $this->generator, '#/different/prefix.*#'); + $router->addRouteEnhancer($this->enhancer); + + $this->matcher->expects($this->never()) + ->method('match') + ; + + $this->enhancer->expects($this->never()) + ->method('enhance') + ; + + $router->match($this->url); + } + + /** + * @expectedException \Symfony\Component\Routing\Exception\ResourceNotFoundException + */ + public function testMatchRequestFilter() + { + $matcher = $this->buildMock('Symfony\\Component\\Routing\\Matcher\\RequestMatcherInterface', array('matchRequest', 'setContext', 'getContext')); + + $router = new DynamicRouter($this->context, $matcher, $this->generator, '#/different/prefix.*#'); + $router->addRouteEnhancer($this->enhancer); + + $matcher->expects($this->never()) + ->method('matchRequest') + ; + + $this->enhancer->expects($this->never()) + ->method('enhance') + ; + + $router->matchRequest($this->request); + } + + /** + * @expectedException \InvalidArgumentException + */ + public function testMatchUrlWithRequestMatcher() + { + $matcher = $this->buildMock('Symfony\\Component\\Routing\\Matcher\\RequestMatcherInterface', array('matchRequest', 'setContext', 'getContext')); + $router = new DynamicRouter($this->context, $matcher, $this->generator); + + $router->match($this->url); + } + + /** + * @expectedException \InvalidArgumentException + */ + public function testInvalidMatcher() + { + new DynamicRouter($this->context, $this, $this->generator); + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/ProviderBasedGeneratorTest.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/ProviderBasedGeneratorTest.php new file mode 100644 index 000000000000..509232dc140f --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/ProviderBasedGeneratorTest.php @@ -0,0 +1,94 @@ +<?php + +namespace Symfony\Cmf\Component\Routing\Tests\Routing; + + +use Symfony\Component\Routing\RouteCollection; +use Symfony\Component\Routing\Route; + +use Symfony\Cmf\Component\Routing\ProviderBasedGenerator; +use Symfony\Cmf\Component\Routing\Test\CmfUnitTestCase; + +class ProviderBasedGeneratorTest extends CmfUnitTestCase +{ + protected $routeDocument; + protected $routeCompiled; + protected $provider; + + protected $generator; + protected $context; + + public function setUp() + { + $this->routeDocument = $this->buildMock('Symfony\\Component\\Routing\\Route', array('getDefaults', 'compile')); + $this->routeCompiled = $this->buildMock('Symfony\\Component\\Routing\\CompiledRoute'); + $this->provider = $this->buildMock("Symfony\\Cmf\\Component\\Routing\\RouteProviderInterface"); + $this->context = $this->buildMock('Symfony\\Component\\Routing\\RequestContext'); + + $this->generator= new TestableProviderBasedGenerator($this->provider); + } + + public function testGenerateFromName() + { + $name = 'foo/bar'; + + $this->provider->expects($this->once()) + ->method('getRouteByName') + ->with($name) + ->will($this->returnValue($this->routeDocument)) + ; + $this->routeDocument->expects($this->once()) + ->method('compile') + ->will($this->returnValue($this->routeCompiled)) + ; + + $this->assertEquals('result_url', $this->generator->generate($name)); + } + + /** + * @expectedException \Symfony\Component\Routing\Exception\RouteNotFoundException + */ + public function testGenerateNotFound() + { + $name = 'foo/bar'; + + $this->provider->expects($this->once()) + ->method('getRouteByName') + ->with($name) + ->will($this->returnValue(null)) + ; + + $this->generator->generate($name); + } + + public function testGenerateFromRoute() + { + $this->provider->expects($this->never()) + ->method('getRouteByName') + ; + $this->routeDocument->expects($this->once()) + ->method('compile') + ->will($this->returnValue($this->routeCompiled)) + ; + + $this->assertEquals('result_url', $this->generator->generate($this->routeDocument)); + } + + public function testSupports() + { + $this->assertTrue($this->generator->supports('foo/bar')); + $this->assertTrue($this->generator->supports($this->routeDocument)); + $this->assertFalse($this->generator->supports($this)); + } +} + +/** + * Overwrite doGenerate to reduce amount of mocking needed + */ +class TestableProviderBasedGenerator extends ProviderBasedGenerator +{ + protected function doGenerate($variables, $defaults, $requirements, $tokens, $parameters, $name, $absolute, $hostnameTokens = null) + { + return 'result_url'; + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/RouteMock.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/RouteMock.php new file mode 100644 index 000000000000..e028d2c6adc8 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/Routing/RouteMock.php @@ -0,0 +1,39 @@ +<?php +namespace Symfony\Cmf\Component\Routing\Tests\Routing; + +use Symfony\Component\Routing\Route as SymfonyRoute; + +use Symfony\Cmf\Component\Routing\RouteObjectInterface; + +class RouteMock extends SymfonyRoute implements RouteObjectInterface +{ + private $locale; + + public function setLocale($locale) + { + $this->locale = $locale; + } + public function getRouteContent() + { + return null; + } + public function getDefaults() + { + $defaults = array(); + if (! is_null($this->locale)) { + $defaults['_locale'] = $this->locale; + } + return $defaults; + } + public function getRequirement($key) + { + if (! $key == '_locale') { + throw new \Exception; + } + return $this->locale; + } + public function getRouteKey() + { + return null; + } +} \ No newline at end of file diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/bootstrap.php b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/bootstrap.php new file mode 100644 index 000000000000..cd9b88e38bd5 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/Tests/bootstrap.php @@ -0,0 +1,19 @@ +<?php + +$file = __DIR__.'/../vendor/autoload.php'; +if (!file_exists($file)) { + throw new RuntimeException('Install dependencies to run test suite.'); +} + +require_once $file; + +spl_autoload_register(function($class) { + if (0 === strpos($class, 'Symfony\\Cmf\\Component\\Routing\\')) { + $path = __DIR__.'/../'.implode('/', array_slice(explode('\\', $class), 4)).'.php'; + if (!stream_resolve_include_path($path)) { + return false; + } + require_once $path; + return true; + } +}); diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/composer.json b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/composer.json new file mode 100644 index 000000000000..7f0a263980d8 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/composer.json @@ -0,0 +1,29 @@ +{ + "name": "symfony-cmf/routing", + "type": "library", + "description": "Extends the Symfony2 routing component for dynamic routes and chaining several routers", + "keywords": ["routing", "database"], + "homepage": "http://cmf.symfony.com", + "license": "MIT", + "authors": [ + { + "name": "Symfony CMF Community", + "homepage": "https://github.com/symfony-cmf/Routing/contributors" + } + ], + "minimum-stability": "dev", + "require": { + "php": ">=5.3.2", + "symfony/routing": ">=2.1,<2.3-dev", + "symfony/http-kernel": ">=2.1,<2.3-dev" + }, + "autoload": { + "psr-0": { "Symfony\\Cmf\\Component\\Routing": "" } + }, + "target-dir": "Symfony/Cmf/Component/Routing", + "extra": { + "branch-alias": { + "dev-master": "1.0-dev" + } + } +} diff --git a/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/phpunit.xml.dist b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/phpunit.xml.dist new file mode 100644 index 000000000000..2564ef1819c9 --- /dev/null +++ b/core/vendor/symfony-cmf/routing/Symfony/Cmf/Component/Routing/phpunit.xml.dist @@ -0,0 +1,26 @@ +<?xml version="1.0" encoding="UTF-8"?> + +<!-- http://www.phpunit.de/manual/current/en/appendixes.configuration.html --> +<phpunit + colors="true" + bootstrap="Tests/bootstrap.php" +> + + <testsuites> + <testsuite name="Symfony Cmf Routing Test Suite"> + <directory>./Tests</directory> + </testsuite> + </testsuites> + + <filter> + <whitelist addUncoveredFilesFromWhitelist="true"> + <directory>.</directory> + <exclude> + <directory>Test/</directory> + <directory>Tests/</directory> + <directory>vendor/</directory> + </exclude> + </whitelist> + </filter> + +</phpunit> -- GitLab