Allow deriving multiple subdiagnostics using one SessionSubdiagnostic

This reimplements ac638c1, which had to be reverted in the previous
commit because it contains a rebase accident that itself reverted
significant unrelated changes to SessionSubdiagnostic.
This commit is contained in:
Xiretza 2022-09-01 20:53:59 +02:00
parent 9df75ee254
commit d9b874c083
3 changed files with 154 additions and 144 deletions

View File

@ -211,11 +211,39 @@ impl<'a> HasFieldMap for SessionSubdiagnosticDeriveBuilder<'a> {
} }
} }
/// Provides frequently-needed information about the diagnostic kinds being derived for this type.
#[derive(Clone, Copy, Debug)]
struct KindsStatistics {
has_multipart_suggestion: bool,
all_multipart_suggestions: bool,
has_normal_suggestion: bool,
}
impl<'a> FromIterator<&'a SubdiagnosticKind> for KindsStatistics {
fn from_iter<T: IntoIterator<Item = &'a SubdiagnosticKind>>(kinds: T) -> Self {
let mut ret = Self {
has_multipart_suggestion: false,
all_multipart_suggestions: true,
has_normal_suggestion: false,
};
for kind in kinds {
if let SubdiagnosticKind::MultipartSuggestion { .. } = kind {
ret.has_multipart_suggestion = true;
} else {
ret.all_multipart_suggestions = false;
}
if let SubdiagnosticKind::Suggestion { .. } = kind {
ret.has_normal_suggestion = true;
}
}
ret
}
}
impl<'a> SessionSubdiagnosticDeriveBuilder<'a> { impl<'a> SessionSubdiagnosticDeriveBuilder<'a> {
fn identify_kind( fn identify_kind(&mut self) -> Result<Vec<(SubdiagnosticKind, Path)>, DiagnosticDeriveError> {
&mut self, let mut kind_slugs = vec![];
) -> Result<Option<(SubdiagnosticKind, Path)>, DiagnosticDeriveError> {
let mut kind_slug = None;
for attr in self.variant.ast().attrs { for attr in self.variant.ast().attrs {
let span = attr.span().unwrap(); let span = attr.span().unwrap();
@ -362,10 +390,10 @@ impl<'a> SessionSubdiagnosticDeriveBuilder<'a> {
| SubdiagnosticKind::MultipartSuggestion { .. } => {} | SubdiagnosticKind::MultipartSuggestion { .. } => {}
} }
kind_slug.set_once(((kind, slug), span)) kind_slugs.push((kind, slug))
} }
Ok(kind_slug.map(|(kind_slug, _)| kind_slug)) Ok(kind_slugs)
} }
/// Generates the code for a field with no attributes. /// Generates the code for a field with no attributes.
@ -387,7 +415,7 @@ impl<'a> SessionSubdiagnosticDeriveBuilder<'a> {
fn generate_field_attr_code( fn generate_field_attr_code(
&mut self, &mut self,
binding: &BindingInfo<'_>, binding: &BindingInfo<'_>,
kind: &SubdiagnosticKind, kind_stats: KindsStatistics,
) -> TokenStream { ) -> TokenStream {
let ast = binding.ast(); let ast = binding.ast();
assert!(ast.attrs.len() > 0, "field without attributes generating attr code"); assert!(ast.attrs.len() > 0, "field without attributes generating attr code");
@ -405,7 +433,7 @@ impl<'a> SessionSubdiagnosticDeriveBuilder<'a> {
}; };
let generated = self let generated = self
.generate_field_code_inner(kind, attr, info) .generate_field_code_inner(kind_stats, attr, info)
.unwrap_or_else(|v| v.to_compile_error()); .unwrap_or_else(|v| v.to_compile_error());
inner_ty.with(binding, generated) inner_ty.with(binding, generated)
@ -415,15 +443,15 @@ impl<'a> SessionSubdiagnosticDeriveBuilder<'a> {
fn generate_field_code_inner( fn generate_field_code_inner(
&mut self, &mut self,
kind: &SubdiagnosticKind, kind_stats: KindsStatistics,
attr: &Attribute, attr: &Attribute,
info: FieldInfo<'_>, info: FieldInfo<'_>,
) -> Result<TokenStream, DiagnosticDeriveError> { ) -> Result<TokenStream, DiagnosticDeriveError> {
let meta = attr.parse_meta()?; let meta = attr.parse_meta()?;
match meta { match meta {
Meta::Path(path) => self.generate_field_code_inner_path(kind, attr, info, path), Meta::Path(path) => self.generate_field_code_inner_path(kind_stats, attr, info, path),
Meta::List(list @ MetaList { .. }) => { Meta::List(list @ MetaList { .. }) => {
self.generate_field_code_inner_list(kind, attr, info, list) self.generate_field_code_inner_list(kind_stats, attr, info, list)
} }
_ => throw_invalid_attr!(attr, &meta), _ => throw_invalid_attr!(attr, &meta),
} }
@ -432,7 +460,7 @@ impl<'a> SessionSubdiagnosticDeriveBuilder<'a> {
/// Generates the code for a `[Meta::Path]`-like attribute on a field (e.g. `#[primary_span]`). /// Generates the code for a `[Meta::Path]`-like attribute on a field (e.g. `#[primary_span]`).
fn generate_field_code_inner_path( fn generate_field_code_inner_path(
&mut self, &mut self,
kind: &SubdiagnosticKind, kind_stats: KindsStatistics,
attr: &Attribute, attr: &Attribute,
info: FieldInfo<'_>, info: FieldInfo<'_>,
path: Path, path: Path,
@ -445,7 +473,7 @@ impl<'a> SessionSubdiagnosticDeriveBuilder<'a> {
match name { match name {
"skip_arg" => Ok(quote! {}), "skip_arg" => Ok(quote! {}),
"primary_span" => { "primary_span" => {
if matches!(kind, SubdiagnosticKind::MultipartSuggestion { .. }) { if kind_stats.has_multipart_suggestion {
throw_invalid_attr!(attr, &Meta::Path(path), |diag| { throw_invalid_attr!(attr, &Meta::Path(path), |diag| {
diag.help( diag.help(
"multipart suggestions use one or more `#[suggestion_part]`s rather \ "multipart suggestions use one or more `#[suggestion_part]`s rather \
@ -464,32 +492,20 @@ impl<'a> SessionSubdiagnosticDeriveBuilder<'a> {
"suggestion_part" => { "suggestion_part" => {
self.has_suggestion_parts = true; self.has_suggestion_parts = true;
match kind { if kind_stats.has_multipart_suggestion {
SubdiagnosticKind::MultipartSuggestion { .. } => { span_err(span, "`#[suggestion_part(...)]` attribute without `code = \"...\"`")
span_err(
span,
"`#[suggestion_part(...)]` attribute without `code = \"...\"`",
)
.emit(); .emit();
Ok(quote! {}) Ok(quote! {})
} } else {
SubdiagnosticKind::Label throw_invalid_attr!(attr, &Meta::Path(path), |diag| {
| SubdiagnosticKind::Note diag.help(
| SubdiagnosticKind::Help
| SubdiagnosticKind::Warn
| SubdiagnosticKind::Suggestion { .. } => {
throw_invalid_attr!(attr, &Meta::Path(path), |diag| {
diag.help(
"`#[suggestion_part(...)]` is only valid in multipart suggestions, use `#[primary_span]` instead", "`#[suggestion_part(...)]` is only valid in multipart suggestions, use `#[primary_span]` instead",
) )
}); });
}
} }
} }
"applicability" => { "applicability" => {
if let SubdiagnosticKind::Suggestion { .. } if kind_stats.has_multipart_suggestion || kind_stats.has_normal_suggestion {
| SubdiagnosticKind::MultipartSuggestion { .. } = kind
{
report_error_if_not_applied_to_applicability(attr, &info)?; report_error_if_not_applied_to_applicability(attr, &info)?;
let binding = info.binding.binding.clone(); let binding = info.binding.binding.clone();
@ -501,13 +517,16 @@ impl<'a> SessionSubdiagnosticDeriveBuilder<'a> {
Ok(quote! {}) Ok(quote! {})
} }
_ => throw_invalid_attr!(attr, &Meta::Path(path), |diag| { _ => throw_invalid_attr!(attr, &Meta::Path(path), |diag| {
let span_attr = if let SubdiagnosticKind::MultipartSuggestion { .. } = kind { let mut span_attrs = vec![];
"suggestion_part" if kind_stats.has_multipart_suggestion {
} else { span_attrs.push("suggestion_part");
"primary_span" }
}; if !kind_stats.all_multipart_suggestions {
span_attrs.push("primary_span")
}
diag.help(format!( diag.help(format!(
"only `{span_attr}`, `applicability` and `skip_arg` are valid field attributes", "only `{}`, `applicability` and `skip_arg` are valid field attributes",
span_attrs.join(", ")
)) ))
}), }),
} }
@ -517,7 +536,7 @@ impl<'a> SessionSubdiagnosticDeriveBuilder<'a> {
/// `#[suggestion_part(code = "...")]`). /// `#[suggestion_part(code = "...")]`).
fn generate_field_code_inner_list( fn generate_field_code_inner_list(
&mut self, &mut self,
kind: &SubdiagnosticKind, kind_stats: KindsStatistics,
attr: &Attribute, attr: &Attribute,
info: FieldInfo<'_>, info: FieldInfo<'_>,
list: MetaList, list: MetaList,
@ -529,7 +548,7 @@ impl<'a> SessionSubdiagnosticDeriveBuilder<'a> {
match name { match name {
"suggestion_part" => { "suggestion_part" => {
if !matches!(kind, SubdiagnosticKind::MultipartSuggestion { .. }) { if !kind_stats.has_multipart_suggestion {
throw_invalid_attr!(attr, &Meta::List(list), |diag| { throw_invalid_attr!(attr, &Meta::List(list), |diag| {
diag.help( diag.help(
"`#[suggestion_part(...)]` is only valid in multipart suggestions", "`#[suggestion_part(...)]` is only valid in multipart suggestions",
@ -576,35 +595,36 @@ impl<'a> SessionSubdiagnosticDeriveBuilder<'a> {
Ok(quote! { suggestions.push((#binding, #code)); }) Ok(quote! { suggestions.push((#binding, #code)); })
} }
_ => throw_invalid_attr!(attr, &Meta::List(list), |diag| { _ => throw_invalid_attr!(attr, &Meta::List(list), |diag| {
let span_attr = if let SubdiagnosticKind::MultipartSuggestion { .. } = kind { let mut span_attrs = vec![];
"suggestion_part" if kind_stats.has_multipart_suggestion {
} else { span_attrs.push("suggestion_part");
"primary_span" }
}; if !kind_stats.all_multipart_suggestions {
span_attrs.push("primary_span")
}
diag.help(format!( diag.help(format!(
"only `{span_attr}`, `applicability` and `skip_arg` are valid field attributes", "only `{}`, `applicability` and `skip_arg` are valid field attributes",
span_attrs.join(", ")
)) ))
}), }),
} }
} }
pub fn into_tokens(&mut self) -> Result<TokenStream, DiagnosticDeriveError> { pub fn into_tokens(&mut self) -> Result<TokenStream, DiagnosticDeriveError> {
let Some((kind, slug)) = self.identify_kind()? else { let kind_slugs = self.identify_kind()?;
if kind_slugs.is_empty() {
throw_span_err!( throw_span_err!(
self.variant.ast().ident.span().unwrap(), self.variant.ast().ident.span().unwrap(),
"subdiagnostic kind not specified" "subdiagnostic kind not specified"
); );
}; };
let init = match &kind { let kind_stats: KindsStatistics = kind_slugs.iter().map(|(kind, _slug)| kind).collect();
SubdiagnosticKind::Label
| SubdiagnosticKind::Note let init = if kind_stats.has_multipart_suggestion {
| SubdiagnosticKind::Help quote! { let mut suggestions = Vec::new(); }
| SubdiagnosticKind::Warn } else {
| SubdiagnosticKind::Suggestion { .. } => quote! {}, quote! {}
SubdiagnosticKind::MultipartSuggestion { .. } => {
quote! { let mut suggestions = Vec::new(); }
}
}; };
let attr_args: TokenStream = self let attr_args: TokenStream = self
@ -612,7 +632,7 @@ impl<'a> SessionSubdiagnosticDeriveBuilder<'a> {
.bindings() .bindings()
.iter() .iter()
.filter(|binding| !binding.ast().attrs.is_empty()) .filter(|binding| !binding.ast().attrs.is_empty())
.map(|binding| self.generate_field_attr_code(binding, &kind)) .map(|binding| self.generate_field_attr_code(binding, kind_stats))
.collect(); .collect();
let span_field = self.span_field.as_ref().map(|(span, _)| span); let span_field = self.span_field.as_ref().map(|(span, _)| span);
@ -622,48 +642,52 @@ impl<'a> SessionSubdiagnosticDeriveBuilder<'a> {
); );
let diag = &self.diag; let diag = &self.diag;
let name = format_ident!("{}{}", if span_field.is_some() { "span_" } else { "" }, kind); let mut calls = TokenStream::new();
let message = quote! { rustc_errors::fluent::#slug }; for (kind, slug) in kind_slugs {
let call = match kind { let name = format_ident!("{}{}", if span_field.is_some() { "span_" } else { "" }, kind);
SubdiagnosticKind::Suggestion { suggestion_kind, code } => { let message = quote! { rustc_errors::fluent::#slug };
if let Some(span) = span_field { let call = match kind {
SubdiagnosticKind::Suggestion { suggestion_kind, code } => {
if let Some(span) = span_field {
let style = suggestion_kind.to_suggestion_style();
quote! { #diag.#name(#span, #message, #code, #applicability, #style); }
} else {
span_err(self.span, "suggestion without `#[primary_span]` field").emit();
quote! { unreachable!(); }
}
}
SubdiagnosticKind::MultipartSuggestion { suggestion_kind } => {
if !self.has_suggestion_parts {
span_err(
self.span,
"multipart suggestion without any `#[suggestion_part(...)]` fields",
)
.emit();
}
let style = suggestion_kind.to_suggestion_style(); let style = suggestion_kind.to_suggestion_style();
quote! { #diag.#name(#span, #message, #code, #applicability, #style); } quote! { #diag.#name(#message, suggestions, #applicability, #style); }
} else {
span_err(self.span, "suggestion without `#[primary_span]` field").emit();
quote! { unreachable!(); }
} }
} SubdiagnosticKind::Label => {
SubdiagnosticKind::MultipartSuggestion { suggestion_kind } => { if let Some(span) = span_field {
if !self.has_suggestion_parts { quote! { #diag.#name(#span, #message); }
span_err( } else {
self.span, span_err(self.span, "label without `#[primary_span]` field").emit();
"multipart suggestion without any `#[suggestion_part(...)]` fields", quote! { unreachable!(); }
) }
.emit();
} }
_ => {
let style = suggestion_kind.to_suggestion_style(); if let Some(span) = span_field {
quote! { #diag.#name(#span, #message); }
quote! { #diag.#name(#message, suggestions, #applicability, #style); } } else {
} quote! { #diag.#name(#message); }
SubdiagnosticKind::Label => { }
if let Some(span) = span_field {
quote! { #diag.#name(#span, #message); }
} else {
span_err(self.span, "label without `#[primary_span]` field").emit();
quote! { unreachable!(); }
} }
} };
_ => { calls.extend(call);
if let Some(span) = span_field { }
quote! { #diag.#name(#span, #message); }
} else {
quote! { #diag.#name(#message); }
}
}
};
let plain_args: TokenStream = self let plain_args: TokenStream = self
.variant .variant
@ -676,7 +700,7 @@ impl<'a> SessionSubdiagnosticDeriveBuilder<'a> {
Ok(quote! { Ok(quote! {
#init #init
#attr_args #attr_args
#call #calls
#plain_args #plain_args
}) })
} }

View File

@ -309,9 +309,7 @@ union AC {
#[derive(SessionSubdiagnostic)] #[derive(SessionSubdiagnostic)]
#[label(parser::add_paren)] #[label(parser::add_paren)]
//~^ NOTE previously specified here
#[label(parser::add_paren)] #[label(parser::add_paren)]
//~^ ERROR specified multiple times
struct AD { struct AD {
#[primary_span] #[primary_span]
span: Span, span: Span,

View File

@ -174,20 +174,8 @@ LL | | b: u64
LL | | } LL | | }
| |_^ | |_^
error: specified multiple times
--> $DIR/subdiagnostic-derive.rs:313:1
|
LL | #[label(parser::add_paren)]
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
note: previously specified here
--> $DIR/subdiagnostic-derive.rs:311:1
|
LL | #[label(parser::add_paren)]
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^
error: `#[label(parser::add_paren)]` is not a valid attribute error: `#[label(parser::add_paren)]` is not a valid attribute
--> $DIR/subdiagnostic-derive.rs:321:28 --> $DIR/subdiagnostic-derive.rs:319:28
| |
LL | #[label(parser::add_paren, parser::add_paren)] LL | #[label(parser::add_paren, parser::add_paren)]
| ^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^
@ -195,67 +183,67 @@ LL | #[label(parser::add_paren, parser::add_paren)]
= help: a diagnostic slug must be the first argument to the attribute = help: a diagnostic slug must be the first argument to the attribute
error: specified multiple times error: specified multiple times
--> $DIR/subdiagnostic-derive.rs:334:5 --> $DIR/subdiagnostic-derive.rs:332:5
| |
LL | #[primary_span] LL | #[primary_span]
| ^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^
| |
note: previously specified here note: previously specified here
--> $DIR/subdiagnostic-derive.rs:331:5 --> $DIR/subdiagnostic-derive.rs:329:5
| |
LL | #[primary_span] LL | #[primary_span]
| ^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^
error: subdiagnostic kind not specified error: subdiagnostic kind not specified
--> $DIR/subdiagnostic-derive.rs:340:8 --> $DIR/subdiagnostic-derive.rs:338:8
| |
LL | struct AG { LL | struct AG {
| ^^ | ^^
error: specified multiple times error: specified multiple times
--> $DIR/subdiagnostic-derive.rs:377:47 --> $DIR/subdiagnostic-derive.rs:375:47
| |
LL | #[suggestion(parser::add_paren, code = "...", code = "...")] LL | #[suggestion(parser::add_paren, code = "...", code = "...")]
| ^^^^^^^^^^^^ | ^^^^^^^^^^^^
| |
note: previously specified here note: previously specified here
--> $DIR/subdiagnostic-derive.rs:377:33 --> $DIR/subdiagnostic-derive.rs:375:33
| |
LL | #[suggestion(parser::add_paren, code = "...", code = "...")] LL | #[suggestion(parser::add_paren, code = "...", code = "...")]
| ^^^^^^^^^^^^ | ^^^^^^^^^^^^
error: specified multiple times error: specified multiple times
--> $DIR/subdiagnostic-derive.rs:395:5 --> $DIR/subdiagnostic-derive.rs:393:5
| |
LL | #[applicability] LL | #[applicability]
| ^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^
| |
note: previously specified here note: previously specified here
--> $DIR/subdiagnostic-derive.rs:392:5 --> $DIR/subdiagnostic-derive.rs:390:5
| |
LL | #[applicability] LL | #[applicability]
| ^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^
error: the `#[applicability]` attribute can only be applied to fields of type `Applicability` error: the `#[applicability]` attribute can only be applied to fields of type `Applicability`
--> $DIR/subdiagnostic-derive.rs:405:5 --> $DIR/subdiagnostic-derive.rs:403:5
| |
LL | #[applicability] LL | #[applicability]
| ^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^
error: suggestion without `code = "..."` error: suggestion without `code = "..."`
--> $DIR/subdiagnostic-derive.rs:418:1 --> $DIR/subdiagnostic-derive.rs:416:1
| |
LL | #[suggestion(parser::add_paren)] LL | #[suggestion(parser::add_paren)]
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
error: invalid applicability error: invalid applicability
--> $DIR/subdiagnostic-derive.rs:428:46 --> $DIR/subdiagnostic-derive.rs:426:46
| |
LL | #[suggestion(parser::add_paren, code ="...", applicability = "foo")] LL | #[suggestion(parser::add_paren, code ="...", applicability = "foo")]
| ^^^^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^^^^
error: suggestion without `#[primary_span]` field error: suggestion without `#[primary_span]` field
--> $DIR/subdiagnostic-derive.rs:446:1 --> $DIR/subdiagnostic-derive.rs:444:1
| |
LL | / #[suggestion(parser::add_paren, code = "...")] LL | / #[suggestion(parser::add_paren, code = "...")]
LL | | LL | |
@ -265,25 +253,25 @@ LL | | }
| |_^ | |_^
error: unsupported type attribute for subdiagnostic enum error: unsupported type attribute for subdiagnostic enum
--> $DIR/subdiagnostic-derive.rs:460:1 --> $DIR/subdiagnostic-derive.rs:458:1
| |
LL | #[label] LL | #[label]
| ^^^^^^^^ | ^^^^^^^^
error: `var` doesn't refer to a field on this type error: `var` doesn't refer to a field on this type
--> $DIR/subdiagnostic-derive.rs:480:39 --> $DIR/subdiagnostic-derive.rs:478:39
| |
LL | #[suggestion(parser::add_paren, code ="{var}", applicability = "machine-applicable")] LL | #[suggestion(parser::add_paren, code ="{var}", applicability = "machine-applicable")]
| ^^^^^^^ | ^^^^^^^
error: `var` doesn't refer to a field on this type error: `var` doesn't refer to a field on this type
--> $DIR/subdiagnostic-derive.rs:499:43 --> $DIR/subdiagnostic-derive.rs:497:43
| |
LL | #[suggestion(parser::add_paren, code ="{var}", applicability = "machine-applicable")] LL | #[suggestion(parser::add_paren, code ="{var}", applicability = "machine-applicable")]
| ^^^^^^^ | ^^^^^^^
error: `#[suggestion_part]` is not a valid attribute error: `#[suggestion_part]` is not a valid attribute
--> $DIR/subdiagnostic-derive.rs:522:5 --> $DIR/subdiagnostic-derive.rs:520:5
| |
LL | #[suggestion_part] LL | #[suggestion_part]
| ^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^
@ -291,7 +279,7 @@ LL | #[suggestion_part]
= help: `#[suggestion_part(...)]` is only valid in multipart suggestions, use `#[primary_span]` instead = help: `#[suggestion_part(...)]` is only valid in multipart suggestions, use `#[primary_span]` instead
error: `#[suggestion_part(...)]` is not a valid attribute error: `#[suggestion_part(...)]` is not a valid attribute
--> $DIR/subdiagnostic-derive.rs:525:5 --> $DIR/subdiagnostic-derive.rs:523:5
| |
LL | #[suggestion_part(code = "...")] LL | #[suggestion_part(code = "...")]
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
@ -299,7 +287,7 @@ LL | #[suggestion_part(code = "...")]
= help: `#[suggestion_part(...)]` is only valid in multipart suggestions = help: `#[suggestion_part(...)]` is only valid in multipart suggestions
error: suggestion without `#[primary_span]` field error: suggestion without `#[primary_span]` field
--> $DIR/subdiagnostic-derive.rs:519:1 --> $DIR/subdiagnostic-derive.rs:517:1
| |
LL | / #[suggestion(parser::add_paren, code = "...")] LL | / #[suggestion(parser::add_paren, code = "...")]
LL | | LL | |
@ -311,13 +299,13 @@ LL | | }
| |_^ | |_^
error: `code` is not a valid nested attribute of a `multipart_suggestion` attribute error: `code` is not a valid nested attribute of a `multipart_suggestion` attribute
--> $DIR/subdiagnostic-derive.rs:534:43 --> $DIR/subdiagnostic-derive.rs:532:43
| |
LL | #[multipart_suggestion(parser::add_paren, code = "...", applicability = "machine-applicable")] LL | #[multipart_suggestion(parser::add_paren, code = "...", applicability = "machine-applicable")]
| ^^^^^^^^^^^^ | ^^^^^^^^^^^^
error: multipart suggestion without any `#[suggestion_part(...)]` fields error: multipart suggestion without any `#[suggestion_part(...)]` fields
--> $DIR/subdiagnostic-derive.rs:534:1 --> $DIR/subdiagnostic-derive.rs:532:1
| |
LL | / #[multipart_suggestion(parser::add_paren, code = "...", applicability = "machine-applicable")] LL | / #[multipart_suggestion(parser::add_paren, code = "...", applicability = "machine-applicable")]
LL | | LL | |
@ -328,19 +316,19 @@ LL | | }
| |_^ | |_^
error: `#[suggestion_part(...)]` attribute without `code = "..."` error: `#[suggestion_part(...)]` attribute without `code = "..."`
--> $DIR/subdiagnostic-derive.rs:544:5 --> $DIR/subdiagnostic-derive.rs:542:5
| |
LL | #[suggestion_part] LL | #[suggestion_part]
| ^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^
error: `#[suggestion_part(...)]` attribute without `code = "..."` error: `#[suggestion_part(...)]` attribute without `code = "..."`
--> $DIR/subdiagnostic-derive.rs:552:5 --> $DIR/subdiagnostic-derive.rs:550:5
| |
LL | #[suggestion_part()] LL | #[suggestion_part()]
| ^^^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^^^
error: `#[primary_span]` is not a valid attribute error: `#[primary_span]` is not a valid attribute
--> $DIR/subdiagnostic-derive.rs:561:5 --> $DIR/subdiagnostic-derive.rs:559:5
| |
LL | #[primary_span] LL | #[primary_span]
| ^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^
@ -348,7 +336,7 @@ LL | #[primary_span]
= help: multipart suggestions use one or more `#[suggestion_part]`s rather than one `#[primary_span]` = help: multipart suggestions use one or more `#[suggestion_part]`s rather than one `#[primary_span]`
error: multipart suggestion without any `#[suggestion_part(...)]` fields error: multipart suggestion without any `#[suggestion_part(...)]` fields
--> $DIR/subdiagnostic-derive.rs:558:1 --> $DIR/subdiagnostic-derive.rs:556:1
| |
LL | / #[multipart_suggestion(parser::add_paren)] LL | / #[multipart_suggestion(parser::add_paren)]
LL | | LL | |
@ -360,19 +348,19 @@ LL | | }
| |_^ | |_^
error: `#[suggestion_part(...)]` attribute without `code = "..."` error: `#[suggestion_part(...)]` attribute without `code = "..."`
--> $DIR/subdiagnostic-derive.rs:569:5 --> $DIR/subdiagnostic-derive.rs:567:5
| |
LL | #[suggestion_part] LL | #[suggestion_part]
| ^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^
error: `#[suggestion_part(...)]` attribute without `code = "..."` error: `#[suggestion_part(...)]` attribute without `code = "..."`
--> $DIR/subdiagnostic-derive.rs:572:5 --> $DIR/subdiagnostic-derive.rs:570:5
| |
LL | #[suggestion_part()] LL | #[suggestion_part()]
| ^^^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^^^
error: `#[suggestion_part(foo = ...)]` is not a valid attribute error: `#[suggestion_part(foo = ...)]` is not a valid attribute
--> $DIR/subdiagnostic-derive.rs:575:23 --> $DIR/subdiagnostic-derive.rs:573:23
| |
LL | #[suggestion_part(foo = "bar")] LL | #[suggestion_part(foo = "bar")]
| ^^^^^^^^^^^ | ^^^^^^^^^^^
@ -380,37 +368,37 @@ LL | #[suggestion_part(foo = "bar")]
= help: `code` is the only valid nested attribute = help: `code` is the only valid nested attribute
error: the `#[suggestion_part(...)]` attribute can only be applied to fields of type `Span` or `MultiSpan` error: the `#[suggestion_part(...)]` attribute can only be applied to fields of type `Span` or `MultiSpan`
--> $DIR/subdiagnostic-derive.rs:578:5 --> $DIR/subdiagnostic-derive.rs:576:5
| |
LL | #[suggestion_part(code = "...")] LL | #[suggestion_part(code = "...")]
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
error: the `#[suggestion_part(...)]` attribute can only be applied to fields of type `Span` or `MultiSpan` error: the `#[suggestion_part(...)]` attribute can only be applied to fields of type `Span` or `MultiSpan`
--> $DIR/subdiagnostic-derive.rs:581:5 --> $DIR/subdiagnostic-derive.rs:579:5
| |
LL | #[suggestion_part()] LL | #[suggestion_part()]
| ^^^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^^^
error: specified multiple times error: specified multiple times
--> $DIR/subdiagnostic-derive.rs:589:37 --> $DIR/subdiagnostic-derive.rs:587:37
| |
LL | #[suggestion_part(code = "...", code = ",,,")] LL | #[suggestion_part(code = "...", code = ",,,")]
| ^^^^^^^^^^^^ | ^^^^^^^^^^^^
| |
note: previously specified here note: previously specified here
--> $DIR/subdiagnostic-derive.rs:589:23 --> $DIR/subdiagnostic-derive.rs:587:23
| |
LL | #[suggestion_part(code = "...", code = ",,,")] LL | #[suggestion_part(code = "...", code = ",,,")]
| ^^^^^^^^^^^^ | ^^^^^^^^^^^^
error: specified multiple times error: specified multiple times
--> $DIR/subdiagnostic-derive.rs:619:5 --> $DIR/subdiagnostic-derive.rs:617:5
| |
LL | #[applicability] LL | #[applicability]
| ^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^
| |
note: previously specified here note: previously specified here
--> $DIR/subdiagnostic-derive.rs:616:43 --> $DIR/subdiagnostic-derive.rs:614:43
| |
LL | #[multipart_suggestion(parser::add_paren, applicability = "machine-applicable")] LL | #[multipart_suggestion(parser::add_paren, applicability = "machine-applicable")]
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
@ -475,6 +463,6 @@ error[E0425]: cannot find value `slug` in module `rustc_errors::fluent`
LL | #[label(slug)] LL | #[label(slug)]
| ^^^^ not found in `rustc_errors::fluent` | ^^^^ not found in `rustc_errors::fluent`
error: aborting due to 64 previous errors error: aborting due to 63 previous errors
For more information about this error, try `rustc --explain E0425`. For more information about this error, try `rustc --explain E0425`.