9943: fix: Do not replace items annotated with builtin attrs with the attr input r=Veykril a=Veykril

This causes runnables to work for paths that actually resolve to the `test` attribute now.
![Code_aUhX1KQfw3](https://user-images.githubusercontent.com/3757771/129917168-bd9ed3f8-3a08-49d2-930a-4b93efaa8acf.png)

Prior to this we replaced items annotated with builtin attributes by the attribute input instead of the item in our dummy expansion which is why the fully written path didn't work as we actually discarded the item while `test` was just ignored.

Fixes #9935

Co-authored-by: Lukas Wirth <lukastw97@gmail.com>
This commit is contained in:
bors[bot] 2021-08-20 11:57:28 +00:00 committed by GitHub
commit da5a5ba378
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
7 changed files with 56 additions and 27 deletions

View file

@ -173,6 +173,10 @@ impl<'db, DB: HirDatabase> Semantics<'db, DB> {
self.imp.descend_node_at_offset(node, offset).find_map(N::cast) self.imp.descend_node_at_offset(node, offset).find_map(N::cast)
} }
pub fn hir_file_for(&self, syntax_node: &SyntaxNode) -> HirFileId {
self.imp.find_file(syntax_node.clone()).file_id
}
pub fn original_range(&self, node: &SyntaxNode) -> FileRange { pub fn original_range(&self, node: &SyntaxNode) -> FileRange {
self.imp.original_range(node) self.imp.original_range(node)
} }

View file

