SNode.C
Loading...
Searching...
No Matches
ApplicationDispatcher.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/ApplicationDispatcher.h"
43
44#include "express/Controller.h"
45#include "express/Request.h"
46#include "express/Route.h"
47#include "express/dispatcher/regex_utils.h"
48
49#ifndef DOXYGEN_SHOULD_SKIP_THIS
50
51#include "log/Logger.h"
52
53#include <list>
54
55#endif /* DOXYGEN_SHOULD_SKIP_THIS */
56
57namespace express::dispatcher {
58
60 const std::function<void(const std::shared_ptr<Request>&, const std::shared_ptr<Response>&)>& lambda)
61 : lambda(lambda) {
62 }
63
64 bool ApplicationDispatcher::dispatch(express::Controller& controller,
65 const std::string& parentMountPath,
66 const express::MountPoint& mountPoint) {
67 bool requestMatched = false;
68
69 if ((controller.getFlags() & Controller::NEXT) == 0) {
70 const std::string absoluteMountPath = parentMountPath + mountPoint.relativeMountPath;
71
72 const std::string requestMethod = controller.getRequest()->method;
73 const std::string requestUrl = controller.getRequest()->url;
74 const std::string requestPath = controller.getRequest()->path;
75
76 // clang-format off
77 requestMatched =
78 (
79 (
80 (
81 mountPoint.method == requestMethod
82 ) || (
83 mountPoint.method == "all"
84 )
85 ) && (
86 (
87 requestUrl == absoluteMountPath
88 ) || (
89 (
90 !controller.getStrictRouting()
91 ) && (
92 requestUrl.starts_with(absoluteMountPath)
93 )
94 ) || (
95 checkForUrlMatch(absoluteMountPath, requestUrl)
96 )
97 )
98 ) || (
99 (
100 mountPoint.method == "use"
101 ) && (
102 (
103 requestUrl.starts_with(absoluteMountPath)
104 )
105 )
106 );
107 // clang-format on
108
109 LOG(TRACE) << "Express: A - RequestUrl: " << controller.getRequest()->url;
110 LOG(TRACE) << "Express: A - RequestPath: " << controller.getRequest()->path;
111 LOG(TRACE) << "Express: A - AbsoluteMountPath: " << absoluteMountPath;
112 LOG(TRACE) << "Express: A - StrictRouting: " << controller.getStrictRouting();
113
114 if (requestMatched) {
115 LOG(TRACE) << " MATCH";
116 if (hasResult(absoluteMountPath)) {
117 setParams(absoluteMountPath, *controller.getRequest());
118 }
119
120 lambda(controller.getRequest(), controller.getResponse());
121 } else {
122 LOG(TRACE) << " NO MQTCH";
123 }
124 }
125
126 return requestMatched;
127 }
128
129 std::list<std::string>
130 ApplicationDispatcher::getRoutes(const std::string& parentMountPath, const MountPoint& mountPoint, bool strictRouting) const {
131 std::list<std::string> routes{"A " + parentMountPath + mountPoint.relativeMountPath + (!strictRouting ? "*" : "")};
132 routes.push_back(" " + mountPoint.method + " " + mountPoint.relativeMountPath);
133
134 if (nextRoute) {
135 routes.splice(routes.end(), nextRoute->getRoute(parentMountPath, strictRouting));
136 }
137
138 return routes;
139 }
140
141} // namespace express::dispatcher
const std::shared_ptr< Request > & getRequest()
bool getStrictRouting() const
const std::shared_ptr< Response > & getResponse()
std::shared_ptr< Route > nextRoute
Definition Dispatcher.h:78
std::string url
Definition Request.h:96
std::string method
Definition Request.h:95
std::string path
Definition Request.h:77
std::list< std::string > getRoute(const std::string &parentMountPath, bool strictRouting) const
Definition Route.cpp:114
const std::function< void(const std::shared_ptr< Request > &, const std::shared_ptr< Response > &)> lambda
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, bool strictRouting) const override
ApplicationDispatcher(const std::function< void(const std::shared_ptr< Request > &, const std::shared_ptr< Response > &)> &lambda)
bool checkForUrlMatch(const std::string &cpath, const std::string &reqpath)
void setParams(const std::string &cpath, express::Request &req)
bool hasResult(const std::string &cpath)
std::string method
Definition MountPoint.h:56
std::string relativeMountPath
Definition MountPoint.h:57