123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352 |
- #[macro_use]
- mod macros;
- use proc_macro2::{Delimiter, Group, Ident, Punct, Spacing, Span, TokenStream, TokenTree};
- use quote::quote;
- use std::iter::FromIterator;
- use syn::Type;
- #[test]
- fn test_mut_self() {
- syn::parse_str::<Type>("fn(mut self)").unwrap();
- syn::parse_str::<Type>("fn(mut self,)").unwrap();
- syn::parse_str::<Type>("fn(mut self: ())").unwrap();
- syn::parse_str::<Type>("fn(mut self: ...)").unwrap_err();
- syn::parse_str::<Type>("fn(mut self: mut self)").unwrap_err();
- syn::parse_str::<Type>("fn(mut self::T)").unwrap_err();
- }
- #[test]
- fn test_macro_variable_type() {
- // mimics the token stream corresponding to `$ty<T>`
- let tokens = TokenStream::from_iter(vec![
- TokenTree::Group(Group::new(Delimiter::None, quote! { ty })),
- TokenTree::Punct(Punct::new('<', Spacing::Alone)),
- TokenTree::Ident(Ident::new("T", Span::call_site())),
- TokenTree::Punct(Punct::new('>', Spacing::Alone)),
- ]);
- snapshot!(tokens as Type, @r###"
- Type::Path {
- path: Path {
- segments: [
- PathSegment {
- ident: "ty",
- arguments: PathArguments::AngleBracketed {
- args: [
- Type(Type::Path {
- path: Path {
- segments: [
- PathSegment {
- ident: "T",
- arguments: None,
- },
- ],
- },
- }),
- ],
- },
- },
- ],
- },
- }
- "###);
- // mimics the token stream corresponding to `$ty::<T>`
- let tokens = TokenStream::from_iter(vec![
- TokenTree::Group(Group::new(Delimiter::None, quote! { ty })),
- TokenTree::Punct(Punct::new(':', Spacing::Joint)),
- TokenTree::Punct(Punct::new(':', Spacing::Alone)),
- TokenTree::Punct(Punct::new('<', Spacing::Alone)),
- TokenTree::Ident(Ident::new("T", Span::call_site())),
- TokenTree::Punct(Punct::new('>', Spacing::Alone)),
- ]);
- snapshot!(tokens as Type, @r###"
- Type::Path {
- path: Path {
- segments: [
- PathSegment {
- ident: "ty",
- arguments: PathArguments::AngleBracketed {
- colon2_token: Some,
- args: [
- Type(Type::Path {
- path: Path {
- segments: [
- PathSegment {
- ident: "T",
- arguments: None,
- },
- ],
- },
- }),
- ],
- },
- },
- ],
- },
- }
- "###);
- }
- #[test]
- fn test_group_angle_brackets() {
- // mimics the token stream corresponding to `Option<$ty>`
- let tokens = TokenStream::from_iter(vec![
- TokenTree::Ident(Ident::new("Option", Span::call_site())),
- TokenTree::Punct(Punct::new('<', Spacing::Alone)),
- TokenTree::Group(Group::new(Delimiter::None, quote! { Vec<u8> })),
- TokenTree::Punct(Punct::new('>', Spacing::Alone)),
- ]);
- snapshot!(tokens as Type, @r###"
- Type::Path {
- path: Path {
- segments: [
- PathSegment {
- ident: "Option",
- arguments: PathArguments::AngleBracketed {
- args: [
- Type(Type::Group {
- elem: Type::Path {
- path: Path {
- segments: [
- PathSegment {
- ident: "Vec",
- arguments: PathArguments::AngleBracketed {
- args: [
- Type(Type::Path {
- path: Path {
- segments: [
- PathSegment {
- ident: "u8",
- arguments: None,
- },
- ],
- },
- }),
- ],
- },
- },
- ],
- },
- },
- }),
- ],
- },
- },
- ],
- },
- }
- "###);
- }
- #[test]
- fn test_group_colons() {
- // mimics the token stream corresponding to `$ty::Item`
- let tokens = TokenStream::from_iter(vec![
- TokenTree::Group(Group::new(Delimiter::None, quote! { Vec<u8> })),
- TokenTree::Punct(Punct::new(':', Spacing::Joint)),
- TokenTree::Punct(Punct::new(':', Spacing::Alone)),
- TokenTree::Ident(Ident::new("Item", Span::call_site())),
- ]);
- snapshot!(tokens as Type, @r###"
- Type::Path {
- path: Path {
- segments: [
- PathSegment {
- ident: "Vec",
- arguments: PathArguments::AngleBracketed {
- args: [
- Type(Type::Path {
- path: Path {
- segments: [
- PathSegment {
- ident: "u8",
- arguments: None,
- },
- ],
- },
- }),
- ],
- },
- },
- PathSegment {
- ident: "Item",
- arguments: None,
- },
- ],
- },
- }
- "###);
- let tokens = TokenStream::from_iter(vec![
- TokenTree::Group(Group::new(Delimiter::None, quote! { [T] })),
- TokenTree::Punct(Punct::new(':', Spacing::Joint)),
- TokenTree::Punct(Punct::new(':', Spacing::Alone)),
- TokenTree::Ident(Ident::new("Element", Span::call_site())),
- ]);
- snapshot!(tokens as Type, @r###"
- Type::Path {
- qself: Some(QSelf {
- ty: Type::Slice {
- elem: Type::Path {
- path: Path {
- segments: [
- PathSegment {
- ident: "T",
- arguments: None,
- },
- ],
- },
- },
- },
- position: 0,
- }),
- path: Path {
- leading_colon: Some,
- segments: [
- PathSegment {
- ident: "Element",
- arguments: None,
- },
- ],
- },
- }
- "###);
- }
- #[test]
- fn test_trait_object() {
- let tokens = quote!(dyn for<'a> Trait<'a> + 'static);
- snapshot!(tokens as Type, @r###"
- Type::TraitObject {
- dyn_token: Some,
- bounds: [
- Trait(TraitBound {
- modifier: None,
- lifetimes: Some(BoundLifetimes {
- lifetimes: [
- LifetimeDef {
- lifetime: Lifetime {
- ident: "a",
- },
- },
- ],
- }),
- path: Path {
- segments: [
- PathSegment {
- ident: "Trait",
- arguments: PathArguments::AngleBracketed {
- args: [
- Lifetime(Lifetime {
- ident: "a",
- }),
- ],
- },
- },
- ],
- },
- }),
- Lifetime(Lifetime {
- ident: "static",
- }),
- ],
- }
- "###);
- let tokens = quote!(dyn 'a + Trait);
- snapshot!(tokens as Type, @r###"
- Type::TraitObject {
- dyn_token: Some,
- bounds: [
- Lifetime(Lifetime {
- ident: "a",
- }),
- Trait(TraitBound {
- modifier: None,
- path: Path {
- segments: [
- PathSegment {
- ident: "Trait",
- arguments: None,
- },
- ],
- },
- }),
- ],
- }
- "###);
- // None of the following are valid Rust types.
- syn::parse_str::<Type>("for<'a> dyn Trait<'a>").unwrap_err();
- syn::parse_str::<Type>("dyn for<'a> 'a + Trait").unwrap_err();
- }
- #[test]
- fn test_trailing_plus() {
- #[rustfmt::skip]
- let tokens = quote!(impl Trait +);
- snapshot!(tokens as Type, @r###"
- Type::ImplTrait {
- bounds: [
- Trait(TraitBound {
- modifier: None,
- path: Path {
- segments: [
- PathSegment {
- ident: "Trait",
- arguments: None,
- },
- ],
- },
- }),
- ],
- }
- "###);
- #[rustfmt::skip]
- let tokens = quote!(dyn Trait +);
- snapshot!(tokens as Type, @r###"
- Type::TraitObject {
- dyn_token: Some,
- bounds: [
- Trait(TraitBound {
- modifier: None,
- path: Path {
- segments: [
- PathSegment {
- ident: "Trait",
- arguments: None,
- },
- ],
- },
- }),
- ],
- }
- "###);
- #[rustfmt::skip]
- let tokens = quote!(Trait +);
- snapshot!(tokens as Type, @r###"
- Type::TraitObject {
- bounds: [
- Trait(TraitBound {
- modifier: None,
- path: Path {
- segments: [
- PathSegment {
- ident: "Trait",
- arguments: None,
- },
- ],
- },
- }),
- ],
- }
- "###);
- }
|