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/token_bucket.hpp>
14
15#include <userver/server/handlers/exceptions.hpp>
16#include <userver/server/handlers/formatted_error_data.hpp>
17#include <userver/server/handlers/handler_base.hpp>
18#include <userver/server/http/http_request.hpp>
19#include <userver/server/http/http_response.hpp>
20#include <userver/server/http/http_response_body_stream_fwd.hpp>
21// Not needed here, but a lot of code depends on it being included transitively
22#include <userver/tracing/span.hpp>
23
24USERVER_NAMESPACE_BEGIN
25
26namespace server::middlewares {
28class HandlerAdapter;
29class HandlerMetrics;
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 HttpHandlerStatisticsAggregate;
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 HttpHandlerStatisticsAggregate& GetHandlerStatistics() const;
79 /// @endcond
80
81 /// Override it if you need a custom logging level for messages about finish
82 /// of request handling for some http statuses.
83 virtual logging::Level GetLogLevelForResponseStatus(http::HttpStatus status) const;
84
85 virtual FormattedErrorData GetFormattedExternalErrorBody(const CustomHandlerException& exc) const;
86
87 std::string GetResponseDataForLoggingChecked(
88 const http::HttpRequest& request,
89 request::RequestContext& context,
90 const std::string& response_data
91 ) const;
92
93 std::string GetUrlForLoggingChecked(const http::HttpRequest& request, request::RequestContext& context) const;
94
95 /// Takes the exception and formats it into response, as specified by
96 /// exception.
98 const http::HttpRequest& request,
99 request::RequestContext& context,
100 const CustomHandlerException& ex
101 ) const;
102
103 /// Takes the exception and formats it into response as an internal server
104 /// error.
106 const http::HttpRequest& request,
107 request::RequestContext& context,
108 const std::exception& ex
109 ) const;
110
111 /// Helper function to log an unknown exception
112 void LogUnknownException(const std::exception& ex, std::optional<logging::Level> log_level_override = {}) const;
113
114 /// Returns the default log level for the handler
115 const std::optional<logging::Level>& GetLogLevel() const;
116
117 static yaml_config::Schema GetStaticConfigSchema();
118
119 /// Override it if you need a custom streamed logic based on request and context.
120 /// @note The default implementation returns the cached value of
121 /// "response-body-streamed" value from static config.
122 virtual bool IsStreamed(const http::HttpRequest&, server::request::RequestContext&) const { return IsStreamed(); }
123
124protected:
125 [[noreturn]] void ThrowUnsupportedHttpMethod(const http::HttpRequest& request) const;
126
127 /// Same as `HandleRequest`.
128 virtual std::string HandleRequestThrow(const http::HttpRequest& request, request::RequestContext& context) const;
129
130 /// The core method for HTTP request handling.
131 /// `request` arg contains HTTP headers, full body, etc.
132 /// The method should return response body.
133 /// @note It is used only if IsStreamed() returned `false`.
134 virtual std::string HandleRequest(http::HttpRequest& request, request::RequestContext& context) const;
135
136 /// The core method for HTTP request handling.
137 /// `request` arg contains HTTP headers, full body, etc.
138 /// The response body is passed in parts to `ResponseBodyStream`.
139 /// Stream transmission is useful when:
140 /// 1) The body size is unknown beforehand.
141 /// 2) The client may take advantage of early body transmission
142 /// (e.g. a Web Browser may start rendering the HTML page
143 /// or downloading dependant resources).
144 /// 3) The body size is huge and we want to have only a part of it
145 /// in memory.
146 /// @note It is used only if IsStreamed() returned `true`.
147 virtual void
148 HandleStreamRequest(server::http::HttpRequest&, server::request::RequestContext&, server::http::ResponseBodyStream&)
149 const;
150
151 /// If IsStreamed() returns `true`, call HandleStreamRequest()
152 /// for request handling, HandleRequest() is not called.
153 /// If it returns `false`, HandleRequest() is called instead,
154 /// and HandleStreamRequest() is not called.
155 /// @note The default implementation returns the cached value of
156 /// "response-body-streamed" value from static config.
157 virtual bool IsStreamed() const { return is_body_streamed_; }
158
159 /// Override it to show per HTTP-method statistics besides statistics for all
160 /// methods
161 virtual bool IsMethodStatisticIncluded() const { return false; }
162
163 /// Override it if you want to disable auth checks in handler by some
164 /// condition
165 virtual bool NeedCheckAuth() const { return true; }
166
167 /// Override it if you need a custom request body logging.
168 virtual std::string GetRequestBodyForLogging(
169 const http::HttpRequest& request,
170 request::RequestContext& context,
171 const std::string& request_body
172 ) const;
173
174 /// Override it if you need a custom response data logging.
175 virtual std::string GetResponseDataForLogging(
176 const http::HttpRequest& request,
177 request::RequestContext& context,
178 const std::string& response_data
179 ) const;
180
181 /// Override it if you need a custom request url logging.
182 virtual std::string GetUrlForLogging(const http::HttpRequest& request, request::RequestContext& context) const;
183
184 /// For internal use. You don't need to override it. This method is overridden
185 /// in format-specific base handlers.
186 virtual void ParseRequestData(const http::HttpRequest&, request::RequestContext&) const {}
187
188 virtual std::string GetMetaType(const http::HttpRequest&) const;
189
190private:
191 friend class middlewares::HandlerAdapter;
192 friend class middlewares::HandlerMetrics;
193 friend class middlewares::Auth;
194
195 void HandleHttpRequest(http::HttpRequest& request, request::RequestContext& context) const;
196
197 void HandleRequestStream(http::HttpRequest& http_request, request::RequestContext& context) const;
198
199 std::string GetRequestBodyForLoggingChecked(
200 const http::HttpRequest& request,
201 request::RequestContext& context,
202 const std::string& request_body
203 ) const;
204
205 template <typename HttpStatistics>
206 void FormatStatistics(utils::statistics::Writer result, const HttpStatistics& stats);
207
208 void FormatPerLabelStatistics(utils::statistics::Writer result) const;
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 std::optional<logging::Level> log_level_;
218 std::unordered_map<int, logging::Level> log_level_for_status_codes_;
219
220 std::unique_ptr<HttpHandlerStatisticsAggregate> handler_statistics_;
221
222 bool set_response_server_hostname_;
223 bool is_body_streamed_;
224
225 std::unique_ptr<middlewares::HttpMiddlewareBase> first_middleware_;
226};
227
228} // namespace server::handlers
229
230USERVER_NAMESPACE_END