mod diagnostic; mod error; mod fluent; mod subdiagnostic; mod utils; use diagnostic::SessionDiagnosticDerive; pub(crate) use fluent::fluent_messages; use proc_macro2::TokenStream; use quote::format_ident; use subdiagnostic::SessionSubdiagnosticDerive; use synstructure::Structure; /// Implements `#[derive(SessionDiagnostic)]`, which allows for errors to be specified as a struct, /// independent from the actual diagnostics emitting code. /// /// ```ignore (rust) /// # extern crate rustc_errors; /// # use rustc_errors::Applicability; /// # extern crate rustc_span; /// # use rustc_span::{symbol::Ident, Span}; /// # extern crate rust_middle; /// # use rustc_middle::ty::Ty; /// #[derive(SessionDiagnostic)] /// #[error(borrowck::move_out_of_borrow, code = "E0505")] /// pub struct MoveOutOfBorrowError<'tcx> { /// pub name: Ident, /// pub ty: Ty<'tcx>, /// #[primary_span] /// #[label] /// pub span: Span, /// #[label(borrowck::first_borrow_label)] /// pub first_borrow_span: Span, /// #[suggestion(code = "{name}.clone()")] /// pub clone_sugg: Option<(Span, Applicability)> /// } /// ``` /// /// ```fluent /// move-out-of-borrow = cannot move out of {$name} because it is borrowed /// .label = cannot move out of borrow /// .first-borrow-label = `{$ty}` first borrowed here /// .suggestion = consider cloning here /// ``` /// /// Then, later, to emit the error: /// /// ```ignore (rust) /// sess.emit_err(MoveOutOfBorrowError { /// expected, /// actual, /// span, /// first_borrow_span, /// clone_sugg: Some(suggestion, Applicability::MachineApplicable), /// }); /// ``` /// /// See rustc dev guide for more examples on using the `#[derive(SessionDiagnostic)]`: /// pub fn session_diagnostic_derive(s: Structure<'_>) -> TokenStream { // Names for the diagnostic we build and the session we build it from. let diag = format_ident!("diag"); let sess = format_ident!("sess"); SessionDiagnosticDerive::new(diag, sess, s).into_tokens() } /// Implements `#[derive(SessionSubdiagnostic)]`, which allows for labels, notes, helps and /// suggestions to be specified as a structs or enums, independent from the actual diagnostics /// emitting code or diagnostic derives. /// /// ```ignore (rust) /// #[derive(SessionSubdiagnostic)] /// pub enum ExpectedIdentifierLabel<'tcx> { /// #[label(parser::expected_identifier)] /// WithoutFound { /// #[primary_span] /// span: Span, /// } /// #[label(parser::expected_identifier_found)] /// WithFound { /// #[primary_span] /// span: Span, /// found: String, /// } /// } /// /// #[derive(SessionSubdiagnostic)] /// #[suggestion_verbose(parser::raw_identifier)] /// pub struct RawIdentifierSuggestion<'tcx> { /// #[primary_span] /// span: Span, /// #[applicability] /// applicability: Applicability, /// ident: Ident, /// } /// ``` /// /// ```fluent /// parser-expected-identifier = expected identifier /// /// parser-expected-identifier-found = expected identifier, found {$found} /// /// parser-raw-identifier = escape `{$ident}` to use it as an identifier /// ``` /// /// Then, later, to add the subdiagnostic: /// /// ```ignore (rust) /// diag.subdiagnostic(ExpectedIdentifierLabel::WithoutFound { span }); /// /// diag.subdiagnostic(RawIdentifierSuggestion { span, applicability, ident }); /// ``` pub fn session_subdiagnostic_derive(s: Structure<'_>) -> TokenStream { SessionSubdiagnosticDerive::new(s).into_tokens() }