Your IP : 216.73.217.13


Current Path : /var/www/surf/TYPO3/vendor/typo3/cms-backend/Classes/Routing/
Upload File :
Current File : /var/www/surf/TYPO3/vendor/typo3/cms-backend/Classes/Routing/Router.php

<?php

/*
 * This file is part of the TYPO3 CMS project.
 *
 * It is free software; you can redistribute it and/or modify it under
 * the terms of the GNU General Public License, either version 2
 * of the License, or any later version.
 *
 * For the full copyright and license information, please read the
 * LICENSE.txt file that was distributed with this source code.
 *
 * The TYPO3 project - inspiring people to share!
 */

namespace TYPO3\CMS\Backend\Routing;

use Psr\Http\Message\ServerRequestInterface;
use Symfony\Component\Routing\Matcher\UrlMatcher;
use Symfony\Component\Routing\Route as SymfonyRoute;
use TYPO3\CMS\Backend\Routing\Exception\MethodNotAllowedException;
use TYPO3\CMS\Backend\Routing\Exception\ResourceNotFoundException;
use TYPO3\CMS\Core\Routing\RequestContextFactory;
use TYPO3\CMS\Core\Routing\RouteCollection;
use TYPO3\CMS\Core\SingletonInterface;

/**
 * Implementation of a class for adding routes, collecting throughout the Bootstrap
 * to register all sorts of Backend Routes, and to fetch the main Collection in order
 * to resolve a route (see ->match() and ->matchRequest()).
 *
 * Ideally, the Router is solely instantiated and accessed via the Bootstrap, the RequestHandler and the UriBuilder.
 *
 * See \TYPO3\CMS\Backend\Http\RequestHandler for more details on route matching() and Bootstrap->initializeBackendRouting().
 *
 * The architecture is inspired by the Symfony Routing Component.
 */
class Router implements SingletonInterface
{
    /**
     * All routes used in the TYPO3 Backend
     */
    protected RouteCollection $routeCollection;

    public function __construct(
        protected readonly RequestContextFactory $requestContextFactory
    ) {
        $this->routeCollection = new RouteCollection();
    }

    /**
     * Adds a new route with a given identifier
     */
    public function addRoute(string $routeIdentifier, Route $route, array $aliases = []): void
    {
        $symfonyRoute = new SymfonyRoute($route->getPath(), [], [], $route->getOptions());
        $symfonyRoute->setMethods($route->getMethods());
        $this->routeCollection->add($routeIdentifier, $symfonyRoute);
        foreach ($aliases as $aliasName) {
            $this->routeCollection->addAlias($aliasName, $routeIdentifier);
        }
    }

    public function addRouteCollection(RouteCollection $routeCollection): void
    {
        $this->routeCollection->addCollection($routeCollection);
    }

    /**
     * Fetch all registered routes, only use in UriBuilder. Does not care about aliases,
     * so be careful with using this method.
     *
     * @return Route[]
     */
    public function getRoutes(): iterable
    {
        return $this->routeCollection->getIterator();
    }

    public function hasRoute(string $routeName): bool
    {
        return $this->routeCollection->get($routeName) !== null;
    }

    /**
     * Returns a route by its identifier, or null if nothing found.
     *
     * @return SymfonyRoute|Route|null
     */
    public function getRoute(string $routeName)
    {
        return $this->routeCollection->get($routeName);
    }

    /**
     * @internal only use in Core, this should not be exposed
     */
    public function getRouteCollection(): RouteCollection
    {
        return $this->routeCollection;
    }

    /**
     * Tries to match a URL path with a set of routes.
     *
     * @param string $pathInfo The path info to be parsed
     * @return Route the first Route object found
     * @throws ResourceNotFoundException If the resource could not be found
     */
    public function match($pathInfo): Route
    {
        foreach ($this->routeCollection->getIterator() as $routeIdentifier => $route) {
            // This check is done in a simple way as there are no parameters yet (get parameters only)
            if ($route->getPath() === $pathInfo) {
                $routeResult = new Route($route->getPath(), $route->getOptions());
                // Store the name of the Route in the _identifier option so the token can be checked against that
                $routeResult->setOption('_identifier', $routeIdentifier);
                return $routeResult;
            }
        }

        throw new ResourceNotFoundException('The requested resource "' . $pathInfo . '" was not found.', 1425389240);
    }

    /**
     * Matches a PSR-7 Request and returns a RouteResult with parameters and the resolved route.
     */
    public function matchResult(ServerRequestInterface $request): RouteResult
    {
        $path = $request->getUri()->getPath();
        if (($normalizedParams = $request->getAttribute('normalizedParams')) !== null) {
            // Remove the directory name of the script from the path. This will usually be `/typo3` in this context.
            $path = substr($path, strlen(dirname($normalizedParams->getScriptName())));
        }
        if ($path === '' || $path === '/' || $path === '/index.php') {
            // Allow the login page to be displayed if routing is not used and on index.php
            // (consolidate RouteDispatcher::evaluateReferrer() when changing 'login' to something different)
            $path = '/login';
        }
        $requestContext = $this->requestContextFactory->fromBackendRequest($request);
        try {
            $result = (new UrlMatcher($this->routeCollection, $requestContext))->match($path);
            $matchedSymfonyRoute = $this->routeCollection->get($result['_route']);
            if ($matchedSymfonyRoute === null) {
                throw new ResourceNotFoundException('The requested resource "' . $path . '" was not found.', 1607596900);
            }
        } catch (\Symfony\Component\Routing\Exception\MethodNotAllowedException $e) {
            throw new MethodNotAllowedException($e->getMessage(), 1612649842);
        } catch (\Symfony\Component\Routing\Exception\ResourceNotFoundException $e) {
            throw new ResourceNotFoundException('The requested resource "' . $path . '" was not found.', 1612649840);
        }
        // Apply matched method to route
        $matchedOptions = $matchedSymfonyRoute->getOptions();
        $methods = $matchedOptions['methods'] ?? [];
        unset($matchedOptions['methods']);
        $route = new Route($matchedSymfonyRoute->getPath(), $matchedOptions);
        if (count($methods) > 0) {
            $route->setMethods($methods);
        }
        $route->setOption('_identifier', $result['_route']);
        unset($result['_route']);
        return new RouteResult($route, $result);
    }

    /**
     * Tries to match a URI against the registered routes.
     * Use ->matchResult() instead, as this method will be deprecated in the future.
     *
     * @return Route the first Route object found
     */
    public function matchRequest(ServerRequestInterface $request): Route
    {
        return $this->matchResult($request)->getRoute();
    }
}