SNode.C
Loading...
Searching...
No Matches
regex_utils.h
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#ifndef EXPRESS_DISPATCHER_REGEX_UTILS_H
43#define EXPRESS_DISPATCHER_REGEX_UTILS_H
44
45namespace express {
46 class Request;
47 class Controller;
48 struct MountPoint;
49} // namespace express
50
51#ifndef DOXYGEN_SHOULD_SKIP_THIS
52
53#include <cctype>
54#include <cstddef>
55#include <map>
56#include <regex>
57#include <string>
58#include <string_view>
59#include <unordered_map>
60#include <vector>
61
62#endif /* DOXYGEN_SHOULD_SKIP_THIS */
63
64namespace express::dispatcher {
65
66 inline bool ieq(char a, char b) {
67 return std::tolower(static_cast<unsigned char>(a)) == std::tolower(static_cast<unsigned char>(b));
68 }
69
70 // ---------- shared mount-point matching (used by Router/Application/Middleware dispatchers) ----------
71 //
72 // Supports Express v4 *string* route wildcards/operators: '*', '?', '+', and grouping '()',
73 // plus ':param', ':param(<re>)' and ':param' modifiers ('?', '+', '*').
74
75 bool methodMatches(std::string_view requestMethod, const std::string& mountMethod);
76
78 bool requestMatched{false};
79 bool decodeError{false};
80 bool isPrefix{false};
81 std::size_t consumedLength{0};
82 std::string_view requestPath;
83 std::map<std::string, std::string> params;
84 std::unordered_map<std::string, std::string> requestQueryPairs;
85 };
86
87 // Join parent + relative mount paths without producing double slashes.
88 // Examples:
89 // joinMountPath("", "/api") -> "/api"
90 // joinMountPath("/api", "/v1") -> "/api/v1"
91 // joinMountPath("/", "/root/test") -> "/root/test" (NOT "//root/test")
92 std::string joinMountPath(std::string_view parentMountPath, std::string_view relativeMountPath);
93
94 MountMatchResult matchMountPoint(express::Controller& controller,
95 const std::string& absoluteMountPath,
96 const express::MountPoint& mountPoint,
97 std::regex& cachedRegex,
98 std::vector<std::string>& cachedNames,
99 bool strictRouting,
100 bool caseInsensitiveRouting);
101
103 public:
104 ScopedPathStrip(express::Request& req, bool enabled, std::size_t consumedLength);
106
111
112 private:
113 express::Request* req_{nullptr};
114 std::string backupUrl_;
115 std::string backupBaseUrl_;
116 std::string backupPath_;
117 std::string backupFile_;
118 bool enabled_{false};
119 };
120
122 public:
123 ScopedParams(express::Request& req, const std::map<std::string, std::string>& params, bool mergeWithParent);
124 ~ScopedParams();
125
126 ScopedParams(const ScopedParams&) = delete;
130
131 private:
132 express::Request* req_{nullptr};
133 std::map<std::string, std::string> backup_;
134 };
135
136} // namespace express::dispatcher
137
138#endif // EXPRESS_DISPATCHER_REGEX_UTILS_H
const std::string & getConnectionName() const
SocketConnection * getSocketConnection() const
const std::shared_ptr< Request > & getRequest() const
const std::shared_ptr< Response > & getResponse() const
std::shared_ptr< Route > nextRoute
Definition Dispatcher.h:81
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
bool dispatch(express::Controller &controller, const express::MountPoint &mountPoint, bool strictRouting, bool caseInsensitiveRouting, bool mergeParams) override
const std::function< void(const std::shared_ptr< Request > &, const std::shared_ptr< Response > &)> lambda
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)
std::map< std::string, std::string > backup_
ScopedParams & operator=(const ScopedParams &)=delete
ScopedParams(express::Request &req, const std::map< std::string, std::string > &params, bool mergeWithParent)
ScopedParams & operator=(ScopedParams &&)=delete
ScopedParams(const ScopedParams &)=delete
ScopedParams(ScopedParams &&)=delete
ScopedPathStrip(const ScopedPathStrip &)=delete
ScopedPathStrip(express::Request &req, bool enabled, std::size_t consumedLength)
ScopedPathStrip & operator=(ScopedPathStrip &&)=delete
ScopedPathStrip & operator=(const ScopedPathStrip &)=delete
ScopedPathStrip(ScopedPathStrip &&)=delete
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 ieq(char a, char b)
Definition regex_utils.h:66
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