Skip to content

Commit 998bf64

Browse files
authored
error messsages: convert most hints to submessage (#13838)
This commit * converts most hints in error messages to separate submessages * removes some non-modular boxing * inlines grammatical contents in format strings in order to improve the indentation of such hints and pave the way to reunifying the error and warning formats for compiler report.
1 parent 156ddad commit 998bf64

33 files changed

+604
-544
lines changed

.depend

-2
Original file line numberDiff line numberDiff line change
@@ -1884,7 +1884,6 @@ typing/typedecl.cmi : \
18841884
parsing/location.cmi \
18851885
typing/includecore.cmi \
18861886
typing/ident.cmi \
1887-
utils/format_doc.cmi \
18881887
typing/errortrace.cmi \
18891888
typing/env.cmi \
18901889
parsing/asttypes.cmi
@@ -2258,7 +2257,6 @@ typing/typetexp.cmi : \
22582257
parsing/parsetree.cmi \
22592258
parsing/longident.cmi \
22602259
parsing/location.cmi \
2261-
utils/format_doc.cmi \
22622260
typing/errortrace.cmi \
22632261
typing/env.cmi \
22642262
parsing/asttypes.cmi

Changes

+3
Original file line numberDiff line numberDiff line change
@@ -348,6 +348,9 @@ Working version
348348
Hint: Did you mean "array"?
349349
(Florian Angeletti, suggestion by Daniel Bünzli, review by Gabriel Scherer)
350350

351+
- #13818: better delimited hints in error message
352+
(Florian Angeletti, review by Gabriel Scherer)
353+
351354
### Internal/compiler-libs changes:
352355

353356
- #13314: Comment the code of Translclass

parsing/lexer.mll

+4-1
Original file line numberDiff line numberDiff line change
@@ -400,7 +400,10 @@ let prepare_error loc = function
400400
let msg = "Illegal empty character literal ''" in
401401
let sub =
402402
[Location.msg
403-
"@{<hint>Hint@}: Did you mean ' ' or a type variable 'a?"] in
403+
"@{<hint>Hint@}: Did you mean %a or a type variable %a?"
404+
Style.inline_code "' '"
405+
Style.inline_code "'a"
406+
] in
404407
Location.error ~loc ~sub msg
405408
| Keyword_as_label kwd ->
406409
Location.errorf ~loc

parsing/location.ml

+17-4
Original file line numberDiff line numberDiff line change
@@ -739,7 +739,7 @@ let batch_mode_printer : report_printer =
739739
| Misc.Error_style.Short ->
740740
()
741741
in
742-
Format.fprintf ppf "@[<v>%a:@ %a@]" print_loc loc
742+
Format.fprintf ppf "%a:@ %a" print_loc loc
743743
(Fmt.compat highlight) loc
744744
in
745745
let pp_txt ppf txt = Format.fprintf ppf "%a" Fmt.Doc.format txt in
@@ -802,9 +802,12 @@ let batch_mode_printer : report_printer =
802802
) msgs
803803
in
804804
let pp_submsg self report ppf { loc; txt } =
805-
Format.fprintf ppf "@[%a @[%a@]@]"
806-
(self.pp_submsg_loc self report) loc
807-
(self.pp_submsg_txt self report) txt
805+
if loc.loc_ghost then
806+
Format.fprintf ppf "@[%a@]" (self.pp_submsg_txt self report) txt
807+
else
808+
Format.fprintf ppf "%a @[%a@]"
809+
(self.pp_submsg_loc self report) loc
810+
(self.pp_submsg_txt self report) txt
808811
in
809812
let pp_submsg_loc self report ppf loc =
810813
if not loc.loc_ghost then
@@ -870,6 +873,16 @@ let mkerror loc sub footnote txt =
870873
let errorf ?(loc = none) ?(sub = []) ?(footnote=Fun.const None) =
871874
Fmt.kdoc_printf (mkerror loc sub footnote)
872875

876+
let aligned_error_hint
877+
?(loc = none) ?(sub = []) ?(footnote=Fun.const None) fmt =
878+
Fmt.kdoc_printf (fun main hint ->
879+
match hint with
880+
| None -> mkerror loc sub footnote main
881+
| Some hint ->
882+
let main, hint = Misc.align_error_hint ~main ~hint in
883+
mkerror loc (mknoloc hint :: sub) footnote main
884+
) fmt
885+
873886
let error ?(loc = none) ?(sub = []) ?(footnote=Fun.const None) msg_str =
874887
mkerror loc sub footnote Fmt.Doc.(string msg_str empty)
875888

