Skip to content

Commit eb8ed3d

Browse files
authored
Merge pull request #1326 from jyn514/delete-database-handler
Remove `database_file_handler`; give precendence to router error messages
2 parents f552d45 + 8010082 commit eb8ed3d

File tree

5 files changed

+44
-67
lines changed

5 files changed

+44
-67
lines changed

src/web/error.rs

+4-7
Original file line numberDiff line numberDiff line change
@@ -163,9 +163,6 @@ mod tests {
163163

164164
#[test]
165165
fn check_404_page_content_resource() {
166-
// Resources with a `.js` and `.ico` extension are special cased in the
167-
// routes_handler which is currently run last. This means that `get("resource.exe")` will
168-
// fail with a `no so such crate` instead of 'no such resource'
169166
wrapper(|env| {
170167
let page = kuchiki::parse_html().one(
171168
env.frontend()
@@ -180,7 +177,7 @@ mod tests {
180177
.next()
181178
.unwrap()
182179
.text_contents(),
183-
"The requested crate does not exist",
180+
"The requested resource does not exist",
184181
);
185182

186183
Ok(())
@@ -200,7 +197,7 @@ mod tests {
200197
.next()
201198
.unwrap()
202199
.text_contents(),
203-
"The requested crate does not exist",
200+
"The requested version does not exist",
204201
);
205202

206203
Ok(())
@@ -219,7 +216,7 @@ mod tests {
219216
.next()
220217
.unwrap()
221218
.text_contents(),
222-
"The requested crate does not exist",
219+
"The requested version does not exist",
223220
);
224221

225222
Ok(())
@@ -242,7 +239,7 @@ mod tests {
242239
.next()
243240
.unwrap()
244241
.text_contents(),
245-
"The requested crate does not exist",
242+
"The requested version does not exist",
246243
);
247244

248245
Ok(())

src/web/file.rs

+5-32
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,15 @@
11
//! Database based file handler
22
33
use crate::storage::{Blob, Storage};
4-
use crate::{error::Result, Config, Metrics};
5-
use iron::{status, Handler, IronResult, Request, Response};
4+
use crate::{error::Result, Config};
5+
use iron::{status, Response};
66

77
#[derive(Debug)]
88
pub(crate) struct File(pub(crate) Blob);
99

1010
impl File {
1111
/// Gets file from database
12-
pub fn from_path(storage: &Storage, path: &str, config: &Config) -> Result<File> {
12+
pub(super) fn from_path(storage: &Storage, path: &str, config: &Config) -> Result<File> {
1313
let max_size = if path.ends_with(".html") {
1414
config.max_file_size_html
1515
} else {
@@ -20,7 +20,7 @@ impl File {
2020
}
2121

2222
/// Consumes File and creates a iron response
23-
pub fn serve(self) -> Response {
23+
pub(super) fn serve(self) -> Response {
2424
use iron::headers::{CacheControl, CacheDirective, ContentType, HttpDate, LastModified};
2525

2626
let mut response = Response::with((status::Ok, self.0.content));
@@ -44,38 +44,11 @@ impl File {
4444
}
4545

4646
/// Checks if mime type of file is "application/x-empty"
47-
pub fn is_empty(&self) -> bool {
47+
pub(super) fn is_empty(&self) -> bool {
4848
self.0.mime == "application/x-empty"
4949
}
5050
}
5151

52-
/// Database based file handler for iron
53-
///
54-
/// This is similar to staticfile crate, but its using getting files from database.
55-
pub struct DatabaseFileHandler;
56-
57-
impl Handler for DatabaseFileHandler {
58-
fn handle(&self, req: &mut Request) -> IronResult<Response> {
59-
let path = req.url.path().join("/");
60-
let storage = extension!(req, Storage);
61-
let config = extension!(req, Config);
62-
if let Ok(file) = File::from_path(&storage, &path, &config) {
63-
let metrics = extension!(req, Metrics);
64-
65-
// Because all requests that don't hit another handler go through here, we will get all
66-
// requests successful or not recorded by the RequestRecorder, so we inject an extra
67-
// "database success" route to keep track of how often we succeed vs 404
68-
metrics
69-
.routes_visited
70-
.with_label_values(&["database success"])
71-
.inc();
72-
Ok(file.serve())
73-
} else {
74-
Err(super::error::Nope::CrateNotFound.into())
75-
}
76-
}
77-
}
78-
7952
#[cfg(test)]
8053
mod tests {
8154
use super::*;

src/web/metrics.rs

+30-15
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ use iron::status::Status;
77
use prometheus::{Encoder, HistogramVec, TextEncoder};
88
use std::time::{Duration, Instant};
99

10-
pub fn metrics_handler(req: &mut Request) -> IronResult<Response> {
10+
pub(super) fn metrics_handler(req: &mut Request) -> IronResult<Response> {
1111
let metrics = extension!(req, Metrics);
1212
let pool = extension!(req, Pool);
1313
let queue = extension!(req, BuildQueue);
@@ -30,7 +30,7 @@ fn duration_to_seconds(d: Duration) -> f64 {
3030
d.as_secs() as f64 + nanos
3131
}
3232

33-
pub struct RequestRecorder {
33+
pub(super) struct RequestRecorder {
3434
handler: Box<dyn iron::Handler>,
3535
route_name: String,
3636
}
@@ -108,6 +108,7 @@ impl Drop for RenderingTimesRecorder<'_> {
108108
#[cfg(test)]
109109
mod tests {
110110
use crate::test::{assert_success, wrapper};
111+
use crate::Context;
111112
use std::collections::HashMap;
112113

113114
#[test]
@@ -133,8 +134,8 @@ mod tests {
133134
("/sitemap.xml", "static resource"),
134135
("/-/static/style.css", "static resource"),
135136
("/-/static/vendored.css", "static resource"),
136-
("/rustdoc/rcc/0.0.0/rcc/index.html", "database"),
137-
("/rustdoc/gcc/0.0.0/gcc/index.html", "database"),
137+
("/rustdoc/rcc/0.0.0/rcc/index.html", "rustdoc page"),
138+
("/rustdoc/gcc/0.0.0/gcc/index.html", "rustdoc page"),
138139
];
139140

140141
wrapper(|env| {
@@ -167,29 +168,43 @@ mod tests {
167168
*entry += 2;
168169
}
169170

171+
// this shows what the routes were *actually* recorded as, making it easier to update ROUTES if the name changes.
172+
let metrics_serialized = metrics.gather(&env.pool()?, &env.build_queue())?;
173+
let all_routes_visited = metrics_serialized
174+
.iter()
175+
.find(|x| x.get_name() == "docsrs_routes_visited")
176+
.unwrap();
177+
let routes_visited_pretty: Vec<_> = all_routes_visited
178+
.get_metric()
179+
.iter()
180+
.map(|metric| {
181+
let labels = metric.get_label();
182+
assert_eq!(labels.len(), 1); // not sure when this would be false
183+
let route = labels[0].get_value();
184+
let count = metric.get_counter().get_value();
185+
format!("{}: {}", route, count)
186+
})
187+
.collect();
188+
println!("routes: {:?}", routes_visited_pretty);
189+
170190
for (label, count) in expected.iter() {
171191
assert_eq!(
172192
metrics.routes_visited.with_label_values(&[*label]).get(),
173-
*count
193+
*count,
194+
"routes_visited metrics for {} are incorrect",
195+
label,
174196
);
175197
assert_eq!(
176198
metrics
177199
.response_time
178200
.with_label_values(&[*label])
179201
.get_sample_count(),
180-
*count as u64
202+
*count as u64,
203+
"response_time metrics for {} are incorrect",
204+
label,
181205
);
182206
}
183207

184-
// extra metrics for the "database success" hack
185-
assert_eq!(
186-
metrics
187-
.routes_visited
188-
.with_label_values(&["database success"])
189-
.get(),
190-
2
191-
);
192-
193208
Ok(())
194209
})
195210
}

src/web/mod.rs

+4-12
Original file line numberDiff line numberDiff line change
@@ -105,7 +105,6 @@ use iron::{
105105
status::Status,
106106
Chain, Handler, Iron, IronError, IronResult, Listening, Request, Response, Url,
107107
};
108-
use metrics::RequestRecorder;
109108
use page::TemplateData;
110109
use postgres::Client;
111110
use router::NoRoute;
@@ -121,7 +120,6 @@ const DEFAULT_BIND: &str = "0.0.0.0:3000";
121120
struct CratesfyiHandler {
122121
shared_resource_handler: Box<dyn Handler>,
123122
router_handler: Box<dyn Handler>,
124-
database_file_handler: Box<dyn Handler>,
125123
inject_extensions: InjectExtensions,
126124
}
127125

@@ -140,19 +138,13 @@ impl CratesfyiHandler {
140138
let inject_extensions = InjectExtensions::new(context, template_data)?;
141139

142140
let routes = routes::build_routes();
143-
let blacklisted_prefixes = routes.page_prefixes();
144-
145141
let shared_resources =
146142
Self::chain(inject_extensions.clone(), rustdoc::SharedResourceHandler);
147143
let router_chain = Self::chain(inject_extensions.clone(), routes.iron_router());
148144

149145
Ok(CratesfyiHandler {
150146
shared_resource_handler: Box::new(shared_resources),
151147
router_handler: Box::new(router_chain),
152-
database_file_handler: Box::new(routes::BlockBlacklistedPrefixes::new(
153-
blacklisted_prefixes,
154-
Box::new(RequestRecorder::new(file::DatabaseFileHandler, "database")),
155-
)),
156148
inject_extensions,
157149
})
158150
}
@@ -165,7 +157,9 @@ impl Handler for CratesfyiHandler {
165157
handle: impl FnOnce() -> IronResult<Response>,
166158
) -> IronResult<Response> {
167159
if e.response.status == Some(status::NotFound) {
168-
handle()
160+
// the routes are ordered from most specific to least; give precedence to the
161+
// original error message.
162+
handle().or(Err(e))
169163
} else {
170164
Err(e)
171165
}
@@ -174,8 +168,7 @@ impl Handler for CratesfyiHandler {
174168
// This is kind of a mess.
175169
//
176170
// Almost all files should be served through the `router_handler`; eventually
177-
// `shared_resource_handler` should go through the router too, and `database_file_handler`
178-
// could be removed altogether.
171+
// `shared_resource_handler` should go through the router too.
179172
//
180173
// Unfortunately, combining `shared_resource_handler` with the `router_handler` breaks
181174
// things, because right now `shared_resource_handler` allows requesting files from *any*
@@ -188,7 +181,6 @@ impl Handler for CratesfyiHandler {
188181
self.router_handler
189182
.handle(req)
190183
.or_else(|e| if_404(e, || self.shared_resource_handler.handle(req)))
191-
.or_else(|e| if_404(e, || self.database_file_handler.handle(req)))
192184
.or_else(|e| {
193185
let err = if let Some(err) = e.error.downcast_ref::<error::Nope>() {
194186
*err

src/web/rustdoc.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -516,7 +516,7 @@ pub fn target_redirect_handler(req: &mut Request) -> IronResult<Response> {
516516

517517
let crate_details = match CrateDetails::new(&mut conn, &name, &version) {
518518
Some(krate) => krate,
519-
None => return Err(Nope::ResourceNotFound.into()),
519+
None => return Err(Nope::VersionNotFound.into()),
520520
};
521521

522522
// [crate, :name, :version, target-redirect, :target, *path]

0 commit comments

Comments
 (0)