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, 2026
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 <list>
57#include <unordered_map>
58
59#endif /* DOXYGEN_SHOULD_SKIP_THIS */
60
61namespace express::dispatcher {
62
64 const std::function<void(const std::shared_ptr<Request>&, const std::shared_ptr<Response>&, express::Next&)>& lambda)
65 : lambda(lambda) {
66 }
67
68 bool MiddlewareDispatcher::dispatch(express::Controller& controller, //
69 const MountPoint& mountPoint,
70 bool strictRouting,
71 bool caseInsensitiveRouting,
72 bool mergeParams) {
73 LOG(TRACE) << "======================= MIDDLEWARE DISPATCH =======================";
75 LOG(TRACE) << " Request Method: " << controller.getRequest()->method;
76 LOG(TRACE) << " Request Url: " << controller.getRequest()->url;
77 LOG(TRACE) << " Request Path: " << controller.getRequest()->path;
78 LOG(TRACE) << " Mountpoint Method: " << mountPoint.method;
79 LOG(TRACE) << " Mountpoint Path: " << mountPoint.relativeMountPath;
80 LOG(TRACE) << " StrictRouting: " << strictRouting;
81 LOG(TRACE) << " CaseInsensitiveRouting: " << caseInsensitiveRouting;
82 LOG(TRACE) << " MergeParams: " << mergeParams;
83
84 bool dispatched = false;
85
86 const bool methodMatchesResult = methodMatches(controller.getRequest()->method, mountPoint.method);
87
88 if (methodMatchesResult && ((controller.getFlags() & Controller::NEXT) == 0)) {
89 const MountMatchResult match =
90 matchMountPoint(controller, mountPoint.relativeMountPath, mountPoint, regex, names, strictRouting, caseInsensitiveRouting);
91
92 if (match.requestMatched) {
93 LOG(TRACE) << "----------------------- MIDDLEWARE MATCH -----------------------";
94
95 dispatched = true;
96
97 if (!match.decodeError) {
98 express::Request& request = *controller.getRequest();
99 request.queries.insert(match.requestQueryPairs.begin(), match.requestQueryPairs.end());
100
101 // Express-style mount path stripping is only applied for use()
102 const ScopedPathStrip pathStrip(request, match.isPrefix, match.consumedLength);
103 const ScopedParams scopedParams(request, match.params, mergeParams);
104
105 Next next(controller);
106 lambda(controller.getRequest(), controller.getResponse(), next);
107
108 // If next() was called synchronously continue current route-tree traversal
109 if ((next.controller.getFlags() & express::Controller::NEXT) != 0) {
110 LOG(TRACE) << "Express: M - Next called - set to NO MATCH";
111 dispatched = false;
112 controller = next.controller;
113 }
114 } else {
115 controller.getResponse()->sendStatus(400);
116 }
117
118 } else {
119 LOG(TRACE) << "----------------------- MIDDLEWARE NOMATCH -----------------------";
120 }
121 } else {
122 LOG(TRACE) << "----------------------- MIDDLEWARE NOMATCH -----------------------";
123 }
124
125 return dispatched;
126 }
127
128 std::list<std::string>
129 MiddlewareDispatcher::getRoutes(const std::string& parentMountPath, const MountPoint& mountPoint, bool strictRouting) const {
130 std::list<std::string> routes{"M " + joinMountPath(parentMountPath, mountPoint.relativeMountPath) + (!strictRouting ? "*" : "")};
131 routes.push_back(" " + mountPoint.method + " " + mountPoint.relativeMountPath);
132
133 if (nextRoute) {
134 routes.splice(routes.end(), nextRoute->getRoute(parentMountPath, strictRouting));
135 }
136
137 return routes;
138 }
139
140} // namespace express::dispatcher
const std::string & getConnectionName() const
SocketConnection * getSocketConnection() const
const std::shared_ptr< Request > & getRequest()
Controller & operator=(const Controller &controller) noexcept
const std::shared_ptr< Response > & getResponse()
std::shared_ptr< Route > nextRoute
Definition Dispatcher.h:81
Next(Controller &controller)
Definition Next.cpp:52
Controller controller
Definition Next.h:66
std::string url
Definition Request.h:98
std::string method
Definition Request.h:97
std::map< std::string, std::string > queries
Definition Request.h:103
std::string path
Definition Request.h:79
void sendStatus(int state, const std::string &html={})
Definition Response.cpp:95
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:98
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 dispatch(express::Controller &controller, const express::MountPoint &mountPoint, bool strictRouting, bool caseInsensitiveRouting, bool mergeParams) override
ScopedParams(express::Request &req, const std::map< std::string, std::string > &params, bool mergeWithParent)
ScopedPathStrip(express::Request &req, bool enabled, std::size_t consumedLength)
std::string joinMountPath(std::string_view parentMountPath, std::string_view relativeMountPath)
MountMatchResult matchMountPoint(express::Controller &controller, const std::string &absoluteMountPath, const express::MountPoint &mountPoint, std::regex &cachedRegex, std::vector< std::string > &cachedNames, bool strictRouting, bool caseInsensitiveRouting)
bool methodMatches(std::string_view requestMethod, const std::string &mountMethod)
std::string method
Definition MountPoint.h:56
std::string relativeMountPath
Definition MountPoint.h:57
std::map< std::string, std::string > params
Definition regex_utils.h:83
std::unordered_map< std::string, std::string > requestQueryPairs
Definition regex_utils.h:84