Skip to content

Commit

Permalink
Merge pull request #1020 from jelford/support_partial_downloads
Browse files Browse the repository at this point in the history
Support partial downloads
  • Loading branch information
brson authored Apr 8, 2017
2 parents 9a99cf4 + c63b275 commit 4cdbe1b
Show file tree
Hide file tree
Showing 10 changed files with 243 additions and 51 deletions.
1 change: 1 addition & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions ci/run.sh
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ cargo -vV
cargo build -v --release --target $TARGET

if [ -z "$SKIP_TESTS" ]; then
cargo test --release -p download --target $TARGET
cargo test --release -p rustup-dist --target $TARGET
cargo test --release --target $TARGET
fi
3 changes: 3 additions & 0 deletions src/download/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,9 @@ curl = { version = "0.4", optional = true }
lazy_static = { version = "0.2", optional = true }
native-tls = { version = "0.1", optional = true }

[dev-dependencies]
tempdir = "0.3.4"

[dependencies.hyper]
version = "0.9.8"
default-features = false
Expand Down
6 changes: 5 additions & 1 deletion src/download/src/errors.rs
Original file line number Diff line number Diff line change
@@ -1,7 +1,11 @@


error_chain! {
links { }

foreign_links { }
foreign_links {
Io(::std::io::Error);
}

errors {
HttpStatus(e: u32) {
Expand Down
112 changes: 82 additions & 30 deletions src/download/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ pub enum Backend { Curl, Hyper, Rustls }

#[derive(Debug, Copy, Clone)]
pub enum Event<'a> {
ResumingPartialDownload,
/// Received the Content-Length of the to-be downloaded data.
DownloadContentLengthReceived(u64),
/// Received some data.
Expand All @@ -33,47 +34,89 @@ const BACKENDS: &'static [Backend] = &[
Backend::Rustls
];

pub fn download(url: &Url,
callback: &Fn(Event) -> Result<()>)
-> Result<()> {
for &backend in BACKENDS {
match download_with_backend(backend, url, callback) {
Err(Error(ErrorKind::BackendUnavailable(_), _)) => (),
Err(e) => return Err(e),
Ok(()) => return Ok(()),
}
}

Err("no working backends".into())
}

pub fn download_with_backend(backend: Backend,
fn download_with_backend(backend: Backend,
url: &Url,
resume_from: u64,
callback: &Fn(Event) -> Result<()>)
-> Result<()> {
match backend {
Backend::Curl => curl::download(url, callback),
Backend::Curl => curl::download(url, resume_from, callback),
Backend::Hyper => hyper::download(url, callback),
Backend::Rustls => rustls::download(url, callback),
}
}

fn supports_partial_download(backend: &Backend) -> bool {
match backend {
&Backend::Curl => true,
_ => false
}
}

pub fn download_to_path_with_backend(
backend: Backend,
url: &Url,
path: &Path,
resume_from_partial: bool,
callback: Option<&Fn(Event) -> Result<()>>)
-> Result<()>
{
use std::cell::RefCell;
use std::fs::{self, File};
use std::io::Write;
use std::fs::{OpenOptions};
use std::io::{Read, Write, Seek, SeekFrom};

|| -> Result<()> {
let file = RefCell::new(try!(File::create(&path).chain_err(
|| "error creating file for download")));
let (file, resume_from) = if resume_from_partial && supports_partial_download(&backend) {
let possible_partial = OpenOptions::new()
.read(true)
.open(&path);

let downloaded_so_far = if let Ok(mut partial) = possible_partial {
if let Some(cb) = callback {
try!(cb(Event::ResumingPartialDownload));

let mut buf = vec![0; 32768];
let mut downloaded_so_far = 0;
loop {
let n = try!(partial.read(&mut buf));
downloaded_so_far += n as u64;
if n == 0 {
break;
}
try!(cb(Event::DownloadDataReceived(&buf[..n])));
}

downloaded_so_far
} else {
let file_info = try!(partial.metadata());
file_info.len()
}
} else {
0
};

try!(download_with_backend(backend, url, &|event| {
let mut possible_partial =
try!(OpenOptions::new()
.write(true)
.create(true)
.open(&path)
.chain_err(|| "error opening file for download"));

try!(possible_partial.seek(SeekFrom::End(0)));

(possible_partial, downloaded_so_far)
} else {
(try!(OpenOptions::new()
.write(true)
.create(true)
.open(&path)
.chain_err(|| "error creating file for download")), 0)
};

let file = RefCell::new(file);

try!(download_with_backend(backend, url, resume_from, &|event| {
if let Event::DownloadDataReceived(data) = event {
try!(file.borrow_mut().write_all(data)
.chain_err(|| "unable to write download to disk"));
Expand All @@ -89,11 +132,8 @@ pub fn download_to_path_with_backend(

Ok(())
}().map_err(|e| {
if path.is_file() {
// FIXME ignoring compound errors
let _ = fs::remove_file(path);
}

// TODO is there any point clearing up here? What kind of errors will leave us with an unusable partial?
e
})
}
Expand All @@ -114,6 +154,7 @@ pub mod curl {
use super::Event;

pub fn download(url: &Url,
resume_from: u64,
callback: &Fn(Event) -> Result<()> )
-> Result<()> {
// Fetch either a cached libcurl handle (which will preserve open
Expand All @@ -128,6 +169,15 @@ pub mod curl {
try!(handle.url(&url.to_string()).chain_err(|| "failed to set url"));
try!(handle.follow_location(true).chain_err(|| "failed to set follow redirects"));

if resume_from > 0 {
try!(handle.resume_from(resume_from)
.chain_err(|| "setting the range header for download resumption"));
} else {
// an error here indicates that the range header isn't supported by underlying curl,
// so there's nothing to "clear" - safe to ignore this error.
let _ = handle.resume_from(0);
}

// Take at most 30s to connect
try!(handle.connect_timeout(Duration::new(30, 0)).chain_err(|| "failed to set connect timeout"));

Expand All @@ -154,8 +204,8 @@ pub mod curl {
if let Ok(data) = str::from_utf8(header) {
let prefix = "Content-Length: ";
if data.starts_with(prefix) {
if let Ok(s) = data[prefix.len()..].trim().parse() {
let msg = Event::DownloadContentLengthReceived(s);
if let Ok(s) = data[prefix.len()..].trim().parse::<u64>() {
let msg = Event::DownloadContentLengthReceived(s + resume_from);
match callback(msg) {
Ok(()) => (),
Err(e) => {
Expand Down Expand Up @@ -188,11 +238,12 @@ pub mod curl {
}));
}

// If we didn't get a 200 or 0 ("OK" for files) then return an error
// If we didn't get a 20x or 0 ("OK" for files) then return an error
let code = try!(handle.response_code().chain_err(|| "failed to get response code"));
if code != 200 && code != 0 {
return Err(ErrorKind::HttpStatus(code).into());
}
match code {
0 | 200 ... 299 => {},
_ => { return Err(ErrorKind::HttpStatus(code).into()); }
};

Ok(())
})
Expand Down Expand Up @@ -639,6 +690,7 @@ pub mod curl {
use super::Event;

pub fn download(_url: &Url,
_resume_from: u64,
_callback: &Fn(Event) -> Result<()> )
-> Result<()> {
Err(ErrorKind::BackendUnavailable("curl").into())
Expand Down
98 changes: 98 additions & 0 deletions src/download/tests/download-curl-resume.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,98 @@
#![cfg(feature = "curl-backend")]

extern crate download;
extern crate tempdir;
extern crate url;

use std::sync::{Arc, Mutex};
use std::fs::{self, File};
use std::io::{self, Read};
use std::path::Path;

use tempdir::TempDir;
use url::Url;

use download::*;

fn tmp_dir() -> TempDir {
TempDir::new("rustup-download-test-").expect("creating tempdir for test")
}

fn file_contents(path: &Path) -> String {
let mut result = String::new();
File::open(&path).unwrap().read_to_string(&mut result).expect("reading test result file");
result
}


pub fn write_file(path: &Path, contents: &str) {
let mut file = fs::OpenOptions::new()
.write(true)
.truncate(true)
.create(true)
.open(path)
.expect("writing test data");

io::Write::write_all(&mut file, contents.as_bytes()).expect("writing test data");

file.sync_data().expect("writing test data");
}

#[test]
fn partially_downloaded_file_gets_resumed_from_byte_offset() {
let tmpdir = tmp_dir();
let from_path = tmpdir.path().join("download-source");
write_file(&from_path, "xxx45");

let target_path = tmpdir.path().join("downloaded");
write_file(&target_path, "123");

let from_url = Url::from_file_path(&from_path).unwrap();
download_to_path_with_backend(
Backend::Curl,
&from_url,
&target_path,
true,
None)
.expect("Test download failed");

assert_eq!(file_contents(&target_path), "12345");
}

#[test]
fn callback_gets_all_data_as_if_the_download_happened_all_at_once() {
let tmpdir = tmp_dir();

let from_path = tmpdir.path().join("download-source");
write_file(&from_path, "xxx45");

let target_path = tmpdir.path().join("downloaded");
write_file(&target_path, "123");

let from_url = Url::from_file_path(&from_path).unwrap();

let received_in_callback = Arc::new(Mutex::new(Vec::new()));

download_to_path_with_backend(Backend::Curl,
&from_url,
&target_path,
true,
Some(&|msg| {
match msg {
Event::DownloadDataReceived(data) => {
for b in data.iter() {
received_in_callback.lock().unwrap().push(b.clone());
}
}
_ => {}
}


Ok(())
}))
.expect("Test download failed");

let ref observed_bytes = *received_in_callback.lock().unwrap();
assert_eq!(observed_bytes, &vec![b'1', b'2', b'3', b'4', b'5']);
assert_eq!(file_contents(&target_path), "12345");
}
Loading

0 comments on commit 4cdbe1b

Please sign in to comment.