From ba6980a9edd3c13cca6e1d643fae8c43fea730fd Mon Sep 17 00:00:00 2001 From: Jay Oster Date: Wed, 29 May 2024 18:29:37 -0700 Subject: [PATCH] Support tuples and arrays in path parser (#17) --- Cargo.lock | 2 +- Cargo.toml | 2 +- src/ty.rs | 82 ++++++++++++++++++++++++++++++++++++++++++++++++++++-- 3 files changed, 82 insertions(+), 4 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 4872b7f..e7a2ef7 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -4,7 +4,7 @@ version = 3 [[package]] name = "myn" -version = "0.2.1" +version = "0.2.2" dependencies = [ "proc-macro2", ] diff --git a/Cargo.toml b/Cargo.toml index 68c635d..2c663a4 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,7 +1,7 @@ [package] name = "myn" description = "Minimalist Rust syntax parsing for procedural macros" -version = "0.2.1" +version = "0.2.2" authors = ["Jay Oster "] repository = "https://github.com/parasyte/myn" edition = "2021" diff --git a/src/ty.rs b/src/ty.rs index 8a46f2d..39814e1 100644 --- a/src/ty.rs +++ b/src/ty.rs @@ -66,7 +66,11 @@ impl TokenIterExt for TokenIter { while let Some(tree) = self.peek() { match tree { - TokenTree::Punct(punct) if punct.as_char() == ',' && nesting == 0 => break, + TokenTree::Punct(punct) + if [',', ';'].contains(&punct.as_char()) && nesting == 0 => + { + break + } TokenTree::Punct(punct) => { let ch = punct.as_char(); @@ -84,7 +88,43 @@ impl TokenIterExt for TokenIter { span.get_or_insert_with(|| ident.span()); path.push_str(&ident.to_string()); } - _ => return Err(spanned_error("Unexpected token", self.next().as_span())), + TokenTree::Group(group) => { + span.get_or_insert(group.span()); + let mut stream = group.stream().into_token_iter(); + + match group.delimiter() { + Delimiter::Parenthesis => { + // Tuples are comma-separated paths. + path.push('('); + while stream.peek().is_some() { + let (inner, _span) = stream.parse_path()?; + path.push_str(&inner); + if stream.peek().is_some() { + stream.expect_punct(',')?; + path.push_str(", "); + } + } + if path.ends_with(' ') { + path.pop(); + } + path.push(')'); + } + Delimiter::Bracket => { + // Arrays are in `[path; size]` form. + path.push('['); + let (inner, _span) = stream.parse_path()?; + path.push_str(&inner); + stream.expect_punct(';')?; + path.push_str("; "); + path.push_str(&stream.try_lit()?.to_string()); + path.push(']'); + } + _ => return Err(spanned_error("Unexpected token", group.span())), + } + } + TokenTree::Literal(_) => { + return Err(spanned_error("Unexpected token", self.next().as_span())) + } } self.next(); @@ -192,6 +232,44 @@ mod tests { use super::*; use std::str::FromStr; + #[test] + fn test_tokeniter_parse_path() { + let mut input = TokenStream::from_str("foo::bar").unwrap().into_token_iter(); + assert_eq!(input.parse_path().unwrap().0, "foo::bar"); + assert!(input.next().is_none()); + + let mut input = TokenStream::from_str("foo::bar") + .unwrap() + .into_token_iter(); + assert_eq!(input.parse_path().unwrap().0, "foo::bar"); + assert!(input.next().is_none()); + + let mut input = TokenStream::from_str("foo::bar<()>") + .unwrap() + .into_token_iter(); + assert_eq!(input.parse_path().unwrap().0, "foo::bar<()>"); + assert!(input.next().is_none()); + + let mut input = TokenStream::from_str("foo::bar<(foo, bar::baz)>") + .unwrap() + .into_token_iter(); + assert_eq!( + input.parse_path().unwrap().0, + "foo::bar<(foo, bar::baz)>" + ); + assert!(input.next().is_none()); + + let mut input = TokenStream::from_str("foo<(bar, [i32; 4])>") + .unwrap() + .into_token_iter(); + assert_eq!(input.parse_path().unwrap().0, "foo<(bar, [i32; 4])>"); + assert!(input.next().is_none()); + + let mut input = TokenStream::from_str("(u8, )").unwrap().into_token_iter(); + assert_eq!(input.parse_path().unwrap().0, "(u8,)"); + assert!(input.next().is_none()); + } + #[test] fn test_tokeniter_expect_group() { let mut input = TokenStream::from_str("{ foo }").unwrap().into_token_iter();