Skip to main content

vtcode_commons/
ansi.rs

1//! Shared ANSI escape parser and stripping utilities for VT Code.
2//!
3//! See `docs/reference/ansi-in-vtcode.md` for the workspace usage map.
4
5use crate::ansi_codes::{BEL_BYTE, ESC_BYTE};
6use memchr::memchr;
7
8const ESC: u8 = ESC_BYTE;
9const BEL: u8 = BEL_BYTE;
10const DEL: u8 = 0x7f;
11const C1_ST: u8 = 0x9c;
12const C1_DCS: u8 = 0x90;
13const C1_SOS: u8 = 0x98;
14const C1_CSI: u8 = 0x9b;
15const C1_OSC: u8 = 0x9d;
16const C1_PM: u8 = 0x9e;
17const C1_APC: u8 = 0x9f;
18const CAN: u8 = 0x18;
19const SUB: u8 = 0x1a;
20const MAX_STRING_SEQUENCE_BYTES: usize = 4096;
21const MAX_CSI_SEQUENCE_BYTES: usize = 64;
22
23#[derive(Clone, Copy)]
24enum StringSequenceTerminator {
25    StOnly,
26    BelOrSt,
27}
28
29impl StringSequenceTerminator {
30    #[inline]
31    const fn allows_bel(self) -> bool {
32        matches!(self, Self::BelOrSt)
33    }
34}
35
36#[inline]
37fn parse_c1_at(bytes: &[u8], start: usize) -> Option<(u8, usize)> {
38    let first = *bytes.get(start)?;
39    if (0x80..=0x9f).contains(&first) {
40        return Some((first, 1));
41    }
42    None
43}
44
45#[inline]
46fn parse_csi(bytes: &[u8], start: usize) -> Option<usize> {
47    // ECMA-48 / ISO 6429 CSI grammar:
48    // - parameter bytes: 0x30..0x3F
49    // - intermediate bytes: 0x20..0x2F
50    // - final byte: 0x40..0x7E
51    // (See ANSI escape code article on Wikipedia, CSI section.)
52    let mut index = start;
53    let mut phase = 0u8; // 0=parameter, 1=intermediate
54    let mut consumed = 0usize;
55
56    while index < bytes.len() {
57        let byte = bytes[index];
58        if byte == ESC {
59            // VT100: ESC aborts current control sequence and starts a new one.
60            return Some(index);
61        }
62        if byte == CAN || byte == SUB {
63            // VT100: CAN/SUB abort current control sequence.
64            return Some(index + 1);
65        }
66
67        consumed += 1;
68        if consumed > MAX_CSI_SEQUENCE_BYTES {
69            // Bound malformed or hostile input.
70            return Some(index + 1);
71        }
72
73        if phase == 0 && (0x30..=0x3f).contains(&byte) {
74            index += 1;
75            continue;
76        }
77        if (0x20..=0x2f).contains(&byte) {
78            phase = 1;
79            index += 1;
80            continue;
81        }
82        if (0x40..=0x7e).contains(&byte) {
83            return Some(index + 1);
84        }
85
86        // Invalid CSI byte: abort sequence without consuming this byte.
87        return Some(index);
88    }
89
90    None
91}
92
93#[inline]
94fn parse_string_sequence(
95    bytes: &[u8],
96    start: usize,
97    terminator: StringSequenceTerminator,
98) -> Option<usize> {
99    let mut consumed = 0usize;
100    for index in start..bytes.len() {
101        if bytes[index] == ESC && !(index + 1 < bytes.len() && bytes[index + 1] == b'\\') {
102            // VT100: ESC aborts current sequence and begins a new one.
103            return Some(index);
104        }
105        if bytes[index] == CAN || bytes[index] == SUB {
106            return Some(index + 1);
107        }
108
109        if let Some((c1, len)) = parse_c1_at(bytes, index)
110            && c1 == C1_ST
111        {
112            return Some(index + len);
113        }
114
115        match bytes[index] {
116            BEL if terminator.allows_bel() => return Some(index + 1),
117            ESC if index + 1 < bytes.len() && bytes[index + 1] == b'\\' => return Some(index + 2),
118            _ => {}
119        }
120
121        consumed += 1;
122        if consumed > MAX_STRING_SEQUENCE_BYTES {
123            // Cap unbounded strings when terminator is missing.
124            return Some(index + 1);
125        }
126    }
127    None
128}
129
130#[inline]
131fn push_visible_byte(output: &mut Vec<u8>, byte: u8) {
132    if matches!(byte, b'\n' | b'\r' | b'\t') || !(byte < 32 || byte == DEL) {
133        output.push(byte);
134    }
135}
136
137#[inline]
138fn parse_ansi_sequence_bytes(bytes: &[u8]) -> Option<usize> {
139    if bytes.is_empty() {
140        return None;
141    }
142
143    if let Some((c1, c1_len)) = parse_c1_at(bytes, 0) {
144        return match c1 {
145            C1_CSI => parse_csi(bytes, c1_len),
146            C1_OSC => parse_string_sequence(bytes, c1_len, StringSequenceTerminator::BelOrSt),
147            C1_DCS | C1_SOS | C1_PM | C1_APC => {
148                parse_string_sequence(bytes, c1_len, StringSequenceTerminator::StOnly)
149            }
150            _ => Some(c1_len),
151        };
152    }
153
154    match bytes[0] {
155        ESC => {
156            if bytes.len() < 2 {
157                return None;
158            }
159
160            match bytes[1] {
161                b'[' => parse_csi(bytes, 2),
162                b']' => parse_string_sequence(bytes, 2, StringSequenceTerminator::BelOrSt),
163                b'P' | b'^' | b'_' | b'X' => {
164                    parse_string_sequence(bytes, 2, StringSequenceTerminator::StOnly)
165                }
166                // Three-byte sequences: ESC + intermediate + final
167                // ESC SP {F,G,L,M,N} — 7/8-bit controls, ANSI conformance
168                // ESC # {3,4,5,6,8} — DEC line attributes / screen alignment
169                // ESC % {@ ,G} — character set selection (ISO 2022)
170                // ESC ( C / ESC ) C / ESC * C / ESC + C — G0-G3 designation
171                b' ' | b'#' | b'%' | b'(' | b')' | b'*' | b'+' => {
172                    if bytes.len() > 2 {
173                        Some(3)
174                    } else {
175                        None
176                    }
177                }
178                next if next < 128 => Some(2),
179                _ => Some(1),
180            }
181        }
182        _ => None,
183    }
184}
185
186/// Strip ANSI escape codes from text, keeping only plain text
187pub fn strip_ansi(text: &str) -> String {
188    let mut output = Vec::with_capacity(text.len());
189    let bytes = text.as_bytes();
190    let mut i = 0;
191
192    while i < bytes.len() {
193        let next_esc = memchr(ESC, &bytes[i..]).map_or(bytes.len(), |offset| i + offset);
194        // Pre-slice to avoid bounds checks in the inner loop — the range
195        // i..next_esc is provably within bytes[..].
196        for &b in &bytes[i..next_esc] {
197            push_visible_byte(&mut output, b);
198        }
199        i = next_esc;
200
201        if i >= bytes.len() {
202            break;
203        }
204
205        if let Some(len) = parse_ansi_sequence_bytes(&bytes[i..]) {
206            i += len;
207            continue;
208        } else {
209            // Incomplete/unterminated control sequence at end of available text.
210            break;
211        }
212    }
213
214    String::from_utf8_lossy(&output).into_owned()
215}
216
217/// Strip ANSI escape codes from arbitrary bytes, preserving non-control bytes.
218///
219/// This is the preferred API when input may contain raw C1 (8-bit) controls.
220pub fn strip_ansi_bytes(input: &[u8]) -> Vec<u8> {
221    let mut output = Vec::with_capacity(input.len());
222    let bytes = input;
223    let mut i = 0;
224
225    while i < bytes.len() {
226        // Pre-slice to the remaining portion so all indexing below shares one bounds edge.
227        let rest = &bytes[i..];
228
229        if (rest[0] == ESC || parse_c1_at(bytes, i).is_some())
230            && let Some(len) = parse_ansi_sequence_bytes(rest)
231        {
232            i += len;
233            continue;
234        }
235        if rest[0] == ESC || parse_c1_at(bytes, i).is_some() {
236            // Incomplete/unterminated control sequence at end of available text.
237            break;
238        }
239
240        push_visible_byte(&mut output, rest[0]);
241        i += 1;
242    }
243    output
244}
245
246/// Parse and determine the length of the ANSI escape sequence at the start of text
247pub fn parse_ansi_sequence(text: &str) -> Option<usize> {
248    let bytes = text.as_bytes();
249    parse_ansi_sequence_bytes(bytes)
250}
251
252/// Fast ASCII-only ANSI stripping for performance-critical paths
253pub fn strip_ansi_ascii_only(text: &str) -> String {
254    let mut output = String::with_capacity(text.len());
255    let bytes = text.as_bytes();
256    let mut search_start = 0;
257    let mut copy_start = 0;
258
259    while let Some(offset) = memchr(ESC, &bytes[search_start..]) {
260        let esc_index = search_start + offset;
261        if let Some(len) = parse_ansi_sequence_bytes(&bytes[esc_index..]) {
262            if copy_start < esc_index {
263                output.push_str(&text[copy_start..esc_index]);
264            }
265            copy_start = esc_index + len;
266            search_start = copy_start;
267        } else {
268            search_start = esc_index + 1;
269        }
270    }
271
272    if copy_start < text.len() {
273        output.push_str(&text[copy_start..]);
274    }
275
276    output
277}
278
279/// Detect if text contains unicode characters that need special handling
280#[must_use]
281pub fn contains_unicode(text: &str) -> bool {
282    text.bytes().any(|b| b >= 0x80)
283}
284
285#[cfg(test)]
286mod tests {
287    use super::{CAN, SUB, strip_ansi, strip_ansi_ascii_only};
288
289    #[test]
290    fn strips_esc_csi_sequences() {
291        let input = "a\x1b[31mred\x1b[0mz";
292        assert_eq!(strip_ansi(input), "aredz");
293        assert_eq!(strip_ansi_ascii_only(input), "aredz");
294    }
295
296    #[test]
297    fn utf8_encoded_c1_is_not_reprocessed_as_control() {
298        // XTerm/ECMA-48: controls are processed once; decoded UTF-8 text is not reprocessed as C1.
299        let input = "a\u{009b}31mred";
300        assert_eq!(strip_ansi(input), input);
301    }
302
303    #[test]
304    fn strip_removes_ascii_del_control() {
305        let input = format!("a{}b", char::from(0x7f));
306        assert_eq!(strip_ansi(&input), "ab");
307    }
308
309    #[test]
310    fn csi_aborts_on_esc_then_new_sequence_parses() {
311        let input = "a\x1b[31\x1b[32mgreen\x1b[0mz";
312        assert_eq!(strip_ansi(input), "agreenz");
313    }
314
315    #[test]
316    fn csi_aborts_on_can_and_sub() {
317        let can = format!("a\x1b[31{}b", char::from(CAN));
318        let sub = format!("a\x1b[31{}b", char::from(SUB));
319        assert_eq!(strip_ansi(&can), "ab");
320        assert_eq!(strip_ansi(&sub), "ab");
321    }
322
323    #[test]
324    fn osc_aborts_on_esc_non_st() {
325        let input = "a\x1b]title\x1b[31mred\x1b[0mz";
326        assert_eq!(strip_ansi(input), "aredz");
327    }
328
329    #[test]
330    fn incomplete_sequence_drops_tail() {
331        let input = "text\x1b[31";
332        assert_eq!(strip_ansi(input), "text");
333    }
334
335    #[test]
336    fn ascii_only_incomplete_sequence_keeps_tail() {
337        let input = "text\x1b[31";
338        assert_eq!(strip_ansi_ascii_only(input), input);
339    }
340
341    #[test]
342    fn strips_common_progress_redraw_sequences() {
343        // Common pattern for dynamic CLI updates:
344        // carriage return + erase line + redraw text.
345        let input = "\r\x1b[2KProgress 10%\r\x1b[2KDone\n";
346        assert_eq!(strip_ansi(input), "\rProgress 10%\rDone\n");
347    }
348
349    #[test]
350    fn strips_cursor_navigation_sequences() {
351        let input = "left\x1b[1D!\nup\x1b[1Arow";
352        assert_eq!(strip_ansi(input), "left!\nuprow");
353    }
354
355    #[test]
356    fn strip_ansi_bytes_supports_raw_c1_csi() {
357        let input = [
358            b'a', 0x9b, b'3', b'1', b'm', b'r', b'e', b'd', 0x9b, b'0', b'm', b'z',
359        ];
360        let out = super::strip_ansi_bytes(&input);
361        assert_eq!(out, b"aredz");
362    }
363
364    #[test]
365    fn strip_ansi_bytes_supports_raw_c1_osc_and_st() {
366        let mut input = b"pre".to_vec();
367        input.extend_from_slice(&[0x9d]);
368        input.extend_from_slice(b"8;;https://example.com");
369        input.extend_from_slice(&[0x9c]);
370        input.extend_from_slice(b"link");
371        input.extend_from_slice(&[0x9d]);
372        input.extend_from_slice(b"8;;");
373        input.extend_from_slice(&[0x9c]);
374        input.extend_from_slice(b"post");
375        let out = super::strip_ansi_bytes(&input);
376        assert_eq!(out, b"prelinkpost");
377    }
378
379    #[test]
380    fn csi_respects_parameter_intermediate_final_grammar() {
381        // Parameter bytes ("1;2"), intermediate bytes (" "), then final ("m")
382        let input = "a\x1b[1;2 mred\x1b[0mz";
383        assert_eq!(strip_ansi(input), "aredz");
384    }
385
386    #[test]
387    fn malformed_csi_does_not_consume_following_text() {
388        // 0x10 is not valid CSI parameter/intermediate/final.
389        let malformed = format!("a\x1b[12{}visible", char::from(0x10));
390        assert_eq!(strip_ansi(&malformed), "avisible");
391    }
392
393    #[test]
394    fn strips_wikipedia_sgr_8bit_color_pattern() {
395        let input = "x\x1b[38;5;196mred\x1b[0my";
396        assert_eq!(strip_ansi(input), "xredy");
397    }
398
399    #[test]
400    fn strips_wikipedia_sgr_truecolor_pattern() {
401        let input = "x\x1b[48;2;12;34;56mblock\x1b[0my";
402        assert_eq!(strip_ansi(input), "xblocky");
403    }
404
405    #[test]
406    fn strips_wikipedia_osc8_hyperlink_pattern() {
407        let input = "go \x1b]8;;https://example.com\x1b\\here\x1b]8;;\x1b\\ now";
408        assert_eq!(strip_ansi(input), "go here now");
409    }
410
411    #[test]
412    fn strips_dec_private_mode_csi() {
413        let input = "a\x1b[?25lb\x1b[?25hc";
414        assert_eq!(strip_ansi(input), "abc");
415    }
416
417    #[test]
418    fn strips_three_byte_esc_sequences() {
419        // ESC # 8 = DEC screen alignment test
420        let input = "a\x1b#8b";
421        assert_eq!(strip_ansi(input), "ab");
422
423        // ESC ( B = designate US ASCII as G0
424        let input2 = "a\x1b(Bb";
425        assert_eq!(strip_ansi(input2), "ab");
426
427        // ESC SP F = 7-bit controls
428        let input3 = "a\x1b Fb";
429        assert_eq!(strip_ansi(input3), "ab");
430
431        // ESC % G = select UTF-8
432        let input4 = "a\x1b%Gb";
433        assert_eq!(strip_ansi(input4), "ab");
434    }
435
436    #[test]
437    fn incomplete_three_byte_esc_sequence_drops_tail() {
438        // ESC # at end — incomplete, should not consume past end
439        let input = "text\x1b#";
440        assert_eq!(strip_ansi(input), "text");
441    }
442}