macros: add args for non-subdiagnostic fields

Non-subdiagnostic fields (i.e. those that don't have `#[label]`
attributes or similar and are just additional context) have to be added
as arguments for Fluent messages to refer them. This commit extends the
`SessionDiagnostic` derive to do this for all fields that do not have
attributes and introduces an `IntoDiagnosticArg` trait that is
implemented on all types that can be converted to a argument for Fluent.

Signed-off-by: David Wood <david.wood@huawei.com>
This commit is contained in:
David Wood 2022-03-30 09:45:36 +01:00
parent 8677fef192
commit 9956d4f99d
6 changed files with 122 additions and 30 deletions

View File

@ -8,6 +8,7 @@ use rustc_error_messages::FluentValue;
use rustc_lint_defs::{Applicability, LintExpectationId}; use rustc_lint_defs::{Applicability, LintExpectationId};
use rustc_serialize::json::Json; use rustc_serialize::json::Json;
use rustc_span::edition::LATEST_STABLE_EDITION; use rustc_span::edition::LATEST_STABLE_EDITION;
use rustc_span::symbol::{Ident, Symbol};
use rustc_span::{Span, DUMMY_SP}; use rustc_span::{Span, DUMMY_SP};
use std::borrow::Cow; use std::borrow::Cow;
use std::fmt; use std::fmt;
@ -31,6 +32,44 @@ pub enum DiagnosticArgValue<'source> {
Number(usize), Number(usize),
} }
/// Converts a value of a type into a `DiagnosticArg` (typically a field of a `SessionDiagnostic`
/// struct). Implemented as a custom trait rather than `From` so that it is implemented on the type
/// being converted rather than on `DiagnosticArgValue`, which enables types from other `rustc_*`
/// crates to implement this.
pub trait IntoDiagnosticArg {
fn into_diagnostic_arg(self) -> DiagnosticArgValue<'static>;
}
impl IntoDiagnosticArg for String {
fn into_diagnostic_arg(self) -> DiagnosticArgValue<'static> {
DiagnosticArgValue::Str(Cow::Owned(self))
}
}
impl IntoDiagnosticArg for Symbol {
fn into_diagnostic_arg(self) -> DiagnosticArgValue<'static> {
self.to_ident_string().into_diagnostic_arg()
}
}
impl IntoDiagnosticArg for Ident {
fn into_diagnostic_arg(self) -> DiagnosticArgValue<'static> {
self.to_string().into_diagnostic_arg()
}
}
impl<'a> IntoDiagnosticArg for &'a str {
fn into_diagnostic_arg(self) -> DiagnosticArgValue<'static> {
self.to_string().into_diagnostic_arg()
}
}
impl IntoDiagnosticArg for usize {
fn into_diagnostic_arg(self) -> DiagnosticArgValue<'static> {
DiagnosticArgValue::Number(self)
}
}
impl<'source> Into<FluentValue<'source>> for DiagnosticArgValue<'source> { impl<'source> Into<FluentValue<'source>> for DiagnosticArgValue<'source> {
fn into(self) -> FluentValue<'source> { fn into(self) -> FluentValue<'source> {
match self { match self {
@ -788,6 +827,15 @@ impl Diagnostic {
&self.args &self.args
} }
pub fn set_arg(
&mut self,
name: impl Into<Cow<'static, str>>,
arg: DiagnosticArgValue<'static>,
) -> &mut Self {
self.args.push((name.into(), arg.into()));
self
}
pub fn styled_message(&self) -> &Vec<(DiagnosticMessage, Style)> { pub fn styled_message(&self) -> &Vec<(DiagnosticMessage, Style)> {
&self.message &self.message
} }

View File

@ -1,8 +1,10 @@
use crate::diagnostic::DiagnosticArgValue;
use crate::{Diagnostic, DiagnosticId, DiagnosticMessage, DiagnosticStyledString, ErrorGuaranteed}; use crate::{Diagnostic, DiagnosticId, DiagnosticMessage, DiagnosticStyledString, ErrorGuaranteed};
use crate::{Handler, Level, MultiSpan, StashKey}; use crate::{Handler, Level, MultiSpan, StashKey};
use rustc_lint_defs::Applicability; use rustc_lint_defs::Applicability;
use rustc_span::Span; use rustc_span::Span;
use std::borrow::Cow;
use std::fmt::{self, Debug}; use std::fmt::{self, Debug};
use std::marker::PhantomData; use std::marker::PhantomData;
use std::ops::{Deref, DerefMut}; use std::ops::{Deref, DerefMut};
@ -536,6 +538,11 @@ impl<'a, G: EmissionGuarantee> DiagnosticBuilder<'a, G> {
forward!(pub fn set_primary_message(&mut self, msg: impl Into<String>) -> &mut Self); forward!(pub fn set_primary_message(&mut self, msg: impl Into<String>) -> &mut Self);
forward!(pub fn set_span(&mut self, sp: impl Into<MultiSpan>) -> &mut Self); forward!(pub fn set_span(&mut self, sp: impl Into<MultiSpan>) -> &mut Self);
forward!(pub fn code(&mut self, s: DiagnosticId) -> &mut Self); forward!(pub fn code(&mut self, s: DiagnosticId) -> &mut Self);
forward!(pub fn set_arg(
&mut self,
name: impl Into<Cow<'static, str>>,
arg: DiagnosticArgValue<'static>,
) -> &mut Self);
} }
impl<G: EmissionGuarantee> Debug for DiagnosticBuilder<'_, G> { impl<G: EmissionGuarantee> Debug for DiagnosticBuilder<'_, G> {

View File

@ -406,7 +406,8 @@ impl fmt::Display for ExplicitBug {
impl error::Error for ExplicitBug {} impl error::Error for ExplicitBug {}
pub use diagnostic::{ pub use diagnostic::{
Diagnostic, DiagnosticArg, DiagnosticId, DiagnosticStyledString, SubDiagnostic, Diagnostic, DiagnosticArg, DiagnosticArgValue, DiagnosticId, DiagnosticStyledString,
IntoDiagnosticArg, SubDiagnostic,
}; };
pub use diagnostic_builder::{DiagnosticBuilder, EmissionGuarantee}; pub use diagnostic_builder::{DiagnosticBuilder, EmissionGuarantee};
use std::backtrace::Backtrace; use std::backtrace::Backtrace;

View File

@ -157,7 +157,7 @@ impl<'a> SessionDiagnosticDerive<'a> {
} }
} }
fn into_tokens(self) -> proc_macro2::TokenStream { fn into_tokens(self) -> proc_macro2::TokenStream {
let SessionDiagnosticDerive { structure, mut builder } = self; let SessionDiagnosticDerive { mut structure, mut builder } = self;
let ast = structure.ast(); let ast = structure.ast();
let attrs = &ast.attrs; let attrs = &ast.attrs;
@ -175,11 +175,17 @@ impl<'a> SessionDiagnosticDerive<'a> {
} }
}; };
let body = structure.each(|field_binding| { // Generates calls to `span_label` and similar functions based on the attributes
// on fields. Code for suggestions uses formatting machinery and the value of
// other fields - because any given field can be referenced multiple times, it
// should be accessed through a borrow. When passing fields to `set_arg` (which
// happens below) for Fluent, we want to move the data, so that has to happen
// in a separate pass over the fields.
let attrs = structure.each(|field_binding| {
let field = field_binding.ast(); let field = field_binding.ast();
let result = field.attrs.iter().map(|attr| { let result = field.attrs.iter().map(|attr| {
builder builder
.generate_field_code( .generate_field_attr_code(
attr, attr,
FieldInfo { FieldInfo {
vis: &field.vis, vis: &field.vis,
@ -190,10 +196,30 @@ impl<'a> SessionDiagnosticDerive<'a> {
) )
.unwrap_or_else(|v| v.to_compile_error()) .unwrap_or_else(|v| v.to_compile_error())
}); });
return quote! {
#(#result);* quote! { #(#result);* }
};
}); });
// When generating `set_arg` calls, move data rather than borrow it to avoid
// requiring clones - this must therefore be the last use of each field (for
// example, any formatting machinery that might refer to a field should be
// generated already).
structure.bind_with(|_| synstructure::BindStyle::Move);
let args = structure.each(|field_binding| {
let field = field_binding.ast();
// When a field has attributes like `#[label]` or `#[note]` then it doesn't
// need to be passed as an argument to the diagnostic. But when a field has no
// attributes then it must be passed as an argument to the diagnostic so that
// it can be referred to by Fluent messages.
if field.attrs.is_empty() {
let diag = &builder.diag;
let ident = &field_binding.binding;
quote! { #diag.set_arg(stringify!(#ident), #field_binding.into_diagnostic_arg()); }
} else {
quote! {}
}
});
// Finally, putting it altogether. // Finally, putting it altogether.
match builder.kind { match builder.kind {
None => { None => {
@ -210,7 +236,10 @@ impl<'a> SessionDiagnosticDerive<'a> {
let mut #diag = #sess.struct_err_with_code("", rustc_errors::DiagnosticId::Error(#code)); let mut #diag = #sess.struct_err_with_code("", rustc_errors::DiagnosticId::Error(#code));
#preamble #preamble
match self { match self {
#body #attrs
}
match self {
#args
} }
#diag #diag
} }
@ -236,6 +265,7 @@ impl<'a> SessionDiagnosticDerive<'a> {
self, self,
#sess: &'__session_diagnostic_sess rustc_session::Session #sess: &'__session_diagnostic_sess rustc_session::Session
) -> rustc_errors::DiagnosticBuilder<'__session_diagnostic_sess, rustc_errors::ErrorGuaranteed> { ) -> rustc_errors::DiagnosticBuilder<'__session_diagnostic_sess, rustc_errors::ErrorGuaranteed> {
use rustc_errors::IntoDiagnosticArg;
#implementation #implementation
} }
} }
@ -345,15 +375,13 @@ impl<'a> SessionDiagnosticDeriveBuilder<'a> {
} }
} }
fn generate_field_code( fn generate_field_attr_code(
&mut self, &mut self,
attr: &syn::Attribute, attr: &syn::Attribute,
info: FieldInfo<'_>, info: FieldInfo<'_>,
) -> Result<proc_macro2::TokenStream, SessionDiagnosticDeriveError> { ) -> Result<proc_macro2::TokenStream, SessionDiagnosticDeriveError> {
let field_binding = &info.binding.binding; let field_binding = &info.binding.binding;
let option_ty = option_inner_ty(&info.ty); let option_ty = option_inner_ty(&info.ty);
let generated_code = self.generate_non_option_field_code( let generated_code = self.generate_non_option_field_code(
attr, attr,
FieldInfo { FieldInfo {
@ -363,15 +391,16 @@ impl<'a> SessionDiagnosticDeriveBuilder<'a> {
span: info.span, span: info.span,
}, },
)?; )?;
Ok(if option_ty.is_none() {
quote! { #generated_code } if option_ty.is_none() {
Ok(quote! { #generated_code })
} else { } else {
quote! { Ok(quote! {
if let Some(#field_binding) = #field_binding { if let Some(#field_binding) = #field_binding {
#generated_code #generated_code
} }
} })
}) }
} }
fn generate_non_option_field_code( fn generate_non_option_field_code(
@ -383,19 +412,20 @@ impl<'a> SessionDiagnosticDeriveBuilder<'a> {
let field_binding = &info.binding.binding; let field_binding = &info.binding.binding;
let name = attr.path.segments.last().unwrap().ident.to_string(); let name = attr.path.segments.last().unwrap().ident.to_string();
let name = name.as_str(); let name = name.as_str();
// At this point, we need to dispatch based on the attribute key + the // At this point, we need to dispatch based on the attribute key + the
// type. // type.
let meta = attr.parse_meta()?; let meta = attr.parse_meta()?;
Ok(match meta { match meta {
syn::Meta::NameValue(syn::MetaNameValue { lit: syn::Lit::Str(s), .. }) => { syn::Meta::NameValue(syn::MetaNameValue { lit: syn::Lit::Str(s), .. }) => {
let formatted_str = self.build_format(&s.value(), attr.span()); let formatted_str = self.build_format(&s.value(), attr.span());
match name { match name {
"message" => { "message" => {
if type_matches_path(&info.ty, &["rustc_span", "Span"]) { if type_matches_path(&info.ty, &["rustc_span", "Span"]) {
quote! { return Ok(quote! {
#diag.set_span(*#field_binding); #diag.set_span(*#field_binding);
#diag.set_primary_message(#formatted_str); #diag.set_primary_message(#formatted_str);
} });
} else { } else {
throw_span_err!( throw_span_err!(
attr.span().unwrap(), attr.span().unwrap(),
@ -405,9 +435,9 @@ impl<'a> SessionDiagnosticDeriveBuilder<'a> {
} }
"label" => { "label" => {
if type_matches_path(&info.ty, &["rustc_span", "Span"]) { if type_matches_path(&info.ty, &["rustc_span", "Span"]) {
quote! { return Ok(quote! {
#diag.span_label(*#field_binding, #formatted_str); #diag.span_label(*#field_binding, #formatted_str);
} });
} else { } else {
throw_span_err!( throw_span_err!(
attr.span().unwrap(), attr.span().unwrap(),
@ -480,11 +510,11 @@ impl<'a> SessionDiagnosticDeriveBuilder<'a> {
); );
}; };
let code = code.unwrap_or_else(|| quote! { String::new() }); let code = code.unwrap_or_else(|| quote! { String::new() });
// Now build it out:
let suggestion_method = format_ident!("span_{}", suggestion_kind); let suggestion_method = format_ident!("span_{}", suggestion_kind);
quote! { return Ok(quote! {
#diag.#suggestion_method(#span, #msg, #code, #applicability); #diag.#suggestion_method(#span, #msg, #code, #applicability);
} });
} }
other => throw_span_err!( other => throw_span_err!(
list.span().unwrap(), list.span().unwrap(),
@ -493,7 +523,7 @@ impl<'a> SessionDiagnosticDeriveBuilder<'a> {
} }
} }
_ => panic!("unhandled meta kind"), _ => panic!("unhandled meta kind"),
}) }
} }
fn span_and_applicability_of_ty( fn span_and_applicability_of_ty(

View File

@ -8,12 +8,18 @@ use crate::ty::{
}; };
use rustc_data_structures::fx::FxHashMap; use rustc_data_structures::fx::FxHashMap;
use rustc_errors::{Applicability, Diagnostic}; use rustc_errors::{Applicability, Diagnostic, DiagnosticArgValue, IntoDiagnosticArg};
use rustc_hir as hir; use rustc_hir as hir;
use rustc_hir::def_id::DefId; use rustc_hir::def_id::DefId;
use rustc_hir::{QPath, TyKind, WhereBoundPredicate, WherePredicate}; use rustc_hir::{QPath, TyKind, WhereBoundPredicate, WherePredicate};
use rustc_span::Span; use rustc_span::Span;
impl<'tcx> IntoDiagnosticArg for Ty<'tcx> {
fn into_diagnostic_arg(self) -> DiagnosticArgValue<'static> {
format!("{}", self).into_diagnostic_arg()
}
}
impl<'tcx> Ty<'tcx> { impl<'tcx> Ty<'tcx> {
/// Similar to `Ty::is_primitive`, but also considers inferred numeric values to be primitive. /// Similar to `Ty::is_primitive`, but also considers inferred numeric values to be primitive.
pub fn is_primitive_ty(self) -> bool { pub fn is_primitive_ty(self) -> bool {

View File

@ -11,8 +11,8 @@
#![crate_type = "lib"] #![crate_type = "lib"]
extern crate rustc_span; extern crate rustc_span;
use rustc_span::Span;
use rustc_span::symbol::Ident; use rustc_span::symbol::Ident;
use rustc_span::Span;
extern crate rustc_macros; extern crate rustc_macros;
use rustc_macros::SessionDiagnostic; use rustc_macros::SessionDiagnostic;
@ -108,7 +108,7 @@ struct ErrorWithMessageAppliedToField {
#[message = "This error has a field, and references {name}"] #[message = "This error has a field, and references {name}"]
//~^ ERROR `name` doesn't refer to a field on this type //~^ ERROR `name` doesn't refer to a field on this type
struct ErrorWithNonexistentField { struct ErrorWithNonexistentField {
span: Span descr: String,
} }
#[derive(SessionDiagnostic)] #[derive(SessionDiagnostic)]
@ -117,7 +117,7 @@ struct ErrorWithNonexistentField {
//~^ ERROR invalid format string: expected `'}'` //~^ ERROR invalid format string: expected `'}'`
struct ErrorMissingClosingBrace { struct ErrorMissingClosingBrace {
name: String, name: String,
span: Span val: usize,
} }
#[derive(SessionDiagnostic)] #[derive(SessionDiagnostic)]
@ -126,7 +126,7 @@ struct ErrorMissingClosingBrace {
//~^ ERROR invalid format string: unmatched `}` //~^ ERROR invalid format string: unmatched `}`
struct ErrorMissingOpeningBrace { struct ErrorMissingOpeningBrace {
name: String, name: String,
span: Span val: usize,
} }
#[derive(SessionDiagnostic)] #[derive(SessionDiagnostic)]