|  | //! utils used in proc-macro tests | 
|  |  | 
|  | use expect_test::Expect; | 
|  | use span::{EditionedFileId, FileId, ROOT_ERASED_FILE_AST_ID, Span, SpanAnchor, SyntaxContext}; | 
|  | use tt::TextRange; | 
|  |  | 
|  | use crate::{EnvSnapshot, ProcMacroSrv, SpanId, dylib, proc_macro_test_dylib_path}; | 
|  |  | 
|  | fn parse_string(call_site: SpanId, src: &str) -> crate::server_impl::TokenStream<SpanId> { | 
|  | 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(&temp_dir::TempDir::new().unwrap(), &path).unwrap(); | 
|  |  | 
|  | let def_site = SpanId(0); | 
|  | let call_site = SpanId(1); | 
|  | let mixed_site = SpanId(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).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).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() | 
|  | } |