userver: userver/server/handlers/http_handler_base.hpp Source File
Loading...
Searching...
No Matches
http_handler_base.hpp
Go to the documentation of this file.
1#pragma once
2
3/// @file userver/server/handlers/http_handler_base.hpp
4/// @brief @copybrief server::handlers::HttpHandlerBase
5
6#include <memory>
7#include <optional>
8#include <string>
9#include <vector>
10
11#include <userver/dynamic_config/source.hpp>
12#include <userver/logging/level.hpp>
13#include <userver/utils/statistics/entry.hpp>
14#include <userver/utils/token_bucket.hpp>
15
16#include <userver/server/handlers/exceptions.hpp>
17#include <userver/server/handlers/formatted_error_data.hpp>
18#include <userver/server/handlers/handler_base.hpp>
19#include <userver/server/http/http_request.hpp>
20#include <userver/server/http/http_response.hpp>
21#include <userver/server/http/http_response_body_stream_fwd.hpp>
22// Not needed here, but a lot of code depends on it being included transitively
23#include <userver/tracing/span.hpp>
24
25USERVER_NAMESPACE_BEGIN
26
27namespace server::middlewares {
28class HttpMiddlewareBase;
29class HandlerAdapter;
30class Auth;
31} // namespace server::middlewares
32
33/// @brief Most common \ref userver_http_handlers "userver HTTP handlers"
34namespace server::handlers {
35
36class HttpHandlerStatistics;
37class HttpRequestStatistics;
38class HttpHandlerMethodStatistics;
39class HttpHandlerStatisticsScope;
40
41/// @ingroup userver_components userver_http_handlers userver_base_classes
42///
43/// @brief Base class for all the @ref userver_http_handlers "Userver HTTP Handlers".
44///
45/// ## Static options of server::handlers::HttpHandlerBase :
46/// @include{doc} scripts/docs/en/components_schema/core/src/server/handlers/http_handler_base.md
47///
48/// Options inherited from @ref server::handlers::HandlerBase :
49/// @include{doc} scripts/docs/en/components_schema/core/src/server/handlers/handler_base.md
50///
51/// Options inherited from @ref components::ComponentBase :
52/// @include{doc} scripts/docs/en/components_schema/core/src/components/impl/component_base.md
53///
54/// ## Example usage:
55///
56/// @include samples/hello_service/src/hello_handler.hpp
57/// @include samples/hello_service/src/hello_handler.cpp
59public:
60 HttpHandlerBase(
61 const components::ComponentConfig& config,
62 const components::ComponentContext& component_context,
63 bool is_monitor = false
64 );
65
66 ~HttpHandlerBase() override;
67
68 void PrepareAndHandleRequest(http::HttpRequest& request, request::RequestContext& context) const override;
69
70 void ReportMalformedRequest(http::HttpRequest& request) const final;
71
72 virtual const std::string& HandlerName() const;
73
74 const std::vector<http::HttpMethod>& GetAllowedMethods() const;
75
76 /// @cond
77 // For internal use only.
78 HttpHandlerStatistics& GetHandlerStatistics() const;
79
80 // For internal use only.
81 HttpRequestStatistics& GetRequestStatistics() const;
82 /// @endcond
83
84 /// Override it if you need a custom logging level for messages about finish
85 /// of request handling for some http statuses.
86 virtual logging::Level GetLogLevelForResponseStatus(http::HttpStatus status) const;
87
88 virtual FormattedErrorData GetFormattedExternalErrorBody(const CustomHandlerException& exc) const;
89
90 std::string GetResponseDataForLoggingChecked(
91 const http::HttpRequest& request,
92 request::RequestContext& context,
93 const std::string& response_data
94 ) const;
95
96 std::string GetUrlForLoggingChecked(const http::HttpRequest& request, request::RequestContext& context) const;
97
98 /// Takes the exception and formats it into response, as specified by
99 /// exception.
101 const http::HttpRequest& request,
102 request::RequestContext& context,
103 const CustomHandlerException& ex
104 ) const;
105
106 /// Takes the exception and formats it into response as an internal server
107 /// error.
109 const http::HttpRequest& request,
110 request::RequestContext& context,
111 const std::exception& ex
112 ) const;
113
114 /// Helper function to log an unknown exception
115 void LogUnknownException(const std::exception& ex, std::optional<logging::Level> log_level_override = {}) const;
116
117 /// Returns the default log level for the handler
118 const std::optional<logging::Level>& GetLogLevel() const;
119
120 static yaml_config::Schema GetStaticConfigSchema();
121
122protected:
123 [[noreturn]] void ThrowUnsupportedHttpMethod(const http::HttpRequest& request) const;
124
125 /// Same as `HandleRequest`.
126 virtual std::string HandleRequestThrow(const http::HttpRequest& request, request::RequestContext& context) const;
127
128 /// The core method for HTTP request handling.
129 /// `request` arg contains HTTP headers, full body, etc.
130 /// The method should return response body.
131 /// @note It is used only if IsStreamed() returned `false`.
132 virtual std::string HandleRequest(http::HttpRequest& request, request::RequestContext& context) const;
133
134 /// The core method for HTTP request handling.
135 /// `request` arg contains HTTP headers, full body, etc.
136 /// The response body is passed in parts to `ResponseBodyStream`.
137 /// Stream transmission is useful when:
138 /// 1) The body size is unknown beforehand.
139 /// 2) The client may take advantage of early body transmission
140 /// (e.g. a Web Browser may start rendering the HTML page
141 /// or downloading dependant resources).
142 /// 3) The body size is huge and we want to have only a part of it
143 /// in memory.
144 /// @note It is used only if IsStreamed() returned `true`.
145 virtual void
146 HandleStreamRequest(server::http::HttpRequest&, server::request::RequestContext&, server::http::ResponseBodyStream&)
147 const;
148
149 /// If IsStreamed() returns `true`, call HandleStreamRequest()
150 /// for request handling, HandleRequest() is not called.
151 /// If it returns `false`, HandleRequest() is called instead,
152 /// and HandleStreamRequest() is not called.
153 /// @note The default implementation returns the cached value of
154 /// "response-body-streamed" value from static config.
155 virtual bool IsStreamed() const { return is_body_streamed_; }
156
157 /// Override it if you need a custom streamed logic based on request and context.
158 /// @note The default implementation returns the cached value of
159 /// "response-body-streamed" value from static config.
160 virtual bool IsStreamed(const http::HttpRequest&, server::request::RequestContext&) const { return IsStreamed(); }
161
162 /// Override it to show per HTTP-method statistics besides statistics for all
163 /// methods
164 virtual bool IsMethodStatisticIncluded() const { return false; }
165
166 /// Override it if you want to disable auth checks in handler by some
167 /// condition
168 virtual bool NeedCheckAuth() const { return true; }
169
170 /// Override it if you need a custom request body logging.
171 virtual std::string GetRequestBodyForLogging(
172 const http::HttpRequest& request,
173 request::RequestContext& context,
174 const std::string& request_body
175 ) const;
176
177 /// Override it if you need a custom response data logging.
178 virtual std::string GetResponseDataForLogging(
179 const http::HttpRequest& request,
180 request::RequestContext& context,
181 const std::string& response_data
182 ) const;
183
184 /// Override it if you need a custom request url logging.
185 virtual std::string GetUrlForLogging(const http::HttpRequest& request, request::RequestContext& context) const;
186
187 /// For internal use. You don't need to override it. This method is overridden
188 /// in format-specific base handlers.
189 virtual void ParseRequestData(const http::HttpRequest&, request::RequestContext&) const {}
190
191 virtual std::string GetMetaType(const http::HttpRequest&) const;
192
193private:
194 friend class middlewares::HandlerAdapter;
195 friend class middlewares::Auth;
196
197 void HandleHttpRequest(http::HttpRequest& request, request::RequestContext& context) const;
198
199 void HandleRequestStream(http::HttpRequest& http_request, request::RequestContext& context) const;
200
201 std::string GetRequestBodyForLoggingChecked(
202 const http::HttpRequest& request,
203 request::RequestContext& context,
204 const std::string& request_body
205 ) const;
206
207 template <typename HttpStatistics>
208 void FormatStatistics(utils::statistics::Writer result, const HttpStatistics& stats);
209
210 void SetResponseServerHostname(http::HttpResponse& response) const;
211
212 void BuildMiddlewarePipeline(const components::ComponentConfig&, const components::ComponentContext&);
213
214 const dynamic_config::Source config_source_;
215 const std::vector<http::HttpMethod> allowed_methods_;
216 const std::string handler_name_;
217 utils::statistics::Entry statistics_holder_;
218 std::optional<logging::Level> log_level_;
219 std::unordered_map<int, logging::Level> log_level_for_status_codes_;
220
221 std::unique_ptr<HttpHandlerStatistics> handler_statistics_;
222 std::unique_ptr<HttpRequestStatistics> request_statistics_;
223
224 bool set_response_server_hostname_;
225 bool is_body_streamed_;
226
227 std::unique_ptr<middlewares::HttpMiddlewareBase> first_middleware_;
228};
229
230} // namespace server::handlers
231
232USERVER_NAMESPACE_END