@@ -32,6 +32,12 @@ struct Transaction {
32
32
bins : Vec < PathBuf > ,
33
33
}
34
34
35
+ impl Transaction {
36
+ fn success ( mut self ) {
37
+ self . bins . clear ( ) ;
38
+ }
39
+ }
40
+
35
41
impl Drop for Transaction {
36
42
fn drop ( & mut self ) {
37
43
for bin in self . bins . iter ( ) {
@@ -44,7 +50,8 @@ pub fn install(root: Option<&str>,
44
50
krate : Option < & str > ,
45
51
source_id : & SourceId ,
46
52
vers : Option < & str > ,
47
- opts : & ops:: CompileOptions ) -> CargoResult < ( ) > {
53
+ opts : & ops:: CompileOptions ,
54
+ force : bool ) -> CargoResult < ( ) > {
48
55
let config = opts. config ;
49
56
let root = try!( resolve_root ( root, config) ) ;
50
57
let ( pkg, source) = if source_id. is_git ( ) {
@@ -77,7 +84,7 @@ pub fn install(root: Option<&str>,
77
84
let metadata = try!( metadata ( config, & root) ) ;
78
85
let list = try!( read_crate_list ( metadata. file ( ) ) ) ;
79
86
let dst = metadata. parent ( ) . join ( "bin" ) ;
80
- try!( check_overwrites ( & dst, & pkg, & opts. filter , & list) ) ;
87
+ try!( check_overwrites ( & dst, & pkg, & opts. filter , & list, force ) ) ;
81
88
}
82
89
83
90
let mut td_opt = None ;
@@ -102,40 +109,122 @@ pub fn install(root: Option<&str>,
102
109
human ( format ! ( "failed to compile `{}`, intermediate artifacts can be \
103
110
found at `{}`", pkg, target_dir. display( ) ) )
104
111
} ) ) ;
112
+ let binaries: Vec < ( & str , & Path ) > = try!( compile. binaries . iter ( ) . map ( |bin| {
113
+ let name = bin. file_name ( ) . unwrap ( ) ;
114
+ if let Some ( s) = name. to_str ( ) {
115
+ Ok ( ( s, bin. as_ref ( ) ) )
116
+ } else {
117
+ bail ! ( "Binary `{:?}` name can't be serialized into string" , name)
118
+ }
119
+ } ) . collect :: < CargoResult < _ > > ( ) ) ;
105
120
106
121
let metadata = try!( metadata ( config, & root) ) ;
107
122
let mut list = try!( read_crate_list ( metadata. file ( ) ) ) ;
108
123
let dst = metadata. parent ( ) . join ( "bin" ) ;
109
- try!( check_overwrites ( & dst, & pkg, & opts. filter , & list) ) ;
124
+ let duplicates = try!( check_overwrites ( & dst, & pkg, & opts. filter , & list, force ) ) ;
110
125
111
- let mut t = Transaction { bins : Vec :: new ( ) } ;
112
126
try!( fs:: create_dir_all ( & dst) ) ;
113
- for bin in compile. binaries . iter ( ) {
114
- let dst = dst. join ( bin. file_name ( ) . unwrap ( ) ) ;
127
+
128
+ // Copy all binaries to a temporary directory under `dst` first, catching
129
+ // some failure modes (e.g. out of space) before touching the existing
130
+ // binaries. This directory will get cleaned up via RAII.
131
+ let staging_dir = try!( TempDir :: new_in ( & dst, "cargo-install" ) ) ;
132
+ for & ( bin, src) in binaries. iter ( ) {
133
+ let dst = staging_dir. path ( ) . join ( bin) ;
134
+ // Try to move if `target_dir` is transient.
135
+ if !source_id. is_path ( ) {
136
+ if fs:: rename ( src, & dst) . is_ok ( ) {
137
+ continue
138
+ }
139
+ }
140
+ try!( fs:: copy ( src, & dst) . chain_error ( || {
141
+ human ( format ! ( "failed to copy `{}` to `{}`" , src. display( ) ,
142
+ dst. display( ) ) )
143
+ } ) ) ;
144
+ }
145
+
146
+ let ( to_replace, to_install) : ( Vec < & str > , Vec < & str > ) =
147
+ binaries. iter ( ) . map ( |& ( bin, _) | bin)
148
+ . partition ( |& bin| duplicates. contains_key ( bin) ) ;
149
+
150
+ let mut installed = Transaction { bins : Vec :: new ( ) } ;
151
+
152
+ // Move the temporary copies into `dst` starting with new binaries.
153
+ for bin in to_install. iter ( ) {
154
+ let src = staging_dir. path ( ) . join ( bin) ;
155
+ let dst = dst. join ( bin) ;
115
156
try!( config. shell ( ) . status ( "Installing" , dst. display ( ) ) ) ;
116
- try!( fs:: copy ( & bin , & dst) . chain_error ( || {
117
- human ( format ! ( "failed to copy `{}` to `{}`" , bin . display( ) ,
157
+ try!( fs:: rename ( & src , & dst) . chain_error ( || {
158
+ human ( format ! ( "failed to move `{}` to `{}`" , src . display( ) ,
118
159
dst. display( ) ) )
119
160
} ) ) ;
120
- t. bins . push ( dst) ;
161
+ installed. bins . push ( dst) ;
162
+ }
163
+
164
+ // Repeat for binaries which replace existing ones but don't pop the error
165
+ // up until after updating metadata.
166
+ let mut replaced_names = Vec :: new ( ) ;
167
+ let result = {
168
+ let mut try_install = || -> CargoResult < ( ) > {
169
+ for & bin in to_replace. iter ( ) {
170
+ let src = staging_dir. path ( ) . join ( bin) ;
171
+ let dst = dst. join ( bin) ;
172
+ try!( config. shell ( ) . status ( "Replacing" , dst. display ( ) ) ) ;
173
+ try!( fs:: rename ( & src, & dst) . chain_error ( || {
174
+ human ( format ! ( "failed to move `{}` to `{}`" , src. display( ) ,
175
+ dst. display( ) ) )
176
+ } ) ) ;
177
+ replaced_names. push ( bin) ;
178
+ }
179
+ Ok ( ( ) )
180
+ } ;
181
+ try_install ( )
182
+ } ;
183
+
184
+ // Update records of replaced binaries.
185
+ for & bin in replaced_names. iter ( ) {
186
+ if let Some ( & Some ( ref p) ) = duplicates. get ( bin) {
187
+ if let Some ( set) = list. v1 . get_mut ( p) {
188
+ set. remove ( bin) ;
189
+ }
190
+ }
191
+ list. v1 . entry ( pkg. package_id ( ) . clone ( ) )
192
+ . or_insert_with ( || BTreeSet :: new ( ) )
193
+ . insert ( bin. to_string ( ) ) ;
194
+ }
195
+
196
+ // Remove empty metadata lines.
197
+ let pkgs = list. v1 . iter ( )
198
+ . filter_map ( |( p, set) | if set. is_empty ( ) { Some ( p. clone ( ) ) } else { None } )
199
+ . collect :: < Vec < _ > > ( ) ;
200
+ for p in pkgs. iter ( ) {
201
+ list. v1 . remove ( p) ;
121
202
}
122
203
204
+ // If installation was successful record newly installed binaries.
205
+ if result. is_ok ( ) {
206
+ list. v1 . entry ( pkg. package_id ( ) . clone ( ) )
207
+ . or_insert_with ( || BTreeSet :: new ( ) )
208
+ . extend ( to_install. iter ( ) . map ( |s| s. to_string ( ) ) ) ;
209
+ }
210
+
211
+ let write_result = write_crate_list ( metadata. file ( ) , list) ;
212
+ match write_result {
213
+ // Replacement error (if any) isn't actually caused by write error
214
+ // but this seems to be the only way to show both.
215
+ Err ( err) => try!( result. chain_error ( || err) ) ,
216
+ Ok ( _) => try!( result) ,
217
+ }
218
+
219
+ // Reaching here means all actions have succeeded. Clean up.
220
+ installed. success ( ) ;
123
221
if !source_id. is_path ( ) {
124
222
// Don't bother grabbing a lock as we're going to blow it all away
125
223
// anyway.
126
224
let target_dir = target_dir. into_path_unlocked ( ) ;
127
225
try!( fs:: remove_dir_all ( & target_dir) ) ;
128
226
}
129
227
130
- list. v1 . entry ( pkg. package_id ( ) . clone ( ) ) . or_insert_with ( || {
131
- BTreeSet :: new ( )
132
- } ) . extend ( t. bins . iter ( ) . map ( |t| {
133
- t. file_name ( ) . unwrap ( ) . to_string_lossy ( ) . into_owned ( )
134
- } ) ) ;
135
- try!( write_crate_list ( metadata. file ( ) , list) ) ;
136
-
137
- t. bins . truncate ( 0 ) ;
138
-
139
228
// Print a warning that if this directory isn't in PATH that they won't be
140
229
// able to run these commands.
141
230
let path = env:: var_os ( "PATH" ) . unwrap_or ( OsString :: new ( ) ) ;
@@ -225,38 +314,61 @@ fn one<I, F>(mut i: I, f: F) -> CargoResult<Option<I::Item>>
225
314
fn check_overwrites ( dst : & Path ,
226
315
pkg : & Package ,
227
316
filter : & ops:: CompileFilter ,
228
- prev : & CrateListingV1 ) -> CargoResult < ( ) > {
317
+ prev : & CrateListingV1 ,
318
+ force : bool ) -> CargoResult < BTreeMap < String , Option < PackageId > > > {
319
+ if let CompileFilter :: Everything = * filter {
320
+ // If explicit --bin or --example flags were passed then those'll
321
+ // get checked during cargo_compile, we only care about the "build
322
+ // everything" case here
323
+ if pkg. targets ( ) . iter ( ) . filter ( |t| t. is_bin ( ) ) . next ( ) . is_none ( ) {
324
+ bail ! ( "specified package has no binaries" )
325
+ }
326
+ }
327
+ let duplicates = find_duplicates ( dst, pkg, filter, prev) ;
328
+ if force || duplicates. is_empty ( ) {
329
+ return Ok ( duplicates)
330
+ }
331
+ // Format the error message.
332
+ let mut msg = String :: new ( ) ;
333
+ for ( ref bin, p) in duplicates. iter ( ) {
334
+ msg. push_str ( & format ! ( "binary `{}` already exists in destination" , bin) ) ;
335
+ if let Some ( p) = p. as_ref ( ) {
336
+ msg. push_str ( & format ! ( " as part of `{}`\n " , p) ) ;
337
+ } else {
338
+ msg. push_str ( "\n " ) ;
339
+ }
340
+ }
341
+ msg. push_str ( "Add --force to overwrite" ) ;
342
+ Err ( human ( msg) )
343
+ }
344
+
345
+ fn find_duplicates ( dst : & Path ,
346
+ pkg : & Package ,
347
+ filter : & ops:: CompileFilter ,
348
+ prev : & CrateListingV1 ) -> BTreeMap < String , Option < PackageId > > {
229
349
let check = |name| {
230
350
let name = format ! ( "{}{}" , name, env:: consts:: EXE_SUFFIX ) ;
231
351
if fs:: metadata ( dst. join ( & name) ) . is_err ( ) {
232
- return Ok ( ( ) )
233
- }
234
- let mut msg = format ! ( "binary `{}` already exists in destination" , name ) ;
235
- if let Some ( ( p , _ ) ) = prev . v1 . iter ( ) . find ( | & ( _ , v ) | v . contains ( & name ) ) {
236
- msg . push_str ( & format ! ( " as part of `{}`" , p ) ) ;
352
+ None
353
+ } else if let Some ( ( p , _ ) ) = prev . v1 . iter ( ) . find ( | & ( _ , v ) | v . contains ( & name ) ) {
354
+ Some ( ( name , Some ( p . clone ( ) ) ) )
355
+ } else {
356
+ Some ( ( name , None ) )
237
357
}
238
- Err ( human ( msg) )
239
358
} ;
240
359
match * filter {
241
360
CompileFilter :: Everything => {
242
- // If explicit --bin or --example flags were passed then those'll
243
- // get checked during cargo_compile, we only care about the "build
244
- // everything" case here
245
- if pkg. targets ( ) . iter ( ) . filter ( |t| t. is_bin ( ) ) . next ( ) . is_none ( ) {
246
- bail ! ( "specified package has no binaries" )
247
- }
248
-
249
- for target in pkg. targets ( ) . iter ( ) . filter ( |t| t. is_bin ( ) ) {
250
- try!( check ( target. name ( ) ) ) ;
251
- }
361
+ pkg. targets ( ) . iter ( )
362
+ . filter ( |t| t. is_bin ( ) )
363
+ . filter_map ( |t| check ( t. name ( ) ) )
364
+ . collect ( )
252
365
}
253
366
CompileFilter :: Only { bins, examples, .. } => {
254
- for bin in bins. iter ( ) . chain ( examples) {
255
- try! ( check ( bin ) ) ;
256
- }
367
+ bins. iter ( ) . chain ( examples)
368
+ . filter_map ( |t| check ( t ) )
369
+ . collect ( )
257
370
}
258
371
}
259
- Ok ( ( ) )
260
372
}
261
373
262
374
fn read_crate_list ( mut file : & File ) -> CargoResult < CrateListingV1 > {
0 commit comments