mirror of
https://github.com/moodle/moodle.git
synced 2025-04-22 08:55:15 +02:00
263 lines
8.8 KiB
PHP
263 lines
8.8 KiB
PHP
<?php
|
|
// This file is part of Moodle - http://moodle.org/
|
|
//
|
|
// Moodle is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// Moodle is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with Moodle. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
namespace core;
|
|
|
|
use core\output\routed_error_handler;
|
|
use core\router\middleware\cors_middleware;
|
|
use core\router\middleware\error_handling_middleware;
|
|
use core\router\middleware\moodle_bootstrap_middleware;
|
|
use core\router\middleware\moodle_route_attribute_middleware;
|
|
use core\router\middleware\uri_normalisation_middleware;
|
|
use core\router\middleware\validation_middleware;
|
|
use core\router\request_validator_interface;
|
|
use core\router\response_handler;
|
|
use core\router\response_validator_interface;
|
|
use core\router\route_loader_interface;
|
|
use Psr\Http\Message\ResponseFactoryInterface;
|
|
use Psr\Http\Message\ResponseInterface;
|
|
use Psr\Http\Message\ServerRequestInterface;
|
|
use Slim\App;
|
|
use Slim\Interfaces\RouteGroupInterface;
|
|
|
|
/**
|
|
* Moodle Router.
|
|
*
|
|
* This class represents the Moodle Router, which handles all aspects of Routing in Moodle.
|
|
*
|
|
* It should not normally be accessed or used outside of its own unit tests, the route_testcase, and the `r.php` handler.
|
|
*
|
|
* @package core
|
|
* @copyright Andrew Lyons <andrew@nicols.co.uk>
|
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
|
*/
|
|
class router {
|
|
/** @var string The base path to use for all requests */
|
|
public readonly string $basepath;
|
|
|
|
/** @var App The SlimPHP App */
|
|
protected readonly App $app;
|
|
|
|
/**
|
|
* Create a new Router.
|
|
*
|
|
* @param response_handler $responsehandler
|
|
* @param route_loader_interface $routeloader
|
|
* @param request_validator_interface $requestvalidator
|
|
* @param response_validator_interface $responsevalidator
|
|
* @param null|string $basepath
|
|
*/
|
|
public function __construct(
|
|
/** @var response_handler */
|
|
protected response_handler $responsehandler,
|
|
|
|
/** @var route_loader_interface The router loader to use */
|
|
protected readonly route_loader_interface $routeloader,
|
|
|
|
/** @var request_validator_interface */
|
|
protected request_validator_interface $requestvalidator,
|
|
|
|
/** @var response_validator_interface */
|
|
protected response_validator_interface $responsevalidator,
|
|
|
|
?string $basepath = null,
|
|
) {
|
|
if ($basepath === null) {
|
|
$basepath = $this->guess_basepath();
|
|
}
|
|
$this->basepath = $basepath;
|
|
}
|
|
|
|
/**
|
|
* Guess the basepath for the Router.
|
|
*
|
|
* @return string
|
|
*/
|
|
protected function guess_basepath(): string {
|
|
global $CFG;
|
|
|
|
// Moodle is not guaranteed to exist at the domain root.
|
|
// Strip out the current script.
|
|
$scriptroot = parse_url($CFG->wwwroot, PHP_URL_PATH) ?? '';
|
|
$scriptfile = str_replace(
|
|
realpath($CFG->dirroot),
|
|
'',
|
|
realpath($_SERVER['SCRIPT_FILENAME']),
|
|
);
|
|
// Replace occurrences of backslashes with forward slashes, especially on Windows.
|
|
$scriptfile = str_replace('\\', '/', $scriptfile);
|
|
$relativeroot = sprintf(
|
|
'%s%s',
|
|
$scriptroot,
|
|
$scriptfile,
|
|
);
|
|
|
|
// The server is not configured to rewrite unknown requests to automatically use the router.
|
|
if ($_SERVER && array_key_exists('REQUEST_URI', $_SERVER)) {
|
|
if (str_starts_with($_SERVER['REQUEST_URI'], $relativeroot)) {
|
|
$scriptroot .= '/r.php';
|
|
}
|
|
}
|
|
|
|
return $scriptroot;
|
|
}
|
|
|
|
/**
|
|
* Get the configured SlimPHP Application.
|
|
*
|
|
* @return App
|
|
*/
|
|
public function get_app(): App {
|
|
if (!isset($this->app)) {
|
|
$this->create_app($this->basepath);
|
|
}
|
|
|
|
return $this->app;
|
|
}
|
|
|
|
/**
|
|
* Get the Response Factory for the Router.
|
|
*
|
|
* @return ResponseFactoryInterface
|
|
*/
|
|
public function get_response_factory(): ResponseFactoryInterface {
|
|
return $this->get_app()->getResponseFactory();
|
|
}
|
|
|
|
/**
|
|
* Create the configured SlimPHP Application.
|
|
*
|
|
* @param string $basepath The base path of the Moodle instance
|
|
*/
|
|
protected function create_app(
|
|
string $basepath = '',
|
|
): void {
|
|
global $CFG;
|
|
|
|
// PHP Does not support autoloading functions.
|
|
require_once("{$CFG->libdir}/nikic/fast-route/src/functions.php");
|
|
|
|
// Create an App using the DI Bridge.
|
|
$this->app = router\bridge::create();
|
|
|
|
// Add Middleware to the App.
|
|
// Note: App Middleware is called before any Group or Route middleware.
|
|
$this->add_middleware();
|
|
$this->configure_caching();
|
|
$this->configure_routes();
|
|
|
|
// Configure the basepath for Moodle.
|
|
$this->app->setBasePath($basepath);
|
|
}
|
|
|
|
/**
|
|
* Add Middleware to the App.
|
|
*/
|
|
protected function add_middleware(): void {
|
|
// Middleware is added like an onion.
|
|
// For a Response, the outer-most middleware is executed first, and the inner-most middleware is executed last.
|
|
// For a Request, the inner-most middleware is executed first, and the outer-most middleware is executed last.
|
|
|
|
// Add the body parsing middleware from Slim.
|
|
// See https://www.slimframework.com/docs/v4/middleware/body-parsing.html for further information.
|
|
$this->app->addBodyParsingMiddleware();
|
|
|
|
// Add Middleware to Bootstrap Moodle from a request.
|
|
$this->app->add(di::get(moodle_bootstrap_middleware::class));
|
|
|
|
// Add the Moodle route attribute to the request.
|
|
// This must be processed after the Routing Middleware has been processed on the request.
|
|
$this->app->add(di::get(moodle_route_attribute_middleware::class));
|
|
|
|
// Add the Routing Middleware as one of the outer-most middleware.
|
|
// This allows the Route to be accessed before it is handled.
|
|
// See https://www.slimframework.com/docs/v4/cookbook/retrieving-current-route.html for further information.
|
|
$this->app->addRoutingMiddleware();
|
|
|
|
// Add request normalisation middleware to standardise the URI.
|
|
// This must be done before the Routing Middleware to ensure that the route is matched correctly.
|
|
$this->app->add(di::get(uri_normalisation_middleware::class));
|
|
|
|
// Add the Error Handling Middleware and configure it to show Moodle Errors for HTML pages.
|
|
$errormiddleware = $this->app->addErrorMiddleware(true, true, true);
|
|
$errorhandler = $errormiddleware->getDefaultErrorHandler();
|
|
$errorhandler->registerErrorRenderer('text/html', routed_error_handler::class);
|
|
}
|
|
|
|
/**
|
|
* Configure the API routes.
|
|
*/
|
|
protected function configure_routes(): void {
|
|
$routegroups = $this->routeloader->configure_routes($this->app);
|
|
foreach ($routegroups as $name => $collection) {
|
|
match ($name) {
|
|
route_loader_interface::ROUTE_GROUP_API => $this->configure_api_route($collection),
|
|
default => null,
|
|
};
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Configure the API Route Middleware.
|
|
*
|
|
* @param RouteGroupInterface $group
|
|
*/
|
|
protected function configure_api_route(RouteGroupInterface $group): void {
|
|
$group
|
|
->add(di::get(error_handling_middleware::class))
|
|
// Add a Middleware to set the CORS headers for all REST Responses.
|
|
->add(di::get(cors_middleware::class))
|
|
->add(di::get(validation_middleware::class));
|
|
}
|
|
|
|
/**
|
|
* Configure caching for the routes.
|
|
*/
|
|
protected function configure_caching(): void {
|
|
global $CFG;
|
|
|
|
// Note: Slim uses a file cache and is not compatible with MUC.
|
|
$this->app->getRouteCollector()->setCacheFile(
|
|
sprintf(
|
|
"%s/routes.%s.cache",
|
|
$CFG->cachedir,
|
|
sha1($this->basepath),
|
|
),
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Handle the specified Request.
|
|
*
|
|
* @param ServerRequestInterface $request
|
|
* @return ResponseInterface
|
|
*/
|
|
public function handle_request(
|
|
ServerRequestInterface $request,
|
|
): ResponseInterface {
|
|
return $this->get_app()->handle($request);
|
|
}
|
|
|
|
/**
|
|
* Serve the current request using global variables.
|
|
*
|
|
* @codeCoverageIgnore
|
|
*/
|
|
public function serve(): void {
|
|
$this->get_app()->run();
|
|
}
|
|
}
|