| //! utils used in proc-macro tests |
| |
| use expect_test::Expect; |
| use span::{ |
| EditionedFileId, FIXUP_ERASED_FILE_AST_ID_MARKER, FileId, ROOT_ERASED_FILE_AST_ID, Span, |
| SpanAnchor, SyntaxContext, TokenId, |
| }; |
| use tt::TextRange; |
| |
| use crate::{EnvSnapshot, ProcMacroSrv, dylib, proc_macro_test_dylib_path}; |
| |
| fn parse_string(call_site: TokenId, src: &str) -> crate::server_impl::TokenStream<TokenId> { |
| crate::server_impl::TokenStream::with_subtree(crate::server_impl::TopSubtree( |
| syntax_bridge::parse_to_token_tree_static_span(span::Edition::CURRENT, call_site, src) |
| .unwrap() |
| .0 |
| .into_vec(), |
| )) |
| } |
| |
| fn parse_string_spanned( |
| anchor: SpanAnchor, |
| call_site: SyntaxContext, |
| src: &str, |
| ) -> crate::server_impl::TokenStream<Span> { |
| crate::server_impl::TokenStream::with_subtree(crate::server_impl::TopSubtree( |
| syntax_bridge::parse_to_token_tree(span::Edition::CURRENT, anchor, call_site, src) |
| .unwrap() |
| .0 |
| .into_vec(), |
| )) |
| } |
| |
| pub fn assert_expand( |
| macro_name: &str, |
| #[rust_analyzer::rust_fixture] ra_fixture: &str, |
| expect: Expect, |
| expect_spanned: Expect, |
| ) { |
| assert_expand_impl(macro_name, ra_fixture, None, expect, expect_spanned); |
| } |
| |
| pub fn assert_expand_attr( |
| macro_name: &str, |
| #[rust_analyzer::rust_fixture] ra_fixture: &str, |
| attr_args: &str, |
| expect: Expect, |
| expect_spanned: Expect, |
| ) { |
| assert_expand_impl(macro_name, ra_fixture, Some(attr_args), expect, expect_spanned); |
| } |
| |
| fn assert_expand_impl( |
| macro_name: &str, |
| input: &str, |
| attr: Option<&str>, |
| expect: Expect, |
| expect_spanned: Expect, |
| ) { |
| let path = proc_macro_test_dylib_path(); |
| let expander = dylib::Expander::new(&path).unwrap(); |
| |
| let def_site = TokenId(0); |
| let call_site = TokenId(1); |
| let mixed_site = TokenId(2); |
| let input_ts = parse_string(call_site, input).into_subtree(call_site); |
| let attr_ts = attr.map(|attr| parse_string(call_site, attr).into_subtree(call_site)); |
| let input_ts_string = format!("{input_ts:?}"); |
| let attr_ts_string = attr_ts.as_ref().map(|it| format!("{it:?}")); |
| |
| let res = expander |
| .expand( |
| macro_name, |
| input_ts, |
| attr_ts, |
| def_site, |
| call_site, |
| mixed_site, |
| FIXUP_ERASED_FILE_AST_ID_MARKER, |
| ) |
| .unwrap(); |
| expect.assert_eq(&format!( |
| "{input_ts_string}\n\n{}\n\n{res:?}", |
| attr_ts_string.unwrap_or_default() |
| )); |
| |
| let def_site = Span { |
| range: TextRange::new(0.into(), 150.into()), |
| anchor: SpanAnchor { |
| file_id: EditionedFileId::current_edition(FileId::from_raw(41)), |
| ast_id: ROOT_ERASED_FILE_AST_ID, |
| }, |
| ctx: SyntaxContext::root(span::Edition::CURRENT), |
| }; |
| let call_site = Span { |
| range: TextRange::new(0.into(), 100.into()), |
| anchor: SpanAnchor { |
| file_id: EditionedFileId::current_edition(FileId::from_raw(42)), |
| ast_id: ROOT_ERASED_FILE_AST_ID, |
| }, |
| ctx: SyntaxContext::root(span::Edition::CURRENT), |
| }; |
| let mixed_site = call_site; |
| |
| let fixture = |
| parse_string_spanned(call_site.anchor, call_site.ctx, input).into_subtree(call_site); |
| let attr = attr.map(|attr| { |
| parse_string_spanned(call_site.anchor, call_site.ctx, attr).into_subtree(call_site) |
| }); |
| let fixture_string = format!("{fixture:?}"); |
| let attr_string = attr.as_ref().map(|it| format!("{it:?}")); |
| |
| let res = expander |
| .expand( |
| macro_name, |
| fixture, |
| attr, |
| def_site, |
| call_site, |
| mixed_site, |
| FIXUP_ERASED_FILE_AST_ID_MARKER, |
| ) |
| .unwrap(); |
| expect_spanned |
| .assert_eq(&format!("{fixture_string}\n\n{}\n\n{res:#?}", attr_string.unwrap_or_default())); |
| } |
| |
| pub(crate) fn list() -> Vec<String> { |
| let dylib_path = proc_macro_test_dylib_path(); |
| let env = EnvSnapshot::default(); |
| let srv = ProcMacroSrv::new(&env); |
| let res = srv.list_macros(&dylib_path).unwrap(); |
| res.into_iter().map(|(name, kind)| format!("{name} [{kind:?}]")).collect() |
| } |