2019-05-01 00:32:33 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Friendica\App;
|
|
|
|
|
|
|
|
|
|
|
|
use FastRoute\DataGenerator\GroupCountBased;
|
|
|
|
use FastRoute\Dispatcher;
|
|
|
|
use FastRoute\RouteCollector;
|
|
|
|
use FastRoute\RouteParser\Std;
|
2019-09-26 21:18:01 +02:00
|
|
|
use Friendica\Core\Hook;
|
2019-10-11 17:55:02 +02:00
|
|
|
use Friendica\Core\L10n;
|
|
|
|
use Friendica\Network\HTTPException;
|
2019-05-01 00:32:33 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Wrapper for FastRoute\Router
|
|
|
|
*
|
|
|
|
* This wrapper only makes use of a subset of the router features, mainly parses a route rule to return the relevant
|
|
|
|
* module class.
|
|
|
|
*
|
|
|
|
* Actual routes are defined in App->collectRoutes.
|
|
|
|
*
|
|
|
|
* @package Friendica\App
|
|
|
|
*/
|
|
|
|
class Router
|
|
|
|
{
|
2019-09-26 21:18:01 +02:00
|
|
|
const POST = 'POST';
|
|
|
|
const GET = 'GET';
|
|
|
|
|
|
|
|
const ALLOWED_METHODS = [
|
|
|
|
self::POST,
|
|
|
|
self::GET,
|
|
|
|
];
|
|
|
|
|
2019-05-01 00:32:33 +02:00
|
|
|
/** @var RouteCollector */
|
|
|
|
protected $routeCollector;
|
|
|
|
|
|
|
|
/**
|
2019-09-26 21:18:01 +02:00
|
|
|
* @var string The HTTP method
|
2019-05-01 00:32:33 +02:00
|
|
|
*/
|
2019-09-26 21:18:01 +02:00
|
|
|
private $httpMethod;
|
2019-05-02 22:03:27 +02:00
|
|
|
|
2019-09-26 21:18:01 +02:00
|
|
|
/**
|
|
|
|
* @param array $server The $_SERVER variable
|
|
|
|
* @param RouteCollector|null $routeCollector Optional the loaded Route collector
|
|
|
|
*/
|
|
|
|
public function __construct(array $server, RouteCollector $routeCollector = null)
|
|
|
|
{
|
|
|
|
$httpMethod = $server['REQUEST_METHOD'] ?? self::GET;
|
|
|
|
$this->httpMethod = in_array($httpMethod, self::ALLOWED_METHODS) ? $httpMethod : self::GET;
|
2019-05-02 22:03:27 +02:00
|
|
|
|
2019-09-26 21:18:01 +02:00
|
|
|
$this->routeCollector = isset($routeCollector) ?
|
|
|
|
$routeCollector :
|
|
|
|
new RouteCollector(new Std(), new GroupCountBased());
|
|
|
|
}
|
2019-05-02 22:03:27 +02:00
|
|
|
|
2019-09-26 21:18:01 +02:00
|
|
|
/**
|
|
|
|
* @param array $routes The routes to add to the Router
|
|
|
|
*
|
|
|
|
* @return self The router instance with the loaded routes
|
|
|
|
*
|
2019-10-11 17:55:02 +02:00
|
|
|
* @throws HTTPException\InternalServerErrorException In case of invalid configs
|
2019-09-26 21:18:01 +02:00
|
|
|
*/
|
|
|
|
public function addRoutes(array $routes)
|
|
|
|
{
|
|
|
|
$routeCollector = (isset($this->routeCollector) ?
|
|
|
|
$this->routeCollector :
|
|
|
|
new RouteCollector(new Std(), new GroupCountBased()));
|
|
|
|
|
|
|
|
foreach ($routes as $route => $config) {
|
|
|
|
if ($this->isGroup($config)) {
|
|
|
|
$this->addGroup($route, $config, $routeCollector);
|
|
|
|
} elseif ($this->isRoute($config)) {
|
|
|
|
$routeCollector->addRoute($config[1], $route, $config[0]);
|
|
|
|
} else {
|
2019-10-11 17:55:02 +02:00
|
|
|
throw new HTTPException\InternalServerErrorException("Wrong route config for route '" . print_r($route, true) . "'");
|
2019-09-26 21:18:01 +02:00
|
|
|
}
|
|
|
|
}
|
2019-05-02 22:03:27 +02:00
|
|
|
|
2019-09-26 21:18:01 +02:00
|
|
|
$this->routeCollector = $routeCollector;
|
2019-05-02 22:03:27 +02:00
|
|
|
|
2019-09-26 21:18:01 +02:00
|
|
|
return $this;
|
|
|
|
}
|
2019-05-01 21:29:04 +02:00
|
|
|
|
2019-09-26 21:18:01 +02:00
|
|
|
/**
|
|
|
|
* Adds a group of routes to a given group
|
|
|
|
*
|
|
|
|
* @param string $groupRoute The route of the group
|
|
|
|
* @param array $routes The routes of the group
|
|
|
|
* @param RouteCollector $routeCollector The route collector to add this group
|
|
|
|
*/
|
|
|
|
private function addGroup(string $groupRoute, array $routes, RouteCollector $routeCollector)
|
|
|
|
{
|
|
|
|
$routeCollector->addGroup($groupRoute, function (RouteCollector $routeCollector) use ($routes) {
|
|
|
|
foreach ($routes as $route => $config) {
|
|
|
|
if ($this->isGroup($config)) {
|
|
|
|
$this->addGroup($route, $config, $routeCollector);
|
|
|
|
} elseif ($this->isRoute($config)) {
|
|
|
|
$routeCollector->addRoute($config[1], $route, $config[0]);
|
|
|
|
}else {
|
2019-10-11 17:55:02 +02:00
|
|
|
throw new HTTPException\InternalServerErrorException("Wrong route config for route '" . print_r($route, true) . "'");
|
2019-09-26 21:18:01 +02:00
|
|
|
}
|
|
|
|
}
|
2019-05-01 21:29:04 +02:00
|
|
|
});
|
2019-09-26 21:18:01 +02:00
|
|
|
}
|
2019-05-13 07:38:15 +02:00
|
|
|
|
2019-09-26 21:18:01 +02:00
|
|
|
/**
|
|
|
|
* Returns true in case the config is a group config
|
|
|
|
*
|
|
|
|
* @param array $config
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
private function isGroup(array $config)
|
|
|
|
{
|
|
|
|
return
|
|
|
|
is_array($config) &&
|
|
|
|
is_string(array_keys($config)[0]) &&
|
|
|
|
// This entry should NOT be a BaseModule
|
|
|
|
(substr(array_keys($config)[0], 0, strlen('Friendica\Module')) !== 'Friendica\Module') &&
|
|
|
|
// The second argument is an array (another routes)
|
|
|
|
is_array(array_values($config)[0]);
|
2019-05-01 00:32:33 +02:00
|
|
|
}
|
|
|
|
|
2019-09-26 21:18:01 +02:00
|
|
|
/**
|
|
|
|
* Returns true in case the config is a route config
|
|
|
|
*
|
|
|
|
* @param array $config
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
private function isRoute(array $config)
|
2019-05-01 00:32:33 +02:00
|
|
|
{
|
2019-09-26 21:18:01 +02:00
|
|
|
return
|
|
|
|
// The config array should at least have one entry
|
|
|
|
!empty($config[0]) &&
|
|
|
|
// This entry should be a BaseModule
|
|
|
|
(substr($config[0], 0, strlen('Friendica\Module')) === 'Friendica\Module') &&
|
|
|
|
// Either there is no other argument
|
|
|
|
(empty($config[1]) ||
|
|
|
|
// Or the second argument is an array (HTTP-Methods)
|
|
|
|
is_array($config[1]));
|
2019-05-01 00:32:33 +02:00
|
|
|
}
|
|
|
|
|
2019-09-26 21:18:01 +02:00
|
|
|
/**
|
|
|
|
* The current route collector
|
|
|
|
*
|
|
|
|
* @return RouteCollector|null
|
|
|
|
*/
|
2019-05-01 00:32:33 +02:00
|
|
|
public function getRouteCollector()
|
|
|
|
{
|
|
|
|
return $this->routeCollector;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the relevant module class name for the given page URI or NULL if no route rule matched.
|
|
|
|
*
|
|
|
|
* @param string $cmd The path component of the request URL without the query string
|
2019-09-26 21:18:01 +02:00
|
|
|
*
|
2019-10-11 17:55:02 +02:00
|
|
|
* @return string A Friendica\BaseModule-extending class name if a route rule matched
|
|
|
|
*
|
|
|
|
* @throws HTTPException\InternalServerErrorException
|
|
|
|
* @throws HTTPException\MethodNotAllowedException If a rule matched but the method didn't
|
|
|
|
* @throws HTTPException\NotFoundException If no rule matched
|
2019-05-01 00:32:33 +02:00
|
|
|
*/
|
|
|
|
public function getModuleClass($cmd)
|
|
|
|
{
|
2019-09-26 21:18:01 +02:00
|
|
|
// Add routes from addons
|
|
|
|
Hook::callAll('route_collection', $this->routeCollector);
|
|
|
|
|
2019-05-01 00:32:33 +02:00
|
|
|
$cmd = '/' . ltrim($cmd, '/');
|
|
|
|
|
|
|
|
$dispatcher = new \FastRoute\Dispatcher\GroupCountBased($this->routeCollector->getData());
|
|
|
|
|
|
|
|
$moduleClass = null;
|
|
|
|
|
2019-09-26 21:18:01 +02:00
|
|
|
$routeInfo = $dispatcher->dispatch($this->httpMethod, $cmd);
|
2019-05-01 00:32:33 +02:00
|
|
|
if ($routeInfo[0] === Dispatcher::FOUND) {
|
|
|
|
$moduleClass = $routeInfo[1];
|
2019-10-11 17:55:02 +02:00
|
|
|
} elseif ($routeInfo[0] === Dispatcher::METHOD_NOT_ALLOWED) {
|
|
|
|
throw new HTTPException\MethodNotAllowedException(L10n::t('Method not allowed for this module. Allowed method(s): %s', implode(', ', $routeInfo[1])));
|
|
|
|
} else {
|
|
|
|
throw new HTTPException\NotFoundException(L10n::t('Page not found.'));
|
2019-05-01 00:32:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return $moduleClass;
|
|
|
|
}
|
|
|
|
}
|