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
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
|
use std::borrow::Cow;
use std::io::{self, Read};
use std::time::Duration;
use reqwest::header::{CONNECTION, HeaderValue};
use reqwest::{Client, ClientBuilder};
use self::error::{DownloadError, ErrorKind};
use failure::ResultExt;
#[derive(Clone, Debug)]
pub struct Url<'a> {
url: Cow<'a, str>,
max_kib: Option<usize>,
timeout: Option<Duration>,
}
impl<'a> From<String> for Url<'a> {
fn from(url: String) -> Self {
Url {
url: Cow::from(url),
max_kib: None,
timeout: None,
}
}
}
impl<'a> From<&'a str> for Url<'a> {
fn from(url: &'a str) -> Self {
Url {
url: Cow::from(url),
max_kib: None,
timeout: None,
}
}
}
impl<'a> Url<'a> {
pub fn max_kib(mut self, limit: usize) -> Self {
self.max_kib = Some(limit);
self
}
pub fn timeout(mut self, timeout: Duration) -> Self {
self.timeout = Some(timeout);
self
}
/// Downloads the file and converts it to a String.
/// Any invalid bytes are converted to a replacement character.
///
/// The error indicated either a failed download or
/// that the limit set by max_kib() was reached.
pub fn request(&self) -> Result<String, DownloadError> {
let client = if let Some(timeout) = self.timeout {
ClientBuilder::new().timeout(timeout).build().unwrap()
} else {
Client::new()
};
let mut response = client
.get(self.url.as_ref())
.header(CONNECTION, HeaderValue::from_static("close"))
.send()
.context(ErrorKind::Connection)?;
// 100 kibibyte buffer
let mut buf = [0; 100 * 1024];
let mut written = 0;
let mut bytes = Vec::new();
// Read until we reach EOF or max_kib KiB
loop {
let len = match response.read(&mut buf) {
Ok(0) => break,
Ok(len) => len,
Err(ref e) if e.kind() == io::ErrorKind::Interrupted => continue,
Err(e) => Err(e).context(ErrorKind::Read)?,
};
bytes.extend_from_slice(&buf[..len]);
written += len;
// Check if the file is too large to download
if let Some(max_kib) = self.max_kib {
if written > max_kib * 1024 {
Err(ErrorKind::DownloadLimit)?;
}
}
}
Ok(String::from_utf8_lossy(&bytes).into_owned())
}
pub fn as_str(&self) -> &str {
&self.url
}
}
pub mod error {
#[derive(Copy, Clone, Eq, PartialEq, Debug, Fail, Error)]
#[error = "DownloadError"]
pub enum ErrorKind {
/// Connection Error
#[fail(display = "A connection error has occured")]
Connection,
/// Read Error
#[fail(display = "A read error has occured")]
Read,
/// Reached download limit error
#[fail(display = "Reached download limit")]
DownloadLimit,
}
}
|