SNode.C
Loading...
Searching...
No Matches
MiddlewareDispatcher.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/MiddlewareDispatcher.h"
43
44#include "core/socket/stream/SocketConnection.h"
45#include "express/Next.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 <algorithm>
57#include <cstddef>
58#include <list>
59#include <regex>
60#include <string_view>
61#include <tuple>
62#include <unordered_map>
63
64#endif /* DOXYGEN_SHOULD_SKIP_THIS */
65
66namespace express::dispatcher {
67
69 const std::function<void(const std::shared_ptr<Request>&, const std::shared_ptr<Response>&, express::Next&)>& lambda)
70 : lambda(lambda) {
71 }
72
73 bool MiddlewareDispatcher::dispatch(express::Controller& controller, const std::string& parentMountPath, const MountPoint& mountPoint) {
74 bool requestMatched = false;
75 const std::string absoluteMountPath = parentMountPath + mountPoint.relativeMountPath;
76
77 if ((controller.getFlags() & Controller::NEXT) == 0) {
78 // Split mount & request into path + query
79 std::string_view mountPath;
80 std::string_view mountQueryString;
81 splitPathAndQuery(absoluteMountPath, mountPath, mountQueryString);
82 auto requiredQueryPairs = parseQuery(mountQueryString);
83
84 std::string_view requestPath;
85 std::string_view requestQueryString;
86 splitPathAndQuery(controller.getRequest()->originalUrl, requestPath, requestQueryString);
87 auto requestQueryPairs = parseQuery(requestQueryString);
88
89 // Normalize single trailing slash if not strict
90 if (!controller.getStrictRouting()) {
91 mountPath = trimOneTrailingSlash(mountPath);
92 requestPath = trimOneTrailingSlash(requestPath);
93 }
94 if (mountPath.empty()) {
95 mountPath = "/";
96 }
97
98 // Middleware is **prefix** with boundary (Express: app.use)
99 bool pathMatches = false;
100 std::size_t consumedLength = 0;
101 if (absoluteMountPath.find(':') != std::string::npos) {
102 // Param mount: compile once, match once, fill params, and record matched prefix length
103 if (regex.mark_count() == 0) {
104 if (regex.mark_count() == 0) {
105 LOG(TRACE) << "MiddlewarDispatchere: precompiled regex";
106 std::tie(regex, names) = compileParamRegex(mountPath,
107 /*isPrefix*/ true,
108 controller.getStrictRouting(),
110 } else {
111 LOG(TRACE) << "MiddlewareDispatcher: using precompiled regex";
112 }
113 }
114 std::cmatch regexMatches;
115 pathMatches = std::regex_search(requestPath.begin(), requestPath.end(), regexMatches, regex);
116 if (pathMatches) {
117 // regexMatches[0] = full matched prefix starting at 0 (anchored by compile_param_regex)
118 consumedLength = static_cast<std::size_t>(regexMatches.length(0));
119
120 // fill named params
121 const std::size_t groups = !regexMatches.empty() ? (regexMatches.size() - 1) : 0;
122 const std::size_t n = std::min(names.size(), groups);
123 for (std::size_t i = 0; i < n; ++i) {
124 controller.getRequest()->params[names[i]] = regexMatches[i + 1].str();
125 }
126 }
127 } else {
128 // Literal boundary prefix
129 pathMatches = boundaryPrefix(requestPath, mountPath, controller.getCaseInsensitiveRouting());
130 if (pathMatches) {
131 consumedLength = mountPath.size(); // IMPORTANT: literal consumes exactly its base length
132 }
133 }
134
135 const bool queryMatches = querySupersetMatches(requestQueryPairs, requiredQueryPairs);
136 requestMatched = (pathMatches && queryMatches);
137
139 << " HTTP Express: middleware -> " << (requestMatched ? "MATCH" : "NO MATCH");
140 LOG(TRACE) << " RequestMethod: " << controller.getRequest()->method;
141 LOG(TRACE) << " RequestUrl: " << controller.getRequest()->url;
142 LOG(TRACE) << " RequestPath: " << controller.getRequest()->path;
143 LOG(TRACE) << " AbsoluteMountPath: " << absoluteMountPath;
144 LOG(TRACE) << " StrictRouting: " << controller.getStrictRouting();
145 LOG(TRACE) << " CaseInsensitiveRouting: " << controller.getCaseInsensitiveRouting();
146
147 if (requestMatched) {
148 // Compute remainder and temporarily **rewrite req.path**
149 std::string_view remainderPath{};
150 if (requestPath.size() > consumedLength) {
151 remainderPath = requestPath.substr(consumedLength);
152 if (!remainderPath.empty() && remainderPath.front() == '/') {
153 remainderPath.remove_prefix(1);
154 }
155 }
156 auto& req = *controller.getRequest();
157 const std::string previousPathBackup = req.path;
158 req.path = remainderPath.empty() ? "/" : ("/" + std::string(remainderPath));
159
160 req.queries.insert(requestQueryPairs.begin(), requestQueryPairs.end());
161
162 if (hasResult(absoluteMountPath)) {
163 setParams(absoluteMountPath, *controller.getRequest());
164 }
165
166 Next next(controller);
167 lambda(controller.getRequest(), controller.getResponse(), next);
168
169 // If next() was called synchronously continue current route-tree traversal
170 if ((next.controller.getFlags() & express::Controller::NEXT) != 0) {
171 LOG(TRACE) << "Express: M - Next called - set to NO MATCH";
172 requestMatched = false;
173 controller = next.controller;
174 }
175
176 // Restore
177 req.path = previousPathBackup;
178 }
179 } else {
181 << " HTTP Express: middleware -> next(...) called";
182 LOG(TRACE) << " RequestMethod: " << controller.getRequest()->method;
183 LOG(TRACE) << " RequestUrl: " << controller.getRequest()->url;
184 LOG(TRACE) << " RequestPath: " << controller.getRequest()->path;
185 LOG(TRACE) << " AbsoluteMountPath: " << absoluteMountPath;
186 }
187
188 return requestMatched;
189 }
190
191 std::list<std::string>
192 MiddlewareDispatcher::getRoutes(const std::string& parentMountPath, const MountPoint& mountPoint, bool strictRouting) const {
193 std::list<std::string> routes{"M " + parentMountPath + mountPoint.relativeMountPath + (!strictRouting ? "*" : "")};
194 routes.push_back(" " + mountPoint.method + " " + mountPoint.relativeMountPath);
195
196 if (nextRoute) {
197 routes.splice(routes.end(), nextRoute->getRoute(parentMountPath, strictRouting));
198 }
199
200 return routes;
201 }
202
203} // namespace express::dispatcher
const std::string & getConnectionName() const
SocketConnection * getSocketConnection() const
const std::shared_ptr< Request > & getRequest()
bool getStrictRouting() const
bool getCaseInsensitiveRouting() const
Controller & operator=(const Controller &controller) noexcept
const std::shared_ptr< Response > & getResponse()
std::shared_ptr< Route > nextRoute
Definition Dispatcher.h:80
Next(Controller &controller)
Definition Next.cpp:52
Controller controller
Definition Next.h:66
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
std::list< std::string > getRoute(const std::string &parentMountPath, bool strictRouting) const
Definition Route.cpp:114
bool dispatch(express::Controller &controller, const std::string &parentMountPath, const express::MountPoint &mountPoint) override
MiddlewareDispatcher(const std::function< void(const std::shared_ptr< Request > &, const std::shared_ptr< Response > &, express::Next &)> &lambda)
std::list< std::string > getRoutes(const std::string &parentMountPath, const MountPoint &mountPoint, bool strictRouting) const override
const std::function< void(const std::shared_ptr< Request > &, const std::shared_ptr< Response > &, express::Next &)> lambda
bool querySupersetMatches(const std::unordered_map< std::string, std::string > &rq, const std::unordered_map< std::string, std::string > &need)
void setParams(const std::string &cpath, Request &req)
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)
bool hasResult(const std::string &cpath)
std::string_view trimOneTrailingSlash(std::string_view s)
std::string method
Definition MountPoint.h:56
std::string relativeMountPath
Definition MountPoint.h:57