73 {
74 bool requestMatched = false;
75 const std::string absoluteMountPath = parentMountPath + mountPoint.
relativeMountPath;
76
78
79 std::string_view mountPath;
80 std::string_view mountQueryString;
82 auto requiredQueryPairs =
parseQuery(mountQueryString);
83
84 std::string_view requestPath;
85 std::string_view requestQueryString;
87 auto requestQueryPairs =
parseQuery(requestQueryString);
88
89
93 }
94 if (mountPath.empty()) {
95 mountPath = "/";
96 }
97
98
99 bool pathMatches = false;
100 std::size_t consumedLength = 0;
101 if (absoluteMountPath.find(':') != std::string::npos) {
102
103 if (
regex.mark_count() == 0) {
104 if (
regex.mark_count() == 0) {
105 LOG(TRACE) << "MiddlewarDispatchere: precompiled regex";
107 true,
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
118 consumedLength = static_cast<std::size_t>(regexMatches.length(0));
119
120
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) {
125 }
126 }
127 } else {
128
130 if (pathMatches) {
131 consumedLength = mountPath.size();
132 }
133 }
134
136 requestMatched = (pathMatches && queryMatches);
137
138 LOG(TRACE) << controller.
getResponse()->getSocketContext()->getSocketConnection()->getConnectionName()
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;
146
147 if (requestMatched) {
148
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 }
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
164 }
165
166 Next next(controller);
168
169
171 LOG(TRACE) << "Express: M - Next called - set to NO MATCH";
172 requestMatched = false;
173 controller = next.controller;
174 }
175
176
177 req.path = previousPathBackup;
178 }
179 } else {
180 LOG(TRACE) << controller.
getResponse()->getSocketContext()->getSocketConnection()->getConnectionName()
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 }
const std::shared_ptr< Request > & getRequest()
bool getStrictRouting() const
bool getCaseInsensitiveRouting() const
const std::shared_ptr< Response > & getResponse()
std::vector< std::string > names
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 relativeMountPath