Skip to content

Commit bbafcf2

Browse files
committed
new chapter with examples of diagnostic translation PRs
1 parent 9d76913 commit bbafcf2

File tree

2 files changed

+204
-0
lines changed

2 files changed

+204
-0
lines changed

src/SUMMARY.md

+1
Original file line numberDiff line numberDiff line change
@@ -149,6 +149,7 @@
149149
- [Diagnostic codes](./diagnostics/diagnostic-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,203 @@
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 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 #108323] as an example*
35+
36+
Suppose you have the following code:
37+
38+
```rust
39+
tcx.sess
40+
.struct_span_err(attr.span, "`main` function is not allowed to be `#[track_caller]`")
41+
.span_label(main_span, "`main` function is not allowed to be `#[track_caller]`")
42+
.emit();
43+
```
44+
45+
> Note that `tcx.sess.struct_span_err` is the [Session::struct_span_err] function.
46+
47+
Referring to the [general comments](#general-comments) section above,
48+
follow these changes:
49+
50+
- Replace the code above with this:
51+
52+
```rust
53+
tcx.sess.emit_err(errors::TrackCallerOnMain {
54+
span: attr.span,
55+
annotated: main_span,
56+
});
57+
```
58+
59+
- Create the above type, `errors::TrackCallerOnMain`,
60+
in `src/errors.rs` (relative to the crate directory root):
61+
62+
```rust
63+
#[derive(Diagnostic)]
64+
#[diag(hir_analysis_track_caller_on_main)]
65+
pub(crate) struct TrackCallerOnMain {
66+
#[primary_span]
67+
pub span: Span,
68+
#[label]
69+
pub annotated: Span,
70+
}
71+
```
72+
73+
- Create the actual text of the diagnostic in `locales/en-US.ftl`:
74+
75+
```fluent
76+
hir_analysis_track_caller_on_main =
77+
`main` function is not allowed to be `#[track_caller]`
78+
.label = `main` function is not allowed to be `#[track_caller]`
79+
```
80+
81+
Once that is done, a PR may be submitted,
82+
and the following comment on the PR page will label the PR accordingly,
83+
as well as alert the right people to review it:
84+
85+
```
86+
@rustbot label +A-translation
87+
r? rust-lang/diagnostics
88+
```
89+
90+
## Example with variable interpolation
91+
92+
*This uses [issue #108436] as an example*
93+
94+
Suppose you have the following code:
95+
96+
```rust
97+
let mut err = ecx.struct_span_err(span, "proc macro panicked");
98+
if let Some(s) = e.as_str() {
99+
err.help(&format!("message: {}", s));
100+
}
101+
err.emit()
102+
```
103+
104+
> Note that `ecx.struct_span_err` indirectly calls [Session::struct_span_err] function.
105+
106+
- Replace the code above with this:
107+
108+
```rust
109+
ecx.sess.emit_err(errors::ProcMacroPanicked {
110+
span,
111+
message: e
112+
.as_str()
113+
.map(|message| errors::ProcMacroPanickedHelp { message: message.into() }),
114+
})
115+
```
116+
117+
- Create the above type, `errors::ProcMacroPanickedHelp``,
118+
in `src/errors.rs` (relative to the crate directory root):
119+
120+
```rust
121+
#[derive(Diagnostic)]
122+
#[diag(expand_proc_macro_panicked)]
123+
pub(crate) struct ProcMacroPanicked {
124+
#[primary_span]
125+
pub span: Span,
126+
#[subdiagnostic]
127+
pub message: Option<ProcMacroPanickedHelp>,
128+
}
129+
130+
#[derive(Subdiagnostic)]
131+
#[help(expand_help)]
132+
pub(crate) struct ProcMacroPanickedHelp {
133+
pub message: String,
134+
}
135+
```
136+
137+
- Create the actual text of the diagnostic in `locales/en-US.ftl`:
138+
139+
```fluent
140+
expand_proc_macro_panicked =
141+
proc macro panicked
142+
.help = message: {$message}
143+
```
144+
145+
## Example with a macro, `struct_span_err!`
146+
147+
*This uses [issue #108373] as an example*
148+
149+
Suppose you have the following code:
150+
151+
```rust
152+
let mut diag = struct_span_err!(
153+
tcx.sess,
154+
generics_where_clauses_span.unwrap_or(main_span),
155+
E0646,
156+
"`main` function is not allowed to have a `where` clause"
157+
);
158+
if let Some(generics_where_clauses_span) = generics_where_clauses_span {
159+
diag.span_label(generics_where_clauses_span, "`main` cannot have a `where` clause");
160+
}
161+
diag.emit();
162+
```
163+
164+
> Note that `struct_span_err!` ultimately calls [Session::struct_span_err_with_code] function.
165+
166+
- Replace the code above with this:
167+
168+
```rust
169+
tcx.sess.emit_err(errors::WhereClauseOnMain {
170+
span: generics_where_clauses_span.unwrap_or(main_span),
171+
generics_span: generics_where_clauses_span,
172+
});
173+
```
174+
175+
- Create the above type, `errors::WhereClauseOnMain`,
176+
in `src/errors.rs` (relative to the crate directory root):
177+
178+
```rust
179+
#[derive(Diagnostic)]
180+
#[diag(hir_analysis_where_clause_on_main, code = "E0646")]
181+
pub(crate) struct WhereClauseOnMain {
182+
#[primary_span]
183+
pub span: Span,
184+
#[label]
185+
pub generics_span: Option<Span>,
186+
}
187+
```
188+
189+
- Create the actual text of the diagnostic in `locales/en-US.ftl`:
190+
191+
```fluent
192+
hir_analysis_where_clause_on_main =
193+
`main` function is not allowed to have a `where` clause
194+
.label = `main` cannot have a `where` clause
195+
```
196+
197+
[Session::struct_span_err]: https://doc.rust-lang.org/stable/nightly-rustc/rustc_session/session/struct.Session.html#method.struct_span_err
198+
[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
199+
[the announcement of the initiative]: https://blog.rust-lang.org/inside-rust/2022/08/16/diagnostic-effort.html#manually-implementing-sessiondiagnostic
200+
[on GitHub]: https://github.com/rust-lang/rust/issues/100717
201+
[issue #108323]: https://github.com/rust-lang/rust/pull/108323
202+
[issue #108373]: https://github.com/rust-lang/rust/pull/108373
203+
[issue #108436]: https://github.com/rust-lang/rust/pull/108436

0 commit comments

Comments
 (0)