Skip to content

Commit c00e56d

Browse files
committed
Auto merge of #4032 - alexcrichton:retry-500, r=matklad
Automatically retry HTTP requests returning 5xx This commit implements auto-retry for downloading crates from crates.io whenever a 5xx response is returned. This should help assist with automatic retries whenever Cargo attempts to download directly from S3 but S3 returns a 500 error, which is defined as "please retry again". This logic may be a little eager to retry *all* 500 errors, but there's a maximum cap on all retries regardless, so hopefully it doesn't result in too many problems. Closes #3962
2 parents 58fe1a8 + 6155653 commit c00e56d

File tree

2 files changed

+77
-12
lines changed

2 files changed

+77
-12
lines changed

src/cargo/sources/registry/remote.rs

+19-12
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ use util::network;
1616
use util::paths;
1717
use util::{FileLock, Filesystem};
1818
use util::{Config, CargoResult, ChainError, human, Sha256, ToUrl};
19+
use util::errors::HttpError;
1920

2021
pub struct RemoteRegistry<'cfg> {
2122
index_path: Filesystem,
@@ -153,26 +154,32 @@ impl<'cfg> RegistryData for RemoteRegistry<'cfg> {
153154
// TODO: don't download into memory, but ensure that if we ctrl-c a
154155
// download we should resume either from the start or the middle
155156
// on the next time
157+
let url = url.to_string();
156158
handle.get(true)?;
157-
handle.url(&url.to_string())?;
159+
handle.url(&url)?;
158160
handle.follow_location(true)?;
159161
let mut state = Sha256::new();
160162
let mut body = Vec::new();
161163
network::with_retry(self.config, || {
162164
state = Sha256::new();
163165
body = Vec::new();
164-
let mut handle = handle.transfer();
165-
handle.write_function(|buf| {
166-
state.update(buf);
167-
body.extend_from_slice(buf);
168-
Ok(buf.len())
169-
})?;
170-
handle.perform()
166+
{
167+
let mut handle = handle.transfer();
168+
handle.write_function(|buf| {
169+
state.update(buf);
170+
body.extend_from_slice(buf);
171+
Ok(buf.len())
172+
})?;
173+
handle.perform()?;
174+
}
175+
let code = handle.response_code()?;
176+
if code != 200 && code != 0 {
177+
let url = handle.effective_url()?.unwrap_or(&url);
178+
Err(HttpError::Not200(code, url.to_string()))
179+
} else {
180+
Ok(())
181+
}
171182
})?;
172-
let code = handle.response_code()?;
173-
if code != 200 && code != 0 {
174-
bail!("failed to get 200 response from `{}`, got {}", url, code)
175-
}
176183

177184
// Verify what we just downloaded
178185
if state.finish().to_hex() != checksum {

src/cargo/util/errors.rs

+58
Original file line numberDiff line numberDiff line change
@@ -334,6 +334,7 @@ impl NetworkError for git2::Error {
334334
}
335335
}
336336
}
337+
337338
impl NetworkError for curl::Error {
338339
fn maybe_spurious(&self) -> bool {
339340
self.is_couldnt_connect() ||
@@ -344,6 +345,63 @@ impl NetworkError for curl::Error {
344345
}
345346
}
346347

348+
#[derive(Debug)]
349+
pub enum HttpError {
350+
Not200(u32, String),
351+
Curl(curl::Error),
352+
}
353+
354+
impl fmt::Display for HttpError {
355+
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
356+
match *self {
357+
HttpError::Not200(code, ref url) => {
358+
write!(f, "failed to get 200 response from `{}`, got {}",
359+
url, code)
360+
}
361+
HttpError::Curl(ref e) => e.fmt(f),
362+
}
363+
}
364+
}
365+
366+
impl Error for HttpError {
367+
fn description(&self) -> &str {
368+
match *self {
369+
HttpError::Not200(..) => "failed to get a 200 response",
370+
HttpError::Curl(ref e) => e.description(),
371+
}
372+
}
373+
374+
fn cause(&self) -> Option<&Error> {
375+
match *self {
376+
HttpError::Not200(..) => None,
377+
HttpError::Curl(ref e) => e.cause(),
378+
}
379+
}
380+
}
381+
382+
impl CargoError for HttpError {
383+
fn is_human(&self) -> bool {
384+
true
385+
}
386+
}
387+
388+
impl NetworkError for HttpError {
389+
fn maybe_spurious(&self) -> bool {
390+
match *self {
391+
HttpError::Not200(code, ref _url) => {
392+
500 <= code && code < 600
393+
}
394+
HttpError::Curl(ref e) => e.maybe_spurious(),
395+
}
396+
}
397+
}
398+
399+
impl From<curl::Error> for HttpError {
400+
fn from(err: curl::Error) -> HttpError {
401+
HttpError::Curl(err)
402+
}
403+
}
404+
347405
// =============================================================================
348406
// various impls
349407

0 commit comments

Comments
 (0)