| // vim: tw=80 |
| use proc_macro2::Span; |
| use quote::{format_ident, quote, ToTokens}; |
| use std::{ |
| collections::hash_map::DefaultHasher, |
| hash::{Hash, Hasher}, |
| }; |
| use syn::{spanned::Spanned, *}; |
| |
| use crate::{ |
| compile_error, |
| mock_function::{self, MockFunction}, |
| AttrFormatter, |
| }; |
| |
| pub(crate) struct MockTrait { |
| pub attrs: Vec<Attribute>, |
| pub consts: Vec<ImplItemConst>, |
| pub generics: Generics, |
| pub methods: Vec<MockFunction>, |
| /// Internally-used name of the trait used. |
| pub ss_name: Ident, |
| /// Fully-qualified name of the trait |
| pub trait_path: Path, |
| /// Path on which the trait is implemented. Usually will be the same as |
| /// structname, but might include concrete generic parameters. |
| self_path: PathSegment, |
| pub types: Vec<ImplItemType>, |
| pub unsafety: Option<Token![unsafe]>, |
| } |
| |
| impl MockTrait { |
| fn ss_name_priv(trait_path: &Path) -> Ident { |
| let path_args = &trait_path.segments.last().unwrap().arguments; |
| if path_args.is_empty() { |
| // Skip the hashing step for easie debugging of generated code |
| format_ident!("{}", trait_path.segments.last().unwrap().ident) |
| } else { |
| // Hash the path args to permit mocking structs that implement |
| // multiple traits distinguished only by their path args |
| let mut hasher = DefaultHasher::new(); |
| path_args.hash(&mut hasher); |
| format_ident!( |
| "{}_{}", |
| trait_path.segments.last().unwrap().ident, |
| hasher.finish() |
| ) |
| } |
| } |
| |
| pub fn ss_name(&self) -> &Ident { |
| &self.ss_name |
| } |
| |
| /// Create a new MockTrait |
| /// |
| /// # Arguments |
| /// * `structname` - name of the struct that implements this trait |
| /// * `struct_generics` - Generics of the parent structure |
| /// * `impl_` - Mockable ItemImpl for a trait |
| /// * `vis` - Visibility of the struct |
| pub fn new( |
| structname: &Ident, |
| struct_generics: &Generics, |
| impl_: ItemImpl, |
| vis: &Visibility, |
| ) -> Self { |
| let mut consts = Vec::new(); |
| let mut methods = Vec::new(); |
| let mut types = Vec::new(); |
| let trait_path = if let Some((_, path, _)) = impl_.trait_ { |
| path |
| } else { |
| compile_error(impl_.span(), "impl block must implement a trait"); |
| Path::from(format_ident!("__mockall_invalid")) |
| }; |
| let ss_name = MockTrait::ss_name_priv(&trait_path); |
| let self_path = match *impl_.self_ty { |
| Type::Path(mut type_path) => type_path.path.segments.pop().unwrap().into_value(), |
| x => { |
| compile_error( |
| x.span(), |
| "mockall_derive only supports mocking traits and structs", |
| ); |
| PathSegment::from(Ident::new("", Span::call_site())) |
| } |
| }; |
| |
| for ii in impl_.items.into_iter() { |
| match ii { |
| ImplItem::Const(iic) => { |
| consts.push(iic); |
| } |
| ImplItem::Method(iim) => { |
| let mf = mock_function::Builder::new(&iim.sig, vis) |
| .attrs(&iim.attrs) |
| .levels(2) |
| .call_levels(0) |
| .struct_(structname) |
| .struct_generics(struct_generics) |
| .trait_(&ss_name) |
| .build(); |
| methods.push(mf); |
| } |
| ImplItem::Type(iit) => { |
| types.push(iit); |
| } |
| _ => { |
| compile_error(ii.span(), "This impl item is not yet supported by MockAll"); |
| } |
| } |
| } |
| MockTrait { |
| attrs: impl_.attrs, |
| consts, |
| generics: impl_.generics, |
| methods, |
| ss_name, |
| trait_path, |
| self_path, |
| types, |
| unsafety: impl_.unsafety, |
| } |
| } |
| |
| /// Generate code for the trait implementation on the mock struct |
| /// |
| /// # Arguments |
| /// |
| /// * `modname`: Name of the parent struct's private module |
| // Supplying modname is an unfortunately hack. Ideally MockTrait |
| // wouldn't need to know that. |
| pub fn trait_impl(&self, modname: &Ident) -> impl ToTokens { |
| let trait_impl_attrs = &self.attrs; |
| let impl_attrs = AttrFormatter::new(&self.attrs) |
| .async_trait(false) |
| .doc(false) |
| .format(); |
| let (ig, _tg, wc) = self.generics.split_for_impl(); |
| let consts = &self.consts; |
| let path_args = &self.self_path.arguments; |
| let calls = self |
| .methods |
| .iter() |
| .map(|meth| meth.call(Some(modname))) |
| .collect::<Vec<_>>(); |
| let contexts = self |
| .methods |
| .iter() |
| .filter(|meth| meth.is_static()) |
| .map(|meth| meth.context_fn(Some(modname))) |
| .collect::<Vec<_>>(); |
| let expects = self |
| .methods |
| .iter() |
| .filter(|meth| !meth.is_static()) |
| .map(|meth| { |
| if meth.is_method_generic() { |
| // Specific impls with generic methods are TODO. |
| meth.expect(modname, None) |
| } else { |
| meth.expect(modname, Some(path_args)) |
| } |
| }) |
| .collect::<Vec<_>>(); |
| let trait_path = &self.trait_path; |
| let self_path = &self.self_path; |
| let types = &self.types; |
| let unsafety = &self.unsafety; |
| quote!( |
| #(#trait_impl_attrs)* |
| #unsafety impl #ig #trait_path for #self_path #wc { |
| #(#consts)* |
| #(#types)* |
| #(#calls)* |
| } |
| #(#impl_attrs)* |
| impl #ig #self_path #wc { |
| #(#expects)* |
| #(#contexts)* |
| } |
| ) |
| } |
| } |