parsing/location.mli

+7
Original file line numberDiff line numberDiff line change
@@ -335,6 +335,13 @@ val error: ?loc:t -> ?sub:msg list -> ?footnote:delayed_msg -> string -> error
335335
val errorf: ?loc:t -> ?sub:msg list -> ?footnote:delayed_msg ->
336336
('a, Format_doc.formatter, unit, error) format4 -> 'a
337337

338+
val aligned_error_hint:
339+
?loc:t -> ?sub:msg list -> ?footnote:delayed_msg ->
340+
('a, Format_doc.formatter, unit, Format_doc.t option -> error) format4 -> 'a
341+
(** [aligned_error_hint ?loc ?sub ?footnote fmt ... aligned_hint] produces an
342+
error report where the potential [aligned_hint] message has been aligned
343+
with the main error message before being added to the list of submessages.*)
344+
338345
val error_of_printer: ?loc:t -> ?sub:msg list -> ?footnote:delayed_msg ->
339346
(Format_doc.formatter -> 'a -> unit) -> 'a -> error
340347

parsing/parse.ml

+13-7
Original file line numberDiff line numberDiff line change
@@ -119,11 +119,11 @@ let prepare_error err =
119119
| Unclosed(opening_loc, opening, closing_loc, closing) ->
120120
Location.errorf
121121
~loc:closing_loc
122+
"Syntax error: %a expected" Style.inline_code closing
122123
~sub:[
123124
Location.msg ~loc:opening_loc
124125
"This %a might be unmatched" Style.inline_code opening
125126
]
126-
"Syntax error: %a expected" Style.inline_code closing
127127

128128
| Expecting (loc, nonterm) ->
129129
Location.errorf ~loc "Syntax error: %a expected."
@@ -167,12 +167,18 @@ let prepare_error err =
167167
Location.errorf ~loc "Syntax error: invalid package type: %a" invalid ipt
168168
| Removed_string_set loc ->
169169
Location.errorf ~loc
170-
"Syntax error: strings are immutable, there is no assignment \
171-
syntax for them.\n\
172-
@{<hint>Hint@}: Mutable sequences of bytes are available in \
173-
the Bytes module.\n\
174-
@{<hint>Hint@}: Did you mean to use %a?"
175-
Style.inline_code "Bytes.set"
170+
"Syntax error: strings are immutable,@ there@ is@ no@ assignment@ \
171+
syntax@ for@ them."
172+
~sub:[
173+
Location.msg
174+
"@{<hint>Hint@}: Mutable sequences of bytes are available in \
175+
the %a module."
176+
Style.inline_code "Bytes";
177+
Location.msg
178+
"@{<hint>Hint@}: Did you mean to use %a?"
179+
Style.inline_code "Bytes.set"
180+
]
181+
176182
let () =
177183
Location.register_error_of_exn
178184
(function

testsuite/tests/formatting/errors_batch.reference

+2-2
Original file line numberDiff line numberDiff line change
@@ -6,5 +6,5 @@ File "hello.ml", line 20, characters 4-8:
66
File "hello.ml", line 20, characters 6-8:
77
Longer sub-messages that do not fit on the same line as the location get
88
indented.
9-
This second sub-message does not have a location;
10-
ghost locations of submessages are not printed.
9+
This second sub-message does not have a location;
10+
ghost locations of submessages are not printed.

testsuite/tests/generated-parse-errors/errors.compilers.reference

+2-1
Original file line numberDiff line numberDiff line change
@@ -1945,7 +1945,8 @@ File "implementation: TRUE DOT LBRACE UIDENT WITH", line 1, characters 7-8:
19451945
File "implementation: TRUE DOT LBRACE WITH", line 1, characters 9-13:
19461946
Error: Syntax error
19471947
File "implementation: TRUE DOT LBRACKET UIDENT RBRACKET LESSMINUS OBJECT END WHILE", line 1, characters 0-31:
1948-
Error: Syntax error: strings are immutable, there is no assignment syntax for them.
1948+
Error: Syntax error: strings are immutable, there is no assignment syntax for
1949+
them.
19491950
Hint: Mutable sequences of bytes are available in the Bytes module.
19501951
Hint: Did you mean to use Bytes.set?
19511952
File "implementation: TRUE DOT LBRACKET UIDENT RBRACKET LESSMINUS WITH", line 1, characters 21-25:

testsuite/tests/let-syntax/let_syntax.ml

+1-1
Original file line numberDiff line numberDiff line change
@@ -217,7 +217,7 @@ Line 3, characters 13-14:
217217
^
218218
Error: The constant "1" has type "int" but an expression was expected of type
219219
"float"
220-
Hint: Did you mean "1."?
220+
Hint: Did you mean "1."?
221221
|}];;
222222

223223
module Ill_typed_3 = struct

testsuite/tests/tool-toplevel/error_highlighting.compilers.reference

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ Line 3, characters 8-9:
2424
^
2525
Error: The constant 1 has type int but an expression was expected of type
2626
float
27-
Hint: Did you mean 1.?
27+
Hint: Did you mean 1.?
2828
Line 4, characters 2-4:
2929
4 | 2 in
3030
^^

testsuite/tests/typing-core-bugs/const_int_hint.ml

+16-16
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ Line 1, characters 19-20:
99
^
1010
Error: The constant "1" has type "int" but an expression was expected of type
1111
"int32"
12-
Hint: Did you mean "1l"?
12+
Hint: Did you mean "1l"?
1313
|}]
1414

