Proyectos de Subversion Moodle

Rev

| Ultima modificación | Ver Log |

Rev Autor Línea Nro. Línea
1441 ariadna 1
<?php
2
// This file is part of Moodle - http://moodle.org/
3
//
4
// Moodle is free software: you can redistribute it and/or modify
5
// it under the terms of the GNU General Public License as published by
6
// the Free Software Foundation, either version 3 of the License, or
7
// (at your option) any later version.
8
//
9
// Moodle is distributed in the hope that it will be useful,
10
// but WITHOUT ANY WARRANTY; without even the implied warranty of
11
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12
// GNU General Public License for more details.
13
//
14
// You should have received a copy of the GNU General Public License
15
// along with Moodle.  If not, see <http://www.gnu.org/licenses/>.
16
 
17
namespace core\router;
18
 
19
use invalid_parameter_exception;
20
use Psr\Http\Message\ServerRequestInterface;
21
use Slim\Exception\HttpNotFoundException;
22
use Slim\Interfaces\RouteInterface;
23
use Slim\Routing\RouteContext;
24
 
25
/**
26
 * Routing attribute.
27
 *
28
 * @package    core
29
 * @copyright  2024 Andrew Lyons <andrew@nicols.co.uk>
30
 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
31
 */
32
class request_validator implements request_validator_interface {
33
    /**
34
     * Validate the request content.
35
     *
36
     * @param ServerRequestInterface $request
37
     * @return ServerRequestInterface
38
     */
39
    public function validate_request(
40
        ServerRequestInterface $request,
41
    ): ServerRequestInterface {
42
        $moodleroute = $request->getAttribute(route::class);
43
        if (!$moodleroute) {
44
            return $request;
45
        }
46
 
47
        // Add a Route middleware to validate the path, and parameters.
48
        $slimroute = RouteContext::fromRequest($request)->getRoute();
49
 
50
        // Validate that the path arguments are valid.
51
        // If they are not, then an Exception should be thrown.
52
        $request = $this->validate_path($request, $moodleroute, $slimroute);
53
 
54
        // Validate query parameters.
55
        $request = $this->validate_query($request, $moodleroute);
56
 
57
        // Validate request headers.
58
        $request = $this->validate_request_header($request, $moodleroute);
59
 
60
        // Validate request body parameters.
61
        // Found in POST, PUT, DELETE, etc.
62
        $request = $this->validate_request_body($request, $moodleroute);
63
 
64
        return $request;
65
    }
66
 
67
    /**
68
     * Validate that the path arguments match those supplied in the route.
69
     *
70
     * @param ServerRequestInterface $request
71
     * @param route $moodleroute
72
     * @param RouteInterface $slimroute The route to validate.
73
     * @return ServerRequestInterface
74
     * @throws \coding_exception
75
     */
76
    protected function validate_path(
77
        ServerRequestInterface $request,
78
        route $moodleroute,
79
        RouteInterface $slimroute,
80
    ): ServerRequestInterface {
81
        $requiredparams = count(array_filter(
82
            $moodleroute->get_path_parameters(),
83
            fn ($pathtype) => $pathtype->is_required($moodleroute),
84
        ));
85
        if ($requiredparams > count($slimroute->getArguments())) {
86
            throw new \coding_exception(sprintf(
87
                "Route %s has %d arguments, but %d pathtypes were specified.",
88
                $slimroute->getPattern(),
89
                count($slimroute->getArguments()),
90
                count($moodleroute->get_path_parameters()),
91
            ));
92
        }
93
 
94
        foreach ($moodleroute->get_path_parameters() as $pathtype) {
95
            try {
96
                $request = $pathtype->validate($request, $slimroute);
97
            } catch (invalid_parameter_exception $e) {
98
                throw new HttpNotFoundException($request, $e->getMessage());
99
            }
100
        }
101
 
102
        return $request;
103
    }
104
 
105
    /**
106
     * Validate that the query parameters match those supplied in the route.
107
     *
108
     * @param ServerRequestInterface $request
109
     * @param route $moodleroute
110
     * @return ServerRequestInterface
111
     */
112
    protected function validate_query(
113
        ServerRequestInterface $request,
114
        route $moodleroute,
115
    ): ServerRequestInterface {
116
        $requestparams = $request->getQueryParams();
117
        $paramnames = array_map(
118
            fn ($param) => $param->get_name($this),
119
            $moodleroute->get_query_parameters(),
120
        );
121
 
122
        // Check for any undeclared parameters.
123
        $unknownparams = array_diff(
124
            array_keys($requestparams),
125
            $paramnames,
126
        );
127
 
128
        // Remove these from the URL.
129
        // They will still be accessible via optional_param.
130
        $request = $request->withQueryParams(
131
            array_diff_key(
132
                $requestparams,
133
                array_flip($unknownparams),
134
            ),
135
        );
136
 
137
        foreach ($moodleroute->get_query_parameters() as $queryparam) {
138
            $request = $queryparam->validate($request, $request->getQueryParams());
139
        }
140
 
141
        return $request;
142
    }
143
 
144
    /**
145
     * Validate that the request headers match the schema.
146
     *
147
     * @param ServerRequestInterface $request
148
     * @param route $moodleroute
149
     * @return ServerRequestInterface
150
     */
151
    protected function validate_request_header(
152
        ServerRequestInterface $request,
153
        route $moodleroute,
154
    ): ServerRequestInterface {
155
        $headerparams = $moodleroute->get_header_parameters();
156
 
157
        foreach ($headerparams as $headerparam) {
158
            $request = $headerparam->validate($request);
159
        }
160
 
161
        return $request;
162
    }
163
 
164
    /**
165
     * Validate that the request body matches the schema.
166
     *
167
     * @param ServerRequestInterface $request
168
     * @param route $moodleroute
169
     * @return ServerRequestInterface
170
     */
171
    protected function validate_request_body(
172
        ServerRequestInterface $request,
173
        route $moodleroute,
174
    ): ServerRequestInterface {
175
        if ($moodleroute->get_request_body() === null) {
176
            // Clear the parsed body if there should not be one.
177
            return $request->withParsedBody([]);
178
        }
179
 
180
        $bodyconfig = $moodleroute->get_request_body()->get_body_for_request($request);
181
        $bodyschema = $bodyconfig->get_schema();
182
 
183
        $parsedbody = $request->getParsedBody();
184
        if (empty($parsedbody)) {
185
            if ($moodleroute->get_request_body()->is_required()) {
186
                throw new invalid_parameter_exception('Missing request body.');
187
            }
188
 
189
            // No body to validate.
190
            return $request;
191
        }
192
 
193
        return $request->withParsedBody(
194
            $bodyschema->validate_data($request->getParsedBody()),
195
        );
196
    }
197
}