• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2018 Guillaume Pinot (@TeXitoi) <texitoi@texitoi.eu>,
2 // Kevin Knapp (@kbknapp) <kbknapp@gmail.com>, and
3 // Ana Hobden (@hoverbear) <operator@hoverbear.org>
4 //
5 // Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6 // http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7 // <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8 // option. This file may not be copied, modified, or distributed
9 // except according to those terms.
10 //
11 // This work was derived from Structopt (https://github.com/TeXitoi/structopt)
12 // commit#ea76fa1b1b273e65e3b0b1046643715b49bec51f which is licensed under the
13 // MIT/Apache 2.0 license.
14 
15 use proc_macro2::{Ident, Span, TokenStream};
16 use proc_macro_error::{abort, abort_call_site};
17 use quote::{format_ident, quote, quote_spanned};
18 use syn::ext::IdentExt;
19 use syn::{
20     punctuated::Punctuated, spanned::Spanned, token::Comma, Data, DataStruct, DeriveInput, Field,
21     Fields, Generics,
22 };
23 
24 use crate::dummies;
25 use crate::item::{Item, Kind, Name};
26 use crate::utils::{inner_type, sub_type, Sp, Ty};
27 
derive_args(input: &DeriveInput) -> TokenStream28 pub fn derive_args(input: &DeriveInput) -> TokenStream {
29     let ident = &input.ident;
30 
31     dummies::args(ident);
32 
33     match input.data {
34         Data::Struct(DataStruct {
35             fields: Fields::Named(ref fields),
36             ..
37         }) => {
38             let name = Name::Derived(ident.clone());
39             let item = Item::from_args_struct(input, name);
40             let fields = fields
41                 .named
42                 .iter()
43                 .map(|field| {
44                     let item = Item::from_args_field(field, item.casing(), item.env_casing());
45                     (field, item)
46                 })
47                 .collect::<Vec<_>>();
48             gen_for_struct(&item, ident, &input.generics, &fields)
49         }
50         Data::Struct(DataStruct {
51             fields: Fields::Unit,
52             ..
53         }) => {
54             let name = Name::Derived(ident.clone());
55             let item = Item::from_args_struct(input, name);
56             let fields = Punctuated::<Field, Comma>::new();
57             let fields = fields
58                 .iter()
59                 .map(|field| {
60                     let item = Item::from_args_field(field, item.casing(), item.env_casing());
61                     (field, item)
62                 })
63                 .collect::<Vec<_>>();
64             gen_for_struct(&item, ident, &input.generics, &fields)
65         }
66         _ => abort_call_site!("`#[derive(Args)]` only supports non-tuple structs"),
67     }
68 }
69 
gen_for_struct( item: &Item, item_name: &Ident, generics: &Generics, fields: &[(&Field, Item)], ) -> TokenStream70 pub fn gen_for_struct(
71     item: &Item,
72     item_name: &Ident,
73     generics: &Generics,
74     fields: &[(&Field, Item)],
75 ) -> TokenStream {
76     if !matches!(&*item.kind(), Kind::Command(_)) {
77         abort! { item.kind().span(),
78             "`{}` cannot be used with `command`",
79             item.kind().name(),
80         }
81     }
82 
83     let (impl_generics, ty_generics, where_clause) = generics.split_for_impl();
84 
85     let constructor = gen_constructor(fields);
86     let updater = gen_updater(fields, true);
87     let raw_deprecated = raw_deprecated();
88 
89     let app_var = Ident::new("__clap_app", Span::call_site());
90     let augmentation = gen_augment(fields, &app_var, item, false);
91     let augmentation_update = gen_augment(fields, &app_var, item, true);
92 
93     let group_id = if item.skip_group() {
94         quote!(None)
95     } else {
96         let group_id = item.ident().unraw().to_string();
97         quote!(Some(clap::Id::from(#group_id)))
98     };
99 
100     quote! {
101         #[allow(dead_code, unreachable_code, unused_variables, unused_braces)]
102         #[allow(
103             clippy::style,
104             clippy::complexity,
105             clippy::pedantic,
106             clippy::restriction,
107             clippy::perf,
108             clippy::deprecated,
109             clippy::nursery,
110             clippy::cargo,
111             clippy::suspicious_else_formatting,
112         )]
113         #[deny(clippy::correctness)]
114         impl #impl_generics clap::FromArgMatches for #item_name #ty_generics #where_clause {
115             fn from_arg_matches(__clap_arg_matches: &clap::ArgMatches) -> ::std::result::Result<Self, clap::Error> {
116                 Self::from_arg_matches_mut(&mut __clap_arg_matches.clone())
117             }
118 
119             fn from_arg_matches_mut(__clap_arg_matches: &mut clap::ArgMatches) -> ::std::result::Result<Self, clap::Error> {
120                 #raw_deprecated
121                 let v = #item_name #constructor;
122                 ::std::result::Result::Ok(v)
123             }
124 
125             fn update_from_arg_matches(&mut self, __clap_arg_matches: &clap::ArgMatches) -> ::std::result::Result<(), clap::Error> {
126                 self.update_from_arg_matches_mut(&mut __clap_arg_matches.clone())
127             }
128 
129             fn update_from_arg_matches_mut(&mut self, __clap_arg_matches: &mut clap::ArgMatches) -> ::std::result::Result<(), clap::Error> {
130                 #raw_deprecated
131                 #updater
132                 ::std::result::Result::Ok(())
133             }
134         }
135 
136         #[allow(dead_code, unreachable_code, unused_variables, unused_braces)]
137         #[allow(
138             clippy::style,
139             clippy::complexity,
140             clippy::pedantic,
141             clippy::restriction,
142             clippy::perf,
143             clippy::deprecated,
144             clippy::nursery,
145             clippy::cargo,
146             clippy::suspicious_else_formatting,
147         )]
148         #[deny(clippy::correctness)]
149         impl #impl_generics clap::Args for #item_name #ty_generics #where_clause {
150             fn group_id() -> Option<clap::Id> {
151                 #group_id
152             }
153             fn augment_args<'b>(#app_var: clap::Command) -> clap::Command {
154                 #augmentation
155             }
156             fn augment_args_for_update<'b>(#app_var: clap::Command) -> clap::Command {
157                 #augmentation_update
158             }
159         }
160     }
161 }
162 
163 /// Generate a block of code to add arguments/subcommands corresponding to
164 /// the `fields` to an cmd.
gen_augment( fields: &[(&Field, Item)], app_var: &Ident, parent_item: &Item, override_required: bool, ) -> TokenStream165 pub fn gen_augment(
166     fields: &[(&Field, Item)],
167     app_var: &Ident,
168     parent_item: &Item,
169     override_required: bool,
170 ) -> TokenStream {
171     let mut subcommand_specified = false;
172     let args = fields.iter().filter_map(|(field, item)| {
173         let kind = item.kind();
174         match &*kind {
175             Kind::Command(_)
176             | Kind::Value
177             | Kind::Skip(_, _)
178             | Kind::FromGlobal(_)
179             | Kind::ExternalSubcommand => None,
180             Kind::Subcommand(ty) => {
181                 if subcommand_specified {
182                     abort!(field.span(), "`#[command(subcommand)]` can only be used once per container");
183                 }
184                 subcommand_specified = true;
185 
186                 let subcmd_type = match (**ty, sub_type(&field.ty)) {
187                     (Ty::Option, Some(sub_type)) => sub_type,
188                     _ => &field.ty,
189                 };
190                 let implicit_methods = if **ty == Ty::Option {
191                     quote!()
192                 } else {
193                     quote_spanned! { kind.span()=>
194                         .subcommand_required(true)
195                         .arg_required_else_help(true)
196                     }
197                 };
198 
199                 let override_methods = if override_required {
200                     quote_spanned! { kind.span()=>
201                         .subcommand_required(false)
202                         .arg_required_else_help(false)
203                     }
204                 } else {
205                     quote!()
206                 };
207 
208                 Some(quote! {
209                     let #app_var = <#subcmd_type as clap::Subcommand>::augment_subcommands( #app_var );
210                     let #app_var = #app_var
211                         #implicit_methods
212                         #override_methods;
213                 })
214             }
215             Kind::Flatten(ty) => {
216                 let inner_type = match (**ty, sub_type(&field.ty)) {
217                     (Ty::Option, Some(sub_type)) => sub_type,
218                     _ => &field.ty,
219                 };
220 
221                 let next_help_heading = item.next_help_heading();
222                 let next_display_order = item.next_display_order();
223                 if override_required {
224                     Some(quote_spanned! { kind.span()=>
225                         let #app_var = #app_var
226                             #next_help_heading
227                             #next_display_order;
228                         let #app_var = <#inner_type as clap::Args>::augment_args_for_update(#app_var);
229                     })
230                 } else {
231                     Some(quote_spanned! { kind.span()=>
232                         let #app_var = #app_var
233                             #next_help_heading
234                             #next_display_order;
235                         let #app_var = <#inner_type as clap::Args>::augment_args(#app_var);
236                     })
237                 }
238             }
239             Kind::Arg(ty) => {
240                 let value_parser = item.value_parser(&field.ty);
241                 let action = item.action(&field.ty);
242                 let value_name = item.value_name();
243 
244                 let implicit_methods = match **ty {
245                     Ty::Unit => {
246                         // Leaving out `value_parser` as it will always fail
247                         quote_spanned! { ty.span()=>
248                             .value_name(#value_name)
249                             #action
250                         }
251                     }
252                     Ty::Option => {
253                         quote_spanned! { ty.span()=>
254                             .value_name(#value_name)
255                             #value_parser
256                             #action
257                         }
258                     }
259 
260                     Ty::OptionOption => quote_spanned! { ty.span()=>
261                         .value_name(#value_name)
262                         .num_args(0..=1)
263                         #value_parser
264                         #action
265                     },
266 
267                     Ty::OptionVec => {
268                         if item.is_positional() {
269                             quote_spanned! { ty.span()=>
270                                 .value_name(#value_name)
271                                 .num_args(1..)  // action won't be sufficient for getting multiple
272                                 #value_parser
273                                 #action
274                             }
275                         } else {
276                             quote_spanned! { ty.span()=>
277                                 .value_name(#value_name)
278                                 #value_parser
279                                 #action
280                             }
281                         }
282                     }
283 
284                     Ty::Vec => {
285                         if item.is_positional() {
286                             quote_spanned! { ty.span()=>
287                                 .value_name(#value_name)
288                                 .num_args(1..)  // action won't be sufficient for getting multiple
289                                 #value_parser
290                                 #action
291                             }
292                         } else {
293                             quote_spanned! { ty.span()=>
294                                 .value_name(#value_name)
295                                 #value_parser
296                                 #action
297                             }
298                         }
299                     }
300 
301                     Ty::VecVec | Ty::OptionVecVec => {
302                         quote_spanned! { ty.span() =>
303                             .value_name(#value_name)
304                             #value_parser
305                             #action
306                         }
307                     }
308 
309                     Ty::Other => {
310                         let required = item.find_default_method().is_none();
311                         // `ArgAction::takes_values` is assuming `ArgAction::default_value` will be
312                         // set though that won't always be true but this should be good enough,
313                         // otherwise we'll report an "arg required" error when unwrapping.
314                         let action_value = action.args();
315                         quote_spanned! { ty.span()=>
316                             .value_name(#value_name)
317                             .required(#required && #action_value.takes_values())
318                             #value_parser
319                             #action
320                         }
321                     }
322                 };
323 
324                 let id = item.id();
325                 let explicit_methods = item.field_methods();
326                 let deprecations = if !override_required {
327                     item.deprecations()
328                 } else {
329                     quote!()
330                 };
331                 let override_methods = if override_required {
332                     quote_spanned! { kind.span()=>
333                         .required(false)
334                     }
335                 } else {
336                     quote!()
337                 };
338 
339                 Some(quote_spanned! { field.span()=>
340                     let #app_var = #app_var.arg({
341                         #deprecations
342 
343                         #[allow(deprecated)]
344                         let arg = clap::Arg::new(#id)
345                             #implicit_methods;
346 
347                         let arg = arg
348                             #explicit_methods;
349 
350                         let arg = arg
351                             #override_methods;
352 
353                         arg
354                     });
355                 })
356             }
357         }
358     });
359 
360     let deprecations = if !override_required {
361         parent_item.deprecations()
362     } else {
363         quote!()
364     };
365     let initial_app_methods = parent_item.initial_top_level_methods();
366     let final_app_methods = parent_item.final_top_level_methods();
367     let group_app_methods = if parent_item.skip_group() {
368         quote!()
369     } else {
370         let group_id = parent_item.ident().unraw().to_string();
371         let literal_group_members = fields
372             .iter()
373             .filter_map(|(_field, item)| {
374                 let kind = item.kind();
375                 if matches!(*kind, Kind::Arg(_)) {
376                     Some(item.id())
377                 } else {
378                     None
379                 }
380             })
381             .collect::<Vec<_>>();
382         let literal_group_members_len = literal_group_members.len();
383         let mut literal_group_members = quote! {{
384             let members: [clap::Id; #literal_group_members_len] = [#( clap::Id::from(#literal_group_members) ),* ];
385             members
386         }};
387         // HACK: Validation isn't ready yet for nested arg groups, so just don't populate the group in
388         // that situation
389         let possible_group_members_len = fields
390             .iter()
391             .filter(|(_field, item)| {
392                 let kind = item.kind();
393                 matches!(*kind, Kind::Flatten(_))
394             })
395             .count();
396         if 0 < possible_group_members_len {
397             literal_group_members = quote! {{
398                 let members: [clap::Id; 0] = [];
399                 members
400             }};
401         }
402 
403         quote!(
404             .group(
405                 clap::ArgGroup::new(#group_id)
406                     .multiple(true)
407                     .args(#literal_group_members)
408             )
409         )
410     };
411     quote! {{
412         #deprecations
413         let #app_var = #app_var
414             #initial_app_methods
415             #group_app_methods
416             ;
417         #( #args )*
418         #app_var #final_app_methods
419     }}
420 }
421 
gen_constructor(fields: &[(&Field, Item)]) -> TokenStream422 pub fn gen_constructor(fields: &[(&Field, Item)]) -> TokenStream {
423     let fields = fields.iter().map(|(field, item)| {
424         let field_name = field.ident.as_ref().unwrap();
425         let kind = item.kind();
426         let arg_matches = format_ident!("__clap_arg_matches");
427         match &*kind {
428             Kind::Command(_)
429             | Kind::Value
430             | Kind::ExternalSubcommand => {
431                 abort! { kind.span(),
432                     "`{}` cannot be used with `arg`",
433                     kind.name(),
434                 }
435             }
436             Kind::Subcommand(ty) => {
437                 let subcmd_type = match (**ty, sub_type(&field.ty)) {
438                     (Ty::Option, Some(sub_type)) => sub_type,
439                     _ => &field.ty,
440                 };
441                 match **ty {
442                     Ty::Option => {
443                         quote_spanned! { kind.span()=>
444                             #field_name: {
445                                 if #arg_matches.subcommand_name().map(<#subcmd_type as clap::Subcommand>::has_subcommand).unwrap_or(false) {
446                                     Some(<#subcmd_type as clap::FromArgMatches>::from_arg_matches_mut(#arg_matches)?)
447                                 } else {
448                                     None
449                                 }
450                             }
451                         }
452                     },
453                     Ty::Other => {
454                         quote_spanned! { kind.span()=>
455                             #field_name: {
456                                 <#subcmd_type as clap::FromArgMatches>::from_arg_matches_mut(#arg_matches)?
457                             }
458                         }
459                     },
460                     Ty::Unit |
461                     Ty::Vec |
462                     Ty::OptionOption |
463                     Ty::OptionVec |
464                     Ty::VecVec |
465                     Ty::OptionVecVec => {
466                         abort!(
467                             ty.span(),
468                             "{} types are not supported for subcommand",
469                             ty.as_str()
470                         );
471                     }
472                 }
473             }
474 
475             Kind::Flatten(ty) => {
476                 let inner_type = match (**ty, sub_type(&field.ty)) {
477                     (Ty::Option, Some(sub_type)) => sub_type,
478                     _ => &field.ty,
479                 };
480                 match **ty {
481                     Ty::Other => {
482                         quote_spanned! { kind.span()=>
483                             #field_name: <#inner_type as clap::FromArgMatches>::from_arg_matches_mut(#arg_matches)?
484                         }
485                     },
486                     Ty::Option => {
487                         quote_spanned! { kind.span()=>
488                             #field_name: {
489                                 let group_id = <#inner_type as clap::Args>::group_id()
490                                     .expect("`#[arg(flatten)]`ed field type implements `Args::group_id`");
491                                 if #arg_matches.contains_id(group_id.as_str()) {
492                                     Some(
493                                         <#inner_type as clap::FromArgMatches>::from_arg_matches_mut(#arg_matches)?
494                                     )
495                                 } else {
496                                     None
497                                 }
498                             }
499                         }
500                     },
501                     Ty::Unit |
502                     Ty::Vec |
503                     Ty::OptionOption |
504                     Ty::OptionVec |
505                     Ty::VecVec |
506                     Ty::OptionVecVec => {
507                         abort!(
508                             ty.span(),
509                             "{} types are not supported for flatten",
510                             ty.as_str()
511                         );
512                     }
513                 }
514             },
515 
516             Kind::Skip(val, _) => match val {
517                 None => quote_spanned!(kind.span()=> #field_name: Default::default()),
518                 Some(val) => quote_spanned!(kind.span()=> #field_name: (#val).into()),
519             },
520 
521             Kind::Arg(ty) | Kind::FromGlobal(ty) => {
522                 gen_parsers(item, ty, field_name, field, None)
523             }
524         }
525     });
526 
527     quote! {{
528         #( #fields ),*
529     }}
530 }
531 
gen_updater(fields: &[(&Field, Item)], use_self: bool) -> TokenStream532 pub fn gen_updater(fields: &[(&Field, Item)], use_self: bool) -> TokenStream {
533     let fields = fields.iter().map(|(field, item)| {
534         let field_name = field.ident.as_ref().unwrap();
535         let kind = item.kind();
536 
537         let access = if use_self {
538             quote! {
539                 #[allow(non_snake_case)]
540                 let #field_name = &mut self.#field_name;
541             }
542         } else {
543             quote!()
544         };
545         let arg_matches = format_ident!("__clap_arg_matches");
546 
547         match &*kind {
548             Kind::Command(_)
549             | Kind::Value
550             | Kind::ExternalSubcommand => {
551                 abort! { kind.span(),
552                     "`{}` cannot be used with `arg`",
553                     kind.name(),
554                 }
555             }
556             Kind::Subcommand(ty) => {
557                 let subcmd_type = match (**ty, sub_type(&field.ty)) {
558                     (Ty::Option, Some(sub_type)) => sub_type,
559                     _ => &field.ty,
560                 };
561 
562                 let updater = quote_spanned! { ty.span()=>
563                     <#subcmd_type as clap::FromArgMatches>::update_from_arg_matches_mut(#field_name, #arg_matches)?;
564                 };
565 
566                 let updater = match **ty {
567                     Ty::Option => quote_spanned! { kind.span()=>
568                         if let Some(#field_name) = #field_name.as_mut() {
569                             #updater
570                         } else {
571                             *#field_name = Some(<#subcmd_type as clap::FromArgMatches>::from_arg_matches_mut(
572                                 #arg_matches
573                             )?);
574                         }
575                     },
576                     _ => quote_spanned! { kind.span()=>
577                         #updater
578                     },
579                 };
580 
581                 quote_spanned! { kind.span()=>
582                     {
583                         #access
584                         #updater
585                     }
586                 }
587             }
588 
589             Kind::Flatten(ty) => {
590                 let inner_type = match (**ty, sub_type(&field.ty)) {
591                     (Ty::Option, Some(sub_type)) => sub_type,
592                     _ => &field.ty,
593                 };
594 
595                 let updater = quote_spanned! { ty.span()=>
596                     <#inner_type as clap::FromArgMatches>::update_from_arg_matches_mut(#field_name, #arg_matches)?;
597                 };
598 
599                 let updater = match **ty {
600                     Ty::Option => quote_spanned! { kind.span()=>
601                         if let Some(#field_name) = #field_name.as_mut() {
602                             #updater
603                         } else {
604                             *#field_name = Some(<#inner_type as clap::FromArgMatches>::from_arg_matches_mut(
605                                 #arg_matches
606                             )?);
607                         }
608                     },
609                     _ => quote_spanned! { kind.span()=>
610                         #updater
611                     },
612                 };
613 
614                 quote_spanned! { kind.span()=>
615                     {
616                         #access
617                         #updater
618                     }
619                 }
620             },
621 
622             Kind::Skip(_, _) => quote!(),
623 
624             Kind::Arg(ty) | Kind::FromGlobal(ty) => gen_parsers(item, ty, field_name, field, Some(&access)),
625         }
626     });
627 
628     quote! {
629         #( #fields )*
630     }
631 }
632 
gen_parsers( item: &Item, ty: &Sp<Ty>, field_name: &Ident, field: &Field, update: Option<&TokenStream>, ) -> TokenStream633 fn gen_parsers(
634     item: &Item,
635     ty: &Sp<Ty>,
636     field_name: &Ident,
637     field: &Field,
638     update: Option<&TokenStream>,
639 ) -> TokenStream {
640     let span = ty.span();
641     let convert_type = inner_type(&field.ty);
642     let id = item.id();
643     let get_one = quote_spanned!(span=> remove_one::<#convert_type>);
644     let get_many = quote_spanned!(span=> remove_many::<#convert_type>);
645     let get_occurrences = quote_spanned!(span=> remove_occurrences::<#convert_type>);
646 
647     // Give this identifier the same hygiene
648     // as the `arg_matches` parameter definition. This
649     // allows us to refer to `arg_matches` within a `quote_spanned` block
650     let arg_matches = format_ident!("__clap_arg_matches");
651 
652     let field_value = match **ty {
653         Ty::Unit => {
654             quote_spanned! { ty.span()=>
655                 ()
656             }
657         }
658 
659         Ty::Option => {
660             quote_spanned! { ty.span()=>
661                 #arg_matches.#get_one(#id)
662             }
663         }
664 
665         Ty::OptionOption => quote_spanned! { ty.span()=>
666             if #arg_matches.contains_id(#id) {
667                 Some(
668                     #arg_matches.#get_one(#id)
669                 )
670             } else {
671                 None
672             }
673         },
674 
675         Ty::OptionVec => quote_spanned! { ty.span()=>
676             if #arg_matches.contains_id(#id) {
677                 Some(#arg_matches.#get_many(#id)
678                     .map(|v| v.collect::<Vec<_>>())
679                     .unwrap_or_else(Vec::new))
680             } else {
681                 None
682             }
683         },
684 
685         Ty::Vec => {
686             quote_spanned! { ty.span()=>
687                 #arg_matches.#get_many(#id)
688                     .map(|v| v.collect::<Vec<_>>())
689                     .unwrap_or_else(Vec::new)
690             }
691         }
692 
693         Ty::VecVec => quote_spanned! { ty.span()=>
694             #arg_matches.#get_occurrences(#id)
695                 .map(|g| g.map(::std::iter::Iterator::collect).collect::<Vec<Vec<_>>>())
696                 .unwrap_or_else(Vec::new)
697         },
698 
699         Ty::OptionVecVec => quote_spanned! { ty.span()=>
700             #arg_matches.#get_occurrences(#id)
701                 .map(|g| g.map(::std::iter::Iterator::collect).collect::<Vec<Vec<_>>>())
702         },
703 
704         Ty::Other => {
705             quote_spanned! { ty.span()=>
706                 #arg_matches.#get_one(#id)
707                     .ok_or_else(|| clap::Error::raw(clap::error::ErrorKind::MissingRequiredArgument, format!("The following required argument was not provided: {}", #id)))?
708             }
709         }
710     };
711 
712     if let Some(access) = update {
713         quote_spanned! { field.span()=>
714             if #arg_matches.contains_id(#id) {
715                 #access
716                 *#field_name = #field_value
717             }
718         }
719     } else {
720         quote_spanned!(field.span()=> #field_name: #field_value )
721     }
722 }
723 
724 #[cfg(feature = "raw-deprecated")]
raw_deprecated() -> TokenStream725 pub fn raw_deprecated() -> TokenStream {
726     quote! {}
727 }
728 
729 #[cfg(not(feature = "raw-deprecated"))]
raw_deprecated() -> TokenStream730 pub fn raw_deprecated() -> TokenStream {
731     quote! {
732         #![allow(deprecated)]  // Assuming any deprecation in here will be related to a deprecation in `Args`
733 
734     }
735 }
736