1
1
use std:: collections:: BTreeMap ;
2
2
use std:: env;
3
3
use std:: fs;
4
+ use std:: fmt;
4
5
use std:: path:: Path ;
5
6
6
7
use serde:: { Deserialize , Deserializer } ;
@@ -23,12 +24,32 @@ pub enum VersionControl { Git, Hg, Pijul, Fossil, NoVcs }
23
24
#[ derive( Debug ) ]
24
25
pub struct NewOptions < ' a > {
25
26
pub version_control : Option < VersionControl > ,
26
- pub bin : bool ,
27
- pub lib : bool ,
27
+ pub kind : NewProjectKind ,
28
28
pub path : & ' a str ,
29
29
pub name : Option < & ' a str > ,
30
30
}
31
31
32
+ #[ derive( Clone , Copy , Debug , PartialEq , Eq ) ]
33
+ pub enum NewProjectKind {
34
+ Bin ,
35
+ Lib ,
36
+ }
37
+
38
+ impl NewProjectKind {
39
+ fn is_bin ( & self ) -> bool {
40
+ * self == NewProjectKind :: Bin
41
+ }
42
+ }
43
+
44
+ impl fmt:: Display for NewProjectKind {
45
+ fn fmt ( & self , f : & mut fmt:: Formatter ) -> fmt:: Result {
46
+ match * self {
47
+ NewProjectKind :: Bin => "binary (application)" ,
48
+ NewProjectKind :: Lib => "library" ,
49
+ } . fmt ( f)
50
+ }
51
+ }
52
+
32
53
struct SourceFileInformation {
33
54
relative_path : String ,
34
55
target_name : String ,
@@ -62,26 +83,21 @@ impl<'de> Deserialize<'de> for VersionControl {
62
83
63
84
impl < ' a > NewOptions < ' a > {
64
85
pub fn new ( version_control : Option < VersionControl > ,
65
- bin : bool ,
66
- lib : bool ,
67
- path : & ' a str ,
68
- name : Option < & ' a str > ) -> NewOptions < ' a > {
69
-
70
- // default to lib
71
- let is_lib = if !bin {
72
- true
73
- }
74
- else {
75
- lib
86
+ bin : bool ,
87
+ lib : bool ,
88
+ path : & ' a str ,
89
+ name : Option < & ' a str > ) -> CargoResult < NewOptions < ' a > > {
90
+
91
+ let kind = match ( bin , lib) {
92
+ ( true , true ) => bail ! ( "can't specify both lib and binary outputs" ) ,
93
+ ( true , false ) => NewProjectKind :: Bin ,
94
+ ( false , true ) => NewProjectKind :: Lib ,
95
+ // default to bin
96
+ ( false , false ) => NewProjectKind :: Bin ,
76
97
} ;
77
98
78
- NewOptions {
79
- version_control : version_control,
80
- bin : bin,
81
- lib : is_lib,
82
- path : path,
83
- name : name,
84
- }
99
+ let opts = NewOptions { version_control, kind, path, name } ;
100
+ Ok ( opts)
85
101
}
86
102
}
87
103
@@ -127,7 +143,7 @@ fn check_name(name: &str, opts: &NewOptions) -> CargoResult<()> {
127
143
"super" , "test" , "trait" , "true" , "type" , "typeof" ,
128
144
"unsafe" , "unsized" , "use" , "virtual" , "where" ,
129
145
"while" , "yield" ] ;
130
- if blacklist. contains ( & name) || ( opts. bin && is_bad_artifact_name ( name) ) {
146
+ if blacklist. contains ( & name) || ( opts. kind . is_bin ( ) && is_bad_artifact_name ( name) ) {
131
147
bail ! ( "The name `{}` cannot be used as a crate name{}" ,
132
148
name,
133
149
name_help)
@@ -269,19 +285,15 @@ pub fn new(opts: &NewOptions, config: &Config) -> CargoResult<()> {
269
285
)
270
286
}
271
287
272
- if opts. lib && opts. bin {
273
- bail ! ( "can't specify both lib and binary outputs" )
274
- }
275
-
276
288
let name = get_name ( & path, opts) ?;
277
289
check_name ( name, opts) ?;
278
290
279
291
let mkopts = MkOptions {
280
292
version_control : opts. version_control ,
281
293
path : & path,
282
294
name : name,
283
- source_files : vec ! [ plan_new_source_file( opts. bin , name. to_string( ) ) ] ,
284
- bin : opts. bin ,
295
+ source_files : vec ! [ plan_new_source_file( opts. kind . is_bin ( ) , name. to_string( ) ) ] ,
296
+ bin : opts. kind . is_bin ( ) ,
285
297
} ;
286
298
287
299
mk ( config, & mkopts) . chain_err ( || {
@@ -299,10 +311,6 @@ pub fn init(opts: &NewOptions, config: &Config) -> CargoResult<()> {
299
311
bail ! ( "`cargo init` cannot be run on existing Cargo projects" )
300
312
}
301
313
302
- if opts. lib && opts. bin {
303
- bail ! ( "can't specify both lib and binary outputs" ) ;
304
- }
305
-
306
314
let name = get_name ( & path, opts) ?;
307
315
check_name ( name, opts) ?;
308
316
@@ -311,7 +319,7 @@ pub fn init(opts: &NewOptions, config: &Config) -> CargoResult<()> {
311
319
detect_source_paths_and_types ( & path, name, & mut src_paths_types) ?;
312
320
313
321
if src_paths_types. is_empty ( ) {
314
- src_paths_types. push ( plan_new_source_file ( opts. bin , name. to_string ( ) ) ) ;
322
+ src_paths_types. push ( plan_new_source_file ( opts. kind . is_bin ( ) , name. to_string ( ) ) ) ;
315
323
} else {
316
324
// --bin option may be ignored if lib.rs or src/lib.rs present
317
325
// Maybe when doing `cargo init --bin` inside a library project stub,
@@ -353,9 +361,9 @@ pub fn init(opts: &NewOptions, config: &Config) -> CargoResult<()> {
353
361
}
354
362
355
363
let mkopts = MkOptions {
356
- version_control : version_control ,
364
+ version_control,
357
365
path : & path,
358
- name : name ,
366
+ name,
359
367
bin : src_paths_types. iter ( ) . any ( |x|x. bin ) ,
360
368
source_files : src_paths_types,
361
369
} ;
0 commit comments