1use hyper::{Body, Request, Response, StatusCode};
10use std::{
11 future::Future,
12 net::{IpAddr, SocketAddr},
13 path::PathBuf,
14 sync::Arc,
15};
16
17#[cfg(any(
18 feature = "compression",
19 feature = "compression-gzip",
20 feature = "compression-brotli",
21 feature = "compression-zstd",
22 feature = "compression-deflate"
23))]
24use crate::compression;
25
26use crate::compression_static;
27
28#[cfg(feature = "basic-auth")]
29use crate::basic_auth;
30
31#[cfg(feature = "fallback-page")]
32use crate::fallback_page;
33
34#[cfg(all(unix, feature = "experimental"))]
35use crate::metrics;
36
37#[cfg(feature = "experimental")]
38use crate::mem_cache::cache::MemCacheOpts;
39
40use crate::{
41 Error, Result, control_headers, cors, custom_headers, error_page, health,
42 http_ext::MethodExt,
43 log_addr, maintenance_mode, redirects, rewrites, security_headers,
44 settings::Advanced,
45 static_files::{self, HandleOpts},
46 virtual_hosts,
47};
48
49#[cfg(feature = "directory-listing")]
50use crate::directory_listing::DirListFmt;
51
52#[cfg(feature = "directory-listing-download")]
53use crate::directory_listing_download::DirDownloadFmt;
54
55pub struct RequestHandlerOpts {
57 pub root_dir: PathBuf,
60 #[cfg(feature = "experimental")]
61 pub memory_cache: Option<MemCacheOpts>,
63 pub compression: bool,
65 #[cfg(any(
66 feature = "compression",
67 feature = "compression-gzip",
68 feature = "compression-brotli",
69 feature = "compression-zstd",
70 feature = "compression-deflate"
71 ))]
72 pub compression_level: crate::settings::CompressionLevel,
74 pub compression_static: bool,
76 #[cfg(feature = "directory-listing")]
78 #[cfg_attr(docsrs, doc(cfg(feature = "directory-listing")))]
79 pub dir_listing: bool,
80 #[cfg(feature = "directory-listing")]
82 #[cfg_attr(docsrs, doc(cfg(feature = "directory-listing")))]
83 pub dir_listing_order: u8,
84 #[cfg(feature = "directory-listing")]
85 #[cfg_attr(docsrs, doc(cfg(feature = "directory-listing")))]
86 pub dir_listing_format: DirListFmt,
88 #[cfg(feature = "directory-listing-download")]
90 #[cfg_attr(docsrs, doc(cfg(feature = "directory-listing-download")))]
91 pub dir_listing_download: Vec<DirDownloadFmt>,
92 pub cors: Option<cors::Configured>,
94 pub security_headers: bool,
96 pub cache_control_headers: bool,
98 pub page404: PathBuf,
100 pub page50x: PathBuf,
102 #[cfg(feature = "fallback-page")]
104 #[cfg_attr(docsrs, doc(cfg(feature = "fallback-page")))]
105 pub page_fallback: Vec<u8>,
106 #[cfg(feature = "basic-auth")]
108 #[cfg_attr(docsrs, doc(cfg(feature = "basic-auth")))]
109 pub basic_auth: String,
110 pub index_files: Vec<String>,
112 pub log_remote_address: bool,
114 pub log_x_real_ip: bool,
116 pub log_forwarded_for: bool,
118 pub trusted_proxies: Vec<IpAddr>,
120 pub redirect_trailing_slash: bool,
122 pub ignore_hidden_files: bool,
124 pub disable_symlinks: bool,
126 pub accept_markdown: bool,
128 pub health: bool,
130 #[cfg(all(unix, feature = "experimental"))]
132 pub experimental_metrics: bool,
133 pub maintenance_mode: bool,
135 pub maintenance_mode_status: StatusCode,
137 pub maintenance_mode_file: PathBuf,
139
140 pub advanced_opts: Option<Advanced>,
142}
143
144impl Default for RequestHandlerOpts {
145 fn default() -> Self {
146 Self {
147 root_dir: PathBuf::from("./public"),
148 compression: true,
149 compression_static: false,
150 #[cfg(any(
151 feature = "compression",
152 feature = "compression-gzip",
153 feature = "compression-brotli",
154 feature = "compression-zstd",
155 feature = "compression-deflate"
156 ))]
157 compression_level: crate::settings::CompressionLevel::Default,
158 #[cfg(feature = "directory-listing")]
159 dir_listing: false,
160 #[cfg(feature = "directory-listing")]
161 dir_listing_order: 6, #[cfg(feature = "directory-listing")]
163 dir_listing_format: DirListFmt::Html,
164 #[cfg(feature = "directory-listing-download")]
165 dir_listing_download: Vec::new(),
166 cors: None,
167 #[cfg(feature = "experimental")]
168 memory_cache: None,
169 security_headers: false,
170 cache_control_headers: true,
171 page404: PathBuf::from("./404.html"),
172 page50x: PathBuf::from("./50x.html"),
173 #[cfg(feature = "fallback-page")]
174 page_fallback: Vec::new(),
175 #[cfg(feature = "basic-auth")]
176 basic_auth: String::new(),
177 index_files: vec!["index.html".into()],
178 log_remote_address: false,
179 log_x_real_ip: false,
180 log_forwarded_for: false,
181 trusted_proxies: Vec::new(),
182 redirect_trailing_slash: true,
183 ignore_hidden_files: false,
184 disable_symlinks: false,
185 accept_markdown: false,
186 health: false,
187 #[cfg(all(unix, feature = "experimental"))]
188 experimental_metrics: false,
189 maintenance_mode: false,
190 maintenance_mode_status: StatusCode::SERVICE_UNAVAILABLE,
191 maintenance_mode_file: PathBuf::new(),
192 advanced_opts: None,
193 }
194 }
195}
196
197pub struct RequestHandler {
199 pub opts: Arc<RequestHandlerOpts>,
201}
202
203impl RequestHandler {
204 pub fn handle<'a>(
206 &'a self,
207 req: &'a mut Request<Body>,
208 remote_addr: Option<SocketAddr>,
209 ) -> impl Future<Output = Result<Response<Body>, Error>> + Send + 'a {
210 let mut base_path = &self.opts.root_dir;
211 #[cfg(feature = "directory-listing")]
212 let dir_listing = self.opts.dir_listing;
213 #[cfg(feature = "directory-listing")]
214 let dir_listing_order = self.opts.dir_listing_order;
215 #[cfg(feature = "directory-listing")]
216 let dir_listing_format = &self.opts.dir_listing_format;
217 #[cfg(feature = "directory-listing-download")]
218 let dir_listing_download = &self.opts.dir_listing_download;
219 let redirect_trailing_slash = self.opts.redirect_trailing_slash;
220 let compression_static = self.opts.compression_static;
221 let ignore_hidden_files = self.opts.ignore_hidden_files;
222 let disable_symlinks = self.opts.disable_symlinks;
223 let index_files: Vec<&str> = self.opts.index_files.iter().map(|s| s.as_str()).collect();
224 #[cfg(feature = "experimental")]
225 let memory_cache = self.opts.memory_cache.as_ref();
226
227 log_addr::pre_process(&self.opts, req, remote_addr);
228
229 async move {
230 if !req.method().is_allowed() {
232 return error_page::error_response(
233 req.uri(),
234 req.method(),
235 &StatusCode::METHOD_NOT_ALLOWED,
236 &self.opts.page404,
237 &self.opts.page50x,
238 );
239 }
240
241 if let Some(result) = health::pre_process(&self.opts, req) {
243 return result;
244 }
245
246 #[cfg(all(unix, feature = "experimental"))]
248 if let Some(result) = metrics::pre_process(&self.opts, req) {
249 return result;
250 }
251
252 if let Some(result) = cors::pre_process(&self.opts, req) {
254 return result;
255 }
256
257 #[cfg(feature = "basic-auth")]
259 if let Some(response) = basic_auth::pre_process(&self.opts, req) {
260 return response;
261 }
262
263 if let Some(response) = maintenance_mode::pre_process(&self.opts, req) {
265 return response;
266 }
267
268 if let Some(result) = redirects::pre_process(&self.opts, req) {
270 return result;
271 }
272
273 if let Some(result) = rewrites::pre_process(&self.opts, req) {
275 return result;
276 }
277
278 if let Some(advanced) = &self.opts.advanced_opts {
280 if let Some(root) =
282 virtual_hosts::get_real_root(req, advanced.virtual_hosts.as_deref())
283 {
284 base_path = root;
285 }
286 }
287
288 let index_files = index_files.as_ref();
289
290 let uri_path_md = if self.opts.accept_markdown {
292 crate::markdown::pre_process(req, base_path, req.uri().path())
293 } else {
294 None
295 };
296 let uri_path = uri_path_md.as_deref().unwrap_or(req.uri().path());
297
298 let (resp, file_path) = match static_files::handle(&HandleOpts {
300 method: req.method(),
301 headers: req.headers(),
302 #[cfg(feature = "experimental")]
303 memory_cache,
304 base_path,
305 uri_path,
306 uri_query: req.uri().query(),
307 #[cfg(feature = "directory-listing")]
308 dir_listing,
309 #[cfg(feature = "directory-listing")]
310 dir_listing_order,
311 #[cfg(feature = "directory-listing")]
312 dir_listing_format,
313 #[cfg(feature = "directory-listing-download")]
314 dir_listing_download,
315 redirect_trailing_slash,
316 compression_static,
317 ignore_hidden_files,
318 index_files,
319 disable_symlinks,
320 })
321 .await
322 {
323 Ok(result) => (result.resp, Some(result.file_path)),
324 Err(status) => (
325 error_page::error_response(
326 req.uri(),
327 req.method(),
328 &status,
329 &self.opts.page404,
330 &self.opts.page50x,
331 )?,
332 None,
333 ),
334 };
335
336 #[cfg(feature = "fallback-page")]
338 let resp = fallback_page::post_process(&self.opts, req, resp)?;
339
340 let resp = cors::post_process(&self.opts, req, resp)?;
342
343 let resp = crate::markdown::post_process(uri_path_md.is_some(), &self.opts, resp)?;
345
346 let resp = compression_static::post_process(&self.opts, req, resp)?;
348
349 #[cfg(any(
351 feature = "compression",
352 feature = "compression-gzip",
353 feature = "compression-brotli",
354 feature = "compression-zstd",
355 feature = "compression-deflate"
356 ))]
357 let resp = compression::post_process(&self.opts, req, resp)?;
358
359 let resp = control_headers::post_process(&self.opts, req, resp)?;
361
362 let resp = security_headers::post_process(&self.opts, req, resp)?;
364
365 let resp = custom_headers::post_process(&self.opts, req, resp, file_path.as_ref())?;
367
368 Ok(resp)
369 }
370 }
371}