AutorÃa | Ultima modificación | Ver Log |
<?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\router;
use invalid_parameter_exception;
use Psr\Http\Message\ServerRequestInterface;
use Slim\Exception\HttpNotFoundException;
use Slim\Interfaces\RouteInterface;
use Slim\Routing\RouteContext;
/**
* Routing attribute.
*
* @package core
* @copyright 2024 Andrew Lyons <andrew@nicols.co.uk>
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
*/
class request_validator implements request_validator_interface {
/**
* Validate the request content.
*
* @param ServerRequestInterface $request
* @return ServerRequestInterface
*/
public function validate_request(
ServerRequestInterface $request,
): ServerRequestInterface {
$moodleroute = $request->getAttribute(route::class);
if (!$moodleroute) {
return $request;
}
// Add a Route middleware to validate the path, and parameters.
$slimroute = RouteContext::fromRequest($request)->getRoute();
// Validate that the path arguments are valid.
// If they are not, then an Exception should be thrown.
$request = $this->validate_path($request, $moodleroute, $slimroute);
// Validate query parameters.
$request = $this->validate_query($request, $moodleroute);
// Validate request headers.
$request = $this->validate_request_header($request, $moodleroute);
// Validate request body parameters.
// Found in POST, PUT, DELETE, etc.
$request = $this->validate_request_body($request, $moodleroute);
return $request;
}
/**
* Validate that the path arguments match those supplied in the route.
*
* @param ServerRequestInterface $request
* @param route $moodleroute
* @param RouteInterface $slimroute The route to validate.
* @return ServerRequestInterface
* @throws \coding_exception
*/
protected function validate_path(
ServerRequestInterface $request,
route $moodleroute,
RouteInterface $slimroute,
): ServerRequestInterface {
$requiredparams = count(array_filter(
$moodleroute->get_path_parameters(),
fn ($pathtype) => $pathtype->is_required($moodleroute),
));
if ($requiredparams > count($slimroute->getArguments())) {
throw new \coding_exception(sprintf(
"Route %s has %d arguments, but %d pathtypes were specified.",
$slimroute->getPattern(),
count($slimroute->getArguments()),
count($moodleroute->get_path_parameters()),
));
}
foreach ($moodleroute->get_path_parameters() as $pathtype) {
try {
$request = $pathtype->validate($request, $slimroute);
} catch (invalid_parameter_exception $e) {
throw new HttpNotFoundException($request, $e->getMessage());
}
}
return $request;
}
/**
* Validate that the query parameters match those supplied in the route.
*
* @param ServerRequestInterface $request
* @param route $moodleroute
* @return ServerRequestInterface
*/
protected function validate_query(
ServerRequestInterface $request,
route $moodleroute,
): ServerRequestInterface {
$requestparams = $request->getQueryParams();
$paramnames = array_map(
fn ($param) => $param->get_name($this),
$moodleroute->get_query_parameters(),
);
// Check for any undeclared parameters.
$unknownparams = array_diff(
array_keys($requestparams),
$paramnames,
);
// Remove these from the URL.
// They will still be accessible via optional_param.
$request = $request->withQueryParams(
array_diff_key(
$requestparams,
array_flip($unknownparams),
),
);
foreach ($moodleroute->get_query_parameters() as $queryparam) {
$request = $queryparam->validate($request, $request->getQueryParams());
}
return $request;
}
/**
* Validate that the request headers match the schema.
*
* @param ServerRequestInterface $request
* @param route $moodleroute
* @return ServerRequestInterface
*/
protected function validate_request_header(
ServerRequestInterface $request,
route $moodleroute,
): ServerRequestInterface {
$headerparams = $moodleroute->get_header_parameters();
foreach ($headerparams as $headerparam) {
$request = $headerparam->validate($request);
}
return $request;
}
/**
* Validate that the request body matches the schema.
*
* @param ServerRequestInterface $request
* @param route $moodleroute
* @return ServerRequestInterface
*/
protected function validate_request_body(
ServerRequestInterface $request,
route $moodleroute,
): ServerRequestInterface {
if ($moodleroute->get_request_body() === null) {
// Clear the parsed body if there should not be one.
return $request->withParsedBody([]);
}
$bodyconfig = $moodleroute->get_request_body()->get_body_for_request($request);
$bodyschema = $bodyconfig->get_schema();
$parsedbody = $request->getParsedBody();
if (empty($parsedbody)) {
if ($moodleroute->get_request_body()->is_required()) {
throw new invalid_parameter_exception('Missing request body.');
}
// No body to validate.
return $request;
}
return $request->withParsedBody(
$bodyschema->validate_data($request->getParsedBody()),
);
}
}