@ -34,9 +34,6 @@ macro_rules! rustc_attr {
}; };
} }
/// Built-in macro-like attributes.
pub const EXTRA_ATTRIBUTES: &[BuiltinAttribute] = &["test", "bench"];
/// "Inert" built-in attributes that have a special meaning to rustc or rustdoc. /// "Inert" built-in attributes that have a special meaning to rustc or rustdoc.
#[rustfmt::skip] #[rustfmt::skip]
pub const INERT_ATTRIBUTES: &[BuiltinAttribute] = &[ pub const INERT_ATTRIBUTES: &[BuiltinAttribute] = &[

View file

@ -1712,27 +1712,24 @@ impl ModCollector<'_, '_> {
if path.kind == PathKind::Plain { if path.kind == PathKind::Plain {
if let Some(tool_module) = path.segments().first() { if let Some(tool_module) = path.segments().first() {
let tool_module = tool_module.to_string(); let tool_module = tool_module.to_string();
if builtin_attr::TOOL_MODULES let is_tool = builtin_attr::TOOL_MODULES
.iter() .iter()
.copied() .copied()
.chain(self.def_collector.registered_tools.iter().map(|s| &**s)) .chain(self.def_collector.registered_tools.iter().map(AsRef::as_ref))
.any(|m| tool_module == *m) .any(|m| tool_module == *m);
{ if is_tool {
return true; return true;
} }
} }
if let Some(name) = path.as_ident() { if let Some(name) = path.as_ident() {
let name = name.to_string(); let name = name.to_string();
if builtin_attr::INERT_ATTRIBUTES let is_inert = builtin_attr::INERT_ATTRIBUTES
.iter() .iter()
.chain(builtin_attr::EXTRA_ATTRIBUTES)
.copied() .copied()
.chain(self.def_collector.registered_attrs.iter().map(|s| &**s)) .chain(self.def_collector.registered_attrs.iter().map(AsRef::as_ref))
.any(|attr| name == *attr) .any(|attr| name == *attr);
{ return is_inert;
return true;
}
} }
} }

View file

@ -17,11 +17,12 @@ macro_rules! register_builtin {
db: &dyn AstDatabase, db: &dyn AstDatabase,
id: MacroCallId, id: MacroCallId,
tt: &tt::Subtree, tt: &tt::Subtree,
item: &tt::Subtree,
) -> Result<tt::Subtree, mbe::ExpandError> { ) -> Result<tt::Subtree, mbe::ExpandError> {
let expander = match *self { let expander = match *self {
$( BuiltinAttrExpander::$variant => $expand, )* $( BuiltinAttrExpander::$variant => $expand, )*
}; };
expander(db, id, tt) expander(db, id, tt, item)
} }
fn find_by_name(name: &name::Name) -> Option<Self> { fn find_by_name(name: &name::Name) -> Option<Self> {
@ -61,7 +62,8 @@ pub fn find_builtin_attr(
fn dummy_attr_expand( fn dummy_attr_expand(
_db: &dyn AstDatabase, _db: &dyn AstDatabase,
_id: MacroCallId, _id: MacroCallId,
tt: &tt::Subtree, _tt: &tt::Subtree,
item: &tt::Subtree,
) -> Result<tt::Subtree, mbe::ExpandError> { ) -> Result<tt::Subtree, mbe::ExpandError> {
Ok(tt.clone()) Ok(item.clone())
} }

View file

@ -54,7 +54,12 @@ impl TokenExpander {
TokenExpander::MacroDef { mac, .. } => mac.expand(tt), TokenExpander::MacroDef { mac, .. } => mac.expand(tt),
TokenExpander::Builtin(it) => it.expand(db, id, tt), TokenExpander::Builtin(it) => it.expand(db, id, tt),
// FIXME switch these to ExpandResult as well // FIXME switch these to ExpandResult as well
TokenExpander::BuiltinAttr(it) => it.expand(db, id, tt).into(), TokenExpander::BuiltinAttr(it) => match db.macro_arg(id) {
Some(macro_arg) => it.expand(db, id, tt, &macro_arg.0).into(),
None => mbe::ExpandResult::only_err(
mbe::ExpandError::Other("No item argument for attribute".to_string()).into(),
),
},
TokenExpander::BuiltinDerive(it) => it.expand(db, id, tt).into(), TokenExpander::BuiltinDerive(it) => it.expand(db, id, tt).into(),
TokenExpander::ProcMacro(_) => { TokenExpander::ProcMacro(_) => {
// We store the result in salsa db to prevent non-deterministic behavior in // We store the result in salsa db to prevent non-deterministic behavior in

View file

@ -22,7 +22,7 @@ use either::Either;
pub use mbe::{ExpandError, ExpandResult}; pub use mbe::{ExpandError, ExpandResult};
pub use parser::FragmentKind; pub use parser::FragmentKind;
use std::{hash::Hash, sync::Arc}; use std::{hash::Hash, iter, sync::Arc};
use base_db::{impl_intern_key, salsa, CrateId, FileId, FileRange}; use base_db::{impl_intern_key, salsa, CrateId, FileId, FileRange};
use syntax::{ use syntax::{
@ -454,7 +454,7 @@ impl InFile<SyntaxNode> {
self, self,
db: &dyn db::AstDatabase, db: &dyn db::AstDatabase,
) -> impl Iterator<Item = InFile<SyntaxNode>> + '_ { ) -> impl Iterator<Item = InFile<SyntaxNode>> + '_ {
std::iter::successors(Some(self), move |node| match node.value.parent() { iter::successors(Some(self), move |node| match node.value.parent() {
Some(parent) => Some(node.with_value(parent)), Some(parent) => Some(node.with_value(parent)),
None => { None => {
let parent_node = node.file_id.call_node(db)?; let parent_node = node.file_id.call_node(db)?;
@ -562,6 +562,23 @@ impl<N: AstNode> InFile<N> {
pub fn syntax(&self) -> InFile<&SyntaxNode> { pub fn syntax(&self) -> InFile<&SyntaxNode> {
self.with_value(self.value.syntax()) self.with_value(self.value.syntax())
} }
pub fn nodes_with_attributes<'db>(
self,
db: &'db dyn db::AstDatabase,
) -> impl Iterator<Item = InFile<N>> + 'db
where
N: 'db,
{
iter::successors(Some(self), move |node| {
let InFile { file_id, value } = node.file_id.call_node(db)?;
N::cast(value).map(|n| InFile::new(file_id, n))
})
}
pub fn node_with_attributes(self, db: &dyn db::AstDatabase) -> InFile<N> {
self.nodes_with_attributes(db).last().unwrap()
}
} }
/// Given a `MacroCallId`, return what `FragmentKind` it belongs to. /// Given a `MacroCallId`, return what `FragmentKind` it belongs to.

View file

@ -3,7 +3,7 @@ use std::fmt;
use ast::NameOwner; use ast::NameOwner;
use cfg::CfgExpr; use cfg::CfgExpr;
use either::Either; use either::Either;
use hir::{AsAssocItem, HasAttrs, HasSource, HirDisplay, Semantics}; use hir::{AsAssocItem, HasAttrs, HasSource, HirDisplay, InFile, Semantics};
use ide_assists::utils::test_related_attribute; use ide_assists::utils::test_related_attribute;
use ide_db::{ use ide_db::{
base_db::{FilePosition, FileRange}, base_db::{FilePosition, FileRange},
@ -232,22 +232,26 @@ fn find_related_tests(
let functions = refs.iter().filter_map(|(range, _)| { let functions = refs.iter().filter_map(|(range, _)| {
let token = file.token_at_offset(range.start()).next()?; let token = file.token_at_offset(range.start()).next()?;
let token = sema.descend_into_macros(token); let token = sema.descend_into_macros(token);
token.ancestors().find_map(ast::Fn::cast) token
.ancestors()
.find_map(ast::Fn::cast)
.map(|f| hir::InFile::new(sema.hir_file_for(f.syntax()), f))
}); });
for fn_def in functions { for fn_def in functions {
if let Some(runnable) = as_test_runnable(sema, &fn_def) { // #[test/bench] expands to just the item causing us to lose the attribute, so recover them by going out of the attribute
let InFile { value: fn_def, .. } = &fn_def.node_with_attributes(sema.db);
if let Some(runnable) = as_test_runnable(sema, fn_def) {
// direct test // direct test
tests.insert(runnable); tests.insert(runnable);
} else if let Some(module) = parent_test_module(sema, &fn_def) { } else if let Some(module) = parent_test_module(sema, fn_def) {
// indirect test // indirect test
find_related_tests_in_module(sema, &fn_def, &module, tests); find_related_tests_in_module(sema, fn_def, &module, tests);
} }
} }
} }
} }
} }
fn find_related_tests_in_module( fn find_related_tests_in_module(
sema: &Semantics<RootDatabase>, sema: &Semantics<RootDatabase>,
fn_def: &ast::Fn, fn_def: &ast::Fn,
@ -292,7 +296,8 @@ fn parent_test_module(sema: &Semantics<RootDatabase>, fn_def: &ast::Fn) -> Optio
} }
pub(crate) fn runnable_fn(sema: &Semantics<RootDatabase>, def: hir::Function) -> Option<Runnable> { pub(crate) fn runnable_fn(sema: &Semantics<RootDatabase>, def: hir::Function) -> Option<Runnable> {
let func = def.source(sema.db)?; // #[test/bench] expands to just the item causing us to lose the attribute, so recover them by going out of the attribute
let func = def.source(sema.db)?.node_with_attributes(sema.db);
let name_string = def.name(sema.db).to_string(); let name_string = def.name(sema.db).to_string();
let root = def.module(sema.db).krate().root_module(sema.db); let root = def.module(sema.db).krate().root_module(sema.db);
@ -499,6 +504,8 @@ fn has_test_function_or_multiple_test_submodules(
match item { match item {
hir::ModuleDef::Function(f) => { hir::ModuleDef::Function(f) => {
if let Some(it) = f.source(sema.db) { if let Some(it) = f.source(sema.db) {
// #[test/bench] expands to just the item causing us to lose the attribute, so recover them by going out of the attribute
let it = it.node_with_attributes(sema.db);
if test_related_attribute(&it.value).is_some() { if test_related_attribute(&it.value).is_some() {
return true; return true;
} }