@@ -8,13 +8,10 @@ extern crate toml;
8
8
use std:: collections:: BTreeSet ;
9
9
use std:: env;
10
10
use std:: fs;
11
- use std:: io;
12
- use std:: path:: { PathBuf , Path } ;
13
- use std:: process:: Command ;
11
+ use std:: path:: PathBuf ;
14
12
15
- use cargo:: { execute_main_without_stdin, handle_error, shell} ;
16
- use cargo:: core:: MultiShell ;
17
- use cargo:: util:: { CliError , CliResult , lev_distance, Config } ;
13
+ use cargo:: execute_main_without_stdin;
14
+ use cargo:: util:: { self , CliResult , lev_distance, Config , human, CargoResult } ;
18
15
19
16
#[ derive( RustcDecodable ) ]
20
17
struct Flags {
@@ -61,35 +58,37 @@ fn main() {
61
58
execute_main_without_stdin ( execute, true , USAGE )
62
59
}
63
60
64
- macro_rules! each_subcommand{ ( $mac: ident) => ( {
65
- $mac!( bench) ;
66
- $mac!( build) ;
67
- $mac!( clean) ;
68
- $mac!( doc) ;
69
- $mac!( fetch) ;
70
- $mac!( generate_lockfile) ;
71
- $mac!( git_checkout) ;
72
- $mac!( help) ;
73
- $mac!( install) ;
74
- $mac!( locate_project) ;
75
- $mac!( login) ;
76
- $mac!( new) ;
77
- $mac!( owner) ;
78
- $mac!( package) ;
79
- $mac!( pkgid) ;
80
- $mac!( publish) ;
81
- $mac!( read_manifest) ;
82
- $mac!( run) ;
83
- $mac!( rustc) ;
84
- $mac!( rustdoc) ;
85
- $mac!( search) ;
86
- $mac!( test) ;
87
- $mac!( uninstall) ;
88
- $mac!( update) ;
89
- $mac!( verify_project) ;
90
- $mac!( version) ;
91
- $mac!( yank) ;
92
- } ) }
61
+ macro_rules! each_subcommand{
62
+ ( $mac: ident) => ( {
63
+ $mac!( bench) ;
64
+ $mac!( build) ;
65
+ $mac!( clean) ;
66
+ $mac!( doc) ;
67
+ $mac!( fetch) ;
68
+ $mac!( generate_lockfile) ;
69
+ $mac!( git_checkout) ;
70
+ $mac!( help) ;
71
+ $mac!( install) ;
72
+ $mac!( locate_project) ;
73
+ $mac!( login) ;
74
+ $mac!( new) ;
75
+ $mac!( owner) ;
76
+ $mac!( package) ;
77
+ $mac!( pkgid) ;
78
+ $mac!( publish) ;
79
+ $mac!( read_manifest) ;
80
+ $mac!( run) ;
81
+ $mac!( rustc) ;
82
+ $mac!( rustdoc) ;
83
+ $mac!( search) ;
84
+ $mac!( test) ;
85
+ $mac!( uninstall) ;
86
+ $mac!( update) ;
87
+ $mac!( verify_project) ;
88
+ $mac!( version) ;
89
+ $mac!( yank) ;
90
+ } )
91
+ }
93
92
94
93
/**
95
94
The top-level `cargo` command handles configuration and project location
@@ -104,7 +103,7 @@ fn execute(flags: Flags, config: &Config) -> CliResult<Option<()>> {
104
103
105
104
if flags. flag_list {
106
105
println ! ( "Installed Commands:" ) ;
107
- for command in list_commands ( ) . into_iter ( ) {
106
+ for command in list_commands ( config ) {
108
107
println ! ( " {}" , command) ;
109
108
} ;
110
109
return Ok ( None )
@@ -143,8 +142,8 @@ fn execute(flags: Flags, config: &Config) -> CliResult<Option<()>> {
143
142
_ => env:: args ( ) . collect ( ) ,
144
143
} ;
145
144
146
- macro_rules! cmd{ ( $name : ident ) => (
147
- if args[ 1 ] == stringify!( $name) . replace( "_" , "-" ) {
145
+ macro_rules! cmd{
146
+ ( $name : ident ) => ( if args[ 1 ] == stringify!( $name) . replace( "_" , "-" ) {
148
147
mod $name;
149
148
config. shell( ) . set_verbose( true ) ;
150
149
let r = cargo:: call_main_without_stdin( $name:: execute, config,
@@ -153,130 +152,100 @@ fn execute(flags: Flags, config: &Config) -> CliResult<Option<()>> {
153
152
false ) ;
154
153
cargo:: process_executed( r, & mut config. shell( ) ) ;
155
154
return Ok ( None )
156
- }
157
- ) }
155
+ } )
156
+ }
158
157
each_subcommand ! ( cmd) ;
159
158
160
- execute_subcommand ( & args[ 1 ] , & args, & mut config . shell ( ) ) ;
159
+ try! ( execute_subcommand ( config , & args[ 1 ] , & args) ) ;
161
160
Ok ( None )
162
161
}
163
162
164
- fn find_closest ( cmd : & str ) -> Option < String > {
165
- let cmds = list_commands ( ) ;
163
+ fn find_closest ( config : & Config , cmd : & str ) -> Option < String > {
164
+ let cmds = list_commands ( config ) ;
166
165
// Only consider candidates with a lev_distance of 3 or less so we don't
167
166
// suggest out-of-the-blue options.
168
167
let mut filtered = cmds. iter ( ) . map ( |c| ( lev_distance ( & c, cmd) , c) )
169
168
. filter ( |& ( d, _) | d < 4 )
170
169
. collect :: < Vec < _ > > ( ) ;
171
170
filtered. sort_by ( |a, b| a. 0 . cmp ( & b. 0 ) ) ;
172
-
173
- if filtered. len ( ) == 0 {
174
- None
175
- } else {
176
- Some ( filtered[ 0 ] . 1 . to_string ( ) )
177
- }
171
+ filtered. get ( 0 ) . map ( |slot| slot. 1 . to_string ( ) )
178
172
}
179
173
180
- fn execute_subcommand ( cmd : & str , args : & [ String ] , shell : & mut MultiShell ) {
181
- let command = match find_command ( cmd) {
182
- Some ( command) => command,
174
+ fn execute_subcommand ( config : & Config ,
175
+ cmd : & str ,
176
+ args : & [ String ] ) -> CargoResult < ( ) > {
177
+ let command_exe = format ! ( "cargo-{}{}" , cmd, env:: consts:: EXE_SUFFIX ) ;
178
+ let path = search_directories ( config)
179
+ . iter ( )
180
+ . map ( |dir| dir. join ( & command_exe) )
181
+ . filter_map ( |dir| fs:: metadata ( & dir) . ok ( ) . map ( |m| ( dir, m) ) )
182
+ . find ( |& ( _, ref meta) | is_executable ( meta) ) ;
183
+ let command = match path {
184
+ Some ( ( command, _) ) => command,
183
185
None => {
184
- let msg = match find_closest ( cmd) {
185
- Some ( closest) => format ! ( "No such subcommand\n \n \t \
186
+ return Err ( human ( match find_closest ( config , cmd) {
187
+ Some ( closest) => format ! ( "no such subcommand\n \n \t \
186
188
Did you mean `{}`?\n ", closest) ,
187
- None => "No such subcommand" . to_string ( )
188
- } ;
189
- return handle_error ( CliError :: new ( & msg, 127 ) , shell)
189
+ None => "no such subcommand" . to_string ( )
190
+ } ) )
190
191
}
191
192
} ;
192
- match Command :: new ( & command) . args ( & args[ 1 ..] ) . status ( ) {
193
- Ok ( ref status) if status. success ( ) => { }
194
- Ok ( ref status) => {
195
- match status. code ( ) {
196
- Some ( code) => handle_error ( CliError :: new ( "" , code) , shell) ,
197
- None => {
198
- let msg = format ! ( "subcommand failed with: {}" , status) ;
199
- handle_error ( CliError :: new ( & msg, 101 ) , shell)
200
- }
201
- }
202
- }
203
- Err ( ref e) if e. kind ( ) == io:: ErrorKind :: NotFound => {
204
- handle_error ( CliError :: new ( "No such subcommand" , 127 ) , shell)
205
- }
206
- Err ( err) => {
207
- let msg = format ! ( "Subcommand failed to run: {}" , err) ;
208
- handle_error ( CliError :: new ( & msg, 127 ) , shell)
209
- }
210
- }
193
+ try!( util:: process ( & command) . args ( & args[ 1 ..] ) . exec ( ) ) ;
194
+ Ok ( ( ) )
211
195
}
212
196
213
197
/// List all runnable commands. find_command should always succeed
214
198
/// if given one of returned command.
215
- fn list_commands ( ) -> BTreeSet < String > {
216
- let command_prefix = "cargo-" ;
199
+ fn list_commands ( config : & Config ) -> BTreeSet < String > {
200
+ let prefix = "cargo-" ;
201
+ let suffix = env:: consts:: EXE_SUFFIX ;
217
202
let mut commands = BTreeSet :: new ( ) ;
218
- for dir in list_command_directory ( ) . iter ( ) {
203
+ for dir in search_directories ( config ) {
219
204
let entries = match fs:: read_dir ( dir) {
220
205
Ok ( entries) => entries,
221
206
_ => continue
222
207
} ;
223
- for entry in entries {
224
- let entry = match entry { Ok ( e) => e, Err ( ..) => continue } ;
225
- let entry = entry. path ( ) ;
226
- let filename = match entry. file_name ( ) . and_then ( |s| s. to_str ( ) ) {
208
+ for entry in entries. filter_map ( |e| e. ok ( ) ) {
209
+ let path = entry. path ( ) ;
210
+ let filename = match path. file_name ( ) . and_then ( |s| s. to_str ( ) ) {
227
211
Some ( filename) => filename,
228
212
_ => continue
229
213
} ;
230
- if filename. starts_with ( command_prefix) &&
231
- filename. ends_with ( env:: consts:: EXE_SUFFIX ) &&
232
- is_executable ( & entry) {
233
- let command = & filename[
234
- command_prefix. len ( ) ..
235
- filename. len ( ) - env:: consts:: EXE_SUFFIX . len ( ) ] ;
236
- commands. insert ( command. to_string ( ) ) ;
214
+ if !filename. starts_with ( prefix) || !filename. ends_with ( suffix) {
215
+ continue
216
+ }
217
+ if let Ok ( meta) = entry. metadata ( ) {
218
+ if is_executable ( & meta) {
219
+ let end = filename. len ( ) - suffix. len ( ) ;
220
+ commands. insert ( filename[ prefix. len ( ) ..end] . to_string ( ) ) ;
221
+ }
237
222
}
238
223
}
239
224
}
240
225
241
- macro_rules! add_cmd{ ( $cmd : ident ) => ( {
242
- commands. insert( stringify!( $cmd) . replace( "_" , "-" ) ) ;
243
- } ) }
226
+ macro_rules! add_cmd {
227
+ ( $cmd : ident ) => ( commands. insert( stringify!( $cmd) . replace( "_" , "-" ) ) )
228
+ }
244
229
each_subcommand ! ( add_cmd) ;
245
230
commands
246
231
}
247
232
248
233
#[ cfg( unix) ]
249
- fn is_executable ( path : & Path ) -> bool {
234
+ fn is_executable ( metadata : & fs :: Metadata ) -> bool {
250
235
use std:: os:: unix:: prelude:: * ;
251
- fs:: metadata ( path) . map ( |m| {
252
- m. permissions ( ) . mode ( ) & 0o001 == 0o001
253
- } ) . unwrap_or ( false )
236
+ metadata. is_file ( ) && metadata. permissions ( ) . mode ( ) & 0o111 != 0
254
237
}
255
238
#[ cfg( windows) ]
256
- fn is_executable ( path : & Path ) -> bool {
257
- fs :: metadata ( path ) . map ( |m| m . is_file ( ) ) . unwrap_or ( false )
239
+ fn is_executable ( metadata : & fs :: Metadata ) -> bool {
240
+ metadata. is_file ( )
258
241
}
259
242
260
- /// Get `Command` to run given command.
261
- fn find_command ( cmd : & str ) -> Option < PathBuf > {
262
- let command_exe = format ! ( "cargo-{}{}" , cmd, env:: consts:: EXE_SUFFIX ) ;
263
- let dirs = list_command_directory ( ) ;
264
- let mut command_paths = dirs. iter ( ) . map ( |dir| dir. join ( & command_exe) ) ;
265
- command_paths. find ( |path| fs:: metadata ( & path) . is_ok ( ) )
266
- }
267
-
268
- /// List candidate locations where subcommands might be installed.
269
- fn list_command_directory ( ) -> Vec < PathBuf > {
270
- let mut dirs = vec ! [ ] ;
271
- if let Ok ( mut path) = env:: current_exe ( ) {
272
- path. pop ( ) ;
273
- dirs. push ( path. join ( "../lib/cargo" ) ) ;
274
- dirs. push ( path) ;
275
- }
243
+ fn search_directories ( config : & Config ) -> Vec < PathBuf > {
244
+ let mut dirs = vec ! [ config. home( ) . join( "bin" ) ] ;
276
245
if let Some ( val) = env:: var_os ( "PATH" ) {
277
246
dirs. extend ( env:: split_paths ( & val) ) ;
278
247
}
279
- dirs
248
+ return dirs
280
249
}
281
250
282
251
fn init_git_transports ( config : & Config ) {
0 commit comments