diff --git a/CHANGES.md b/CHANGES.md index f972e9c8b0c3..82de6aa360c0 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -82,6 +82,9 @@ - Added an (aliases ...) field to the (rules ...) stanza which allows the specification of multiple aliases per rule (#6194, @Alizter) +- Handle CSI n K code in ANSI escape codes from commands. (#...., fixes #5528, + @emillon) + 3.4.1 (26-07-2022) ------------------ diff --git a/otherlibs/stdune/ansi_color.ml b/otherlibs/stdune/ansi_color.ml index 9a4842d56e64..621915fe0262 100644 --- a/otherlibs/stdune/ansi_color.ml +++ b/otherlibs/stdune/ansi_color.ml @@ -179,6 +179,17 @@ let strip str = in loop 0 +let index_from_any str start chars = + let n = String.length str in + let rec go i = + if i >= n then None + else + match List.find chars ~f:(fun c -> Char.equal str.[i] c) with + | None -> go (i + 1) + | Some c -> Some (i, c) + in + go start + let parse_line str styles = let len = String.length str in let add_chunk acc ~styles ~pos ~len = @@ -201,9 +212,9 @@ let parse_line str styles = let seq_start = seq_start + 2 in if seq_start >= len || str.[seq_start - 1] <> '[' then (styles, acc) else - match String.index_from str seq_start 'm' with + match index_from_any str seq_start [ 'm'; 'K' ] with | None -> (styles, acc) - | Some seq_end -> + | Some (seq_end, 'm') -> let styles = if seq_start = seq_end then (* Some commands output "\027[m", which seems to be interpreted @@ -223,7 +234,8 @@ let parse_line str styles = else s :: styles) |> List.rev in - loop styles (seq_end + 1) acc) + loop styles (seq_end + 1) acc + | Some (seq_end, _) -> loop styles (seq_end + 1) acc) in loop styles 0 Pp.nop diff --git a/test/blackbox-tests/test-cases/github5528.t b/test/blackbox-tests/test-cases/github5528.t new file mode 100644 index 000000000000..0b7eb8f30adc --- /dev/null +++ b/test/blackbox-tests/test-cases/github5528.t @@ -0,0 +1,37 @@ + $ cat > dune-project < (lang dune 1.0) + > EOF + + $ cat > dune < (test + > (name t)) + > EOF + + $ cat > t.ml < type color = Normal | Cyan + > + > let int_of_color = function + > | Normal -> 0 + > | Cyan -> 6 + > + > let in_color c pp out x = + > let n = int_of_color c in + > Printf.fprintf out "\x1b[3%dm" n; + > pp out x; + > Printf.fprintf out "\x1b[0m" + > + > let reset_line = "\x1b[2K\r" + > + > let () = + > Printf.printf "%sVery Secret!\n%!" reset_line; + > Printf.printf "%s\n%!" (String.make 15 '-'); + > Printf.printf "%a\n%!" (in_color Cyan output_string) "Can you see it?" + > EOF + + $ dune runtest -f + Very Secret! + --------------- + Can you see it? + + $ dune exec ./t.exe + Can you see it?