1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
use std::io::Write;
use anyhow::{Context, Result};
use futures_channel::mpsc;
use abstutil::prettyprint_usize;
pub async fn download_bytes<I: AsRef<str>>(
url: I,
post_body: Option<String>,
progress: &mut mpsc::Sender<String>,
) -> Result<Vec<u8>> {
let url = url.as_ref();
info!("Downloading {}", url);
let mut resp = if let Some(body) = post_body {
reqwest::Client::new()
.post(url)
.body(body)
.send()
.await
.unwrap()
} else {
reqwest::get(url).await.unwrap()
};
resp.error_for_status_ref()
.with_context(|| format!("downloading {}", url))?;
let total_size = resp.content_length().map(|x| x as usize);
let mut bytes = Vec::new();
while let Some(chunk) = resp.chunk().await.unwrap() {
let msg = if let Some(n) = total_size {
format!(
"{:.2}% ({} / {} bytes)",
(bytes.len() as f64) / (n as f64) * 100.0,
prettyprint_usize(bytes.len()),
prettyprint_usize(n)
)
} else {
format!(
"{} bytes (unknown total size)",
prettyprint_usize(bytes.len())
)
};
if let Err(err) = progress.try_send(msg) {
warn!("Couldn't send download progress message: {}", err);
}
bytes.write_all(&chunk).unwrap();
}
println!();
Ok(bytes)
}
pub async fn download_to_file<I1: AsRef<str>, I2: AsRef<str>>(
url: I1,
post_body: Option<String>,
path: I2,
) -> Result<()> {
let (mut tx, rx) = futures_channel::mpsc::channel(1000);
print_download_progress(rx);
let bytes = download_bytes(url, post_body, &mut tx).await?;
let path = path.as_ref();
std::fs::create_dir_all(std::path::Path::new(path).parent().unwrap())?;
let mut file = std::fs::File::create(path)?;
file.write_all(&bytes)?;
Ok(())
}
pub fn print_download_progress(mut progress: mpsc::Receiver<String>) {
tokio::task::spawn_blocking(move || loop {
match progress.try_next() {
Ok(Some(msg)) => {
abstutil::clear_current_line();
print!("{}", msg);
std::io::stdout().flush().unwrap();
}
Ok(None) => break,
Err(_) => {}
}
});
}