1515
let _ : int32 * int32 = 42l, 43;;
@@ -19,7 +19,7 @@ Line 1, characters 29-31:
1919
^^
2020
Error: The constant "43" has type "int" but an expression was expected of type
2121
"int32"
22-
Hint: Did you mean "43l"?
22+
Hint: Did you mean "43l"?
2323
|}]
2424

2525
let _ : int32 * nativeint = 42l, 43;;
@@ -29,7 +29,7 @@ Line 1, characters 33-35:
2929
^^
3030
Error: The constant "43" has type "int" but an expression was expected of type
3131
"nativeint"
32-
Hint: Did you mean "43n"?
32+
Hint: Did you mean "43n"?
3333
|}]
3434

3535
let _ = min 6L 7;;
@@ -39,7 +39,7 @@ Line 1, characters 15-16:
3939
^
4040
Error: The constant "7" has type "int" but an expression was expected of type
4141
"int64"
42-
Hint: Did you mean "7L"?
42+
Hint: Did you mean "7L"?
4343
|}]
4444

4545
let _ : float = 123;;
@@ -49,7 +49,7 @@ Line 1, characters 16-19:
4949
^^^
5050
Error: The constant "123" has type "int" but an expression was expected of type
5151
"float"
52-
Hint: Did you mean "123."?
52+
Hint: Did you mean "123."?
5353
|}]
5454

5555
(* no hint *)
@@ -73,7 +73,7 @@ Line 2, characters 4-5:
7373
^
7474
Error: This pattern matches values of type "int"
7575
but a pattern was expected which matches values of type "int32"
76-
Hint: Did you mean "0l"?
76+
Hint: Did you mean "0l"?
7777
|}]
7878

7979
let _ : int64 -> int64 = function
@@ -85,7 +85,7 @@ Line 2, characters 9-10:
8585
^
8686
Error: This pattern matches values of type "int"
8787
but a pattern was expected which matches values of type "int64"
88-
Hint: Did you mean "2L"?
88+
Hint: Did you mean "2L"?
8989
|}]
9090

9191
(* symmetric *)
@@ -96,7 +96,7 @@ Line 1, characters 16-18:
9696
^^
9797
Error: The constant "1L" has type "int64" but an expression was expected of type
9898
"int32"
99-
Hint: Did you mean "1l"?
99+
Hint: Did you mean "1l"?
100100
|}]
101101
let _ : float = 1L;;
102102
[%%expect{|
@@ -105,7 +105,7 @@ Line 1, characters 16-18:
105105
^^
106106
Error: The constant "1L" has type "int64" but an expression was expected of type
107107
"float"
108-
Hint: Did you mean "1."?
108+
Hint: Did you mean "1."?
109109
|}]
110110
let _ : int64 = 1n;;
111111
[%%expect{|
@@ -114,7 +114,7 @@ Line 1, characters 16-18:
114114
^^
115115
Error: The constant "1n" has type "nativeint"
116116
but an expression was expected of type "int64"
117-
Hint: Did you mean "1L"?
117+
Hint: Did you mean "1L"?
118118
|}]
119119
let _ : nativeint = 1l;;
120120
[%%expect{|
@@ -123,7 +123,7 @@ Line 1, characters 20-22:
123123
^^
124124
Error: The constant "1l" has type "int32" but an expression was expected of type
125125
"nativeint"
126-
Hint: Did you mean "1n"?
126+
Hint: Did you mean "1n"?
127127
|}]
128128

