SNode.C
Loading...
Searching...
No Matches
RouterDispatcher.cpp
Go to the documentation of this file.
1/*
2 * SNode.C - A Slim Toolkit for Network Communication
3 * Copyright (C) Volker Christian <me@vchrist.at>
4 * 2020, 2021, 2022, 2023, 2024, 2025
5 *
6 * This program is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU Lesser General Public License as published
8 * by the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public License
17 * along with this program. If not, see <http://www.gnu.org/licenses/>.
18 */
19
20/*
21 * MIT License
22 *
23 * Permission is hereby granted, free of charge, to any person obtaining a copy
24 * of this software and associated documentation files (the "Software"), to deal
25 * in the Software without restriction, including without limitation the rights
26 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
27 * copies of the Software, and to permit persons to whom the Software is
28 * furnished to do so, subject to the following conditions:
29 *
30 * The above copyright notice and this permission notice shall be included in
31 * all copies or substantial portions of the Software.
32 *
33 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
34 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
35 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
36 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
37 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
38 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
39 * THE SOFTWARE.
40 */
41
42#include "express/dispatcher/RouterDispatcher.h"
43
44#include "core/socket/stream/SocketConnection.h"
45#include "express/Controller.h"
46#include "express/Request.h"
47#include "express/Response.h"
48#include "express/Route.h"
49#include "express/dispatcher/regex_utils.h"
50#include "web/http/server/SocketContext.h"
51
52#ifndef DOXYGEN_SHOULD_SKIP_THIS
53
54#include "log/Logger.h"
55
56#include <cstddef>
57#include <string_view>
58#include <tuple>
59#include <unordered_map>
60
61#endif /* DOXYGEN_SHOULD_SKIP_THIS */
62
63namespace express::dispatcher {
64
66 return routes;
67 }
68
69 bool RouterDispatcher::setStrictRouting(bool strictRouting) {
70 const bool oldStrictRouting = this->strictRouting;
71
72 this->strictRouting = strictRouting;
73
74 return oldStrictRouting;
75 }
76
78 return strictRouting;
79 }
80
81 bool RouterDispatcher::setCaseInsensitiveRouting(bool caseInsensitiveRouting) {
82 const bool oldCaseInsensitiveRouting = this->caseInsensitiveRouting;
83
84 this->caseInsensitiveRouting = caseInsensitiveRouting;
85
86 return oldCaseInsensitiveRouting;
87 }
88
92
93 bool
94 RouterDispatcher::dispatch(express::Controller& controller, const std::string& parentMountPath, const express::MountPoint& mountPoint) {
95 bool dispatched = false;
96
97 if (controller.getRequest()->method == mountPoint.method || mountPoint.method == "use" || mountPoint.method == "all") {
98 const std::string absoluteMountPath = parentMountPath + mountPoint.relativeMountPath;
99
100 // Split mount & request into path + query
101 std::string_view mountPath;
102 std::string_view mountQueryString;
103 splitPathAndQuery(absoluteMountPath, mountPath, mountQueryString);
104 const std::unordered_map<std::string, std::string> requiredQueryPairs = parseQuery(mountQueryString);
105
106 std::string_view requestPath;
107 std::string_view requestQueryString;
108 splitPathAndQuery(controller.getRequest()->originalUrl, requestPath, requestQueryString);
109 const std::unordered_map<std::string, std::string> requestQueryPairs = parseQuery(requestQueryString);
110
111 // Normalize single trailing slash if not strict
112 if (!controller.getStrictRouting()) {
113 mountPath = trimOneTrailingSlash(mountPath);
114 requestPath = trimOneTrailingSlash(requestPath);
115 }
116 if (mountPath.empty()) {
117 mountPath = "/";
118 }
119
120 // Router is **prefix** with boundary (like app.use)
121 bool pathMatches = false;
122 if (absoluteMountPath.find(':') != std::string::npos) {
123 if (regex.mark_count() == 0) {
124 LOG(TRACE) << "RouterDispatcher: precompiled regex";
125 std::tie(regex, names) = compileParamRegex(mountPath,
126 /*isPrefix*/ true,
127 controller.getStrictRouting(),
129 } else {
130 LOG(TRACE) << "RouterDispatcher: using precompiled regex";
131 }
132
133 pathMatches = matchAndFillParams(regex, names, requestPath, *controller.getRequest());
134 } else {
135 pathMatches = boundaryPrefix(requestPath, mountPath, controller.getCaseInsensitiveRouting());
136 }
137 const bool queryMatches = querySupersetMatches(requestQueryPairs, requiredQueryPairs);
138
139 const bool requestMatched = (pathMatches && queryMatches);
140
142 << " HTTP Express: router -> " << (requestMatched ? "MATCH" : "NO MATCH");
143 LOG(TRACE) << " RequestMethod: " << controller.getRequest()->method;
144 LOG(TRACE) << " RequestUrl: " << controller.getRequest()->url;
145 LOG(TRACE) << " RequestPath: " << controller.getRequest()->path;
146 LOG(TRACE) << " Mountpoint Method: " << mountPoint.method;
147 LOG(TRACE) << " Mountpoint RelativePath: " << mountPoint.relativeMountPath;
148 LOG(TRACE) << " Mountpoint AbsolutePath: " << absoluteMountPath;
149 LOG(TRACE) << " StrictRouting: " << controller.getStrictRouting();
150 LOG(TRACE) << " CaseInsensitiveRouting: " << controller.getCaseInsensitiveRouting();
151
152 if (requestMatched) {
153 // Compute remainder and temporarily **rewrite req.path** for subtree
154 std::size_t consumedLength = 0;
155 if (absoluteMountPath.find(':') != std::string::npos) {
156 // Run (or reuse) the param regex match and take its full-match length
157 std::cmatch regexMatches;
158 auto [regex, names] = compileParamRegex(
159 mountPath, /*isPrefix*/ true, controller.getStrictRouting(), controller.getCaseInsensitiveRouting());
160 if (!std::regex_search(requestPath.begin(), requestPath.end(), regexMatches, regex)) {
161 return false; // should not happen because pathMatches was true, but be safe
162 }
163 consumedLength = static_cast<std::size_t>(regexMatches.length(0)); // <-- ACTUAL matched prefix
164 } else {
165 // Literal boundary prefix: consume exactly the base length
166 consumedLength = mountPath.size();
167 }
168
169 // Now compute remainder using 'consumedLength'
170 std::string_view remainderPath{};
171 if (requestPath.size() > consumedLength) {
172 remainderPath = requestPath.substr(consumedLength);
173 if (!remainderPath.empty() && remainderPath.front() == '/') {
174 remainderPath.remove_prefix(1);
175 }
176 }
177
178 auto& req = *controller.getRequest();
179 const std::string previousPathBackup = req.path;
180 req.path = remainderPath.empty() ? "/" : ("/" + std::string(remainderPath));
181
182 req.queries.insert(requestQueryPairs.begin(), requestQueryPairs.end());
183
184 for (Route& route : routes) {
185 const bool oldStrictRouting = controller.setStrictRouting(strictRouting);
186
187 dispatched = route.dispatch(controller, absoluteMountPath);
188
189 controller.setStrictRouting(oldStrictRouting);
190
191 if (dispatched) {
192 LOG(TRACE) << "Express: R - Dispatched";
193
194 break;
195 }
196 if (controller.nextRouterCalled()) {
197 LOG(TRACE) << "Express: R - NextRouter called - breaking dispatching";
198
199 break;
200 }
201 }
202
203 req.path = previousPathBackup;
204 }
205 }
206
207 return dispatched;
208 }
209
210 std::list<std::string> RouterDispatcher::getRoutes(const std::string& parentMountPath, const MountPoint& mountPoint) const {
211 return getRoutes(parentMountPath, mountPoint, strictRouting);
212 }
213
214 std::list<std::string>
215 RouterDispatcher::getRoutes(const std::string& parentMountPath, const MountPoint& mountPoint, bool strictRouting) const {
216 std::list<std::string> collectedRoutes;
217
218 for (const Route& route : routes) {
219 collectedRoutes.splice(
220 collectedRoutes.end(),
221 route.getRoute(parentMountPath + "$" + mountPoint.relativeMountPath + "$", this->strictRouting ? true : strictRouting));
222 }
223
224 return collectedRoutes;
225 }
226
227} // namespace express::dispatcher
const std::string & getConnectionName() const
SocketConnection * getSocketConnection() const
const std::shared_ptr< Request > & getRequest()
bool setStrictRouting(bool strictRouting)
bool getStrictRouting() const
bool getCaseInsensitiveRouting() const
const std::shared_ptr< Response > & getResponse()
std::string originalUrl
Definition Request.h:76
std::string url
Definition Request.h:96
web::http::CiStringMap< std::string > queries
Definition Request.h:101
std::string method
Definition Request.h:95
std::string path
Definition Request.h:77
web::http::server::SocketContext * getSocketContext() const
Definition Response.cpp:68
bool dispatch(Controller &controller, const std::string &parentMountPath)
Definition Route.cpp:84
std::list< std::string > getRoute(const std::string &parentMountPath, bool strictRouting) const
Definition Route.cpp:114
bool setCaseInsensitiveRouting(bool caseInsensitiveRouting)
std::list< std::string > getRoutes(const std::string &parentMountPath, const MountPoint &mountPoint, bool strictRouting) const override
bool dispatch(express::Controller &controller, const std::string &parentMountPath, const express::MountPoint &mountPoint) override
std::list< std::string > getRoutes(const std::string &parentMountPath, const MountPoint &mountPoint) const
bool setStrictRouting(bool strictRouting)
std::list< express::Route > & getRoutes()
std::list< express::Route > routes
bool matchAndFillParams(const std::regex &rx, const std::vector< std::string > &names, std::string_view reqPath, RequestLike &req)
bool querySupersetMatches(const std::unordered_map< std::string, std::string > &rq, const std::unordered_map< std::string, std::string > &need)
void splitPathAndQuery(std::string_view url, std::string_view &path, std::string_view &query)
bool boundaryPrefix(std::string_view path, std::string_view base, bool caseInsensitive)
std::pair< std::regex, std::vector< std::string > > compileParamRegex(std::string_view mountPath, bool isPrefix, bool strictRouting, bool caseInsensitive)
std::unordered_map< std::string, std::string > parseQuery(std::string_view qs)
std::string_view trimOneTrailingSlash(std::string_view s)
std::string method
Definition MountPoint.h:56
std::string relativeMountPath
Definition MountPoint.h:57