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
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
use std::process::exit;
use std::path::PathBuf;
use std::io::{self, Read, Write};
use std::fs;
use std::env;
use pbr;
use sha2;
use reqwest::{
header,
Client,
Proxy,
Result as ReqResult,
Url,
};
use tempfile;
use digest::Digest;
use digest::generic_array::functional::FunctionalSequence;
use utils::{
read,
write,
unpack
};
use project_dirs::PROJECT_DIRS;
pub struct PrebuiltPackage {
pub url: &'static str,
pub name: &'static str,
pub version: &'static str,
pub arch: &'static str,
pub hash: &'static str,
pub size: u64,
}
fn create_client() -> ReqResult<Client> {
let mut builder = Client::builder().danger_accept_invalid_certs( true );
match env::var("HTTPS_PROXY") {
Err(_) => {},
Ok(proxy) => { builder = builder.proxy(Proxy::https(&proxy).unwrap()); }
};
match env::var("HTTP_PROXY") {
Err(_) => {},
Ok(proxy) => { builder = builder.proxy(Proxy::http(&proxy).unwrap()); }
};
match env::var("ALL_PROXY") {
Err(_) => {},
Ok(proxy) => { builder = builder.proxy(Proxy::all(&proxy).unwrap()); }
};
builder.build()
}
pub fn download_package( package: &PrebuiltPackage ) -> PathBuf {
let url = Url::parse( package.url ).unwrap();
let package_filename = url.path_segments().unwrap().last().unwrap().to_owned();
let unpack_path = PROJECT_DIRS.data_local_dir().join( package.name ).join( package.arch );
let version_path = unpack_path.join( ".version" );
if let Ok( existing_version ) = read( &version_path ) {
if existing_version == package.version {
return unpack_path;
}
}
if fs::metadata( &unpack_path ).is_ok() {
fs::remove_dir_all( &unpack_path ).unwrap();
}
fs::create_dir_all( &unpack_path ).unwrap();
eprintln!( "Downloading {}...", package_filename );
let client = create_client().unwrap();
let mut response = client.get( url )
.header( header::CONNECTION, "close" )
.send()
.unwrap();
let prefix = format!( "cargo-web-{}-download", package.name );
let tmpdir = tempfile::Builder::new().prefix( &prefix ).tempdir().unwrap();
let dlpath = tmpdir.path().join( &package_filename );
let mut fp = fs::File::create( &dlpath ).unwrap();
let length = response.headers().get( header::CONTENT_LENGTH )
.and_then( |len| len.to_str().ok() )
.and_then( |len| len.parse().ok() )
.unwrap_or( package.size );
let mut pb = pbr::ProgressBar::new( length );
pb.set_units( pbr::Units::Bytes );
let mut buffer = Vec::new();
buffer.resize( 1024 * 1024, 0 );
let mut hasher = sha2::Sha256::default();
loop {
let length = match response.read( &mut buffer ) {
Ok( 0 ) => break,
Ok( length ) => length,
Err( ref err ) if err.kind() == io::ErrorKind::Interrupted => continue,
Err( err ) => panic!( err )
};
let slice = &buffer[ 0..length ];
hasher.input( slice );
fp.write_all( slice ).unwrap();
pb.add( length as u64 );
}
pb.finish();
let actual_hash = hasher.result();
let actual_hash = actual_hash.map( |byte| format!( "{:02x}", byte ) ).join( "" );
if actual_hash != package.hash {
eprintln!( "error: the hash of {} doesn't match the expected hash!", package_filename );
eprintln!( " actual: {}", actual_hash );
eprintln!( " expected: {}", package.hash );
exit( 101 );
}
eprintln!( "Unpacking {}...", package_filename );
unpack( &dlpath, &unpack_path ).unwrap();
write( &version_path, package.version ).unwrap();
eprintln!( "Package {} was successfully installed!", package_filename );
return unpack_path;
}