Skip to content

Commit

Permalink
add FluentRaw to represent fluent raw content
Browse files Browse the repository at this point in the history
  • Loading branch information
chenyukang committed Nov 19, 2023
1 parent 9c27678 commit db6553b
Show file tree
Hide file tree
Showing 9 changed files with 73 additions and 38 deletions.
18 changes: 17 additions & 1 deletion compiler/rustc_error_messages/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -266,6 +266,8 @@ type FluentId = Cow<'static, str>;
pub enum SubdiagnosticMessage {
/// Non-translatable diagnostic message.
Str(Cow<'static, str>),
/// Translatable diagnostic message in Fluent raw format.
FluentRaw(Cow<'static, str>),
/// Translatable message which has already been translated eagerly.
///
/// Some diagnostics have repeated subdiagnostics where the same interpolated variables would
Expand Down Expand Up @@ -313,6 +315,8 @@ impl From<Cow<'static, str>> for SubdiagnosticMessage {
pub enum DiagnosticMessage {
/// Non-translatable diagnostic message.
Str(Cow<'static, str>),
/// Translatable diagnostic message in Fluent raw format.
FluentRaw(Cow<'static, str>),
/// Translatable message which has already been translated eagerly.
///
/// Some diagnostics have repeated subdiagnostics where the same interpolated variables would
Expand Down Expand Up @@ -342,6 +346,7 @@ impl DiagnosticMessage {
pub fn with_subdiagnostic_message(&self, sub: SubdiagnosticMessage) -> Self {
let attr = match sub {
SubdiagnosticMessage::Str(s) => return DiagnosticMessage::Str(s),
SubdiagnosticMessage::FluentRaw(s) => return DiagnosticMessage::FluentRaw(s),
SubdiagnosticMessage::Eager(s) => return DiagnosticMessage::Eager(s),
SubdiagnosticMessage::FluentIdentifier(id) => {
return DiagnosticMessage::FluentIdentifier(id, None);
Expand All @@ -351,6 +356,7 @@ impl DiagnosticMessage {

match self {
DiagnosticMessage::Str(s) => DiagnosticMessage::Str(s.clone()),
DiagnosticMessage::FluentRaw(s) => DiagnosticMessage::FluentRaw(s.clone()),
DiagnosticMessage::Eager(s) => DiagnosticMessage::Eager(s.clone()),
DiagnosticMessage::FluentIdentifier(id, _) => {
DiagnosticMessage::FluentIdentifier(id.clone(), Some(attr))
Expand All @@ -360,7 +366,9 @@ impl DiagnosticMessage {

pub fn as_str(&self) -> Option<&str> {
match self {
DiagnosticMessage::Eager(s) | DiagnosticMessage::Str(s) => Some(s),
DiagnosticMessage::Eager(s)
| DiagnosticMessage::Str(s)
| DiagnosticMessage::FluentRaw(s) => Some(s),
DiagnosticMessage::FluentIdentifier(_, _) => None,
}
}
Expand All @@ -382,6 +390,13 @@ impl From<Cow<'static, str>> for DiagnosticMessage {
}
}

#[macro_export]
macro_rules! fluent_raw {
($str:expr) => {
DiagnosticMessage::FluentRaw(Cow::Borrowed($str))
};
}

/// A workaround for "good path" ICEs when formatting types in disabled lints.
///
/// Delays formatting until `.into(): DiagnosticMessage` is used.
Expand All @@ -402,6 +417,7 @@ impl Into<SubdiagnosticMessage> for DiagnosticMessage {
fn into(self) -> SubdiagnosticMessage {
match self {
DiagnosticMessage::Str(s) => SubdiagnosticMessage::Str(s),
DiagnosticMessage::FluentRaw(s) => SubdiagnosticMessage::FluentRaw(s),
DiagnosticMessage::Eager(s) => SubdiagnosticMessage::Eager(s),
DiagnosticMessage::FluentIdentifier(id, None) => {
SubdiagnosticMessage::FluentIdentifier(id)
Expand Down
2 changes: 1 addition & 1 deletion compiler/rustc_errors/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ use rustc_data_structures::stable_hasher::{Hash128, StableHasher};
use rustc_data_structures::sync::{Lock, Lrc};
use rustc_data_structures::AtomicRef;
pub use rustc_error_messages::{
fallback_fluent_bundle, fluent_bundle, DelayDm, DiagnosticMessage, FluentBundle,
fallback_fluent_bundle, fluent_bundle, fluent_raw, DelayDm, DiagnosticMessage, FluentBundle,
LanguageIdentifier, LazyFallbackBundle, MultiSpan, SpanLabel, SubdiagnosticMessage,
};
use rustc_fluent_macro::fluent_messages;
Expand Down
28 changes: 15 additions & 13 deletions compiler/rustc_errors/src/translation.rs
Original file line number Diff line number Diff line change
Expand Up @@ -65,21 +65,23 @@ pub trait Translate {
trace!(?message, ?args);
let (identifier, attr) = match message {
DiagnosticMessage::Str(msg) | DiagnosticMessage::Eager(msg) => {
return Ok(Cow::Borrowed(msg));
}
DiagnosticMessage::FluentRaw(msg) => {
// FIXME(yukang): A hack for raw fluent content for new diagnostics proc format
let trimed = msg.replace(" ", "");
if trimed.contains("$") || trimed.contains("{\"") || trimed.contains("\"}") {
let fluent_text = format!("dummy = {}", msg);
if let Ok(resource) = FluentResource::try_new(fluent_text) {
let mut bundle = RawFluentBundle::new(vec![langid!("en-US")]);
bundle.add_resource(resource).unwrap();
let mut errors = vec![];
let pattern = bundle.get_message("dummy").unwrap().value().unwrap();
let res = bundle.format_pattern(&pattern, Some(args), &mut errors);
return Ok(Cow::Owned(
res.to_string().replace("\u{2068}", "").replace("\u{2069}", ""),
));
}
let fluent_text = format!("dummy = {}", msg);
if let Ok(resource) = FluentResource::try_new(fluent_text) {
let mut bundle = RawFluentBundle::new(vec![langid!("en-US")]);
bundle.add_resource(resource).unwrap();
let mut errors = vec![];
let pattern = bundle.get_message("dummy").unwrap().value().unwrap();
let res = bundle.format_pattern(&pattern, Some(args), &mut errors);
return Ok(Cow::Owned(
res.to_string().replace("\u{2068}", "").replace("\u{2069}", ""),
));
}

// If the message is not a valid Fluent resource, just return the original
return Ok(Cow::Borrowed(msg));
}
DiagnosticMessage::FluentIdentifier(identifier, attr) => (identifier, attr),
Expand Down
2 changes: 1 addition & 1 deletion compiler/rustc_macros/src/diagnostics/diagnostic.rs
Original file line number Diff line number Diff line change
Expand Up @@ -69,7 +69,7 @@ impl<'a> DiagnosticDerive<'a> {
}
(None, Some(raw_label)) => {
quote! {
let mut #diag = #handler.struct_diagnostic(DiagnosticMessage::Str(#raw_label.into()));
let mut #diag = #handler.struct_diagnostic(DiagnosticMessage::FluentRaw(#raw_label.into()));
}
}
(Some(_slug), Some(_raw_label)) => {
Expand Down
10 changes: 5 additions & 5 deletions compiler/rustc_macros/src/diagnostics/diagnostic_builder.rs
Original file line number Diff line number Diff line change
Expand Up @@ -509,7 +509,7 @@ impl<'a> DiagnosticDeriveVariantBuilder<'a> {

let suggestion_label = if let Some(raw_label) = raw_label {
quote! {
#raw_label
DiagnosticMessage::FluentRaw(Cow::Borrowed(#raw_label))
}
} else {
quote! {
Expand Down Expand Up @@ -547,15 +547,15 @@ impl<'a> DiagnosticDeriveVariantBuilder<'a> {
return quote! {
#diag.#fn_name(
#field_binding,
#raw_label
DiagnosticMessage::FluentRaw(Cow::Borrowed(#raw_label))
);
};
}
if let Some(raw_label) = self.get_attr(kind.to_string().as_str()) {
quote! {
#diag.#fn_name(
#field_binding,
#raw_label
DiagnosticMessage::FluentRaw(Cow::Borrowed(#raw_label))
);
}
} else {
Expand All @@ -579,12 +579,12 @@ impl<'a> DiagnosticDeriveVariantBuilder<'a> {
let diag = &self.parent.diag;
if let Some(raw_label) = raw_label {
return quote! {
#diag.#kind(#raw_label);
#diag.#kind(DiagnosticMessage::FluentRaw(Cow::Borrowed(#raw_label)));
};
}
if let Some(raw_label) = self.get_attr(kind.to_string().as_str()) {
quote! {
#diag.#kind(#raw_label);
#diag.#kind(DiagnosticMessage::FluentRaw(Cow::Borrowed(#raw_label)));
}
} else {
quote! {
Expand Down
2 changes: 1 addition & 1 deletion compiler/rustc_macros/src/diagnostics/subdiagnostic.rs
Original file line number Diff line number Diff line change
Expand Up @@ -527,7 +527,7 @@ impl<'parent, 'a> SubdiagnosticDeriveVariantBuilder<'parent, 'a> {
quote! { let #message = #f(#diag, crate::fluent_generated::#slug.into()); },
);
} else {
calls.extend(quote! { let #message = #f(#diag, #raw_label.into()); });
calls.extend(quote! { let #message = #f(#diag, DiagnosticMessage::FluentRaw(Cow::Borrowed(#raw_label)).into()); });
}

let name = format_ident!(
Expand Down
35 changes: 21 additions & 14 deletions compiler/rustc_parse/src/errors.rs
Original file line number Diff line number Diff line change
@@ -1,18 +1,17 @@
// ignore-tidy-filelength
use std::borrow::Cow;

use crate::parser::{ForbiddenLetReason, TokenDescription};
use rustc_ast::token::Token;
use rustc_ast::{Path, Visibility};
use rustc_errors::DiagnosticMessage;
use rustc_errors::{fluent_raw, DiagnosticMessage};
use rustc_errors::{AddToDiagnostic, Applicability, EmissionGuarantee, IntoDiagnostic};
use rustc_macros::{Diagnostic, Subdiagnostic};
use rustc_session::errors::ExprParenthesesNeeded;
use rustc_span::edition::{Edition, LATEST_STABLE_EDITION};
use rustc_span::symbol::Ident;
use rustc_span::{Span, Symbol};

use crate::parser::{ForbiddenLetReason, TokenDescription};

#[derive(Diagnostic)]
#[diag("ambiguous `+` in a type")]
pub(crate) struct AmbiguousPlus {
Expand Down Expand Up @@ -1205,18 +1204,22 @@ impl<'a, G: EmissionGuarantee> IntoDiagnostic<'a, G> for ExpectedIdentifier {

let mut diag = handler.struct_diagnostic(match token_descr {
Some(TokenDescription::ReservedIdentifier) => {
"expected identifier, found reserved identifier `{$token}`"
fluent_raw!("expected identifier, found reserved identifier `{$token}`")
}
Some(TokenDescription::Keyword) => "expected identifier, found keyword `{$token}`",
Some(TokenDescription::Keyword) => {
fluent_raw!("expected identifier, found keyword `{$token}`")
}

Some(TokenDescription::ReservedKeyword) => {
"expected identifier, found reserved keyword `{$token}`"
fluent_raw!("expected identifier, found reserved keyword `{$token}`")
}

Some(TokenDescription::DocComment) => {
"expected identifier, found doc comment `{$token}`"
fluent_raw!("expected identifier, found doc comment `{$token}`")
}
None => {
fluent_raw!("expected identifier, found `{$token}`")
}

None => "expected identifier, found `{$token}`",
});
diag.set_span(self.span);
diag.set_arg("token", self.token);
Expand Down Expand Up @@ -1264,14 +1267,18 @@ impl<'a, G: EmissionGuarantee> IntoDiagnostic<'a, G> for ExpectedSemi {

let mut diag = handler.struct_diagnostic(match token_descr {
Some(TokenDescription::ReservedIdentifier) => {
"expected `;`, found reserved identifier `{$token}`"
fluent_raw!("expected `;`, found reserved identifier `{$token}`")
}
Some(TokenDescription::Keyword) => {
fluent_raw!("expected `;`, found keyword `{$token}`")
}
Some(TokenDescription::Keyword) => "expected `;`, found keyword `{$token}`",
Some(TokenDescription::ReservedKeyword) => {
"expected `;`, found reserved keyword `{$token}`"
fluent_raw!("expected `;`, found reserved keyword `{$token}`")
}
Some(TokenDescription::DocComment) => {
fluent_raw!("expected `;`, found doc comment `{$token}`")
}
Some(TokenDescription::DocComment) => "expected `;`, found doc comment `{$token}`",
None => "expected `;`, found `{$token}`",
None => fluent_raw!("expected `;`, found `{$token}`"),
});
diag.set_span(self.span);
diag.set_arg("token", self.token);
Expand Down
12 changes: 10 additions & 2 deletions compiler/rustc_parse/src/parser/attr.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,11 @@ use crate::errors::{InvalidMetaItem, SuffixedLiteralInAttribute};
use rustc_ast as ast;
use rustc_ast::attr;
use rustc_ast::token::{self, Delimiter, Nonterminal};
use rustc_errors::fluent_raw;
use rustc_errors::DiagnosticMessage;
use rustc_errors::{error_code, Diagnostic, IntoDiagnostic, PResult};
use rustc_span::{sym, BytePos, Span};
use std::borrow::Cow;
use std::convert::TryInto;
use thin_vec::ThinVec;
use tracing::debug;
Expand Down Expand Up @@ -175,10 +178,15 @@ impl<'a> Parser<'a> {
Ok(Some(item)) => {
// FIXME(#100717)
err.set_arg("item", item.kind.descr());
err.span_label(item.span, "the inner {$item_type} doesn't annotate this {$item}");
err.span_label(
item.span,
fluent_raw!("the inner {$item_type} doesn't annotate this {$item}"),
);
err.span_suggestion_verbose(
replacement_span,
"to annotate the {$item}, change the {$item_type} from inner to outer style",
fluent_raw!(
"to annotate the {$item}, change the {$item_type} from inner to outer style"
),
match attr_type {
OuterAttributeType::Attribute => "",
OuterAttributeType::DocBlockComment => "*",
Expand Down
2 changes: 2 additions & 0 deletions compiler/rustc_parse/src/parser/expr.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ use rustc_ast::{Arm, Async, BlockCheckMode, Expr, ExprKind, Label, Movability, R
use rustc_ast::{ClosureBinder, MetaItemLit, StmtKind};
use rustc_ast_pretty::pprust;
use rustc_data_structures::stack::ensure_sufficient_stack;
use rustc_errors::DiagnosticMessage;
use rustc_errors::{
AddToDiagnostic, Applicability, Diagnostic, DiagnosticBuilder, ErrorGuaranteed, IntoDiagnostic,
PResult, StashKey,
Expand All @@ -36,6 +37,7 @@ use rustc_span::source_map::{self, Spanned};
use rustc_span::symbol::kw::PathRoot;
use rustc_span::symbol::{kw, sym, Ident, Symbol};
use rustc_span::{BytePos, Pos, Span};
use std::borrow::Cow;
use thin_vec::{thin_vec, ThinVec};

/// Possibly accepts an `token::Interpolated` expression (a pre-parsed expression
Expand Down

0 comments on commit db6553b

Please sign in to comment.