129129
(* not implemented *)
@@ -153,7 +153,7 @@ Line 1, characters 23-28:
153153
^^^^^
154154
Error: The constant "1_000" has type "int" but an expression was expected of type
155155
"int64"
156-
Hint: Did you mean "1_000L"?
156+
Hint: Did you mean "1_000L"?
157157
|}]
158158
let _ : nativeint * nativeint = 0n, 0xAA_BBL;;
159159
[%%expect{|
@@ -162,7 +162,7 @@ Line 1, characters 36-44:
162162
^^^^^^^^
163163
Error: The constant "0xAA_BBL" has type "int64"
164164
but an expression was expected of type "nativeint"
165-
Hint: Did you mean "0xAA_BBn"?
165+
Hint: Did you mean "0xAA_BBn"?
166166
|}]
167167
let _ : int32 -> int32 = function
168168
| 1l | 0o2_345 -> 3l
@@ -173,7 +173,7 @@ Line 2, characters 9-16:
173173
^^^^^^^
174174
Error: This pattern matches values of type "int"
175175
but a pattern was expected which matches values of type "int32"
176-
Hint: Did you mean "0o2_345l"?
176+
Hint: Did you mean "0o2_345l"?
177177
|}]
178178
let _ : int32 -> int32 = fun x -> match x with
179179
| 1l | 0b1000_1101 -> 3l
@@ -184,7 +184,7 @@ Line 2, characters 9-20:
184184
^^^^^^^^^^^
185185
Error: This pattern matches values of type "int"
186186
but a pattern was expected which matches values of type "int32"
187-
Hint: Did you mean "0b1000_1101l"?
187+
Hint: Did you mean "0b1000_1101l"?
188188
|}]
189189
type t1 = {f1: int32};; let _ = fun x -> x.f1 <- 1_000n;;
190190
[%%expect{|
@@ -194,5 +194,5 @@ Line 1, characters 49-55:
194194
^^^^^^
195195
Error: The constant "1_000n" has type "nativeint"
196196
but an expression was expected of type "int32"
197-
Hint: Did you mean "1_000l"?
197+
Hint: Did you mean "1_000l"?
198198
|}]

testsuite/tests/typing-core-bugs/type_expected_explanation.ml

+2-2
Original file line numberDiff line numberDiff line change
@@ -168,7 +168,7 @@ Line 3, characters 22-26:
168168
3 | else if x > y then [y;x]
169169
^^^^
170170
Error: This variant expression is expected to have type "unit"
171-
because it is in the result of a conditional with no else branch
171+
because it is in the result of a conditional with no else branch
172172
There is no constructor "::" within type "unit"
173173
|}];;
174174

@@ -192,6 +192,6 @@ Line 1, characters 35-39:
192192
1 | if false then (match () with () -> true);;
193193
^^^^
194194
Error: This variant expression is expected to have type "unit"
195-
because it is in the result of a conditional with no else branch
195+
because it is in the result of a conditional with no else branch
196196
There is no constructor "true" within type "unit"
197197
|}]

testsuite/tests/typing-extensions/iarray.ml

+1-1
Original file line numberDiff line numberDiff line change
@@ -58,7 +58,7 @@ Line 3, characters 4-5:
5858
^
5959
Error: This pattern matches values of type "int"
6060
but a pattern was expected which matches values of type "float"
61-
Hint: Did you mean "1."?
61+
Hint: Did you mean "1."?
6262
|}];;
6363

6464
match marray with

testsuite/tests/typing-gadts/syntactic-arity.ml

+1-2
Original file line numberDiff line numberDiff line change
@@ -33,8 +33,7 @@ Line 1, characters 14-21:
3333
^^^^^^^
3434
Error: This expression has type "'a -> 'b"
3535
but an expression was expected of type "string"
36-
Hint: This function application is partial, maybe some arguments
37-
are missing.
36+
Hint: This function application is partial, maybe some arguments are missing.
3837
|}];;
3938

4039
(* And the fully polymorphic definition is rejected. *)

0 commit comments

Comments
 (0)