Skip to content

Commit 78c3edc

Browse files
committed
new chapter with examples of diagnostic translation PRs
1 parent 7e50a6a commit 78c3edc

File tree

2 files changed

+198
-0
lines changed

2 files changed

+198
-0
lines changed

src/SUMMARY.md

+1
Original file line numberDiff line numberDiff line change
@@ -149,6 +149,7 @@
149149
- [Error codes](./diagnostics/error-codes.md)
150150
- [Diagnostic items](./diagnostics/diagnostic-items.md)
151151
- [`ErrorGuaranteed`](./diagnostics/error-guaranteed.md)
152+
- [Making diagnostics translatable](diagnostics/making-diagnostics-translatable.md)
152153

153154
# MIR to Binaries
154155

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,197 @@
1+
# Making diagnostics translatable
2+
3+
<!-- toc -->
4+
5+
There is an ongoing effort to make diagnostics translatable,
6+
and the coordination happens [on GitHub].
7+
8+
This is intended to be a gentle guide to help with this effort,
9+
with the use of examples.
10+
11+
> A more detailed explanation can be found in [the announcement of the initiative].
12+
> However,
13+
> note that some of the important details are now outdated.
14+
>
15+
> Reference documentation can be found in these chapters:
16+
> - [Diagnostic structs](diagnostic-structs.md)
17+
> - [Translation system](translation.md)
18+
19+
## General comments
20+
21+
For a single diagnostic, 3 files need to be modified:
22+
23+
- One file where the diagnostic is emitted,
24+
typically by calling [Session::struct_span_err].
25+
- One file where the type representing the diagnostic,
26+
typically a `struct`, would be added.
27+
This would be in the `errors` module of the relevant rustc crate.
28+
- One file where the actual text of the diagnostic is located,
29+
located in a file named `locales/en-US.ftl`,
30+
relative to the root path of the relevant rustc crate.
31+
32+
## Simple example
33+
34+
*This uses [issue #108727] for demonstration*
35+
36+
Suppose you have the following code:
37+
38+
```rust
39+
ecx.struct_span_err(span, "proc-macro derive produced unparseable tokens").emit();
40+
```
41+
42+
> Note that `ecx.struct_span_err` is the [Session::struct_span_err].
43+
44+
Referring to the [general comments](#general-comments) section above,
45+
follow these changes:
46+
47+
- Replace the code above with this:
48+
49+
```rust
50+
ecx.sess.emit_err(errors::ProcMacroDeriveTokens { span });
51+
```
52+
53+
- Create the above type, `errors::ProcMacroDeriveTokens`,
54+
in `src/errors.rs` (relative to crate directory root):
55+
56+
```rust
57+
#[derive(Diagnostic)]
58+
#[diag(expand_proc_macro_derive_tokens)]
59+
pub struct ProcMacroDeriveTokens {
60+
#[primary_span]
61+
pub span: Span,
62+
}
63+
```
64+
65+
- Create the actual text of the diagnostic in `locales/en-US.ftl`
66+
(also relative to crate directory root):
67+
68+
```fluent
69+
expand_proc_macro_derive_tokens =
70+
proc-macro derive produced unparseable tokens
71+
```
72+
73+
Once that is done, a PR may be submitted,
74+
and the following comment on the PR page will label the PR accordingly,
75+
as well as alert the right people to review it:
76+
77+
```
78+
@rustbot label +A-translation
79+
r? rust-lang/diagnostics
80+
```
81+
82+
## Example with variable interpolation
83+
84+
*This uses [issue #108436] for demonstration*
85+
86+
Suppose you have the following code:
87+
88+
```rust
89+
let mut err = ecx.struct_span_err(span, "proc macro panicked");
90+
if let Some(s) = e.as_str() {
91+
err.help(&format!("message: {}", s));
92+
}
93+
err.emit()
94+
```
95+
96+
> Note that `ecx.struct_span_err` indirectly calls [Session::struct_span_err].
97+
98+
- Replace the code above with this:
99+
100+
```rust
101+
ecx.sess.emit_err(errors::ProcMacroPanicked {
102+
span,
103+
message: e
104+
.as_str()
105+
.map(|message| errors::ProcMacroPanickedHelp { message: message.into() }),
106+
})
107+
```
108+
109+
- Create the above type, `errors::ProcMacroPanickedHelp`,
110+
in `src/errors.rs` (relative to crate directory root):
111+
112+
```rust
113+
#[derive(Diagnostic)]
114+
#[diag(expand_proc_macro_panicked)]
115+
pub(crate) struct ProcMacroPanicked {
116+
#[primary_span]
117+
pub span: Span,
118+
#[subdiagnostic]
119+
pub message: Option<ProcMacroPanickedHelp>,
120+
}
121+
122+
#[derive(Subdiagnostic)]
123+
#[help(expand_help)]
124+
pub(crate) struct ProcMacroPanickedHelp {
125+
pub message: String,
126+
}
127+
```
128+
129+
- Create the actual text of the diagnostic in `locales/en-US.ftl`
130+
(also relative to crate directory root):
131+
132+
```fluent
133+
expand_proc_macro_panicked =
134+
proc macro panicked
135+
.help = message: {$message}
136+
```
137+
138+
## Example with a macro, `struct_span_err!`
139+
140+
*This uses [issue #108373] for demonstration*
141+
142+
Suppose you have the following code:
143+
144+
```rust
145+
let mut diag = struct_span_err!(
146+
tcx.sess,
147+
generics_where_clauses_span.unwrap_or(main_span),
148+
E0646,
149+
"`main` function is not allowed to have a `where` clause"
150+
);
151+
if let Some(generics_where_clauses_span) = generics_where_clauses_span {
152+
diag.span_label(generics_where_clauses_span, "`main` cannot have a `where` clause");
153+
}
154+
diag.emit();
155+
```
156+
157+
> Note that `struct_span_err!` ultimately calls [Session::struct_span_err_with_code].
158+
159+
- Replace the code above with this:
160+
161+
```rust
162+
tcx.sess.emit_err(errors::WhereClauseOnMain {
163+
span: generics_where_clauses_span.unwrap_or(main_span),
164+
generics_span: generics_where_clauses_span,
165+
});
166+
```
167+
168+
- Create the above type, `errors::WhereClauseOnMain`,
169+
in `src/errors.rs` (relative to crate directory root):
170+
171+
```rust
172+
#[derive(Diagnostic)]
173+
#[diag(hir_analysis_where_clause_on_main, code = "E0646")]
174+
pub(crate) struct WhereClauseOnMain {
175+
#[primary_span]
176+
pub span: Span,
177+
#[label]
178+
pub generics_span: Option<Span>,
179+
}
180+
```
181+
182+
- Create the actual text of the diagnostic in `locales/en-US.ftl`
183+
(also relative to crate directory root):
184+
185+
```fluent
186+
hir_analysis_where_clause_on_main =
187+
`main` function is not allowed to have a `where` clause
188+
.label = `main` cannot have a `where` clause
189+
```
190+
191+
[Session::struct_span_err]: https://doc.rust-lang.org/stable/nightly-rustc/rustc_session/session/struct.Session.html#method.struct_span_err
192+
[Session::struct_span_err_with_code]: https://doc.rust-lang.org/stable/nightly-rustc/rustc_session/session/struct.Session.html#method.struct_span_err_with_code
193+
[the announcement of the initiative]: https://blog.rust-lang.org/inside-rust/2022/08/16/diagnostic-effort.html#manually-implementing-sessiondiagnostic
194+
[on GitHub]: https://github.com/rust-lang/rust/issues/100717
195+
[issue #108373]: https://github.com/rust-lang/rust/pull/108373
196+
[issue #108436]: https://github.com/rust-lang/rust/pull/108436
197+
[issue #108727]: https://github.com/rust-lang/rust/pull/108727

0 commit comments

Comments
 (0)